Merge pull request #6812 from andrethomas/tasmota_slave

Tasmota slave
This commit is contained in:
Theo Arends 2019-11-02 12:41:15 +01:00 committed by GitHub
commit b823a05e5d
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
1 changed files with 141 additions and 37 deletions

View File

@ -43,12 +43,14 @@
#define CMND_FEATURES 0x01 #define CMND_FEATURES 0x01
#define CMND_JSON 0x02 #define CMND_JSON 0x02
#define CMND_SECOND_TICK 0x03 #define CMND_FUNC_EVERY_SECOND 0x03
#define CMND_FUNC_EVERY_100_MSECOND 0x04
#define CMND_SLAVE_SEND 0x05
#define CMND_PUBLISH_TELE 0x06
#define PARAM_DATA_START 0xFE #define PARAM_DATA_START 0xFE
#define PARAM_DATA_END 0xFF #define PARAM_DATA_END 0xFF
#include <TasmotaSerial.h> #include <TasmotaSerial.h>
/* /*
@ -143,24 +145,40 @@ struct TSLAVE {
} TSlave; } TSlave;
typedef union { typedef union {
uint16_t data; uint32_t data;
struct { struct {
uint16_t json : 1; // Slave supports providing a JSON for TELEPERIOD uint32_t func_json_append : 1; // Slave supports providing a JSON for TELEPERIOD
uint16_t second_tick : 1; // Slave supports receiving a FUNC_EVERY_SECOND callback with no response uint32_t func_every_second : 1; // Slave supports receiving a FUNC_EVERY_SECOND callback with no response
uint16_t spare2 : 1; uint32_t func_every_100_msecond : 1; // Slave supports receiving a FUNC_EVERY_100_MSECOND callback with no response
uint16_t spare3 : 1; uint32_t func_slave_send : 1; // Slave supports receiving commands with "slave send xxx"
uint16_t spare4 : 1; uint32_t spare4 : 1;
uint16_t spare5 : 1; uint32_t spare5 : 1;
uint16_t spare6 : 1; uint32_t spare6 : 1;
uint16_t spare7 : 1; uint32_t spare7 : 1;
uint16_t spare8 : 1; uint32_t spare8 : 1;
uint16_t spare9 : 1; uint32_t spare9 : 1;
uint16_t spare10 : 1; uint32_t spare10 : 1;
uint16_t spare11 : 1; uint32_t spare11 : 1;
uint16_t spare12 : 1; uint32_t spare12 : 1;
uint16_t spare13 : 1; uint32_t spare13 : 1;
uint16_t spare14 : 1; uint32_t spare14 : 1;
uint16_t spare15 : 1; uint32_t spare15 : 1;
uint32_t spare16 : 1;
uint32_t spare17 : 1;
uint32_t spare18 : 1;
uint32_t spare19 : 1;
uint32_t spare20 : 1;
uint32_t spare21 : 1;
uint32_t spare22 : 1;
uint32_t spare23 : 1;
uint32_t spare24 : 1;
uint32_t spare25 : 1;
uint32_t spare26 : 1;
uint32_t spare27 : 1;
uint32_t spare28 : 1;
uint32_t spare29 : 1;
uint32_t spare30 : 1;
uint32_t spare31 : 1;
}; };
} TSlaveFeatureCfg; } TSlaveFeatureCfg;
@ -169,18 +187,17 @@ typedef union {
* Tasmota as master * Tasmota as master
*/ */
struct FEATURES { struct TSLAVE_FEATURES {
uint32_t features_version; uint32_t features_version;
TSlaveFeatureCfg features; TSlaveFeatureCfg features;
uint16_t spare4;
} TSlaveSettings; } TSlaveSettings;
struct COMMAND { struct TSLAVE_COMMAND {
uint8_t command; uint8_t command;
uint8_t parameter; uint8_t parameter;
uint8_t unused2; uint8_t unused2;
uint8_t unused3; uint8_t unused3;
} Command; } TSlaveCommand;
TasmotaSerial *TasmotaSlave_Serial; TasmotaSerial *TasmotaSlave_Serial;
@ -444,7 +461,7 @@ void TasmotaSlave_Init(void)
TasmotaSlave_Serial->readBytesUntil(char(PARAM_DATA_START), buffer, sizeof(buffer)); TasmotaSlave_Serial->readBytesUntil(char(PARAM_DATA_START), buffer, sizeof(buffer));
uint8_t len = TasmotaSlave_Serial->readBytesUntil(char(PARAM_DATA_END), buffer, sizeof(buffer)); uint8_t len = TasmotaSlave_Serial->readBytesUntil(char(PARAM_DATA_END), buffer, sizeof(buffer));
memcpy(&TSlaveSettings, &buffer, sizeof(TSlaveSettings)); memcpy(&TSlaveSettings, &buffer, sizeof(TSlaveSettings));
if (20191026 <= TSlaveSettings.features_version) { if (20191101 <= TSlaveSettings.features_version) {
TSlave.type = true; TSlave.type = true;
AddLog_P2(LOG_LEVEL_INFO, PSTR("Tasmota Slave Version %u"), TSlaveSettings.features_version); AddLog_P2(LOG_LEVEL_INFO, PSTR("Tasmota Slave Version %u"), TSlaveSettings.features_version);
} }
@ -453,7 +470,7 @@ void TasmotaSlave_Init(void)
void TasmotaSlave_Show(void) void TasmotaSlave_Show(void)
{ {
if ((TSlave.type) && (TSlaveSettings.features.json)) { if ((TSlave.type) && (TSlaveSettings.features.func_json_append)) {
char buffer[100]; char buffer[100];
TasmotaSlave_sendCmnd(CMND_JSON, 0); TasmotaSlave_sendCmnd(CMND_JSON, 0);
TasmotaSlave_Serial->readBytesUntil(char(PARAM_DATA_START), buffer, sizeof(buffer)-1); TasmotaSlave_Serial->readBytesUntil(char(PARAM_DATA_START), buffer, sizeof(buffer)-1);
@ -465,17 +482,94 @@ void TasmotaSlave_Show(void)
void TasmotaSlave_sendCmnd(uint8_t cmnd, uint8_t param) void TasmotaSlave_sendCmnd(uint8_t cmnd, uint8_t param)
{ {
Command.command = cmnd; TSlaveCommand.command = cmnd;
Command.parameter = param; TSlaveCommand.parameter = param;
char buffer[sizeof(Command)+2]; char buffer[sizeof(TSlaveCommand)+2];
buffer[0] = CMND_START; buffer[0] = CMND_START;
memcpy(&buffer[1], &Command, sizeof(Command)); memcpy(&buffer[1], &TSlaveCommand, sizeof(TSlaveCommand));
buffer[sizeof(Command)+1] = CMND_END; buffer[sizeof(TSlaveCommand)+1] = CMND_END;
for (uint8_t ca = 0; ca < sizeof(buffer); ca++) { for (uint8_t ca = 0; ca < sizeof(buffer); ca++) {
TasmotaSlave_Serial->write(buffer[ca]); TasmotaSlave_Serial->write(buffer[ca]);
} }
} }
const char kTasmotaSlaveCommands[] PROGMEM = "|" // No prefix
"Slave" ;
void (* const TasmotaSlaveCommand[])(void) PROGMEM = {
&CmndTasmotaSlave };
void CmndTasmotaSlave(void)
{
if (XdrvMailbox.data_len > 0) {
uint8_t paramcount = 1;
for (uint8_t idx = 0; idx < strlen(XdrvMailbox.data); idx++) {
if ((' ' == XdrvMailbox.data[idx]) || (',' == XdrvMailbox.data[idx])) {
XdrvMailbox.data[idx] = ','; // Make it a comma irrespective whether its a space or a comma
paramcount++;
}
UpperCase(XdrvMailbox.data,XdrvMailbox.data);
char sub_string[XdrvMailbox.data_len];
if (1 == paramcount) { // Single parameter commands
if (!strcmp(subStr(sub_string, XdrvMailbox.data, ",", 1), "RESET")) {
TasmotaSlave_Reset();
TSlave.type = false; // Force redetection
TSlave.waitstate = 7; // give it at least 3 seconds to restart from bootloader
}
}
if (2 == paramcount) { // Double parameter commands
if ((!strcmp(subStr(sub_string, XdrvMailbox.data, ",", 1), "SEND")) && (TSlaveSettings.features.func_slave_send)) {
char tmp[XdrvMailbox.data_len];
sprintf(tmp, "%s", subStr(sub_string, XdrvMailbox.data, ",", 2));
TasmotaSlave_sendCmnd(CMND_SLAVE_SEND, strlen(tmp));
TasmotaSlave_Serial->write(char(PARAM_DATA_START));
for (uint8_t ci = 0; ci < strlen(tmp); ci++) {
TasmotaSlave_Serial->write(tmp[ci]);
}
TasmotaSlave_Serial->write(char(PARAM_DATA_END));
}
}
}
}
ResponseCmndDone();
}
void TasmotaSlave_ProcessIn(void)
{
uint8_t cmnd = TasmotaSlave_Serial->read();
switch (cmnd) {
case CMND_START:
TasmotaSlave_waitForSerialData(sizeof(TSlaveCommand),50);
uint8_t buffer[sizeof(TSlaveCommand)];
for (uint8_t idx = 0; idx < sizeof(TSlaveCommand); idx++) {
buffer[idx] = TasmotaSlave_Serial->read();
}
TasmotaSlave_Serial->read(); // read trailing byte of command
memcpy(&TSlaveCommand, &buffer, sizeof(TSlaveCommand));
if (CMND_PUBLISH_TELE == TSlaveCommand.command) { // We need to publish stat/ with incoming stream as content
char inbuf[sizeof(TSlaveCommand.parameter)+1];
TasmotaSlave_waitForSerialData(TSlaveCommand.parameter, 50);
TasmotaSlave_Serial->read(); // Read leading byte
for (uint8_t idx = 0; idx < TSlaveCommand.parameter; idx++) {
inbuf[idx] = TasmotaSlave_Serial->read();
}
TasmotaSlave_Serial->read(); // Read trailing byte
inbuf[TSlaveCommand.parameter] = '\0';
Response_P(PSTR("{\"TasmotaSlave\":"));
ResponseAppend_P("%s", inbuf);
ResponseJsonEnd();
MqttPublishPrefixTopic_P(RESULT_OR_TELE, mqtt_data);
XdrvRulesProcess();
}
break;
default:
break;
}
}
/*********************************************************************************************\ /*********************************************************************************************\
* Interface * Interface
\*********************************************************************************************/ \*********************************************************************************************/
@ -485,19 +579,29 @@ bool Xdrv31(uint8_t function)
bool result = false; bool result = false;
switch (function) { switch (function) {
case FUNC_EVERY_SECOND: case FUNC_EVERY_100_MSECOND:
TasmotaSlave_Init(); if ((TSlave.type) && (TSlaveSettings.features.func_every_100_msecond)) {
if (TSlave.type) { if (TasmotaSlave_Serial->available()) {
if (TSlaveSettings.features.second_tick) { TasmotaSlave_ProcessIn();
TasmotaSlave_sendCmnd(CMND_SECOND_TICK, 0);
} }
} }
break; break;
case FUNC_EVERY_SECOND:
if ((TSlave.type) && (TSlaveSettings.features.func_every_second)) {
TasmotaSlave_sendCmnd(CMND_FUNC_EVERY_SECOND, 0);
}
TasmotaSlave_Init();
break;
case FUNC_JSON_APPEND: case FUNC_JSON_APPEND:
TasmotaSlave_Show(); if ((TSlave.type) && (TSlaveSettings.features.func_json_append)) {
TasmotaSlave_Show();
}
break;
case FUNC_COMMAND:
result = DecodeCommand(kTasmotaSlaveCommands, TasmotaSlaveCommand);
break; break;
} }
return result; return result;
} }
#endif // USE_TASMOTA_SLAVE #endif // USE_TASMOTA_SLAVE