11 #if ESP_IDF_VERSION < ESP_IDF_VERSION_VAL(5, 3, 0) 12 #define SOC_HP_I2C_NUM SOC_I2C_NUM 18 static const char *
const TAG =
"i2c.idf";
21 ESP_LOGCONFIG(TAG,
"Setting up I2C bus...");
22 static i2c_port_t next_port = I2C_NUM_0;
24 #if SOC_HP_I2C_NUM > 1 25 next_port = (next_port == I2C_NUM_0) ? I2C_NUM_1 : I2C_NUM_MAX;
27 next_port = I2C_NUM_MAX;
30 if (
port_ == I2C_NUM_MAX) {
31 ESP_LOGE(TAG,
"Too many I2C buses configured. Max %u supported.", SOC_HP_I2C_NUM);
39 memset(&conf, 0,
sizeof(conf));
40 conf.mode = I2C_MODE_MASTER;
46 #ifdef USE_ESP32_VARIANT_ESP32S2 48 conf.clk_flags = I2C_SCLK_SRC_FLAG_AWARE_DFS;
50 esp_err_t err = i2c_param_config(
port_, &conf);
52 ESP_LOGW(TAG,
"i2c_param_config failed: %s", esp_err_to_name(err));
58 ESP_LOGW(TAG,
"i2c timeout of %" PRIu32
"us greater than max of 13ms on esp-idf, setting to max",
timeout_);
63 ESP_LOGW(TAG,
"i2c_set_timeout failed: %s", esp_err_to_name(err));
67 ESP_LOGV(TAG,
"i2c_timeout set to %" PRIu32
" ticks (%" PRIu32
" us)",
timeout_ * 80,
timeout_);
70 err = i2c_driver_install(
port_, I2C_MODE_MASTER, 0, 0, ESP_INTR_FLAG_IRAM);
72 ESP_LOGW(TAG,
"i2c_driver_install failed: %s", esp_err_to_name(err));
78 ESP_LOGV(TAG,
"Scanning i2c bus for active devices...");
83 ESP_LOGCONFIG(TAG,
"I2C Bus:");
84 ESP_LOGCONFIG(TAG,
" SDA Pin: GPIO%u", this->
sda_pin_);
85 ESP_LOGCONFIG(TAG,
" SCL Pin: GPIO%u", this->
scl_pin_);
86 ESP_LOGCONFIG(TAG,
" Frequency: %" PRIu32
" Hz", this->
frequency_);
88 ESP_LOGCONFIG(TAG,
" Timeout: %" PRIu32
"us", this->
timeout_);
90 switch (this->recovery_result_) {
92 ESP_LOGCONFIG(TAG,
" Recovery: bus successfully recovered");
95 ESP_LOGCONFIG(TAG,
" Recovery: failed, SCL is held low on the bus");
98 ESP_LOGCONFIG(TAG,
" Recovery: failed, SDA is held low on the bus");
102 ESP_LOGI(TAG,
"Results from i2c bus scan:");
104 ESP_LOGI(TAG,
"Found no i2c devices!");
108 ESP_LOGI(TAG,
"Found i2c device at address 0x%02X", s.first);
110 ESP_LOGE(TAG,
"Unknown error at address 0x%02X", s.first);
121 ESP_LOGVV(TAG,
"i2c bus not initialized!");
124 i2c_cmd_handle_t
cmd = i2c_cmd_link_create();
125 esp_err_t err = i2c_master_start(cmd);
127 ESP_LOGVV(TAG,
"RX from %02X master start failed: %s", address, esp_err_to_name(err));
128 i2c_cmd_link_delete(cmd);
131 err = i2c_master_write_byte(cmd, (address << 1) | I2C_MASTER_READ,
true);
133 ESP_LOGVV(TAG,
"RX from %02X address write failed: %s", address, esp_err_to_name(err));
134 i2c_cmd_link_delete(cmd);
137 for (
size_t i = 0; i < cnt; i++) {
138 const auto &buf = buffers[i];
141 err = i2c_master_read(cmd, buf.data, buf.len, i == cnt - 1 ? I2C_MASTER_LAST_NACK : I2C_MASTER_ACK);
143 ESP_LOGVV(TAG,
"RX from %02X data read failed: %s", address, esp_err_to_name(err));
144 i2c_cmd_link_delete(cmd);
148 err = i2c_master_stop(cmd);
150 ESP_LOGVV(TAG,
"RX from %02X stop failed: %s", address, esp_err_to_name(err));
151 i2c_cmd_link_delete(cmd);
154 err = i2c_master_cmd_begin(
port_, cmd, 20 / portTICK_PERIOD_MS);
157 i2c_cmd_link_delete(cmd);
158 if (err == ESP_FAIL) {
160 ESP_LOGVV(TAG,
"RX from %02X failed: not acked", address);
162 }
else if (err == ESP_ERR_TIMEOUT) {
163 ESP_LOGVV(TAG,
"RX from %02X failed: timeout", address);
165 }
else if (err != ESP_OK) {
166 ESP_LOGVV(TAG,
"RX from %02X failed: %s", address, esp_err_to_name(err));
170 #ifdef ESPHOME_LOG_HAS_VERY_VERBOSE 172 std::string debug_hex;
174 for (
size_t i = 0; i < cnt; i++) {
175 const auto &buf = buffers[i];
176 for (
size_t j = 0; j < buf.len; j++) {
177 snprintf(debug_buf,
sizeof(debug_buf),
"%02X", buf.data[j]);
178 debug_hex += debug_buf;
181 ESP_LOGVV(TAG,
"0x%02X RX %s", address, debug_hex.c_str());
190 ESP_LOGVV(TAG,
"i2c bus not initialized!");
194 #ifdef ESPHOME_LOG_HAS_VERY_VERBOSE 196 std::string debug_hex;
198 for (
size_t i = 0; i < cnt; i++) {
199 const auto &buf = buffers[i];
200 for (
size_t j = 0; j < buf.len; j++) {
201 snprintf(debug_buf,
sizeof(debug_buf),
"%02X", buf.data[j]);
202 debug_hex += debug_buf;
205 ESP_LOGVV(TAG,
"0x%02X TX %s", address, debug_hex.c_str());
208 i2c_cmd_handle_t
cmd = i2c_cmd_link_create();
209 esp_err_t err = i2c_master_start(cmd);
211 ESP_LOGVV(TAG,
"TX to %02X master start failed: %s", address, esp_err_to_name(err));
212 i2c_cmd_link_delete(cmd);
215 err = i2c_master_write_byte(cmd, (address << 1) | I2C_MASTER_WRITE,
true);
217 ESP_LOGVV(TAG,
"TX to %02X address write failed: %s", address, esp_err_to_name(err));
218 i2c_cmd_link_delete(cmd);
221 for (
size_t i = 0; i < cnt; i++) {
222 const auto &buf = buffers[i];
225 err = i2c_master_write(cmd, buf.data, buf.len,
true);
227 ESP_LOGVV(TAG,
"TX to %02X data write failed: %s", address, esp_err_to_name(err));
228 i2c_cmd_link_delete(cmd);
233 err = i2c_master_stop(cmd);
235 ESP_LOGVV(TAG,
"TX to %02X master stop failed: %s", address, esp_err_to_name(err));
236 i2c_cmd_link_delete(cmd);
240 err = i2c_master_cmd_begin(
port_, cmd, 20 / portTICK_PERIOD_MS);
241 i2c_cmd_link_delete(cmd);
242 if (err == ESP_FAIL) {
244 ESP_LOGVV(TAG,
"TX to %02X failed: not acked", address);
246 }
else if (err == ESP_ERR_TIMEOUT) {
247 ESP_LOGVV(TAG,
"TX to %02X failed: timeout", address);
249 }
else if (err != ESP_OK) {
250 ESP_LOGVV(TAG,
"TX to %02X failed: %s", address, esp_err_to_name(err));
259 void IDFI2CBus::recover_() {
260 ESP_LOGI(TAG,
"Performing I2C bus recovery");
262 const gpio_num_t scl_pin =
static_cast<gpio_num_t
>(
scl_pin_);
263 const gpio_num_t sda_pin =
static_cast<gpio_num_t
>(
sda_pin_);
271 const auto half_period_usec = 7;
274 gpio_set_level(scl_pin, 1);
275 gpio_config_t scl_config{};
276 scl_config.pin_bit_mask = 1ULL <<
scl_pin_;
277 scl_config.mode = GPIO_MODE_INPUT_OUTPUT_OD;
278 scl_config.pull_up_en = GPIO_PULLUP_ENABLE;
279 scl_config.pull_down_en = GPIO_PULLDOWN_DISABLE;
280 scl_config.intr_type = GPIO_INTR_DISABLE;
281 gpio_config(&scl_config);
284 gpio_set_level(sda_pin, 1);
285 gpio_config_t sda_conf{};
286 sda_conf.pin_bit_mask = 1ULL <<
sda_pin_;
287 sda_conf.mode = GPIO_MODE_INPUT_OUTPUT_OD;
288 sda_conf.pull_up_en = GPIO_PULLUP_ENABLE;
289 sda_conf.pull_down_en = GPIO_PULLDOWN_DISABLE;
290 sda_conf.intr_type = GPIO_INTR_DISABLE;
291 gpio_config(&sda_conf);
296 if (gpio_get_level(scl_pin) == 0) {
297 ESP_LOGE(TAG,
"Recovery failed: SCL is held LOW on the I2C bus");
309 for (
auto i = 0; i < 9; i++) {
310 gpio_set_level(scl_pin, 0);
312 gpio_set_level(scl_pin, 1);
323 while (wait-- && gpio_get_level(scl_pin) == 0) {
327 if (gpio_get_level(scl_pin) == 0) {
328 ESP_LOGE(TAG,
"Recovery failed: SCL is held LOW during clock pulse cycle");
337 if (gpio_get_level(sda_pin) == 0) {
338 ESP_LOGE(TAG,
"Recovery failed: SDA is held LOW after clock pulse cycle");
355 gpio_set_level(sda_pin, 0);
364 gpio_set_level(sda_pin, 1);
372 #endif // USE_ESP_IDF the WriteBuffer structure stores a pointer to a write buffer and its length
void dump_config() override
void i2c_scan_()
Scans the I2C bus for devices.
std::vector< std::pair< uint8_t, bool > > scan_results_
array containing scan results
the ReadBuffer structure stores a pointer to a read buffer and its length
ErrorCode writev(uint8_t address, WriteBuffer *buffers, size_t cnt, bool stop) override
timeout while waiting to receive bytes
ErrorCode readv(uint8_t address, ReadBuffer *buffers, size_t cnt) override
No error found during execution of method.
I2C bus acknowledgment not received.
Application App
Global storage of Application pointer - only one Application can exist.
bool scan_
Should we scan ? Can be set in the yaml.
virtual void mark_failed()
Mark this component as failed.
Implementation of SPI Controller mode.
void IRAM_ATTR HOT delayMicroseconds(uint32_t us)
ErrorCode
Error codes returned by I2CBus and I2CDevice methods.
miscellaneous I2C error during execution
call method to a not initialized bus