Files
M5Unit-HEART/examples/UnitUnified/PlotToSerial/main/PlotToSerial.cpp
T

220 lines
7.2 KiB
C++

/*
* SPDX-FileCopyrightText: 2026 M5Stack Technology CO LTD
*
* SPDX-License-Identifier: MIT
*/
/*
Example using M5UnitUnified for UnitHeart / HatHeart
*/
#include <M5Unified.h>
#include <M5UnitUnified.h>
#include <M5UnitUnifiedHEART.h>
#include <Wire.h>
#include <M5HAL.hpp> // For NessoN1
// *************************************************************
// Choose one define symbol to match the unit you are using
// *************************************************************
#if !defined(USING_UNIT_HEART) && !defined(USING_HAT_HEART)
// #define USING_UNIT_HEART
// #define USING_HAT_HEART
#endif
#if defined(USING_UNIT_HEART)
#pragma message "Using UnitHeart"
using namespace m5::unit::max30100;
#elif defined(USING_HAT_HEART)
#pragma message "Using HatHeart"
using namespace m5::unit::max30102;
#else
#error Please choose unit!
#endif
namespace {
auto& lcd = M5.Display;
m5::unit::UnitUnified Units;
#if defined(USING_UNIT_HEART)
m5::unit::UnitHeart unit;
#elif defined(USING_HAT_HEART)
m5::unit::HatHeart unit;
#endif
m5::heart::PulseMonitor monitor;
#if defined(USING_HAT_HEART)
constexpr bool using_multi_led_mode{false}; // Using multiLED mode if true
struct I2cPins {
int sda;
int scl;
};
I2cPins get_hat_i2c_pins(const m5::board_t board)
{
switch (board) {
case m5::board_t::board_M5StickC:
case m5::board_t::board_M5StickCPlus:
case m5::board_t::board_M5StickCPlus2:
return {0, 26};
case m5::board_t::board_M5StickS3:
return {8, 0};
case m5::board_t::board_M5StackCoreInk:
return {25, 26};
case m5::board_t::board_ArduinoNessoN1:
return {6, 7};
default:
return {-1, -1};
}
}
#endif
} // namespace
void setup()
{
auto m5cfg = M5.config();
#if defined(USING_HAT_HEART)
m5cfg.pmic_button = false; // Disable BtnPWR
m5cfg.internal_imu = false; // Disable internal IMU
m5cfg.internal_rtc = false; // Disable internal RTC
#endif
M5.begin(m5cfg);
M5.setTouchButtonHeightByRatio(100);
// The screen shall be in landscape mode
if (lcd.height() > lcd.width()) {
lcd.setRotation(1);
}
auto board = M5.getBoard();
#if defined(USING_HAT_HEART)
const auto pins = get_hat_i2c_pins(board);
M5_LOGI("getHatPin: SDA:%u SCL:%u", pins.sda, pins.scl);
if (pins.sda < 0 || pins.scl < 0) {
M5_LOGE("Illegal pin number");
lcd.fillScreen(TFT_RED);
while (true) {
m5::utility::delay(10000);
}
}
// Setup required to use HatHEART
pinMode(pins.scl, OUTPUT);
// Using MultiLED mode
// In MultiLED mode, you need to set and start them yourself
if (using_multi_led_mode) {
auto cfg = unit.config();
cfg.start_periodic = false; // Ignore auto start
unit.config(cfg);
}
auto& wire = (board == m5::board_t::board_ArduinoNessoN1) ? Wire1 : Wire;
wire.end();
wire.begin(pins.sda, pins.scl, 400 * 1000U);
if (!Units.add(unit, wire) || !Units.begin()) {
M5_LOGE("Failed to begin");
lcd.fillScreen(TFT_RED);
while (true) {
m5::utility::delay(10000);
}
}
#else
// NessoN1: Arduino Wire (I2C_NUM_0) cannot be used for GROVE port.
// Wire is used by M5Unified In_I2C for internal devices (IOExpander etc.).
// Wire1 exists but is reserved for HatPort — cannot be used for GROVE.
// Reconfiguring Wire to GROVE pins breaks In_I2C, causing ESP_ERR_INVALID_STATE in M5.update().
// Solution: Use SoftwareI2C via M5HAL (bit-banging) for the GROVE port.
// NanoC6: Wire.begin() on GROVE pins conflicts with m5::I2C_Class registered by Ex_I2C.setPort()
// on the same I2C_NUM_0, causing sporadic NACK errors.
// Solution: Use M5.Ex_I2C (m5::I2C_Class) directly instead of Arduino Wire.
bool unit_ready{};
if (board == m5::board_t::board_ArduinoNessoN1) {
// NessoN1: GROVE is on port_b (GPIO 5/4), not port_a (which maps to Wire pins 8/10)
auto pin_num_sda = M5.getPin(m5::pin_name_t::port_b_out);
auto pin_num_scl = M5.getPin(m5::pin_name_t::port_b_in);
M5_LOGI("getPin(M5HAL): SDA:%u SCL:%u", pin_num_sda, pin_num_scl);
m5::hal::bus::I2CBusConfig i2c_cfg;
i2c_cfg.pin_sda = m5::hal::gpio::getPin(pin_num_sda);
i2c_cfg.pin_scl = m5::hal::gpio::getPin(pin_num_scl);
auto i2c_bus = m5::hal::bus::i2c::getBus(i2c_cfg);
M5_LOGI("Bus:%d", i2c_bus.has_value());
unit_ready = Units.add(unit, i2c_bus ? i2c_bus.value() : nullptr) && Units.begin();
} else if (board == m5::board_t::board_M5NanoC6) {
// NanoC6: Use M5.Ex_I2C (m5::I2C_Class, not Arduino Wire)
M5_LOGI("Using M5.Ex_I2C");
unit_ready = Units.add(unit, M5.Ex_I2C) && Units.begin();
} else {
auto pin_num_sda = M5.getPin(m5::pin_name_t::port_a_sda);
auto pin_num_scl = M5.getPin(m5::pin_name_t::port_a_scl);
M5_LOGI("getPin: SDA:%u SCL:%u", pin_num_sda, pin_num_scl);
Wire.end();
Wire.begin(pin_num_sda, pin_num_scl, 400 * 1000U);
unit_ready = Units.add(unit, Wire) && Units.begin();
}
if (!unit_ready) {
M5_LOGE("Failed to begin");
lcd.fillScreen(TFT_RED);
while (true) {
m5::utility::delay(10000);
}
}
#endif
M5_LOGI("M5UnitUnified has been begun");
M5_LOGI("%s", Units.debugInfo().c_str());
#if defined(USING_HAT_HEART)
// In MultiLED mode, you need to set and start them yourself
if (using_multi_led_mode) {
M5_LOGI("MultiLED mode");
unit.writeMode(Mode::MultiLED);
unit.writeSpO2Configuration(ADC::Range4096nA, Sampling::Rate400, LEDPulse::Width411);
unit.writeFIFOConfiguration(FIFOSampling::Average4, true, 15);
// unit.writeMultiLEDModeControl(Slot::Red, Slot::IR); // (A)
unit.writeMultiLEDModeControl(Slot::IR, Slot::Red); // (B)
unit.writeLEDCurrent(0, 0x1F); // Red if (A), IR if (B)
unit.writeLEDCurrent(1, 0x1F); // IR if (A), Red if (B)
unit.startPeriodicMeasurement();
}
#endif
monitor.setSamplingRate(unit.calculateSamplingRate());
lcd.fillScreen(TFT_DARKGREEN);
}
void loop()
{
M5.update();
Units.update();
if (unit.updated()) {
// WARNING
// If overflow is occurring, the sampling rate should be reduced because the processing is not up to par
if (unit.overflow()) {
M5_LOGW("OVERFLOW:%u", unit.overflow());
}
bool beat{};
// MAX30100/02 is equipped with a FIFO, so multiple data may be stored
while (unit.available()) {
M5.Log.printf(">IR:%u\n>RED:%u\n", unit.ir(), unit.red());
monitor.push_back(unit.ir(), unit.red()); // Push back the oldest data
M5.Log.printf(">MIR:%f\n", monitor.latestIR());
monitor.update();
beat |= monitor.isBeat();
unit.discard(); // Discard the oldest data
}
M5.Log.printf(">BPM:%f\n>SpO2:%f\n>BEAT:%u\n", monitor.bpm(), monitor.SpO2(), beat);
}
// Measure temperature
if (M5.BtnA.wasClicked()) {
TemperatureData td{};
if (unit.measureTemperatureSingleshot(td)) {
M5.Log.printf(">Temp:%f\n", td.celsius());
}
}
}