/* xlgt_09_sm2335.ino - sm2335 five channel led support for Tasmota Copyright (C) 2021 Theo Arends and Cossid 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 . */ #ifdef USE_LIGHT #ifdef USE_SM2335 /*********************************************************************************************\ * SM2335 RGBCW Led bulbs like the SwitchBot Color Bulb * * SwitchBot Color Bulb * {"NAME":"SwitchBot Color Bulb","GPIO":[0,0,0,0,9129,9088,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0],"FLAG":0,"BASE":1} \*********************************************************************************************/ #define XLGT_09 9 // 11 = identification | 0 = reserved | 00 = Standby | 000 = start at OUT1/5 #define SM2335_ADDR_STANDBY 0xC0 // 11000000 0xC0 // 11 = identification | 0 = reserved | 01 = 3 channels (RGB) | 000 = start at OUT1/5 //#define SM2335_ADDR_START_3CH 0xC8 // 11001000 0xC8 // 11 = identification | 0 = reserved | 10 = 2 channels (CW) | 000 = start at OUT1/5 //#define SM2335_ADDR_START_2CH 0xD0 // 11010000 0xD0 // 11 = identification | 0 = reserved | 11 = 5 channels (RGB+CW) | 000 = start at OUT1/5 #define SM2335_ADDR_START_5CH 0xD8 // 11011000 0xD8 // Current values // 0x0 // 0000 RGB 10mA | CW 5mA // 0x1 // 0001 RGB 20mA | CW 10mA // 0x2 // 0010 RGB 30mA | CW 15mA // 0x3 // 0011 RGB 40mA | CW 20mA // 0x4 // 0100 RGB 50mA | CW 25mA // 0x5 // 0101 RGB 60mA | CW 30mA // 0x6 // 0110 RGB 70mA | CW 35mA // 0x7 // 0111 RGB 80mA | CW 40mA // 0x8 // 1000 RGB 90mA | CW 45mA // 0x9 // 1001 RGB 100mA | CW 50mA // 0xA // 1010 RGB 110mA | CW 55mA // 0xB // 1011 RGB 120mA | CW 60mA // 0xC // 1100 RGB 130mA | CW 65mA // 0xD // 1101 RGB 140mA | CW 70mA // 0xE // 1110 RGB 150mA | CW 75mA // 0xF // 1111 RGB 160mA | CW 80mA struct SM2335 { uint8_t clk = 0; uint8_t data = 0; uint8_t current; } Sm2335; /*********************************************************************************************\ * SM2335 code - inspired by Bp5758d/SM2135 \*********************************************************************************************/ const uint8_t SM2335_DELAY = 2; void SM2335Init(void) { pinMode(Sm2335.data, OUTPUT); pinMode(Sm2335.clk, OUTPUT); SM2335Stop(); } void SM2335Write(uint8_t value) { for (int bit_idx = 7; bit_idx >= 0; bit_idx--) { bool bit = bitRead(value, bit_idx); digitalWrite(Sm2335.data, bit); delayMicroseconds(SM2335_DELAY); digitalWrite(Sm2335.clk, HIGH); delayMicroseconds(SM2335_DELAY); digitalWrite(Sm2335.clk, LOW); delayMicroseconds(SM2335_DELAY); } // Wait for ACK pinMode(Sm2335.data, INPUT); digitalWrite(Sm2335.clk, HIGH); delayMicroseconds(SM2335_DELAY); digitalWrite(Sm2335.clk, LOW); delayMicroseconds(SM2335_DELAY); pinMode(Sm2335.data, OUTPUT); } void SM2335Start(uint8_t addr) { digitalWrite(Sm2335.data, LOW); delayMicroseconds(SM2335_DELAY); digitalWrite(Sm2335.clk, LOW); delayMicroseconds(SM2335_DELAY); SM2335Write(addr); } void SM2335Stop(void) { digitalWrite(Sm2335.clk, HIGH); delayMicroseconds(SM2335_DELAY); digitalWrite(Sm2335.data, HIGH); delayMicroseconds(SM2335_DELAY); } /********************************************************************************************/ bool SM2335SetChannels(void) { uint16_t *cur_col_10 = (uint16_t*)XdrvMailbox.command; // If we receive 0 for all channels, we'll assume that the lightbulb is off, and activate SM2335's standby mode. if (cur_col_10[0] == 0 && cur_col_10[1] == 0 && cur_col_10[2] == 0 && cur_col_10[3] == 0 && cur_col_10[4] == 0) { SM2335Start(SM2335_ADDR_STANDBY); // Clear all remaining data. This clears out Current, Red, Green, Blue, Cold White, Warm White. for (int i = 0; i < 11; i++) { SM2335Write(0); } SM2335Stop(); return true; } // Write the header activating all 5 channels SM2335Start(SM2335_ADDR_START_5CH); // Set the current defined in ModuleSelected. SM2335Write(Sm2335.current); // Set RGB and CW grayscale. for (int i = 0; i < 5; i++) { SM2335Write((uint8_t)(cur_col_10[i] >> 8)); SM2335Write((uint8_t)(cur_col_10[i] & 0xFF)); } SM2335Stop(); return true; } void SM2335ModuleSelected(void) { if (PinUsed(GPIO_SM2335_CLK) && PinUsed(GPIO_SM2335_DAT, GPIO_ANY)) { Sm2335.clk = Pin(GPIO_SM2335_CLK); Sm2335.data = Pin(GPIO_SM2335_DAT, GPIO_ANY); // See #define MAX_SM2335_DAT 16 in tasmota_template.h int currentDat = GetPin(Sm2335.data) - AGPIO(GPIO_SM2335_DAT); // 0 .. 15 // Split RGB and CW current. Sm2335.current = (currentDat << 4) | currentDat; SM2335Init(); TasmotaGlobal.light_type = LT_RGBWC; TasmotaGlobal.light_driver = XLGT_09; AddLog(LOG_LEVEL_DEBUG, PSTR("LGT: SM2335 Found")); } } /*********************************************************************************************\ * Interface \*********************************************************************************************/ bool Xlgt09(uint32_t function) { bool result = false; switch (function) { case FUNC_SET_CHANNELS: result = SM2335SetChannels(); break; case FUNC_MODULE_INIT: SM2335ModuleSelected(); break; } return result; } #endif // USE_SM2335 #endif // USE_LIGHT