ESPHome 2026.1.4
Loading...
Searching...
No Matches
sen5x.cpp
Go to the documentation of this file.
1#include "sen5x.h"
3#include "esphome/core/hal.h"
5#include "esphome/core/log.h"
6#include <cinttypes>
7
8namespace esphome {
9namespace sen5x {
10
11static const char *const TAG = "sen5x";
12
13static const uint16_t SEN5X_CMD_AUTO_CLEANING_INTERVAL = 0x8004;
14static const uint16_t SEN5X_CMD_GET_DATA_READY_STATUS = 0x0202;
15static const uint16_t SEN5X_CMD_GET_FIRMWARE_VERSION = 0xD100;
16static const uint16_t SEN5X_CMD_GET_PRODUCT_NAME = 0xD014;
17static const uint16_t SEN5X_CMD_GET_SERIAL_NUMBER = 0xD033;
18static const uint16_t SEN5X_CMD_NOX_ALGORITHM_TUNING = 0x60E1;
19static const uint16_t SEN5X_CMD_READ_MEASUREMENT = 0x03C4;
20static const uint16_t SEN5X_CMD_RHT_ACCELERATION_MODE = 0x60F7;
21static const uint16_t SEN5X_CMD_START_CLEANING_FAN = 0x5607;
22static const uint16_t SEN5X_CMD_START_MEASUREMENTS = 0x0021;
23static const uint16_t SEN5X_CMD_START_MEASUREMENTS_RHT_ONLY = 0x0037;
24static const uint16_t SEN5X_CMD_STOP_MEASUREMENTS = 0x3f86;
25static const uint16_t SEN5X_CMD_TEMPERATURE_COMPENSATION = 0x60B2;
26static const uint16_t SEN5X_CMD_VOC_ALGORITHM_STATE = 0x6181;
27static const uint16_t SEN5X_CMD_VOC_ALGORITHM_TUNING = 0x60D0;
28
29static const int8_t SEN5X_INDEX_SCALE_FACTOR = 10; // used for VOC and NOx index values
30static const int8_t SEN5X_MIN_INDEX_VALUE = 1 * SEN5X_INDEX_SCALE_FACTOR; // must be adjusted by the scale factor
31static const int16_t SEN5X_MAX_INDEX_VALUE = 500 * SEN5X_INDEX_SCALE_FACTOR; // must be adjusted by the scale factor
32
33static const LogString *rht_accel_mode_to_string(RhtAccelerationMode mode) {
34 switch (mode) {
36 return LOG_STR("LOW");
38 return LOG_STR("MEDIUM");
40 return LOG_STR("HIGH");
41 default:
42 return LOG_STR("UNKNOWN");
43 }
44}
45
47 // the sensor needs 1000 ms to enter the idle state
48 this->set_timeout(1000, [this]() {
49 // Check if measurement is ready before reading the value
50 if (!this->write_command(SEN5X_CMD_GET_DATA_READY_STATUS)) {
51 ESP_LOGE(TAG, "Failed to write data ready status command");
52 this->mark_failed();
53 return;
54 }
55 delay(20); // per datasheet
56
57 uint16_t raw_read_status;
58 if (!this->read_data(raw_read_status)) {
59 ESP_LOGE(TAG, "Failed to read data ready status");
60 this->mark_failed();
61 return;
62 }
63
64 uint32_t stop_measurement_delay = 0;
65 // In order to query the device periodic measurement must be ceased
66 if (raw_read_status) {
67 ESP_LOGD(TAG, "Data is available; stopping periodic measurement");
68 if (!this->write_command(SEN5X_CMD_STOP_MEASUREMENTS)) {
69 ESP_LOGE(TAG, "Failed to stop measurements");
70 this->mark_failed();
71 return;
72 }
73 // According to the SEN5x datasheet the sensor will only respond to other commands after waiting 200 ms after
74 // issuing the stop_periodic_measurement command
75 stop_measurement_delay = 200;
76 }
77 this->set_timeout(stop_measurement_delay, [this]() {
78 uint16_t raw_serial_number[3];
79 if (!this->get_register(SEN5X_CMD_GET_SERIAL_NUMBER, raw_serial_number, 3, 20)) {
80 ESP_LOGE(TAG, "Failed to read serial number");
82 this->mark_failed();
83 return;
84 }
85 this->serial_number_[0] = static_cast<bool>(uint16_t(raw_serial_number[0]) & 0xFF);
86 this->serial_number_[1] = static_cast<uint16_t>(raw_serial_number[0] & 0xFF);
87 this->serial_number_[2] = static_cast<uint16_t>(raw_serial_number[1] >> 8);
88 ESP_LOGV(TAG, "Serial number %02d.%02d.%02d", this->serial_number_[0], this->serial_number_[1],
89 this->serial_number_[2]);
90
91 uint16_t raw_product_name[16];
92 if (!this->get_register(SEN5X_CMD_GET_PRODUCT_NAME, raw_product_name, 16, 20)) {
93 ESP_LOGE(TAG, "Failed to read product name");
95 this->mark_failed();
96 return;
97 }
98 // 2 ASCII bytes are encoded in an int
99 const uint16_t *current_int = raw_product_name;
100 char current_char;
101 uint8_t max = 16;
102 do {
103 // first char
104 current_char = *current_int >> 8;
105 if (current_char) {
106 this->product_name_.push_back(current_char);
107 // second char
108 current_char = *current_int & 0xFF;
109 if (current_char) {
110 this->product_name_.push_back(current_char);
111 }
112 }
113 current_int++;
114 } while (current_char && --max);
115
116 Sen5xType sen5x_type = UNKNOWN;
117 if (this->product_name_ == "SEN50") {
118 sen5x_type = SEN50;
119 } else {
120 if (this->product_name_ == "SEN54") {
121 sen5x_type = SEN54;
122 } else {
123 if (this->product_name_ == "SEN55") {
124 sen5x_type = SEN55;
125 }
126 }
127 }
128 ESP_LOGD(TAG, "Product name: %s", this->product_name_.c_str());
129 if (this->humidity_sensor_ && sen5x_type == SEN50) {
130 ESP_LOGE(TAG, "Relative humidity requires a SEN54 or SEN55");
131 this->humidity_sensor_ = nullptr; // mark as not used
132 }
133 if (this->temperature_sensor_ && sen5x_type == SEN50) {
134 ESP_LOGE(TAG, "Temperature requires a SEN54 or SEN55");
135 this->temperature_sensor_ = nullptr; // mark as not used
136 }
137 if (this->voc_sensor_ && sen5x_type == SEN50) {
138 ESP_LOGE(TAG, "VOC requires a SEN54 or SEN55");
139 this->voc_sensor_ = nullptr; // mark as not used
140 }
141 if (this->nox_sensor_ && sen5x_type != SEN55) {
142 ESP_LOGE(TAG, "NOx requires a SEN55");
143 this->nox_sensor_ = nullptr; // mark as not used
144 }
145
146 if (!this->get_register(SEN5X_CMD_GET_FIRMWARE_VERSION, this->firmware_version_, 20)) {
147 ESP_LOGE(TAG, "Failed to read firmware version");
149 this->mark_failed();
150 return;
151 }
152 this->firmware_version_ >>= 8;
153 ESP_LOGV(TAG, "Firmware version %d", this->firmware_version_);
154
155 if (this->voc_sensor_ && this->store_baseline_) {
156 uint32_t combined_serial =
157 encode_uint24(this->serial_number_[0], this->serial_number_[1], this->serial_number_[2]);
158 // Hash with config hash, version, and serial number
159 // This ensures the baseline storage is cleared after OTA
160 // Serial numbers are unique to each sensor, so multiple sensors can be used without conflict
161 uint32_t hash = fnv1a_hash_extend(App.get_config_version_hash(), combined_serial);
162 this->pref_ = global_preferences->make_preference<uint16_t[4]>(hash, true);
164 if (this->pref_.load(&this->voc_baseline_state_)) {
165 if (!this->write_command(SEN5X_CMD_VOC_ALGORITHM_STATE, this->voc_baseline_state_, 4)) {
166 ESP_LOGE(TAG, "VOC Baseline State write to sensor failed");
167 } else {
168 ESP_LOGV(TAG, "VOC Baseline State loaded");
169 delay(20);
170 }
171 }
172 }
173 bool result;
175 // override default value
176 result = write_command(SEN5X_CMD_AUTO_CLEANING_INTERVAL, this->auto_cleaning_interval_.value());
177 } else {
178 result = write_command(SEN5X_CMD_AUTO_CLEANING_INTERVAL);
179 }
180 if (result) {
181 delay(20);
182 uint16_t secs[2];
183 if (this->read_data(secs, 2)) {
184 this->auto_cleaning_interval_ = secs[0] << 16 | secs[1];
185 }
186 }
187 if (this->acceleration_mode_.has_value()) {
188 result = this->write_command(SEN5X_CMD_RHT_ACCELERATION_MODE, this->acceleration_mode_.value());
189 } else {
190 result = this->write_command(SEN5X_CMD_RHT_ACCELERATION_MODE);
191 }
192 if (!result) {
193 ESP_LOGE(TAG, "Failed to set rh/t acceleration mode");
195 this->mark_failed();
196 return;
197 }
198 delay(20);
199 if (!this->acceleration_mode_.has_value()) {
200 uint16_t mode;
201 if (this->read_data(mode)) {
203 } else {
204 ESP_LOGE(TAG, "Failed to read RHT Acceleration mode");
205 }
206 }
207 if (this->voc_tuning_params_.has_value()) {
208 this->write_tuning_parameters_(SEN5X_CMD_VOC_ALGORITHM_TUNING, this->voc_tuning_params_.value());
209 delay(20);
210 }
211 if (this->nox_tuning_params_.has_value()) {
212 this->write_tuning_parameters_(SEN5X_CMD_NOX_ALGORITHM_TUNING, this->nox_tuning_params_.value());
213 delay(20);
214 }
215
216 if (this->temperature_compensation_.has_value()) {
218 delay(20);
219 }
220
221 // Finally start sensor measurements
222 auto cmd = SEN5X_CMD_START_MEASUREMENTS_RHT_ONLY;
223 if (this->pm_1_0_sensor_ || this->pm_2_5_sensor_ || this->pm_4_0_sensor_ || this->pm_10_0_sensor_) {
224 // if any of the gas sensors are active we need a full measurement
225 cmd = SEN5X_CMD_START_MEASUREMENTS;
226 }
227
228 if (!this->write_command(cmd)) {
229 ESP_LOGE(TAG, "Error starting continuous measurements");
231 this->mark_failed();
232 return;
233 }
234 this->initialized_ = true;
235 });
236 });
237}
238
240 ESP_LOGCONFIG(TAG, "SEN5X:");
241 LOG_I2C_DEVICE(this);
242 if (this->is_failed()) {
243 switch (this->error_code_) {
245 ESP_LOGW(TAG, ESP_LOG_MSG_COMM_FAIL);
246 break;
248 ESP_LOGW(TAG, "Measurement initialization failed");
249 break;
251 ESP_LOGW(TAG, "Unable to read serial ID");
252 break;
254 ESP_LOGW(TAG, "Unable to read product name");
255 break;
256 case FIRMWARE_FAILED:
257 ESP_LOGW(TAG, "Unable to read firmware version");
258 break;
259 default:
260 ESP_LOGW(TAG, "Unknown setup error");
261 break;
262 }
263 }
264 ESP_LOGCONFIG(TAG,
265 " Product name: %s\n"
266 " Firmware version: %d\n"
267 " Serial number %02d.%02d.%02d",
268 this->product_name_.c_str(), this->firmware_version_, this->serial_number_[0], this->serial_number_[1],
269 this->serial_number_[2]);
271 ESP_LOGCONFIG(TAG, " Auto cleaning interval: %" PRId32 "s", this->auto_cleaning_interval_.value());
272 }
273 if (this->acceleration_mode_.has_value()) {
274 ESP_LOGCONFIG(TAG, " RH/T acceleration mode: %s",
275 LOG_STR_ARG(rht_accel_mode_to_string(this->acceleration_mode_.value())));
276 }
277 if (this->voc_sensor_) {
278 char hex_buf[5 * 4];
279 format_hex_pretty_to(hex_buf, this->voc_baseline_state_, 4, 0);
280 ESP_LOGCONFIG(TAG,
281 " Store Baseline: %s\n"
282 " State: %s\n",
283 TRUEFALSE(this->store_baseline_), hex_buf);
284 }
285 LOG_UPDATE_INTERVAL(this);
286 LOG_SENSOR(" ", "PM 1.0", this->pm_1_0_sensor_);
287 LOG_SENSOR(" ", "PM 2.5", this->pm_2_5_sensor_);
288 LOG_SENSOR(" ", "PM 4.0", this->pm_4_0_sensor_);
289 LOG_SENSOR(" ", "PM 10.0", this->pm_10_0_sensor_);
290 LOG_SENSOR(" ", "Temperature", this->temperature_sensor_);
291 LOG_SENSOR(" ", "Humidity", this->humidity_sensor_);
292 LOG_SENSOR(" ", "VOC", this->voc_sensor_); // SEN54 and SEN55 only
293 LOG_SENSOR(" ", "NOx", this->nox_sensor_); // SEN55 only
294}
295
297 if (!this->initialized_) {
298 return;
299 }
300
301 if (!this->write_command(SEN5X_CMD_READ_MEASUREMENT)) {
302 this->status_set_warning();
303 ESP_LOGD(TAG, "Write error: read measurement (%d)", this->last_error_);
304 return;
305 }
306 this->set_timeout(20, [this]() {
307 uint16_t measurements[8];
308
309 if (!this->read_data(measurements, 8)) {
310 this->status_set_warning();
311 ESP_LOGD(TAG, "Read data error (%d)", this->last_error_);
312 return;
313 }
314
315 ESP_LOGVV(TAG, "pm_1_0 = 0x%.4x", measurements[0]);
316 float pm_1_0 = measurements[0] == UINT16_MAX ? NAN : measurements[0] / 10.0f;
317
318 ESP_LOGVV(TAG, "pm_2_5 = 0x%.4x", measurements[1]);
319 float pm_2_5 = measurements[1] == UINT16_MAX ? NAN : measurements[1] / 10.0f;
320
321 ESP_LOGVV(TAG, "pm_4_0 = 0x%.4x", measurements[2]);
322 float pm_4_0 = measurements[2] == UINT16_MAX ? NAN : measurements[2] / 10.0f;
323
324 ESP_LOGVV(TAG, "pm_10_0 = 0x%.4x", measurements[3]);
325 float pm_10_0 = measurements[3] == UINT16_MAX ? NAN : measurements[3] / 10.0f;
326
327 ESP_LOGVV(TAG, "humidity = 0x%.4x", measurements[4]);
328 float humidity = measurements[4] == INT16_MAX ? NAN : static_cast<int16_t>(measurements[4]) / 100.0f;
329
330 ESP_LOGVV(TAG, "temperature = 0x%.4x", measurements[5]);
331 float temperature = measurements[5] == INT16_MAX ? NAN : static_cast<int16_t>(measurements[5]) / 200.0f;
332
333 ESP_LOGVV(TAG, "voc = 0x%.4x", measurements[6]);
334 int16_t voc_idx = static_cast<int16_t>(measurements[6]);
335 float voc = (voc_idx < SEN5X_MIN_INDEX_VALUE || voc_idx > SEN5X_MAX_INDEX_VALUE)
336 ? NAN
337 : static_cast<float>(voc_idx) / 10.0f;
338
339 ESP_LOGVV(TAG, "nox = 0x%.4x", measurements[7]);
340 int16_t nox_idx = static_cast<int16_t>(measurements[7]);
341 float nox = (nox_idx < SEN5X_MIN_INDEX_VALUE || nox_idx > SEN5X_MAX_INDEX_VALUE)
342 ? NAN
343 : static_cast<float>(nox_idx) / 10.0f;
344
345 if (this->pm_1_0_sensor_ != nullptr) {
346 this->pm_1_0_sensor_->publish_state(pm_1_0);
347 }
348 if (this->pm_2_5_sensor_ != nullptr) {
349 this->pm_2_5_sensor_->publish_state(pm_2_5);
350 }
351 if (this->pm_4_0_sensor_ != nullptr) {
352 this->pm_4_0_sensor_->publish_state(pm_4_0);
353 }
354 if (this->pm_10_0_sensor_ != nullptr) {
355 this->pm_10_0_sensor_->publish_state(pm_10_0);
356 }
357 if (this->temperature_sensor_ != nullptr) {
358 this->temperature_sensor_->publish_state(temperature);
359 }
360 if (this->humidity_sensor_ != nullptr) {
361 this->humidity_sensor_->publish_state(humidity);
362 }
363 if (this->voc_sensor_ != nullptr) {
364 this->voc_sensor_->publish_state(voc);
365 }
366 if (this->nox_sensor_ != nullptr) {
367 this->nox_sensor_->publish_state(nox);
368 }
369
370 if (!this->voc_sensor_ || !this->store_baseline_ ||
371 (App.get_loop_component_start_time() - this->voc_baseline_time_) < SHORTEST_BASELINE_STORE_INTERVAL) {
372 this->status_clear_warning();
373 } else {
375 if (!this->write_command(SEN5X_CMD_VOC_ALGORITHM_STATE)) {
376 this->status_set_warning();
377 ESP_LOGW(TAG, ESP_LOG_MSG_COMM_FAIL);
378 } else {
379 this->set_timeout(20, [this]() {
380 if (!this->read_data(this->voc_baseline_state_, 4)) {
381 this->status_set_warning();
382 ESP_LOGW(TAG, ESP_LOG_MSG_COMM_FAIL);
383 } else {
384 if (this->pref_.save(&this->voc_baseline_state_)) {
385 ESP_LOGD(TAG, "VOC Baseline State saved");
386 }
387 this->status_clear_warning();
388 }
389 });
390 }
391 }
392 });
393}
394
395bool SEN5XComponent::write_tuning_parameters_(uint16_t i2c_command, const GasTuning &tuning) {
396 uint16_t params[6];
397 params[0] = tuning.index_offset;
398 params[1] = tuning.learning_time_offset_hours;
399 params[2] = tuning.learning_time_gain_hours;
400 params[3] = tuning.gating_max_duration_minutes;
401 params[4] = tuning.std_initial;
402 params[5] = tuning.gain_factor;
403 auto result = write_command(i2c_command, params, 6);
404 if (!result) {
405 ESP_LOGE(TAG, "Set tuning parameters failed (command=%0xX, err=%d)", i2c_command, this->last_error_);
406 }
407 return result;
408}
409
411 uint16_t params[3];
412 params[0] = compensation.offset;
413 params[1] = compensation.normalized_offset_slope;
414 params[2] = compensation.time_constant;
415 if (!write_command(SEN5X_CMD_TEMPERATURE_COMPENSATION, params, 3)) {
416 ESP_LOGE(TAG, "Set temperature_compensation failed (%d)", this->last_error_);
417 return false;
418 }
419 return true;
420}
421
423 if (!write_command(SEN5X_CMD_START_CLEANING_FAN)) {
424 this->status_set_warning();
425 ESP_LOGE(TAG, "Start fan cleaning failed (%d)", this->last_error_);
426 return false;
427 } else {
428 ESP_LOGD(TAG, "Fan auto clean started");
429 }
430 return true;
431}
432
433} // namespace sen5x
434} // namespace esphome
BedjetMode mode
BedJet operating mode.
constexpr uint32_t get_config_version_hash()
Get the config hash extended with ESPHome version.
uint32_t IRAM_ATTR HOT get_loop_component_start_time() const
Get the cached time in milliseconds from when the current component started its loop execution.
virtual void mark_failed()
Mark this component as failed.
bool is_failed() const
void status_set_warning(const char *message=nullptr)
ESPDEPRECATED("Use const char* or uint32_t overload instead. Removed in 2026.7.0", "2026.1.0") void set_timeout(const std voi set_timeout)(const char *name, uint32_t timeout, std::function< void()> &&f)
Set a timeout function with a unique name.
Definition component.h:445
void status_clear_warning()
bool save(const T *src)
Definition preferences.h:21
virtual ESPPreferenceObject make_preference(size_t length, uint32_t type, bool in_flash)=0
bool has_value() const
Definition optional.h:92
value_type const & value() const
Definition optional.h:94
optional< RhtAccelerationMode > acceleration_mode_
Definition sen5x.h:122
sensor::Sensor * pm_4_0_sensor_
Definition sen5x.h:113
void dump_config() override
Definition sen5x.cpp:239
ESPPreferenceObject pref_
Definition sen5x.h:127
bool write_tuning_parameters_(uint16_t i2c_command, const GasTuning &tuning)
Definition sen5x.cpp:395
sensor::Sensor * pm_2_5_sensor_
Definition sen5x.h:112
sensor::Sensor * temperature_sensor_
Definition sen5x.h:116
optional< uint32_t > auto_cleaning_interval_
Definition sen5x.h:123
sensor::Sensor * pm_1_0_sensor_
Definition sen5x.h:111
sensor::Sensor * voc_sensor_
Definition sen5x.h:118
optional< TemperatureCompensation > temperature_compensation_
Definition sen5x.h:126
sensor::Sensor * nox_sensor_
Definition sen5x.h:120
optional< GasTuning > voc_tuning_params_
Definition sen5x.h:124
sensor::Sensor * pm_10_0_sensor_
Definition sen5x.h:114
sensor::Sensor * humidity_sensor_
Definition sen5x.h:117
bool write_temperature_compensation_(const TemperatureCompensation &compensation)
Definition sen5x.cpp:410
optional< GasTuning > nox_tuning_params_
Definition sen5x.h:125
uint16_t voc_baseline_state_[4]
Definition sen5x.h:103
i2c::ErrorCode last_error_
last error code from I2C operation
bool get_register(uint16_t command, uint16_t *data, uint8_t len, uint8_t delay=0)
get data words from I2C register.
bool write_command(T i2c_register)
Write a command to the I2C device.
bool read_data(uint16_t *data, uint8_t len)
Read data words from I2C device.
void publish_state(float state)
Publish a new state to the front-end.
Definition sensor.cpp:76
@ PRODUCT_NAME_FAILED
Definition sen5x.h:16
@ MEASUREMENT_INIT_FAILED
Definition sen5x.h:15
@ FIRMWARE_FAILED
Definition sen5x.h:17
@ SERIAL_NUMBER_IDENTIFICATION_FAILED
Definition sen5x.h:14
@ COMMUNICATION_FAILED
Definition sen5x.h:13
RhtAccelerationMode
Definition sen5x.h:21
@ LOW_ACCELERATION
Definition sen5x.h:22
@ HIGH_ACCELERATION
Definition sen5x.h:24
@ MEDIUM_ACCELERATION
Definition sen5x.h:23
Providing packet encoding functions for exchanging data with a remote host.
Definition a01nyub.cpp:7
constexpr uint32_t fnv1a_hash_extend(uint32_t hash, const char *str)
Extend a FNV-1a hash with additional string data.
Definition helpers.h:424
constexpr uint32_t encode_uint24(uint8_t byte1, uint8_t byte2, uint8_t byte3)
Encode a 24-bit value given three bytes in most to least significant byte order.
Definition helpers.h:467
char * format_hex_pretty_to(char *buffer, size_t buffer_size, const uint8_t *data, size_t length, char separator)
Format byte array as uppercase hex to buffer (base implementation).
Definition helpers.cpp:334
ESPPreferences * global_preferences
void IRAM_ATTR HOT delay(uint32_t ms)
Definition core.cpp:26
Application App
Global storage of Application pointer - only one Application can exist.
uint16_t learning_time_gain_hours
Definition sen5x.h:30
uint16_t gating_max_duration_minutes
Definition sen5x.h:31
uint16_t learning_time_offset_hours
Definition sen5x.h:29
uint16_t temperature
Definition sun_gtil2.cpp:12