mirror of https://github.com/arendst/Tasmota.git
170 lines
5.4 KiB
Arduino
170 lines
5.4 KiB
Arduino
|
/*
|
||
|
xsns_91_vindriktning.ino - IKEA vindriktning particle concentration sensor support for Tasmota
|
||
|
|
||
|
Copyright (C) 2021 Marcel Ritter and 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_VINDRIKTNING
|
||
|
/*********************************************************************************************\
|
||
|
* IKEA VINDRIKTNING particle concentration sensor
|
||
|
\*********************************************************************************************/
|
||
|
|
||
|
#define XSNS_91 91
|
||
|
|
||
|
#include <TasmotaSerial.h>
|
||
|
|
||
|
#ifndef MIN_INTERVAL_PERIOD
|
||
|
#define MIN_INTERVAL_PERIOD 60 // minimum interval period in seconds required for passive mode
|
||
|
#endif
|
||
|
|
||
|
#define VINDRIKTNING_DATASET_SIZE 20
|
||
|
|
||
|
TasmotaSerial *VindriktningSerial;
|
||
|
|
||
|
struct VINDRIKTNING {
|
||
|
uint16_t pm2_5 = 0;
|
||
|
uint16_t pm1_0 = 0;
|
||
|
uint16_t pm10 = 0;
|
||
|
uint8_t type = 1;
|
||
|
uint8_t valid = 0;
|
||
|
bool discovery_triggered = false;
|
||
|
} Vindriktning;
|
||
|
|
||
|
bool VindriktningReadData(void) {
|
||
|
if (!VindriktningSerial->available()) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
int serial_in_byte_counter = 0;
|
||
|
uint8_t buffer[VINDRIKTNING_DATASET_SIZE];
|
||
|
uint8_t crc = 0;
|
||
|
|
||
|
while (VindriktningSerial->available()) {
|
||
|
uint8_t serial_in_byte = VindriktningSerial->read();
|
||
|
if (serial_in_byte_counter <= VINDRIKTNING_DATASET_SIZE -1) {
|
||
|
buffer[serial_in_byte_counter++] = serial_in_byte;
|
||
|
crc += serial_in_byte;
|
||
|
}
|
||
|
}
|
||
|
VindriktningSerial->flush(); // Make room for another burst
|
||
|
|
||
|
AddLogBuffer(LOG_LEVEL_DEBUG_MORE, buffer, VINDRIKTNING_DATASET_SIZE);
|
||
|
|
||
|
if (serial_in_byte_counter < VINDRIKTNING_DATASET_SIZE) {
|
||
|
AddLog(LOG_LEVEL_DEBUG, PSTR("VDN: Not enough data (%d < 20)"), serial_in_byte_counter);
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
if (crc != 0) {
|
||
|
AddLog(LOG_LEVEL_DEBUG, PSTR("VDN: " D_CHECKSUM_FAILURE));
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// sample data:
|
||
|
// 16 11 0b 00 00 00 0c 00 00 03 cb 00 00 00 0c 01 00 00 00 e7
|
||
|
// |pm2_5| |pm1_0| |pm10 | | CRC |
|
||
|
Vindriktning.pm2_5 = (buffer[5] << 8) | buffer[6];
|
||
|
Vindriktning.pm1_0 = (buffer[9] << 8) | buffer[10];
|
||
|
Vindriktning.pm10 = (buffer[13] << 8) | buffer[14];
|
||
|
|
||
|
if (!Vindriktning.discovery_triggered) {
|
||
|
TasmotaGlobal.discovery_counter = 1; // force TasDiscovery()
|
||
|
Vindriktning.discovery_triggered = true;
|
||
|
}
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
/*********************************************************************************************/
|
||
|
|
||
|
void VindriktningSecond(void) { // Every second
|
||
|
if (VindriktningReadData()) {
|
||
|
Vindriktning.valid = MIN_INTERVAL_PERIOD;
|
||
|
} else {
|
||
|
if (Vindriktning.valid) {
|
||
|
Vindriktning.valid--;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/*********************************************************************************************/
|
||
|
|
||
|
void VindriktningInit(void) {
|
||
|
Vindriktning.type = 0;
|
||
|
if (PinUsed(GPIO_VINDRIKTNING_RX)) {
|
||
|
VindriktningSerial = new TasmotaSerial(Pin(GPIO_VINDRIKTNING_RX), -1, 1);
|
||
|
if (VindriktningSerial->begin(9600)) {
|
||
|
if (VindriktningSerial->hardwareSerial()) { ClaimSerial(); }
|
||
|
Vindriktning.type = 1;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#ifdef USE_WEBSERVER
|
||
|
const char HTTP_VINDRIKTNING_SNS[] PROGMEM =
|
||
|
"{s}VINDRIKTNING " D_ENVIRONMENTAL_CONCENTRATION " 1.0 " D_UNIT_MICROMETER "{m}%d " D_UNIT_MICROGRAM_PER_CUBIC_METER "{e}"
|
||
|
"{s}VINDRIKTNING " D_ENVIRONMENTAL_CONCENTRATION " 2.5 " D_UNIT_MICROMETER "{m}%d " D_UNIT_MICROGRAM_PER_CUBIC_METER "{e}"
|
||
|
"{s}VINDRIKTNING " D_ENVIRONMENTAL_CONCENTRATION " 10 " D_UNIT_MICROMETER "{m}%d " D_UNIT_MICROGRAM_PER_CUBIC_METER "{e}"; // {s} = <tr><th>, {m} = </th><td>, {e} = </td></tr>
|
||
|
#endif // USE_WEBSERVER
|
||
|
|
||
|
void VindriktningShow(bool json) {
|
||
|
if (Vindriktning.valid) {
|
||
|
if (json) {
|
||
|
ResponseAppend_P(PSTR(",\"VINDRIKTNING\":{\"PM1\":%d,\"PM2.5\":%d,\"PM10\":%d}"),
|
||
|
Vindriktning.pm1_0, Vindriktning.pm2_5, Vindriktning.pm10);
|
||
|
#ifdef USE_DOMOTICZ
|
||
|
if (0 == TasmotaGlobal.tele_period) {
|
||
|
DomoticzSensor(DZ_COUNT, Vindriktning.pm1_0); // PM1.0
|
||
|
DomoticzSensor(DZ_VOLTAGE, Vindriktning.pm2_5); // PM2.5
|
||
|
DomoticzSensor(DZ_CURRENT, Vindriktning.pm10); // PM10
|
||
|
}
|
||
|
#endif // USE_DOMOTICZ
|
||
|
#ifdef USE_WEBSERVER
|
||
|
} else {
|
||
|
WSContentSend_PD(HTTP_VINDRIKTNING_SNS, Vindriktning.pm1_0, Vindriktning.pm2_5, Vindriktning.pm10);
|
||
|
#endif // USE_WEBSERVER
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/*********************************************************************************************\
|
||
|
* Interface
|
||
|
\*********************************************************************************************/
|
||
|
|
||
|
bool Xsns91(uint8_t function) {
|
||
|
bool result = false;
|
||
|
|
||
|
if (Vindriktning.type) {
|
||
|
switch (function) {
|
||
|
case FUNC_EVERY_SECOND:
|
||
|
VindriktningSecond();
|
||
|
break;
|
||
|
case FUNC_JSON_APPEND:
|
||
|
VindriktningShow(1);
|
||
|
break;
|
||
|
#ifdef USE_WEBSERVER
|
||
|
case FUNC_WEB_SENSOR:
|
||
|
VindriktningShow(0);
|
||
|
break;
|
||
|
#endif // USE_WEBSERVER
|
||
|
case FUNC_INIT:
|
||
|
VindriktningInit();
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
#endif // USE_VINDRIKTNING
|