xsns_20_novasds.ino 8.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243
  1. /*
  2. xsns_20_novasds.ino - Nova SDS011/SDS021 particle concentration sensor support for Sonoff-Tasmota
  3. Copyright (C) 2018 Theo Arends
  4. This program is free software: you can redistribute it and/or modify
  5. it under the terms of the GNU General Public License as published by
  6. the Free Software Foundation, either version 3 of the License, or
  7. (at your option) any later version.
  8. This program is distributed in the hope that it will be useful,
  9. but WITHOUT ANY WARRANTY; without even the implied warranty of
  10. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  11. GNU General Public License for more details.
  12. You should have received a copy of the GNU General Public License
  13. along with this program. If not, see <http://www.gnu.org/licenses/>.
  14. */
  15. #ifdef USE_NOVA_SDS
  16. /*********************************************************************************************\
  17. * Nova Fitness SDS011 (and possibly SDS021) particle concentration sensor
  18. * For background information see http://aqicn.org/sensor/sds011/
  19. * For protocol specification see
  20. * https://cdn.sparkfun.com/assets/parts/1/2/2/7/5/Laser_Dust_Sensor_Control_Protocol_V1.3.pdf
  21. *
  22. * Hardware Serial will be selected if GPIO3 = [SDS0X01]
  23. \*********************************************************************************************/
  24. #define XSNS_20 20
  25. #include <TasmotaSerial.h>
  26. #ifndef WORKING_PERIOD
  27. #define WORKING_PERIOD 5 // NodaSDS sleep working period in minutes
  28. #endif
  29. #ifndef NOVA_SDS_REINIT_CHECK
  30. #define NOVA_SDS_REINIT_CHECK 80 // NodaSDS reinitalized check in seconds
  31. #endif
  32. #ifndef NOVA_SDS_QUERY_INTERVAL
  33. #define NOVA_SDS_QUERY_INTERVAL 3 // NodaSDS query interval in seconds
  34. #endif
  35. #ifndef NOVA_SDS_RECDATA_TIMEOUT
  36. #define NOVA_SDS_RECDATA_TIMEOUT 150 // NodaSDS query data timeout in ms
  37. #endif
  38. #ifndef NOVA_SDS_DEVICE_ID
  39. #define NOVA_SDS_DEVICE_ID 0xFFFF // NodaSDS all sensor response
  40. #endif
  41. TasmotaSerial *NovaSdsSerial;
  42. uint8_t novasds_type = 1;
  43. uint8_t novasds_valid = 0;
  44. struct sds011data {
  45. uint16_t pm100;
  46. uint16_t pm25;
  47. } novasds_data;
  48. // NovaSDS commands
  49. #define NOVA_SDS_REPORTING_MODE 2 // Cmnd "data reporting mode"
  50. #define NOVA_SDS_QUERY_DATA 4 // Cmnd "Query data"
  51. #define NOVA_SDS_SET_DEVICE_ID 5 // Cmnd "Set Device ID"
  52. #define NOVA_SDS_SLEEP_AND_WORK 6 // Cmnd "sleep and work mode"
  53. #define NOVA_SDS_WORKING_PERIOD 8 // Cmnd "working period"
  54. #define NOVA_SDS_CHECK_FIRMWARE_VER 7 // Cmnd "Check firmware version"
  55. #define NOVA_SDS_QUERY_MODE 0 // Subcmnd "query mode"
  56. #define NOVA_SDS_SET_MODE 1 // Subcmnd "set mode"
  57. #define NOVA_SDS_REPORT_ACTIVE 0 // Subcmnd "report active mode" - Sensor received query data command to report a measurement data
  58. #define NOVA_SDS_REPORT_QUERY 1 // Subcmnd "report query mode" - Sensor automatically reports a measurement data in a work period
  59. #define NOVA_SDS_WORK 0 // Subcmnd "work mode"
  60. #define NOVA_SDS_SLEEP 1 // Subcmnd "sleep mode"
  61. bool NovaSdsCommand(uint8_t byte1, uint8_t byte2, uint8_t byte3, uint16_t sensorid, byte *buffer)
  62. {
  63. uint8_t novasds_cmnd[19] = {0xAA, 0xB4, byte1, byte2, byte3, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, (uint8_t)(sensorid & 0xFF), (uint8_t)((sensorid>>8) & 0xFF), 0x00, 0xAB};
  64. // calc crc
  65. for (byte i = 2; i < 17; i++) {
  66. novasds_cmnd[17] += novasds_cmnd[i];
  67. }
  68. //~ snprintf_P(log_data, sizeof(log_data), PSTR("SDS: Send %02X %02X %02X %02X %02X %02X %02X %02X %02X %02X %02X %02X %02X %02X %02X %02X %02X %02X %02X"),
  69. //~ novasds_cmnd[0],novasds_cmnd[1],novasds_cmnd[2],novasds_cmnd[3],novasds_cmnd[4],novasds_cmnd[5],novasds_cmnd[6],novasds_cmnd[7],novasds_cmnd[8],novasds_cmnd[9],
  70. //~ novasds_cmnd[10],novasds_cmnd[11],novasds_cmnd[12],novasds_cmnd[13],novasds_cmnd[14],novasds_cmnd[15],novasds_cmnd[16],novasds_cmnd[17],novasds_cmnd[18]);
  71. //~ AddLog(LOG_LEVEL_DEBUG);
  72. // send cmnd
  73. NovaSdsSerial->write(novasds_cmnd, sizeof(novasds_cmnd));
  74. NovaSdsSerial->flush();
  75. // wait for any response
  76. unsigned long cmndtime = millis();
  77. while ( (TimePassedSince(cmndtime) < NOVA_SDS_RECDATA_TIMEOUT) && ( ! NovaSdsSerial->available() ) );
  78. if ( ! NovaSdsSerial->available() ) {
  79. // timeout
  80. return false;
  81. }
  82. byte recbuf[10];
  83. memset(recbuf, 0, sizeof(recbuf));
  84. // sync to 0xAA header
  85. while ( (TimePassedSince(cmndtime) < NOVA_SDS_RECDATA_TIMEOUT) && ( NovaSdsSerial->available() > 0) && (0xAA != (recbuf[0] = NovaSdsSerial->read())) );
  86. if ( 0xAA != recbuf[0] ) {
  87. // no head found
  88. return false;
  89. }
  90. // read rest (9 of 10 bytes) of message
  91. NovaSdsSerial->readBytes(&recbuf[1], 9);
  92. AddLogSerial(LOG_LEVEL_DEBUG_MORE, recbuf, sizeof(recbuf));
  93. if ( NULL != buffer ) {
  94. // return data to buffer
  95. memcpy(buffer, recbuf, sizeof(recbuf));
  96. }
  97. // checksum & tail check
  98. if ((0xAB != recbuf[9] ) || (recbuf[8] != ((recbuf[2] + recbuf[3] + recbuf[4] + recbuf[5] + recbuf[6] + recbuf[7]) & 0xFF))) {
  99. AddLog_P(LOG_LEVEL_DEBUG, PSTR("SDS: " D_CHECKSUM_FAILURE));
  100. return false;
  101. }
  102. return true;
  103. }
  104. void NovaSdsSetWorkPeriod(void)
  105. {
  106. // set sensor working period
  107. NovaSdsCommand(NOVA_SDS_WORKING_PERIOD, NOVA_SDS_SET_MODE, WORKING_PERIOD, NOVA_SDS_DEVICE_ID, NULL);
  108. // set sensor report only on query
  109. NovaSdsCommand(NOVA_SDS_REPORTING_MODE, NOVA_SDS_SET_MODE, NOVA_SDS_REPORT_QUERY, NOVA_SDS_DEVICE_ID, NULL);
  110. }
  111. bool NovaSdsReadData(void)
  112. {
  113. byte d[10];
  114. if ( ! NovaSdsCommand(NOVA_SDS_QUERY_DATA, 0, 0, NOVA_SDS_DEVICE_ID, d) ) {
  115. return false;
  116. }
  117. novasds_data.pm25 = (d[2] + 256 * d[3]);
  118. novasds_data.pm100 = (d[4] + 256 * d[5]);
  119. return true;
  120. }
  121. /*********************************************************************************************/
  122. void NovaSdsSecond(void) // Every second
  123. {
  124. if (0 == (uptime % NOVA_SDS_REINIT_CHECK)) {
  125. if (!novasds_valid) {
  126. NovaSdsSetWorkPeriod();
  127. }
  128. } else if (0 == (uptime % NOVA_SDS_QUERY_INTERVAL)) {
  129. if (NovaSdsReadData()) {
  130. novasds_valid = 10;
  131. } else {
  132. if (novasds_valid) {
  133. novasds_valid--;
  134. }
  135. }
  136. }
  137. }
  138. /*********************************************************************************************/
  139. void NovaSdsInit(void)
  140. {
  141. novasds_type = 0;
  142. if (pin[GPIO_SDS0X1_RX] < 99 && pin[GPIO_SDS0X1_TX] < 99) {
  143. NovaSdsSerial = new TasmotaSerial(pin[GPIO_SDS0X1_RX], pin[GPIO_SDS0X1_TX], 1);
  144. if (NovaSdsSerial->begin(9600)) {
  145. if (NovaSdsSerial->hardwareSerial()) {
  146. ClaimSerial();
  147. }
  148. novasds_type = 1;
  149. NovaSdsSetWorkPeriod();
  150. }
  151. }
  152. }
  153. #ifdef USE_WEBSERVER
  154. const char HTTP_SDS0X1_SNS[] PROGMEM = "%s"
  155. "{s}SDS0X1 " D_ENVIRONMENTAL_CONCENTRATION " 2.5 " D_UNIT_MICROMETER "{m}%s " D_UNIT_MICROGRAM_PER_CUBIC_METER "{e}"
  156. "{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>
  157. #endif // USE_WEBSERVER
  158. void NovaSdsShow(boolean json)
  159. {
  160. if (novasds_valid) {
  161. float pm10f = (float)(novasds_data.pm100) / 10.0f;
  162. float pm2_5f = (float)(novasds_data.pm25) / 10.0f;
  163. char pm10[33];
  164. dtostrfd(pm10f, 1, pm10);
  165. char pm2_5[33];
  166. dtostrfd(pm2_5f, 1, pm2_5);
  167. if (json) {
  168. snprintf_P(mqtt_data, sizeof(mqtt_data), PSTR("%s,\"SDS0X1\":{\"PM2.5\":%s,\"PM10\":%s}"), mqtt_data, pm2_5, pm10);
  169. #ifdef USE_DOMOTICZ
  170. if (0 == tele_period) {
  171. DomoticzSensor(DZ_VOLTAGE, pm2_5); // PM2.5
  172. DomoticzSensor(DZ_CURRENT, pm10); // PM10
  173. }
  174. #endif // USE_DOMOTICZ
  175. #ifdef USE_WEBSERVER
  176. } else {
  177. snprintf_P(mqtt_data, sizeof(mqtt_data), HTTP_SDS0X1_SNS, mqtt_data, pm2_5, pm10);
  178. #endif // USE_WEBSERVER
  179. }
  180. }
  181. }
  182. /*********************************************************************************************\
  183. * Interface
  184. \*********************************************************************************************/
  185. boolean Xsns20(byte function)
  186. {
  187. boolean result = false;
  188. if (novasds_type) {
  189. switch (function) {
  190. case FUNC_INIT:
  191. NovaSdsInit();
  192. break;
  193. case FUNC_EVERY_SECOND:
  194. NovaSdsSecond();
  195. break;
  196. case FUNC_JSON_APPEND:
  197. NovaSdsShow(1);
  198. break;
  199. #ifdef USE_WEBSERVER
  200. case FUNC_WEB_APPEND:
  201. NovaSdsShow(0);
  202. break;
  203. #endif // USE_WEBSERVER
  204. }
  205. }
  206. return result;
  207. }
  208. #endif // USE_NOVA_SDS