ESPHome 2025.9.0
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 uint8_t temp_stack[BUFFER_STACK_SIZE];
43 std::unique_ptr<uint8_t[]> temp_heap;
44 uint8_t *temp;
45 size_t required_buffer_len = data_len * 3 + 2;
46
47 // Is a dynamic allocation required ?
48 if (required_buffer_len >= BUFFER_STACK_SIZE) {
49 temp_heap = std::unique_ptr<uint8_t[]>(new uint8_t[required_buffer_len]);
50 temp = temp_heap.get();
51 } else {
52 temp = temp_stack;
53 }
54 // First byte or word is the command
55 uint8_t raw_idx = 0;
56 if (command_len == 1) {
57 temp[raw_idx++] = command & 0xFF;
58 } else {
59 // command is 2 bytes
60#if __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__
61 temp[raw_idx++] = command >> 8;
62 temp[raw_idx++] = command & 0xFF;
63#else
64 temp[raw_idx++] = command & 0xFF;
65 temp[raw_idx++] = command >> 8;
66#endif
67 }
68 // add parameters followed by crc
69 // skipped if len == 0
70 for (size_t i = 0; i < data_len; i++) {
71#if __BYTE_ORDER__ == __ORDER_LITTLE_ENDIAN__
72 temp[raw_idx++] = data[i] >> 8;
73 temp[raw_idx++] = data[i] & 0xFF;
74#else
75 temp[raw_idx++] = data[i] & 0xFF;
76 temp[raw_idx++] = data[i] >> 8;
77#endif
78 // Use MSB first since Sensirion devices use CRC-8 with MSB first
79 temp[raw_idx++] = crc8(&temp[raw_idx - 2], 2, 0xFF, CRC_POLYNOMIAL, true);
80 }
81 this->last_error_ = this->write(temp, raw_idx);
82 return this->last_error_ == i2c::ERROR_OK;
83}
84
85bool SensirionI2CDevice::get_register_(uint16_t reg, CommandLen command_len, uint16_t *data, const uint8_t len,
86 const uint8_t delay_ms) {
87 if (!this->write_command_(reg, command_len, nullptr, 0)) {
88 ESP_LOGE(TAG, "Write failed: reg=0x%X (%d) err=%d,", reg, command_len, this->last_error_);
89 return false;
90 }
91 delay(delay_ms);
92 bool result = this->read_data(data, len);
93 if (!result) {
94 ESP_LOGE(TAG, "Read failed: reg=0x%X err=%d,", reg, this->last_error_);
95 }
96 return result;
97}
98
99} // namespace sensirion_common
100} // namespace esphome
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:40
@ ERROR_OK
No error found during execution of method.
Definition i2c_bus.h:33
Providing packet encoding functions for exchanging data with a remote host.
Definition a01nyub.cpp:7
std::string size_t len
Definition helpers.h:280
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:44
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:174
void IRAM_ATTR HOT delay(uint32_t ms)
Definition core.cpp:29