/* * SPDX-FileCopyrightText: 2025 M5Stack Technology CO LTD * * SPDX-License-Identifier: MIT */ /* Example using M5UnitUnified for UnitThermal2 */ #include #include #include #include #include using namespace m5::unit::thermal2; namespace { auto& lcd = M5.Display; m5::unit::UnitUnified Units; m5::unit::UnitThermal2 unit; void ring_buzzer(const uint16_t freq, const uint8_t duty, const uint16_t count = 1, const uint32_t ms = 100, const uint32_t interval = 50) { unit.writeBuzzerControl(false); for (uint16_t i = 0; i < count; ++i) { unit.writeBuzzer(freq, duty); unit.writeBuzzerControl(true); m5::utility::delay(ms); unit.writeBuzzerControl(false); if (i + 1 != count) { m5::utility::delay(interval); } } unit.writeBuzzerControl(false); } constexpr float low_alarm_temp{10.0f}; constexpr float high_alarm_temp{30.0f}; void dump(const Data& d) { M5.Log.printf("Subpage:%u\n", d.subpage); M5.Log.printf("Med:%.2f Avg:%.2f High:%.2f Low:%.2f\n", d.medianTemperature(), d.averageTemperature(), d.highestTemperature(), d.lowestTemperature()); M5.Log.printf("MostDiff(%u,%u) lowest{%u,%u} highest<%u,%u>\n", d.most_diff_x, d.most_diff_y, d.lowest_diff_x, d.lowest_diff_y, d.highest_diff_x, d.highest_diff_y); M5.Log.printf( " 0000 0001 0002 0003 0004 0005 0006 0007 0008 0009 0010 0011 0012 0013 0014 " "0015 " "0016 0017 0018 0019 0020 0021 0022 0023 0024 0025 0026 0027 0028 0029 0030 " "0031\n"); M5.Log.printf( "--------------------------------------------------------------------------------------------------------------" "------" "-------------------------------------------------------------------------------------------------------------" "\n"); for (int y = 0; y < 24; ++y) { M5.Log.printf("%02d:%s", y, ((y & 1) != d.subpage) ? " " : ""); for (int x = 0; x < 16; ++x) { auto val = d.raw[y * 16 + x]; int xx = x * 2 + ((y & 1) != d.subpage); if (xx == d.most_diff_x && y == d.most_diff_y) { M5.Log.printf("(%04X) ", val); } else if (xx == d.lowest_diff_x && y == d.lowest_diff_y) { M5.Log.printf("{%04X} ", val); } else if (xx == d.highest_diff_x && y == d.highest_diff_y) { M5.Log.printf("<%04X> ", val); } else { M5.Log.printf(" %04X ", val); } if (x != 15) { M5.Log.printf(" "); } } M5.Log.printf("\n"); } } } // namespace void setup() { M5.begin(); M5.setTouchButtonHeightByRatio(100); // The screen shall be in landscape mode if (lcd.height() > lcd.width()) { lcd.setRotation(1); } auto board = M5.getBoard(); auto cfg = unit.config(); cfg.rate = Refresh::Rate0_5Hz; // Measure per 2 second unit.config(cfg); // 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, 100 * 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); } } M5_LOGI("M5UnitUnified has been begun"); M5_LOGI("%s", Units.debugInfo().c_str()); unit.writeAlarmEnabled(0); unit.writeBuzzerControl(false); unit.writeLED(2, 2, 10); lcd.fillScreen(TFT_DARKGREEN); } void loop() { M5.update(); Units.update(); // Periodic if (unit.updated()) { dump(unit.oldest()); } // Button on UnitThermal2 (toggle periodic <-> single) if (unit.wasClicked()) { static bool single{}; single = !single; if (single) { unit.stopPeriodicMeasurement(); unit.writeLED(10, 2, 10); ring_buzzer(2000, 64, 2); Data page0{}, page1{}; if (unit.measureSingleshot(page0, page1)) { ring_buzzer(2000, 64); unit.writeLED(2, 10, 2); dump(page0); dump(page1); } } else { unit.writeLED(2, 2, 10); ring_buzzer(4000, 64); unit.startPeriodicMeasurement(); } } }