2023-01-29 07:06:25 +00:00
|
|
|
/*
|
|
|
|
xsns_103_sen5x.ino - SEN5X gas and air quality sensor support for Tasmota
|
|
|
|
|
|
|
|
Copyright (C) 2022 Tyeth Gundry
|
|
|
|
|
|
|
|
This program is free software: you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation, either version 3 of the License, or
|
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#ifdef USE_I2C
|
|
|
|
#ifdef USE_SEN5X
|
|
|
|
/*********************************************************************************************\
|
2023-02-17 07:37:06 +00:00
|
|
|
* SEN5X - Gas (VOC - Volatile Organic Compounds / NOx - Nitrous Oxides) and Particulates (PM)
|
2023-01-29 07:06:25 +00:00
|
|
|
*
|
|
|
|
* Source: Sensirion SEN5X Driver + Example, and Tasmota Driver 98 by Jean-Pierre Deschamps
|
|
|
|
* Adaption for TASMOTA: Tyeth Gundry
|
|
|
|
*
|
2023-02-17 07:37:06 +00:00
|
|
|
* I2C Address: 0x69
|
2023-01-29 07:06:25 +00:00
|
|
|
\*********************************************************************************************/
|
|
|
|
|
|
|
|
#define XSNS_103 103
|
|
|
|
#define XI2C_76 76 // See I2CDEVICES.md
|
|
|
|
|
|
|
|
#define SEN5X_ADDRESS 0x69
|
2023-08-26 15:14:30 +01:00
|
|
|
#define SEN5X_PASSIVE_MODE_INTERVAL 10
|
2023-01-29 07:06:25 +00:00
|
|
|
|
|
|
|
#include <SensirionI2CSen5x.h>
|
|
|
|
#include <Wire.h>
|
|
|
|
SensirionI2CSen5x *sen5x = nullptr;
|
|
|
|
|
|
|
|
struct SEN5XDATA_s {
|
|
|
|
float massConcentrationPm1p0;
|
|
|
|
float massConcentrationPm2p5;
|
|
|
|
float massConcentrationPm4p0;
|
|
|
|
float massConcentrationPm10p0;
|
|
|
|
float ambientHumidity;
|
|
|
|
float ambientTemperature;
|
|
|
|
float vocIndex;
|
|
|
|
float noxIndex;
|
|
|
|
} *SEN5XDATA = nullptr;
|
2023-03-22 12:25:55 +00:00
|
|
|
|
2023-01-29 07:06:25 +00:00
|
|
|
/********************************************************************************************/
|
|
|
|
|
2023-03-22 12:25:55 +00:00
|
|
|
void sen5x_Init(void) {
|
2023-01-29 07:06:25 +00:00
|
|
|
int usingI2cBus = 0;
|
|
|
|
#ifdef ESP32
|
2023-03-22 12:25:55 +00:00
|
|
|
if (!I2cSetDevice(SEN5X_ADDRESS, 0)) {
|
2023-01-29 07:06:25 +00:00
|
|
|
DEBUG_SENSOR_LOG(PSTR("Sensirion SEN5X not found, i2c bus 0"));
|
2024-10-30 21:23:13 +00:00
|
|
|
if (TasmotaGlobal.i2c_enabled[1] ) {
|
2023-03-22 12:25:55 +00:00
|
|
|
if(!I2cSetDevice(SEN5X_ADDRESS, 1)) {
|
2023-01-29 07:06:25 +00:00
|
|
|
DEBUG_SENSOR_LOG(PSTR("Sensirion SEN5X not found, i2c bus 1"));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
usingI2cBus = 1;
|
2023-03-22 12:25:55 +00:00
|
|
|
} else {
|
2023-01-29 07:06:25 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#else
|
2023-03-22 12:25:55 +00:00
|
|
|
if (!I2cSetDevice(SEN5X_ADDRESS)) {
|
2023-01-29 07:06:25 +00:00
|
|
|
DEBUG_SENSOR_LOG(PSTR("Sensirion SEN5X not found, i2c bus 0"));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
#endif
|
2023-03-22 12:25:55 +00:00
|
|
|
|
|
|
|
sen5x = new SensirionI2CSen5x();
|
|
|
|
if (1 == usingI2cBus) {
|
2024-10-11 13:11:00 +01:00
|
|
|
#if defined(ESP32) && defined(USE_I2C_BUS2)
|
2023-01-29 07:06:25 +00:00
|
|
|
sen5x->begin(Wire1);
|
|
|
|
#else
|
|
|
|
sen5x->begin(Wire);
|
|
|
|
#endif
|
2023-03-22 12:25:55 +00:00
|
|
|
}
|
2023-01-29 07:06:25 +00:00
|
|
|
else {
|
|
|
|
sen5x->begin(Wire);
|
2023-03-22 12:25:55 +00:00
|
|
|
}
|
2023-08-26 15:14:30 +01:00
|
|
|
|
|
|
|
if (!Settings->flag6.sen5x_passive_mode) {
|
|
|
|
int error_stop = sen5x->deviceReset();
|
|
|
|
if (error_stop != 0) {
|
|
|
|
DEBUG_SENSOR_LOG(PSTR("Sensirion SEN5X failed to reset device (I2C Bus %d)"), usingI2cBus);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
// Wait 1 second for sensors to start recording + 100ms for reset command
|
|
|
|
delay(1100);
|
|
|
|
int error_start = sen5x->startMeasurement();
|
|
|
|
if (error_start != 0) {
|
|
|
|
DEBUG_SENSOR_LOG(PSTR("Sensirion SEN5X failed to start measurement (I2C Bus %d)"), usingI2cBus);
|
|
|
|
return;
|
|
|
|
}
|
2023-01-29 07:06:25 +00:00
|
|
|
}
|
|
|
|
|
2023-03-22 12:25:55 +00:00
|
|
|
SEN5XDATA = (SEN5XDATA_s *)calloc(1, sizeof(struct SEN5XDATA_s));
|
|
|
|
I2cSetActiveFound(SEN5X_ADDRESS, "SEN5X", usingI2cBus);
|
2023-01-29 07:06:25 +00:00
|
|
|
}
|
|
|
|
|
2023-03-22 12:25:55 +00:00
|
|
|
void SEN5XUpdate(void) { // Perform every second to ensure proper operation of the baseline compensation algorithm
|
2023-01-29 07:06:25 +00:00
|
|
|
uint16_t error;
|
|
|
|
char errorMessage[256];
|
|
|
|
DEBUG_SENSOR_LOG(PSTR("Running readMeasuredValues for SEN5X..."));
|
|
|
|
|
|
|
|
error = sen5x->readMeasuredValues(
|
|
|
|
SEN5XDATA->massConcentrationPm1p0, SEN5XDATA->massConcentrationPm2p5, SEN5XDATA->massConcentrationPm4p0,
|
|
|
|
SEN5XDATA->massConcentrationPm10p0, SEN5XDATA->ambientHumidity, SEN5XDATA->ambientTemperature, SEN5XDATA->vocIndex,
|
|
|
|
SEN5XDATA->noxIndex);
|
|
|
|
|
2023-03-22 12:25:55 +00:00
|
|
|
if (error) {
|
|
|
|
AddLog(LOG_LEVEL_DEBUG, PSTR("S5X: Failed to retrieve readings"));
|
|
|
|
#ifdef DEBUG_TASMOTA_SENSOR
|
|
|
|
DEBUG_SENSOR_LOG(PSTR("Error trying to execute readMeasuredValues():"));
|
2023-01-29 07:06:25 +00:00
|
|
|
errorToString(error, errorMessage, 256);
|
|
|
|
DEBUG_SENSOR_LOG(errorMessage);
|
2023-03-22 12:25:55 +00:00
|
|
|
} else {
|
2023-01-29 07:06:25 +00:00
|
|
|
DEBUG_SENSOR_LOG(PSTR("SEN5x readings:-"));
|
2023-03-22 12:25:55 +00:00
|
|
|
DEBUG_SENSOR_LOG(PSTR("MassConcentrationPm1p0: %1_f"), &SEN5XDATA->massConcentrationPm1p0);
|
|
|
|
DEBUG_SENSOR_LOG(PSTR("MassConcentrationPm2p5: %1_f"), &SEN5XDATA->massConcentrationPm2p5);
|
|
|
|
DEBUG_SENSOR_LOG(PSTR("MassConcentrationPm4p0: %1_f"), &SEN5XDATA->massConcentrationPm4p0);
|
|
|
|
DEBUG_SENSOR_LOG(PSTR("MassConcentrationPm10p0: %1_f"), &SEN5XDATA->massConcentrationPm10p0);
|
|
|
|
if (isnan(SEN5XDATA->ambientHumidity)) {
|
|
|
|
DEBUG_SENSOR_LOG(PSTR("AmbientHumidity: n/a"));
|
|
|
|
} else {
|
|
|
|
DEBUG_SENSOR_LOG(PSTR("AmbientHumidity: %*_f"), 2, &SEN5XDATA->ambientHumidity);
|
2023-01-29 07:06:25 +00:00
|
|
|
}
|
|
|
|
|
2023-03-22 12:25:55 +00:00
|
|
|
if (isnan(SEN5XDATA->ambientTemperature)) {
|
|
|
|
DEBUG_SENSOR_LOG(PSTR("AmbientTemperature: n/a"));
|
|
|
|
} else {
|
|
|
|
DEBUG_SENSOR_LOG(PSTR("AmbientTemperature: %*_f"), 2, &SEN5XDATA->ambientTemperature);
|
2023-01-29 07:06:25 +00:00
|
|
|
}
|
2023-03-22 12:25:55 +00:00
|
|
|
|
|
|
|
if (isnan(SEN5XDATA->vocIndex)) {
|
|
|
|
DEBUG_SENSOR_LOG(PSTR("VocIndex: n/a"));
|
|
|
|
} else {
|
|
|
|
DEBUG_SENSOR_LOG(PSTR("VocIndex: %*_f"), 0, &SEN5XDATA->vocIndex);
|
2023-01-29 07:06:25 +00:00
|
|
|
}
|
2023-03-22 12:25:55 +00:00
|
|
|
|
|
|
|
if (isnan(SEN5XDATA->noxIndex)) {
|
|
|
|
DEBUG_SENSOR_LOG(PSTR("NoxIndex: n/a"));
|
|
|
|
} else {
|
|
|
|
DEBUG_SENSOR_LOG(PSTR("NoxIndex: %*_f"), 0, &SEN5XDATA->noxIndex);
|
2023-01-29 07:06:25 +00:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-22 12:25:55 +00:00
|
|
|
void SEN5XShow(bool json) {
|
|
|
|
char types[10];
|
|
|
|
strcpy_P(types, PSTR("SEN5X"));
|
|
|
|
|
|
|
|
float temperature = 0;
|
|
|
|
float humidity = 0;
|
|
|
|
float abs_humidity = 0;
|
|
|
|
bool ahum_available = (!isnan(SEN5XDATA->ambientTemperature) && !isnan(SEN5XDATA->ambientHumidity) && (SEN5XDATA->ambientHumidity > 0));
|
|
|
|
if (ahum_available) {
|
|
|
|
temperature = ConvertTemp(SEN5XDATA->ambientTemperature);
|
|
|
|
humidity = ConvertHumidity(SEN5XDATA->ambientHumidity);
|
|
|
|
abs_humidity = CalcTempHumToAbsHum(SEN5XDATA->ambientTemperature, SEN5XDATA->ambientHumidity);
|
|
|
|
}
|
2023-01-29 07:06:25 +00:00
|
|
|
|
2023-03-22 12:25:55 +00:00
|
|
|
if (json) {
|
|
|
|
ResponseAppend_P(PSTR(",\"%s\":{\"PM1\":%1_f,\"PM2.5\":%1_f,\"PM4\":%1_f,\"PM10\":%1_f,"),
|
|
|
|
types,
|
|
|
|
&SEN5XDATA->massConcentrationPm1p0, &SEN5XDATA->massConcentrationPm2p5, &SEN5XDATA->massConcentrationPm4p0, &SEN5XDATA->massConcentrationPm10p0);
|
|
|
|
if (!isnan(SEN5XDATA->noxIndex)) {
|
|
|
|
ResponseAppend_P(PSTR("\"NOx\":%0_f,"), &SEN5XDATA->noxIndex);
|
2023-01-29 07:06:25 +00:00
|
|
|
}
|
2023-03-22 12:25:55 +00:00
|
|
|
if (!isnan(SEN5XDATA->vocIndex)) {
|
|
|
|
ResponseAppend_P(PSTR("\"VOC\":%0_f,"), &SEN5XDATA->vocIndex);
|
2023-01-29 07:06:25 +00:00
|
|
|
}
|
2023-03-22 12:25:55 +00:00
|
|
|
if (ahum_available) {
|
|
|
|
ResponseAppendTHD(temperature, humidity);
|
2023-03-30 07:49:48 +01:00
|
|
|
ResponseAppend_P(PSTR(",\"" D_JSON_AHUM "\":%4_f"), &abs_humidity);
|
2023-03-22 12:25:55 +00:00
|
|
|
}
|
|
|
|
ResponseJsonEnd();
|
2023-01-29 07:06:25 +00:00
|
|
|
#ifdef USE_WEBSERVER
|
2023-03-22 12:25:55 +00:00
|
|
|
} else {
|
|
|
|
WSContentSend_PD(HTTP_SNS_F_ENVIRONMENTAL_CONCENTRATION, types, "1", &SEN5XDATA->massConcentrationPm1p0);
|
|
|
|
WSContentSend_PD(HTTP_SNS_F_ENVIRONMENTAL_CONCENTRATION, types, "2.5", &SEN5XDATA->massConcentrationPm2p5);
|
|
|
|
WSContentSend_PD(HTTP_SNS_F_ENVIRONMENTAL_CONCENTRATION, types, "4", &SEN5XDATA->massConcentrationPm4p0);
|
|
|
|
WSContentSend_PD(HTTP_SNS_F_ENVIRONMENTAL_CONCENTRATION, types, "10", &SEN5XDATA->massConcentrationPm10p0);
|
|
|
|
if (!isnan(SEN5XDATA->noxIndex)) {
|
|
|
|
WSContentSend_PD(HTTP_SNS_F_NOX, types, 0, &SEN5XDATA->noxIndex);
|
|
|
|
}
|
|
|
|
if (!isnan(SEN5XDATA->vocIndex)) {
|
|
|
|
WSContentSend_PD(HTTP_SNS_F_VOC, types, 0, &SEN5XDATA->vocIndex);
|
|
|
|
}
|
|
|
|
if (ahum_available) {
|
|
|
|
WSContentSend_THD(types, temperature, humidity);
|
2023-03-30 07:49:48 +01:00
|
|
|
WSContentSend_PD(HTTP_SNS_F_ABS_HUM, types, 4, &abs_humidity);
|
2023-03-22 12:25:55 +00:00
|
|
|
}
|
2023-01-29 07:06:25 +00:00
|
|
|
#endif
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*********************************************************************************************\
|
|
|
|
* Interface
|
|
|
|
\*********************************************************************************************/
|
|
|
|
|
2023-03-22 12:25:55 +00:00
|
|
|
bool Xsns103(uint32_t function) {
|
|
|
|
if (!I2cEnabled(XI2C_76)) { return false; }
|
2023-01-29 07:06:25 +00:00
|
|
|
|
|
|
|
bool result = false;
|
|
|
|
|
2023-03-22 12:25:55 +00:00
|
|
|
if (FUNC_INIT == function) {
|
2023-01-29 07:06:25 +00:00
|
|
|
sen5x_Init();
|
|
|
|
}
|
2023-03-22 12:25:55 +00:00
|
|
|
else if (SEN5XDATA != nullptr) {
|
|
|
|
switch (function) {
|
2023-01-29 07:06:25 +00:00
|
|
|
case FUNC_EVERY_SECOND:
|
2023-08-26 15:14:30 +01:00
|
|
|
if (Settings->flag6.sen5x_passive_mode) {
|
|
|
|
if (TasmotaGlobal.uptime % SEN5X_PASSIVE_MODE_INTERVAL == 0) {
|
|
|
|
SEN5XUpdate();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
SEN5XUpdate();
|
|
|
|
}
|
2023-01-29 07:06:25 +00:00
|
|
|
break;
|
|
|
|
case FUNC_JSON_APPEND:
|
|
|
|
SEN5XShow(1);
|
|
|
|
break;
|
|
|
|
#ifdef USE_WEBSERVER
|
|
|
|
case FUNC_WEB_SENSOR:
|
|
|
|
SEN5XShow(0);
|
|
|
|
break;
|
|
|
|
#endif // USE_WEBSERVER
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // USE_SEN5X
|
|
|
|
#endif // USE_I2C
|