ESPHome 2026.2.4
Loading...
Searching...
No Matches
i2c_sensirion.cpp
Go to the documentation of this file.
1#include "i2c_sensirion.h"
2#include "esphome/core/hal.h"
4#include "esphome/core/log.h"
5#include <cinttypes>
6
7namespace esphome {
8namespace sensirion_common {
9
10static const char *const TAG = "sensirion_i2c";
11// To avoid memory allocations for small writes a stack buffer is used
12static const size_t BUFFER_STACK_SIZE = 16;
13
14bool SensirionI2CDevice::read_data(uint16_t *data, const uint8_t len) {
15 const uint8_t num_bytes = len * 3;
16 uint8_t buf[num_bytes];
17
18 this->last_error_ = this->read(buf, num_bytes);
19 if (this->last_error_ != i2c::ERROR_OK) {
20 return false;
21 }
22
23 for (uint8_t i = 0; i < len; i++) {
24 const uint8_t j = 3 * i;
25 // Use MSB first since Sensirion devices use CRC-8 with MSB first
26 uint8_t crc = crc8(&buf[j], 2, 0xFF, CRC_POLYNOMIAL, true);
27 if (crc != buf[j + 2]) {
28 ESP_LOGE(TAG, "CRC invalid @ %d! 0x%02X != 0x%02X", i, buf[j + 2], crc);
30 return false;
31 }
32 data[i] = encode_uint16(buf[j], buf[j + 1]);
33 }
34 return true;
35}
36/***
37 * write command with parameters and insert crc
38 * use stack array for less than 4 parameters. Most Sensirion I2C commands have less parameters
39 */
40bool SensirionI2CDevice::write_command_(uint16_t command, CommandLen command_len, const uint16_t *data,
41 const uint8_t data_len) {
42 size_t required_buffer_len = data_len * 3 + 2;
43 SmallBufferWithHeapFallback<BUFFER_STACK_SIZE> buffer(required_buffer_len);
44 uint8_t *temp = buffer.get();
45 // First byte or word is the command
46 uint8_t raw_idx = 0;
47 if (command_len == 1) {
48 temp[raw_idx++] = command & 0xFF;
49 } else {
50 // command is 2 bytes
51 temp[raw_idx++] = command >> 8;
52 temp[raw_idx++] = command & 0xFF;
53 }
54 // add parameters followed by crc
55 // skipped if len == 0
56 for (size_t i = 0; i < data_len; i++) {
57 temp[raw_idx++] = data[i] >> 8;
58 temp[raw_idx++] = data[i] & 0xFF;
59 // Use MSB first since Sensirion devices use CRC-8 with MSB first
60 uint8_t crc = crc8(&temp[raw_idx - 2], 2, 0xFF, CRC_POLYNOMIAL, true);
61 temp[raw_idx++] = crc;
62 }
63 this->last_error_ = this->write(temp, raw_idx);
64 return this->last_error_ == i2c::ERROR_OK;
65}
66
67bool SensirionI2CDevice::get_register_(uint16_t reg, CommandLen command_len, uint16_t *data, const uint8_t len,
68 const uint8_t delay_ms) {
69 if (!this->write_command_(reg, command_len, nullptr, 0)) {
70 ESP_LOGE(TAG, "Write failed: reg=0x%X (%d) err=%d,", reg, command_len, this->last_error_);
71 return false;
72 }
73 delay(delay_ms);
74 bool result = this->read_data(data, len);
75 if (!result) {
76 ESP_LOGE(TAG, "Read failed: reg=0x%X err=%d,", reg, this->last_error_);
77 }
78 return result;
79}
80
81} // namespace sensirion_common
82} // namespace esphome
Helper class for efficient buffer allocation - uses stack for small sizes, heap for large This is use...
Definition helpers.h:411
ErrorCode write(const uint8_t *data, size_t len) const
writes an array of bytes to a device using an I2CBus
Definition i2c.h:184
ErrorCode read(uint8_t *data, size_t len) const
reads an array of bytes from the device using an I2CBus
Definition i2c.h:164
I2CRegister reg(uint8_t a_register)
calls the I2CRegister constructor
Definition i2c.h:153
uint8_t size_t len
Definition i2c.h:273
bool get_register_(uint16_t reg, CommandLen command_len, uint16_t *data, uint8_t len, uint8_t delay)
get data words from I2C register.
i2c::ErrorCode last_error_
last error code from I2C operation
bool read_data(uint16_t *data, uint8_t len)
Read data words from I2C device.
bool write_command_(uint16_t command, CommandLen command_len, const uint16_t *data, uint8_t data_len)
Write a command with arguments as words.
@ ERROR_CRC
bytes received with a CRC error
Definition i2c_bus.h:24
@ ERROR_OK
No error found during execution of method.
Definition i2c_bus.h:17
Providing packet encoding functions for exchanging data with a remote host.
Definition a01nyub.cpp:7
std::string size_t len
Definition helpers.h:692
uint8_t crc8(const uint8_t *data, uint8_t len, uint8_t crc, uint8_t poly, bool msb_first)
Calculate a CRC-8 checksum of data with size len.
Definition helpers.cpp:46
constexpr uint16_t encode_uint16(uint8_t msb, uint8_t lsb)
Encode a 16-bit value given the most and least significant byte.
Definition helpers.h:528
void IRAM_ATTR HOT delay(uint32_t ms)
Definition core.cpp:26