2018-03-03 14:28:30 +00:00
|
|
|
/*
|
|
|
|
xsns_20_novasds.ino - Nova SDS011/SDS021 particle concentration sensor support for Sonoff-Tasmota
|
|
|
|
|
|
|
|
Copyright (C) 2018 Theo Arends
|
|
|
|
|
|
|
|
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_NOVA_SDS
|
|
|
|
/*********************************************************************************************\
|
|
|
|
* Nova Fitness SDS011 (and possibly SDS021) particle concentration sensor
|
|
|
|
* For background information see http://aqicn.org/sensor/sds011/
|
2018-05-10 16:21:26 +01:00
|
|
|
*
|
|
|
|
* Hardware Serial will be selected if GPIO3 = [SDS0X01]
|
2018-03-03 14:28:30 +00:00
|
|
|
\*********************************************************************************************/
|
|
|
|
|
|
|
|
#include <TasmotaSerial.h>
|
|
|
|
|
2018-10-26 16:13:17 +01:00
|
|
|
#ifndef WORKING_PERIOD
|
|
|
|
#define WORKING_PERIOD 5
|
|
|
|
#endif
|
|
|
|
|
2018-03-03 14:28:30 +00:00
|
|
|
TasmotaSerial *NovaSdsSerial;
|
|
|
|
|
|
|
|
uint8_t novasds_type = 1;
|
|
|
|
uint8_t novasds_valid = 0;
|
|
|
|
|
2018-10-26 16:13:17 +01:00
|
|
|
uint8_t novasds_workperiod[19] = {0xAA, 0xB4, 0x08, 0x01, 0x05, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0x0C, 0xAB}; //5 minutes
|
|
|
|
uint8_t novasds_setquerymode[19] = {0xAA, 0xB4, 0x02, 0x01, 0x01, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0x02, 0xAB}; //query mode
|
|
|
|
uint8_t novasds_querydata[19] = {0xAA, 0xB4, 0x04, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0xFF, 0xFF, 0x02, 0xAB}; //query DATA
|
|
|
|
|
|
|
|
|
2018-03-03 14:28:30 +00:00
|
|
|
struct sds011data {
|
|
|
|
uint16_t pm100;
|
|
|
|
uint16_t pm25;
|
|
|
|
} novasds_data;
|
|
|
|
|
2018-10-26 16:13:17 +01:00
|
|
|
void NovaSdsSetWorkPeriod()
|
|
|
|
{
|
|
|
|
|
|
|
|
while (NovaSdsSerial->available() > 0) {
|
|
|
|
NovaSdsSerial->read();
|
|
|
|
}
|
|
|
|
|
|
|
|
novasds_workperiod[4] = WORKING_PERIOD;
|
|
|
|
novasds_workperiod[17] = ((novasds_workperiod[2] + novasds_workperiod[3] + novasds_workperiod[4] + novasds_workperiod[15] + novasds_workperiod[16]) & 0xFF); //checksum
|
|
|
|
|
|
|
|
NovaSdsSerial->write(novasds_workperiod, sizeof(novasds_workperiod));
|
|
|
|
NovaSdsSerial->flush();
|
|
|
|
|
|
|
|
while (NovaSdsSerial->available() > 0) {
|
|
|
|
NovaSdsSerial->read();
|
|
|
|
}
|
|
|
|
|
|
|
|
NovaSdsSerial->write(novasds_setquerymode, sizeof(novasds_setquerymode));
|
|
|
|
NovaSdsSerial->flush();
|
|
|
|
|
|
|
|
while (NovaSdsSerial->available() > 0) {
|
|
|
|
NovaSdsSerial->read();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2018-03-03 14:28:30 +00:00
|
|
|
bool NovaSdsReadData()
|
|
|
|
{
|
|
|
|
if (! NovaSdsSerial->available()) return false;
|
|
|
|
|
2018-10-26 16:13:17 +01:00
|
|
|
NovaSdsSerial->write(novasds_querydata, sizeof(novasds_querydata));
|
|
|
|
NovaSdsSerial->flush();
|
|
|
|
|
2018-03-03 14:28:30 +00:00
|
|
|
while ((NovaSdsSerial->peek() != 0xAA) && NovaSdsSerial->available()) {
|
|
|
|
NovaSdsSerial->read();
|
|
|
|
}
|
|
|
|
|
|
|
|
byte d[8] = { 0 };
|
|
|
|
NovaSdsSerial->read(); // skip 0xAA
|
|
|
|
NovaSdsSerial->readBytes(d, 8);
|
|
|
|
NovaSdsSerial->flush();
|
|
|
|
|
|
|
|
AddLogSerial(LOG_LEVEL_DEBUG_MORE, d, 8);
|
|
|
|
|
|
|
|
if (d[7] == ((d[1] + d[2] + d[3] + d[4] + d[5] + d[6]) & 0xFF)) {
|
|
|
|
novasds_data.pm25 = (d[1] + 256 * d[2]);
|
|
|
|
novasds_data.pm100 = (d[3] + 256 * d[4]);
|
|
|
|
} else {
|
|
|
|
AddLog_P(LOG_LEVEL_DEBUG, PSTR("SDS: " D_CHECKSUM_FAILURE));
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
novasds_valid = 10;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*********************************************************************************************/
|
|
|
|
|
|
|
|
void NovaSdsSecond() // Every second
|
|
|
|
{
|
|
|
|
if (NovaSdsReadData()) {
|
|
|
|
novasds_valid = 10;
|
|
|
|
} else {
|
|
|
|
if (novasds_valid) {
|
|
|
|
novasds_valid--;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*********************************************************************************************/
|
|
|
|
|
|
|
|
void NovaSdsInit()
|
|
|
|
{
|
|
|
|
novasds_type = 0;
|
2018-10-26 16:13:17 +01:00
|
|
|
if (pin[GPIO_SDS0X1_RX] < 99 && pin[GPIO_SDS0X1_TX] < 99) {
|
|
|
|
NovaSdsSerial = new TasmotaSerial(pin[GPIO_SDS0X1_RX], pin[GPIO_SDS0X1_TX], 1);
|
2018-05-10 16:21:26 +01:00
|
|
|
if (NovaSdsSerial->begin(9600)) {
|
2018-10-26 16:13:17 +01:00
|
|
|
if (NovaSdsSerial->hardwareSerial()) {
|
|
|
|
ClaimSerial();
|
|
|
|
}
|
2018-03-03 14:28:30 +00:00
|
|
|
novasds_type = 1;
|
2018-10-26 16:13:17 +01:00
|
|
|
NovaSdsSetWorkPeriod();
|
2018-03-03 14:28:30 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef USE_WEBSERVER
|
|
|
|
const char HTTP_SDS0X1_SNS[] PROGMEM = "%s"
|
2018-03-08 16:53:02 +00:00
|
|
|
"{s}SDS0X1 " D_ENVIRONMENTAL_CONCENTRATION " 2.5 " D_UNIT_MICROMETER "{m}%s " D_UNIT_MICROGRAM_PER_CUBIC_METER "{e}"
|
|
|
|
"{s}SDS0X1 " D_ENVIRONMENTAL_CONCENTRATION " 10 " D_UNIT_MICROMETER "{m}%s " D_UNIT_MICROGRAM_PER_CUBIC_METER "{e}"; // {s} = <tr><th>, {m} = </th><td>, {e} = </td></tr>
|
2018-03-03 14:28:30 +00:00
|
|
|
#endif // USE_WEBSERVER
|
|
|
|
|
|
|
|
void NovaSdsShow(boolean json)
|
|
|
|
{
|
|
|
|
if (novasds_valid) {
|
2018-03-08 16:53:02 +00:00
|
|
|
char pm10[10];
|
|
|
|
char pm2_5[10];
|
|
|
|
float pm10f = (float)(novasds_data.pm100) / 10.0f;
|
|
|
|
float pm2_5f = (float)(novasds_data.pm25) / 10.0f;
|
|
|
|
dtostrfd(pm10f, 1, pm10);
|
|
|
|
dtostrfd(pm2_5f, 1, pm2_5);
|
2018-03-03 14:28:30 +00:00
|
|
|
if (json) {
|
2018-03-08 16:53:02 +00:00
|
|
|
snprintf_P(mqtt_data, sizeof(mqtt_data), PSTR("%s,\"SDS0X1\":{\"PM2.5\":%s,\"PM10\":%s}"), mqtt_data, pm2_5, pm10);
|
2018-03-12 13:56:48 +00:00
|
|
|
#ifdef USE_DOMOTICZ
|
2018-04-11 09:11:20 +01:00
|
|
|
if (0 == tele_period) {
|
|
|
|
DomoticzSensor(DZ_VOLTAGE, pm2_5); // PM2.5
|
|
|
|
DomoticzSensor(DZ_CURRENT, pm10); // PM10
|
|
|
|
}
|
2018-03-12 13:56:48 +00:00
|
|
|
#endif // USE_DOMOTICZ
|
2018-03-03 14:28:30 +00:00
|
|
|
#ifdef USE_WEBSERVER
|
|
|
|
} else {
|
2018-03-08 16:53:02 +00:00
|
|
|
snprintf_P(mqtt_data, sizeof(mqtt_data), HTTP_SDS0X1_SNS, mqtt_data, pm2_5, pm10);
|
2018-03-03 14:28:30 +00:00
|
|
|
#endif // USE_WEBSERVER
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*********************************************************************************************\
|
|
|
|
* Interface
|
|
|
|
\*********************************************************************************************/
|
|
|
|
|
|
|
|
#define XSNS_20
|
|
|
|
|
|
|
|
boolean Xsns20(byte function)
|
|
|
|
{
|
|
|
|
boolean result = false;
|
|
|
|
|
|
|
|
if (novasds_type) {
|
|
|
|
switch (function) {
|
|
|
|
case FUNC_INIT:
|
|
|
|
NovaSdsInit();
|
|
|
|
break;
|
|
|
|
case FUNC_EVERY_SECOND:
|
|
|
|
NovaSdsSecond();
|
|
|
|
break;
|
|
|
|
case FUNC_JSON_APPEND:
|
|
|
|
NovaSdsShow(1);
|
|
|
|
break;
|
|
|
|
#ifdef USE_WEBSERVER
|
|
|
|
case FUNC_WEB_APPEND:
|
|
|
|
NovaSdsShow(0);
|
|
|
|
break;
|
|
|
|
#endif // USE_WEBSERVER
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2018-10-26 16:13:17 +01:00
|
|
|
#endif // USE_NOVA_SDS
|