From 04bf125448fadc6ea9093ad30c116e6196c1ca76 Mon Sep 17 00:00:00 2001 From: Tomasz Duda Date: Mon, 20 May 2024 23:50:11 +0200 Subject: [PATCH] add i2c --- esphome/components/i2c/__init__.py | 48 ++++-- esphome/components/i2c/i2c_bus.h | 3 + esphome/components/i2c/i2c_bus_zephyr.cpp | 157 ++++++++++++++++++ esphome/components/i2c/i2c_bus_zephyr.h | 31 ++++ .../components/{zephyr => nrf52}/power.cpp | 10 +- .../ads1115/test.nrf52-adafruit.yaml | 13 ++ tests/components/ads1115/test.nrf52.yaml | 13 ++ tests/components/i2c/test.nrf52-adafruit.yaml | 3 + tests/components/i2c/test.nrf52.yaml | 3 + tests/test12.2.yaml | 33 +++- tests/test12.3.yaml | 3 +- 11 files changed, 299 insertions(+), 18 deletions(-) create mode 100644 esphome/components/i2c/i2c_bus_zephyr.cpp create mode 100644 esphome/components/i2c/i2c_bus_zephyr.h rename esphome/components/{zephyr => nrf52}/power.cpp (85%) create mode 100644 tests/components/ads1115/test.nrf52-adafruit.yaml create mode 100644 tests/components/ads1115/test.nrf52.yaml create mode 100644 tests/components/i2c/test.nrf52-adafruit.yaml create mode 100644 tests/components/i2c/test.nrf52.yaml diff --git a/esphome/components/i2c/__init__.py b/esphome/components/i2c/__init__.py index f52a0edb9f..8101de0842 100644 --- a/esphome/components/i2c/__init__.py +++ b/esphome/components/i2c/__init__.py @@ -16,14 +16,17 @@ from esphome.const import ( PLATFORM_ESP32, PLATFORM_ESP8266, PLATFORM_RP2040, + PLATFORM_NRF52, ) from esphome.core import coroutine_with_priority, CORE +from esphome.components.zephyr import zephyr_add_overlay, zephyr_add_prj_conf CODEOWNERS = ["@esphome/core"] i2c_ns = cg.esphome_ns.namespace("i2c") I2CBus = i2c_ns.class_("I2CBus") ArduinoI2CBus = i2c_ns.class_("ArduinoI2CBus", I2CBus, cg.Component) IDFI2CBus = i2c_ns.class_("IDFI2CBus", I2CBus, cg.Component) +ZephyrI2CBus = i2c_ns.class_("ZephyrI2CBus", I2CBus, cg.Component) I2CDevice = i2c_ns.class_("I2CDevice") @@ -37,6 +40,8 @@ def _bus_declare_type(value): return cv.declare_id(ArduinoI2CBus)(value) if CORE.using_esp_idf: return cv.declare_id(IDFI2CBus)(value) + if CORE.using_zephyr: + return cv.declare_id(ZephyrI2CBus)(value) raise NotImplementedError @@ -64,7 +69,7 @@ CONFIG_SCHEMA = cv.All( cv.Optional(CONF_SCAN, default=True): cv.boolean, } ).extend(cv.COMPONENT_SCHEMA), - cv.only_on([PLATFORM_ESP32, PLATFORM_ESP8266, PLATFORM_RP2040]), + cv.only_on([PLATFORM_ESP32, PLATFORM_ESP8266, PLATFORM_RP2040, PLATFORM_NRF52]), ) @@ -74,17 +79,40 @@ async def to_code(config): var = cg.new_Pvariable(config[CONF_ID]) await cg.register_component(var, config) - cg.add(var.set_sda_pin(config[CONF_SDA])) - if CONF_SDA_PULLUP_ENABLED in config: - cg.add(var.set_sda_pullup_enabled(config[CONF_SDA_PULLUP_ENABLED])) - cg.add(var.set_scl_pin(config[CONF_SCL])) - if CONF_SCL_PULLUP_ENABLED in config: - cg.add(var.set_scl_pullup_enabled(config[CONF_SCL_PULLUP_ENABLED])) + if CORE.using_zephyr: + zephyr_add_prj_conf("I2C", True) + zephyr_add_overlay( + f""" +&pinctrl {{ + i2c0_default: i2c0_default {{ + group1 {{ + psels = , + ; + }}; + }}; + i2c0_sleep: i2c0_sleep {{ + group1 {{ + psels = , + ; + low-power-enable; + }}; + }}; +}}; +""" + ) - cg.add(var.set_frequency(int(config[CONF_FREQUENCY]))) + else: + cg.add(var.set_sda_pin(config[CONF_SDA])) + if CONF_SDA_PULLUP_ENABLED in config: + cg.add(var.set_sda_pullup_enabled(config[CONF_SDA_PULLUP_ENABLED])) + cg.add(var.set_scl_pin(config[CONF_SCL])) + if CONF_SCL_PULLUP_ENABLED in config: + cg.add(var.set_scl_pullup_enabled(config[CONF_SCL_PULLUP_ENABLED])) + + cg.add(var.set_frequency(int(config[CONF_FREQUENCY]))) + if CONF_TIMEOUT in config: + cg.add(var.set_timeout(int(config[CONF_TIMEOUT].total_microseconds))) cg.add(var.set_scan(config[CONF_SCAN])) - if CONF_TIMEOUT in config: - cg.add(var.set_timeout(int(config[CONF_TIMEOUT].total_microseconds))) if CORE.using_arduino: cg.add_library("Wire", None) diff --git a/esphome/components/i2c/i2c_bus.h b/esphome/components/i2c/i2c_bus.h index fbfc88323e..ac7c13de82 100644 --- a/esphome/components/i2c/i2c_bus.h +++ b/esphome/components/i2c/i2c_bus.h @@ -3,6 +3,7 @@ #include #include #include +#include "esphome/core/hal.h" namespace esphome { namespace i2c { @@ -102,6 +103,8 @@ class I2CBus { } else if (err == ERROR_UNKNOWN) { scan_results_.emplace_back(address, false); } + // it takes 16sec to scan on nrf52. It prevents board reset. + arch_feed_wdt(); } } std::vector> scan_results_; ///< array containing scan results diff --git a/esphome/components/i2c/i2c_bus_zephyr.cpp b/esphome/components/i2c/i2c_bus_zephyr.cpp new file mode 100644 index 0000000000..245b60f798 --- /dev/null +++ b/esphome/components/i2c/i2c_bus_zephyr.cpp @@ -0,0 +1,157 @@ +#ifdef USE_ZEPHYR + +#include "i2c_bus_zephyr.h" +#include +#include "esphome/core/log.h" + +namespace esphome { +namespace i2c { + +static const char *const TAG = "i2c.zephyr"; + +void ZephyrI2CBus::setup() { + const device *i2c_dev = DEVICE_DT_GET(DT_NODELABEL(i2c0)); + if (!device_is_ready(i2c_dev)) { + ESP_LOGE(TAG, "I2C dev is not ready."); + return; + } + i2c_dev_ = i2c_dev; + + this->recovery_result_ = i2c_recover_bus(i2c_dev_); + if (this->recovery_result_ != 0) { + ESP_LOGE(TAG, "I2C recover bus failed, err %d", this->recovery_result_); + } + if (this->scan_) { + // FIXME it should be done one by one since it takes over 18sec + ESP_LOGV(TAG, "Scanning I2C bus for active devices..."); + this->i2c_scan_(); + } +} + +void ZephyrI2CBus::dump_config() { + ESP_LOGCONFIG(TAG, "I2C Bus:"); + // ESP_LOGCONFIG(TAG, " SDA Pin: GPIO%u", this->sda_pin_); + // ESP_LOGCONFIG(TAG, " SCL Pin: GPIO%u", this->scl_pin_); + if (!i2c_dev_) { + ESP_LOGCONFIG(TAG, " Not initialized"); + return; + } + uint32_t dev_config = 0; + // FIXME this function is not implemented in driver + int err = i2c_get_config(i2c_dev_, &dev_config); + if (err != 0) { + ESP_LOGE(TAG, "Cannot get I2C config, err %d", err); + } else { + auto get_speed = [](uint32_t dev_config) { + switch (I2C_SPEED_GET(dev_config)) { + case I2C_SPEED_STANDARD: + return "100 kHz"; + case I2C_SPEED_FAST: + return "400 kHz"; + case I2C_SPEED_FAST_PLUS: + return "1 MHz"; + case I2C_SPEED_HIGH: + return "3.4 MHz"; + case I2C_SPEED_ULTRA: + return "5 MHz"; + } + return "unknown"; + }; + ESP_LOGCONFIG(TAG, " Frequency: %s", get_speed(dev_config)); + } + + // if (timeout_ > 0) { + // ESP_LOGCONFIG(TAG, " Timeout: %" PRIu32 "us", this->timeout_); + // } + if (this->recovery_result_ != 0) { + ESP_LOGCONFIG(TAG, " Recovery: failed, err %d", this->recovery_result_); + } else { + ESP_LOGCONFIG(TAG, " Recovery: bus successfully recovered"); + } + if (this->scan_) { + ESP_LOGI(TAG, "Results from I2C bus scan:"); + if (scan_results_.empty()) { + ESP_LOGI(TAG, "Found no I2C devices!"); + } else { + for (const auto &s : scan_results_) { + if (s.second) { + ESP_LOGI(TAG, "Found I2C device at address 0x%02X", s.first); + } else { + ESP_LOGE(TAG, "Unknown error at address 0x%02X", s.first); + } + } + } + } +} + +ErrorCode ZephyrI2CBus::readv(uint8_t address, ReadBuffer *buffers, size_t cnt) { + if (!i2c_dev_) { + return ERROR_NOT_INITIALIZED; + } + + std::vector msgs(cnt); + + for (size_t i = 0; i < cnt; ++i) { + msgs[i].buf = buffers[i].data; + msgs[i].len = buffers[i].len; + // TODO how to use I2C_MSG_RESTART + msgs[i].flags = I2C_MSG_READ | I2C_MSG_RESTART; + } + msgs[cnt - 1].flags |= I2C_MSG_STOP; + + auto err = i2c_transfer(i2c_dev_, &msgs[0], msgs.size(), address); + + if (err == -EIO) { + return ERROR_NOT_ACKNOWLEDGED; + } + + if (err != 0) { + ESP_LOGE(TAG, "i2c writing error %d", err); + return ERROR_UNKNOWN; + } + + return ERROR_OK; +} + +ErrorCode ZephyrI2CBus::writev(uint8_t address, WriteBuffer *buffers, size_t cnt, bool stop) { + if (!i2c_dev_) { + return ERROR_NOT_INITIALIZED; + } + + uint8_t dst = 0x00; // dummy data to not use random value + + std::vector msgs(cnt == 0 ? 1 : cnt); + + for (size_t i = 0; i < cnt; ++i) { + if (buffers) { + msgs[i].buf = const_cast(buffers[i].data); + msgs[i].len = buffers[i].len; + } else { + msgs[i].buf = &dst; + msgs[i].len = 0U; + } + msgs[i].flags = I2C_MSG_WRITE; + } + + if (stop) { + msgs[cnt - 1].flags |= I2C_MSG_STOP; + } + + auto err = i2c_transfer(i2c_dev_, &msgs[0], msgs.size(), address); + + if (err == -EIO) { + return ERROR_NOT_ACKNOWLEDGED; + } + + if (err != 0) { + ESP_LOGE(TAG, "i2c writing error %d", err); + return ERROR_UNKNOWN; + } + + return ERROR_OK; +} + +} // namespace i2c +} // namespace esphome + +#endif // USE_ESP_IDF diff --git a/esphome/components/i2c/i2c_bus_zephyr.h b/esphome/components/i2c/i2c_bus_zephyr.h new file mode 100644 index 0000000000..656819c6f3 --- /dev/null +++ b/esphome/components/i2c/i2c_bus_zephyr.h @@ -0,0 +1,31 @@ +#pragma once + +#ifdef USE_ZEPHYR + +#include "i2c_bus.h" +#include "esphome/core/component.h" + +struct device; + +namespace esphome { +namespace i2c { + +class ZephyrI2CBus : public I2CBus, public Component { + public: + void setup() override; + void dump_config() override; + ErrorCode readv(uint8_t address, ReadBuffer *buffers, size_t cnt) override; + ErrorCode writev(uint8_t address, WriteBuffer *buffers, size_t cnt, bool stop) override; + float get_setup_priority() const override { return setup_priority::BUS; } + + void set_scan(bool scan) { scan_ = scan; } + + protected: + const struct device *i2c_dev_ = nullptr; + int recovery_result_ = 0; +}; + +} // namespace i2c +} // namespace esphome + +#endif // USE_ESP_IDF diff --git a/esphome/components/zephyr/power.cpp b/esphome/components/nrf52/power.cpp similarity index 85% rename from esphome/components/zephyr/power.cpp rename to esphome/components/nrf52/power.cpp index b0a8dc935d..88e5228aa4 100644 --- a/esphome/components/zephyr/power.cpp +++ b/esphome/components/nrf52/power.cpp @@ -1,7 +1,10 @@ -#ifdef USE_ZEPHYR +#ifdef USE_NRF52 #include #include +namespace esphome { +namespace nrf52 { + static int board_esphome_init(void) { /* if the board is powered from USB * (high voltage mode), GPIO output voltage is set to 1.8 volts by @@ -28,6 +31,11 @@ static int board_esphome_init(void) { return 0; } +} // namespace nrf52 +} // namespace esphome + +static int board_esphome_init(void) { return esphome::nrf52::board_esphome_init(); } SYS_INIT(board_esphome_init, PRE_KERNEL_1, CONFIG_KERNEL_INIT_PRIORITY_DEFAULT); + #endif diff --git a/tests/components/ads1115/test.nrf52-adafruit.yaml b/tests/components/ads1115/test.nrf52-adafruit.yaml new file mode 100644 index 0000000000..f4869ba531 --- /dev/null +++ b/tests/components/ads1115/test.nrf52-adafruit.yaml @@ -0,0 +1,13 @@ +i2c: + - id: i2c_ads1115 + sda: P1.04 + scl: P1.06 + +ads1115: + address: 0x48 + +sensor: + - platform: ads1115 + multiplexer: A0_A1 + gain: 1.024 + id: ads1115_sensor diff --git a/tests/components/ads1115/test.nrf52.yaml b/tests/components/ads1115/test.nrf52.yaml new file mode 100644 index 0000000000..f4869ba531 --- /dev/null +++ b/tests/components/ads1115/test.nrf52.yaml @@ -0,0 +1,13 @@ +i2c: + - id: i2c_ads1115 + sda: P1.04 + scl: P1.06 + +ads1115: + address: 0x48 + +sensor: + - platform: ads1115 + multiplexer: A0_A1 + gain: 1.024 + id: ads1115_sensor diff --git a/tests/components/i2c/test.nrf52-adafruit.yaml b/tests/components/i2c/test.nrf52-adafruit.yaml new file mode 100644 index 0000000000..f4c76e89d7 --- /dev/null +++ b/tests/components/i2c/test.nrf52-adafruit.yaml @@ -0,0 +1,3 @@ +i2c: + sda: P1.04 + scl: P1.06 diff --git a/tests/components/i2c/test.nrf52.yaml b/tests/components/i2c/test.nrf52.yaml new file mode 100644 index 0000000000..f4c76e89d7 --- /dev/null +++ b/tests/components/i2c/test.nrf52.yaml @@ -0,0 +1,3 @@ +i2c: + sda: P1.04 + scl: P1.06 diff --git a/tests/test12.2.yaml b/tests/test12.2.yaml index c2ad48ad7a..ed372d82cb 100644 --- a/tests/test12.2.yaml +++ b/tests/test12.2.yaml @@ -41,6 +41,7 @@ output: id: rest_gpio - platform: gpio pin: + inverted: true number: 13 mode: output: true @@ -88,21 +89,37 @@ debug: sensor: - platform: uptime name: Uptime Sensor - update_interval: 5sec - platform: adc pin: VDDHDIV5 name: "VDDH Voltage" - update_interval: 5sec filters: - multiply: 5 - platform: adc pin: VDD name: "VDD Voltage" - update_interval: 5sec - platform: adc pin: AIN0 name: "AIN0 Voltage" - update_interval: 5sec + - platform: ads1115 + multiplexer: 'A0_GND' + gain: 6.144 + name: "ADS1115 Channel A0-GND" + update_interval: 1s + - platform: ads1115 + multiplexer: 'A1_GND' + gain: 6.144 + name: "ADS1115 Channel A1-GND" + update_interval: 1s + - platform: ads1115 + multiplexer: 'A2_GND' + gain: 6.144 + name: "ADS1115 Channel A2-GND" + update_interval: 1s + - platform: ads1115 + multiplexer: 'A3_GND' + gain: 6.144 + name: "ADS1115 Channel A3-GND" + update_interval: 1s deep_sleep: @@ -111,4 +128,10 @@ text_sensor: reset_reason: name: "Reset Reason" -zephyr_shell: +i2c: + sda: P1.04 + scl: P1.06 + scan: false + +ads1115: + - address: 0x48 diff --git a/tests/test12.3.yaml b/tests/test12.3.yaml index 5d7ebda804..bcf88f042e 100644 --- a/tests/test12.3.yaml +++ b/tests/test12.3.yaml @@ -6,7 +6,7 @@ esphome: name: nrf52-test-nrf-adafruit logger: - level: DEBUG + level: NONE logs: switch: NONE @@ -78,4 +78,3 @@ text_sensor: reset_reason: name: "Reset Reason" -zephyr_shell: