| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186 |
- /*
- xsns_18_pms5003.ino - PMS5003-7003 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_PMS5003
- /*********************************************************************************************\
- * PlanTower PMS5003 and PMS7003 particle concentration sensor
- * For background information see http://aqicn.org/sensor/pms5003-7003/
- *
- * Hardware Serial will be selected if GPIO3 = [PMS5003]
- \*********************************************************************************************/
- #define XSNS_18 18
- #include <TasmotaSerial.h>
- TasmotaSerial *PmsSerial;
- uint8_t pms_type = 1;
- uint8_t pms_valid = 0;
- struct pms5003data {
- uint16_t framelen;
- uint16_t pm10_standard, pm25_standard, pm100_standard;
- uint16_t pm10_env, pm25_env, pm100_env;
- uint16_t particles_03um, particles_05um, particles_10um, particles_25um, particles_50um, particles_100um;
- uint16_t unused;
- uint16_t checksum;
- } pms_data;
- /*********************************************************************************************/
- boolean PmsReadData(void)
- {
- if (! PmsSerial->available()) {
- return false;
- }
- while ((PmsSerial->peek() != 0x42) && PmsSerial->available()) {
- PmsSerial->read();
- }
- if (PmsSerial->available() < 32) {
- return false;
- }
- uint8_t buffer[32];
- uint16_t sum = 0;
- PmsSerial->readBytes(buffer, 32);
- PmsSerial->flush(); // Make room for another burst
- AddLogSerial(LOG_LEVEL_DEBUG_MORE, buffer, 32);
- // get checksum ready
- for (uint8_t i = 0; i < 30; i++) {
- sum += buffer[i];
- }
- // The data comes in endian'd, this solves it so it works on all platforms
- uint16_t buffer_u16[15];
- for (uint8_t i = 0; i < 15; i++) {
- buffer_u16[i] = buffer[2 + i*2 + 1];
- buffer_u16[i] += (buffer[2 + i*2] << 8);
- }
- if (sum != buffer_u16[14]) {
- AddLog_P(LOG_LEVEL_DEBUG, PSTR("PMS: " D_CHECKSUM_FAILURE));
- return false;
- }
- memcpy((void *)&pms_data, (void *)buffer_u16, 30);
- pms_valid = 10;
- return true;
- }
- /*********************************************************************************************/
- void PmsSecond(void) // Every second
- {
- if (PmsReadData()) {
- pms_valid = 10;
- } else {
- if (pms_valid) {
- pms_valid--;
- }
- }
- }
- /*********************************************************************************************/
- void PmsInit(void)
- {
- pms_type = 0;
- if (pin[GPIO_PMS5003] < 99) {
- PmsSerial = new TasmotaSerial(pin[GPIO_PMS5003], -1, 1);
- if (PmsSerial->begin(9600)) {
- if (PmsSerial->hardwareSerial()) { ClaimSerial(); }
- pms_type = 1;
- }
- }
- }
- #ifdef USE_WEBSERVER
- const char HTTP_PMS5003_SNS[] PROGMEM = "%s"
- // "{s}PMS5003 " D_STANDARD_CONCENTRATION " 1 " D_UNIT_MICROMETER "{m}%d " D_UNIT_MICROGRAM_PER_CUBIC_METER "{e}"
- // "{s}PMS5003 " D_STANDARD_CONCENTRATION " 2.5 " D_UNIT_MICROMETER "{m}%d " D_UNIT_MICROGRAM_PER_CUBIC_METER "{e}"
- // "{s}PMS5003 " D_STANDARD_CONCENTRATION " 10 " D_UNIT_MICROMETER "{m}%d " D_UNIT_MICROGRAM_PER_CUBIC_METER "{e}"
- "{s}PMS5003 " D_ENVIRONMENTAL_CONCENTRATION " 1 " D_UNIT_MICROMETER "{m}%d " D_UNIT_MICROGRAM_PER_CUBIC_METER "{e}"
- "{s}PMS5003 " D_ENVIRONMENTAL_CONCENTRATION " 2.5 " D_UNIT_MICROMETER "{m}%d " D_UNIT_MICROGRAM_PER_CUBIC_METER "{e}"
- "{s}PMS5003 " D_ENVIRONMENTAL_CONCENTRATION " 10 " D_UNIT_MICROMETER "{m}%d " D_UNIT_MICROGRAM_PER_CUBIC_METER "{e}"
- "{s}PMS5003 " D_PARTICALS_BEYOND " 0.3 " D_UNIT_MICROMETER "{m}%d " D_UNIT_PARTS_PER_DECILITER "{e}"
- "{s}PMS5003 " D_PARTICALS_BEYOND " 0.5 " D_UNIT_MICROMETER "{m}%d " D_UNIT_PARTS_PER_DECILITER "{e}"
- "{s}PMS5003 " D_PARTICALS_BEYOND " 1 " D_UNIT_MICROMETER "{m}%d " D_UNIT_PARTS_PER_DECILITER "{e}"
- "{s}PMS5003 " D_PARTICALS_BEYOND " 2.5 " D_UNIT_MICROMETER "{m}%d " D_UNIT_PARTS_PER_DECILITER "{e}"
- "{s}PMS5003 " D_PARTICALS_BEYOND " 5 " D_UNIT_MICROMETER "{m}%d " D_UNIT_PARTS_PER_DECILITER "{e}"
- "{s}PMS5003 " D_PARTICALS_BEYOND " 10 " D_UNIT_MICROMETER "{m}%d " D_UNIT_PARTS_PER_DECILITER "{e}"; // {s} = <tr><th>, {m} = </th><td>, {e} = </td></tr>
- #endif // USE_WEBSERVER
- void PmsShow(boolean json)
- {
- if (pms_valid) {
- if (json) {
- snprintf_P(mqtt_data, sizeof(mqtt_data), PSTR("%s,\"PMS5003\":{\"CF1\":%d,\"CF2.5\":%d,\"CF10\":%d,\"PM1\":%d,\"PM2.5\":%d,\"PM10\":%d,\"PB0.3\":%d,\"PB0.5\":%d,\"PB1\":%d,\"PB2.5\":%d,\"PB5\":%d,\"PB10\":%d}"), mqtt_data,
- pms_data.pm10_standard, pms_data.pm25_standard, pms_data.pm100_standard,
- pms_data.pm10_env, pms_data.pm25_env, pms_data.pm100_env,
- pms_data.particles_03um, pms_data.particles_05um, pms_data.particles_10um, pms_data.particles_25um, pms_data.particles_50um, pms_data.particles_100um);
- #ifdef USE_DOMOTICZ
- if (0 == tele_period) {
- DomoticzSensor(DZ_COUNT, pms_data.pm10_env); // PM1
- DomoticzSensor(DZ_VOLTAGE, pms_data.pm25_env); // PM2.5
- DomoticzSensor(DZ_CURRENT, pms_data.pm100_env); // PM10
- }
- #endif // USE_DOMOTICZ
- #ifdef USE_WEBSERVER
- } else {
- snprintf_P(mqtt_data, sizeof(mqtt_data), HTTP_PMS5003_SNS, mqtt_data,
- // pms_data.pm10_standard, pms_data.pm25_standard, pms_data.pm100_standard,
- pms_data.pm10_env, pms_data.pm25_env, pms_data.pm100_env,
- pms_data.particles_03um, pms_data.particles_05um, pms_data.particles_10um, pms_data.particles_25um, pms_data.particles_50um, pms_data.particles_100um);
- #endif // USE_WEBSERVER
- }
- }
- }
- /*********************************************************************************************\
- * Interface
- \*********************************************************************************************/
- boolean Xsns18(byte function)
- {
- boolean result = false;
- if (pms_type) {
- switch (function) {
- case FUNC_INIT:
- PmsInit();
- break;
- case FUNC_EVERY_SECOND:
- PmsSecond();
- break;
- case FUNC_JSON_APPEND:
- PmsShow(1);
- break;
- #ifdef USE_WEBSERVER
- case FUNC_WEB_APPEND:
- PmsShow(0);
- break;
- #endif // USE_WEBSERVER
- }
- }
- return result;
- }
- #endif // USE_PMS5003
|