xdrv_16_tuyadimmer.ino 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433
  1. /*
  2. xdrv_16_tuyadimmer.ino - Tuya dimmer support for Sonoff-Tasmota
  3. Copyright (C) 2018 digiblur, Joel Stein and 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_TUYA_DIMMER
  16. #define XDRV_16 16
  17. #ifndef TUYA_DIMMER_ID
  18. #define TUYA_DIMMER_ID 0
  19. #endif
  20. #define TUYA_POWER_ID 1
  21. #define TUYA_CMD_HEARTBEAT 0x00
  22. #define TUYA_CMD_QUERY_PRODUCT 0x01
  23. #define TUYA_CMD_MCU_CONF 0x02
  24. #define TUYA_CMD_WIFI_STATE 0x03
  25. #define TUYA_CMD_WIFI_RESET 0x04
  26. #define TUYA_CMD_WIFI_SELECT 0x05
  27. #define TUYA_CMD_SET_DP 0x06
  28. #define TUYA_CMD_STATE 0x07
  29. #define TUYA_CMD_QUERY_STATE 0x08
  30. #define TUYA_TYPE_BOOL 0x01
  31. #define TUYA_TYPE_VALUE 0x02
  32. #define TUYA_BUFFER_SIZE 256
  33. #include <TasmotaSerial.h>
  34. TasmotaSerial *TuyaSerial = nullptr;
  35. uint8_t tuya_new_dim = 0; // Tuya dimmer value temp
  36. boolean tuya_ignore_dim = false; // Flag to skip serial send to prevent looping when processing inbound states from the faceplate interaction
  37. uint8_t tuya_cmd_status = 0; // Current status of serial-read
  38. uint8_t tuya_cmd_checksum = 0; // Checksum of tuya command
  39. uint8_t tuya_data_len = 0; // Data lenght of command
  40. int8_t tuya_wifi_state = -2; // Keep MCU wifi-status in sync with WifiState()
  41. char *tuya_buffer = NULL; // Serial receive buffer
  42. int tuya_byte_counter = 0; // Index in serial receive buffer
  43. /*********************************************************************************************\
  44. * Internal Functions
  45. \*********************************************************************************************/
  46. void TuyaSendCmd(uint8_t cmd, uint8_t payload[] = nullptr, uint16_t payload_len = 0){
  47. uint8_t checksum = (0xFF + cmd + (payload_len >> 8) + (payload_len & 0xFF));
  48. TuyaSerial->write(0x55); // Tuya header 55AA
  49. TuyaSerial->write(0xAA);
  50. TuyaSerial->write((uint8_t)0x00); // version 00
  51. TuyaSerial->write(cmd); // Tuya command
  52. TuyaSerial->write(payload_len >> 8); // following data length (Hi)
  53. TuyaSerial->write(payload_len & 0xFF); // following data length (Lo)
  54. snprintf_P(log_data, sizeof(log_data), PSTR("TYA: TX Packet: \"55aa00%02x%02x%02x"), cmd, payload_len >> 8, payload_len & 0xFF);
  55. for(int i = 0; i < payload_len; ++i) {
  56. TuyaSerial->write(payload[i]);
  57. checksum += payload[i];
  58. snprintf_P(log_data, sizeof(log_data), PSTR("%s%02x"), log_data, payload[i]);
  59. }
  60. TuyaSerial->write(checksum);
  61. TuyaSerial->flush();
  62. snprintf_P(log_data, sizeof(log_data), PSTR("%s%02x\""), log_data, checksum);
  63. AddLog(LOG_LEVEL_DEBUG);
  64. }
  65. void TuyaSendState(uint8_t id, uint8_t type, uint8_t* value){
  66. uint16_t payload_len = 4;
  67. uint8_t payload_buffer[8];
  68. payload_buffer[0] = id;
  69. payload_buffer[1] = type;
  70. switch(type){
  71. case TUYA_TYPE_BOOL:
  72. payload_len += 1;
  73. payload_buffer[2] = 0x00;
  74. payload_buffer[3] = 0x01;
  75. payload_buffer[4] = value[0];
  76. break;
  77. case TUYA_TYPE_VALUE:
  78. payload_len += 4;
  79. payload_buffer[2] = 0x00;
  80. payload_buffer[3] = 0x04;
  81. payload_buffer[4] = value[3];
  82. payload_buffer[5] = value[2];
  83. payload_buffer[6] = value[1];
  84. payload_buffer[7] = value[0];
  85. break;
  86. }
  87. TuyaSendCmd(TUYA_CMD_SET_DP, payload_buffer, payload_len);
  88. }
  89. void TuyaSendBool(uint8_t id, boolean value){
  90. TuyaSendState(id, TUYA_TYPE_BOOL, &value);
  91. }
  92. void TuyaSendValue(uint8_t id, uint32_t value){
  93. TuyaSendState(id, TUYA_TYPE_VALUE, (uint8_t*)(&value));
  94. }
  95. boolean TuyaSetPower(void)
  96. {
  97. boolean status = false;
  98. uint8_t rpower = XdrvMailbox.index;
  99. int16_t source = XdrvMailbox.payload;
  100. if (source != SRC_SWITCH && TuyaSerial) { // ignore to prevent loop from pushing state from faceplate interaction
  101. snprintf_P(log_data, sizeof(log_data), PSTR("TYA: SetDevicePower.rpower=%d"), rpower);
  102. AddLog(LOG_LEVEL_DEBUG);
  103. TuyaSendBool(TUYA_POWER_ID, rpower);
  104. status = true;
  105. }
  106. return status;
  107. }
  108. boolean TuyaSetChannels(void)
  109. {
  110. LightSerialDuty(((uint8_t*)XdrvMailbox.data)[0]);
  111. return true;
  112. }
  113. void LightSerialDuty(uint8_t duty)
  114. {
  115. if (duty > 0 && !tuya_ignore_dim && TuyaSerial) {
  116. if (duty < 25) {
  117. duty = 25; // dimming acts odd below 25(10%) - this mirrors the threshold set on the faceplate itself
  118. }
  119. snprintf_P(log_data, sizeof(log_data), PSTR( "TYA: Send Serial Packet Dim Value=%d (id=%d)"), duty, Settings.param[P_TUYA_DIMMER_ID]);
  120. AddLog(LOG_LEVEL_DEBUG);
  121. TuyaSendValue(Settings.param[P_TUYA_DIMMER_ID], duty);
  122. } else {
  123. tuya_ignore_dim = false; // reset flag
  124. snprintf_P(log_data, sizeof(log_data), PSTR( "TYA: Send Dim Level skipped due to 0 or already set. Value=%d"), duty);
  125. AddLog(LOG_LEVEL_DEBUG);
  126. }
  127. }
  128. void TuyaRequestState(void){
  129. if(TuyaSerial) {
  130. // Get current status of MCU
  131. snprintf_P(log_data, sizeof(log_data), "TYA: Request MCU state");
  132. AddLog(LOG_LEVEL_DEBUG);
  133. TuyaSendCmd(TUYA_CMD_QUERY_STATE);
  134. }
  135. }
  136. void TuyaResetWifi(void)
  137. {
  138. if (!Settings.flag.button_restrict) {
  139. char scmnd[20];
  140. snprintf_P(scmnd, sizeof(scmnd), D_CMND_WIFICONFIG " %d", 2);
  141. ExecuteCommand(scmnd, SRC_BUTTON);
  142. }
  143. }
  144. void TuyaPacketProcess(void)
  145. {
  146. char scmnd[20];
  147. switch(tuya_buffer[3]) {
  148. case TUYA_CMD_HEARTBEAT:
  149. AddLog_P(LOG_LEVEL_DEBUG, PSTR("TYA: Heartbeat"));
  150. if(tuya_buffer[6] == 0){
  151. AddLog_P(LOG_LEVEL_DEBUG, PSTR("TYA: Detected MCU restart"));
  152. tuya_wifi_state = -2;
  153. }
  154. break;
  155. case TUYA_CMD_STATE:
  156. if (tuya_buffer[5] == 5) { // on/off packet
  157. snprintf_P(log_data, sizeof(log_data),PSTR("TYA: RX - %s State"),tuya_buffer[10]?"On":"Off");
  158. AddLog(LOG_LEVEL_DEBUG);
  159. if((power || Settings.light_dimmer > 0) && (power != tuya_buffer[10])) {
  160. ExecuteCommandPower(1, tuya_buffer[10], SRC_SWITCH); // send SRC_SWITCH? to use as flag to prevent loop from inbound states from faceplate interaction
  161. }
  162. }
  163. else if (tuya_buffer[5] == 8) { // dim packet
  164. snprintf_P(log_data, sizeof(log_data), PSTR("TYA: RX Dim State=%d"), tuya_buffer[13]);
  165. AddLog(LOG_LEVEL_DEBUG);
  166. if (!Settings.param[P_TUYA_DIMMER_ID]) {
  167. snprintf_P(log_data, sizeof(log_data), PSTR("TYA: Autoconfiguring Dimmer ID %d"), tuya_buffer[6]);
  168. AddLog(LOG_LEVEL_DEBUG);
  169. Settings.param[P_TUYA_DIMMER_ID] = tuya_buffer[6];
  170. }
  171. tuya_new_dim = round(tuya_buffer[13] * (100. / 255.));
  172. if((power || Settings.flag3.tuya_apply_o20) && (tuya_new_dim > 0) && (abs(tuya_new_dim - Settings.light_dimmer) > 1)) {
  173. snprintf_P(scmnd, sizeof(scmnd), PSTR(D_CMND_DIMMER " %d"), tuya_new_dim );
  174. snprintf_P(log_data, sizeof(log_data), PSTR("TYA: Send CMND_DIMMER_STR=%s"), scmnd );
  175. AddLog(LOG_LEVEL_DEBUG);
  176. tuya_ignore_dim = true;
  177. ExecuteCommand(scmnd, SRC_SWITCH);
  178. }
  179. }
  180. break;
  181. case TUYA_CMD_WIFI_RESET:
  182. case TUYA_CMD_WIFI_SELECT:
  183. AddLog_P(LOG_LEVEL_DEBUG, PSTR("TYA: RX WiFi Reset"));
  184. TuyaResetWifi();
  185. break;
  186. case TUYA_CMD_WIFI_STATE:
  187. AddLog_P(LOG_LEVEL_DEBUG, PSTR("TYA: RX WiFi LED set ACK"));
  188. tuya_wifi_state = WifiState();
  189. break;
  190. case TUYA_CMD_MCU_CONF:
  191. AddLog_P(LOG_LEVEL_DEBUG, PSTR("TYA: RX MCU configuration"));
  192. if (tuya_buffer[5] == 2) {
  193. uint8_t led1_gpio = tuya_buffer[6];
  194. uint8_t key1_gpio = tuya_buffer[7];
  195. boolean key1_set = false;
  196. boolean led1_set = false;
  197. for (byte i = 0; i < MAX_GPIO_PIN; i++) {
  198. if (Settings.my_gp.io[i] == GPIO_LED1) led1_set = true;
  199. else if (Settings.my_gp.io[i] == GPIO_KEY1) key1_set = true;
  200. }
  201. if(!Settings.my_gp.io[led1_gpio] && !led1_set){
  202. Settings.my_gp.io[led1_gpio] = GPIO_LED1;
  203. restart_flag = 2;
  204. }
  205. if(!Settings.my_gp.io[key1_gpio] && !key1_set){
  206. Settings.my_gp.io[key1_gpio] = GPIO_KEY1;
  207. restart_flag = 2;
  208. }
  209. }
  210. TuyaRequestState();
  211. break;
  212. default:
  213. AddLog_P(LOG_LEVEL_DEBUG, PSTR("TYA: RX unknown command"));
  214. }
  215. }
  216. /*********************************************************************************************\
  217. * API Functions
  218. \*********************************************************************************************/
  219. boolean TuyaModuleSelected(void)
  220. {
  221. if (!(pin[GPIO_TUYA_RX] < 99) || !(pin[GPIO_TUYA_TX] < 99)) { // fallback to hardware-serial if not explicitly selected
  222. pin[GPIO_TUYA_TX] = 1;
  223. pin[GPIO_TUYA_RX] = 3;
  224. Settings.my_gp.io[1] = GPIO_TUYA_TX;
  225. Settings.my_gp.io[3] = GPIO_TUYA_RX;
  226. restart_flag = 2;
  227. }
  228. light_type = LT_SERIAL1;
  229. return true;
  230. }
  231. void TuyaInit(void)
  232. {
  233. if (!Settings.param[P_TUYA_DIMMER_ID]) {
  234. Settings.param[P_TUYA_DIMMER_ID] = TUYA_DIMMER_ID;
  235. }
  236. tuya_buffer = (char*)(malloc(TUYA_BUFFER_SIZE));
  237. if (tuya_buffer != NULL) {
  238. TuyaSerial = new TasmotaSerial(pin[GPIO_TUYA_RX], pin[GPIO_TUYA_TX], 2);
  239. if (TuyaSerial->begin(9600)) {
  240. if (TuyaSerial->hardwareSerial()) { ClaimSerial(); }
  241. // Get MCU Configuration
  242. snprintf_P(log_data, sizeof(log_data), "TYA: Request MCU configuration");
  243. AddLog(LOG_LEVEL_DEBUG);
  244. TuyaSendCmd(TUYA_CMD_MCU_CONF);
  245. }
  246. }
  247. }
  248. void TuyaSerialInput(void)
  249. {
  250. while (TuyaSerial->available()) {
  251. yield();
  252. byte serial_in_byte = TuyaSerial->read();
  253. if (serial_in_byte == 0x55) { // Start TUYA Packet
  254. tuya_cmd_status = 1;
  255. tuya_buffer[tuya_byte_counter++] = serial_in_byte;
  256. tuya_cmd_checksum += serial_in_byte;
  257. }
  258. else if (tuya_cmd_status == 1 && serial_in_byte == 0xAA){ // Only packtes with header 0x55AA are valid
  259. tuya_cmd_status = 2;
  260. tuya_byte_counter = 0;
  261. tuya_buffer[tuya_byte_counter++] = 0x55;
  262. tuya_buffer[tuya_byte_counter++] = 0xAA;
  263. tuya_cmd_checksum = 0xFF;
  264. }
  265. else if (tuya_cmd_status == 2){
  266. if(tuya_byte_counter == 5){ // Get length of data
  267. tuya_cmd_status = 3;
  268. tuya_data_len = serial_in_byte;
  269. }
  270. tuya_cmd_checksum += serial_in_byte;
  271. tuya_buffer[tuya_byte_counter++] = serial_in_byte;
  272. }
  273. else if ((tuya_cmd_status == 3) && (tuya_byte_counter == (6 + tuya_data_len)) && (tuya_cmd_checksum == serial_in_byte)){ // Compare checksum and process packet
  274. tuya_buffer[tuya_byte_counter++] = serial_in_byte;
  275. snprintf_P(log_data, sizeof(log_data), PSTR("TYA: RX Packet: \""));
  276. for (int i = 0; i < tuya_byte_counter; i++) {
  277. snprintf_P(log_data, sizeof(log_data), PSTR("%s%02x"), log_data, tuya_buffer[i]);
  278. }
  279. snprintf_P(log_data, sizeof(log_data), PSTR("%s\""), log_data);
  280. AddLog(LOG_LEVEL_DEBUG);
  281. TuyaPacketProcess();
  282. tuya_byte_counter = 0;
  283. tuya_cmd_status = 0;
  284. tuya_cmd_checksum = 0;
  285. tuya_data_len = 0;
  286. } // read additional packets from TUYA
  287. else if(tuya_byte_counter < TUYA_BUFFER_SIZE -1) { // add char to string if it still fits
  288. tuya_buffer[tuya_byte_counter++] = serial_in_byte;
  289. tuya_cmd_checksum += serial_in_byte;
  290. } else {
  291. tuya_byte_counter = 0;
  292. tuya_cmd_status = 0;
  293. tuya_cmd_checksum = 0;
  294. tuya_data_len = 0;
  295. }
  296. }
  297. }
  298. boolean TuyaButtonPressed(void)
  299. {
  300. if (!XdrvMailbox.index && ((PRESSED == XdrvMailbox.payload) && (NOT_PRESSED == lastbutton[XdrvMailbox.index]))) {
  301. snprintf_P(log_data, sizeof(log_data), PSTR("TYA: Reset GPIO triggered"));
  302. AddLog(LOG_LEVEL_DEBUG);
  303. TuyaResetWifi();
  304. return true; // Reset GPIO served here
  305. }
  306. return false; // Don't serve other buttons
  307. }
  308. void TuyaSetWifiLed(void){
  309. uint8_t wifi_state = 0x02;
  310. switch(WifiState()){
  311. case WIFI_SMARTCONFIG:
  312. wifi_state = 0x00;
  313. break;
  314. case WIFI_MANAGER:
  315. case WIFI_WPSCONFIG:
  316. wifi_state = 0x01;
  317. break;
  318. case WIFI_RESTART:
  319. wifi_state = 0x03;
  320. break;
  321. }
  322. snprintf_P(log_data, sizeof(log_data), "TYA: Set WiFi LED to state %d (%d)", wifi_state, WifiState());
  323. AddLog(LOG_LEVEL_DEBUG);
  324. TuyaSendCmd(TUYA_CMD_WIFI_STATE, &wifi_state, 1);
  325. }
  326. /*********************************************************************************************\
  327. * Interface
  328. \*********************************************************************************************/
  329. boolean Xdrv16(byte function)
  330. {
  331. boolean result = false;
  332. if (TUYA_DIMMER == Settings.module) {
  333. switch (function) {
  334. case FUNC_MODULE_INIT:
  335. result = TuyaModuleSelected();
  336. break;
  337. case FUNC_INIT:
  338. TuyaInit();
  339. break;
  340. case FUNC_LOOP:
  341. if (TuyaSerial) { TuyaSerialInput(); }
  342. break;
  343. case FUNC_SET_DEVICE_POWER:
  344. result = TuyaSetPower();
  345. break;
  346. case FUNC_BUTTON_PRESSED:
  347. result = TuyaButtonPressed();
  348. break;
  349. case FUNC_EVERY_SECOND:
  350. if(TuyaSerial && tuya_wifi_state!=WifiState()) { TuyaSetWifiLed(); }
  351. break;
  352. case FUNC_SET_CHANNELS:
  353. result = TuyaSetChannels();
  354. break;
  355. }
  356. }
  357. return result;
  358. }
  359. #endif // USE_TUYA_DIMMER