Tasmota/tasmota/tasmota_xlgt_light/xlgt_09_sm2335.ino

177 lines
5.8 KiB
C++

/*
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 <http://www.gnu.org/licenses/>.
*/
#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(uint8_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