2018-10-25 17:17:58 +01:00
|
|
|
/*
|
2019-10-27 10:13:24 +00:00
|
|
|
xdrv_17_rcswitch.ino - RF transceiver using RcSwitch library for Tasmota
|
2018-10-25 17:17:58 +01:00
|
|
|
|
2019-12-31 13:23:34 +00:00
|
|
|
Copyright (C) 2020 Theo Arends
|
2018-10-25 17:17:58 +01:00
|
|
|
|
|
|
|
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_RC_SWITCH
|
|
|
|
/*********************************************************************************************\
|
2018-10-26 11:30:25 +01:00
|
|
|
* RF send and receive using RCSwitch library https://github.com/sui77/rc-switch/
|
2018-10-25 17:17:58 +01:00
|
|
|
\*********************************************************************************************/
|
|
|
|
|
2018-11-07 09:30:03 +00:00
|
|
|
#define XDRV_17 17
|
|
|
|
|
2018-10-25 17:17:58 +01:00
|
|
|
#define D_JSON_RF_PROTOCOL "Protocol"
|
|
|
|
#define D_JSON_RF_BITS "Bits"
|
|
|
|
#define D_JSON_RF_DATA "Data"
|
|
|
|
|
|
|
|
#define D_CMND_RFSEND "RFSend"
|
|
|
|
#define D_JSON_RF_PULSE "Pulse"
|
|
|
|
#define D_JSON_RF_REPEAT "Repeat"
|
|
|
|
|
2019-08-11 17:12:18 +01:00
|
|
|
const char kRfSendCommands[] PROGMEM = "|" // No prefix
|
|
|
|
D_CMND_RFSEND;
|
2019-08-11 14:18:11 +01:00
|
|
|
|
2019-11-24 11:24:35 +00:00
|
|
|
void (* const RfSendCommand[])(void) PROGMEM = {
|
|
|
|
&CmndRfSend };
|
2019-08-11 14:18:11 +01:00
|
|
|
|
2018-10-25 17:17:58 +01:00
|
|
|
#include <RCSwitch.h>
|
|
|
|
|
|
|
|
RCSwitch mySwitch = RCSwitch();
|
|
|
|
|
2018-11-11 13:45:19 +00:00
|
|
|
#define RF_TIME_AVOID_DUPLICATE 1000 // Milliseconds
|
2018-10-25 17:17:58 +01:00
|
|
|
|
2018-10-26 11:30:25 +01:00
|
|
|
uint32_t rf_lasttime = 0;
|
2018-10-25 17:17:58 +01:00
|
|
|
|
2018-11-14 13:32:09 +00:00
|
|
|
void RfReceiveCheck(void)
|
2018-10-25 17:17:58 +01:00
|
|
|
{
|
|
|
|
if (mySwitch.available()) {
|
|
|
|
|
2018-10-26 11:30:25 +01:00
|
|
|
unsigned long data = mySwitch.getReceivedValue();
|
|
|
|
unsigned int bits = mySwitch.getReceivedBitlength();
|
|
|
|
int protocol = mySwitch.getReceivedProtocol();
|
|
|
|
int delay = mySwitch.getReceivedDelay();
|
2018-10-25 17:17:58 +01:00
|
|
|
|
2019-03-11 13:55:14 +00:00
|
|
|
AddLog_P2(LOG_LEVEL_DEBUG, PSTR("RFR: Data 0x%lX (%u), Bits %d, Protocol %d, Delay %d"), data, data, bits, protocol, delay);
|
2018-10-25 17:17:58 +01:00
|
|
|
|
2018-10-26 11:30:25 +01:00
|
|
|
uint32_t now = millis();
|
|
|
|
if ((now - rf_lasttime > RF_TIME_AVOID_DUPLICATE) && (data > 0)) {
|
2018-10-25 17:17:58 +01:00
|
|
|
rf_lasttime = now;
|
|
|
|
|
|
|
|
char stemp[16];
|
2019-11-03 11:33:36 +00:00
|
|
|
if (Settings.flag.rf_receive_decimal) { // SetOption28 - RF receive data format (0 = hexadecimal, 1 = decimal)
|
2018-10-26 11:30:25 +01:00
|
|
|
snprintf_P(stemp, sizeof(stemp), PSTR("%u"), (uint32_t)data);
|
2018-10-25 17:17:58 +01:00
|
|
|
} else {
|
2019-03-11 13:55:14 +00:00
|
|
|
snprintf_P(stemp, sizeof(stemp), PSTR("\"0x%lX\""), (uint32_t)data);
|
2018-10-25 17:17:58 +01:00
|
|
|
}
|
2019-09-04 17:06:34 +01:00
|
|
|
ResponseTime_P(PSTR(",\"" D_JSON_RFRECEIVED "\":{\"" D_JSON_RF_DATA "\":%s,\"" D_JSON_RF_BITS "\":%d,\"" D_JSON_RF_PROTOCOL "\":%d,\"" D_JSON_RF_PULSE "\":%d}}"),
|
2018-11-11 13:45:19 +00:00
|
|
|
stemp, bits, protocol, delay);
|
2020-07-20 16:24:51 +01:00
|
|
|
MqttPublishPrefixTopicRulesProcess_P(RESULT_OR_TELE, PSTR(D_JSON_RFRECEIVED));
|
2018-10-25 17:17:58 +01:00
|
|
|
#ifdef USE_DOMOTICZ
|
2018-10-26 11:30:25 +01:00
|
|
|
DomoticzSensor(DZ_COUNT, data); // Send data as Domoticz Counter value
|
2018-10-25 17:17:58 +01:00
|
|
|
#endif // USE_DOMOTICZ
|
|
|
|
}
|
|
|
|
mySwitch.resetAvailable();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-11-14 13:32:09 +00:00
|
|
|
void RfInit(void)
|
2018-10-25 17:17:58 +01:00
|
|
|
{
|
2020-04-27 11:54:07 +01:00
|
|
|
if (PinUsed(GPIO_RFSEND)) {
|
2020-04-26 16:33:27 +01:00
|
|
|
mySwitch.enableTransmit(Pin(GPIO_RFSEND));
|
2018-10-25 17:17:58 +01:00
|
|
|
}
|
2020-04-27 11:54:07 +01:00
|
|
|
if (PinUsed(GPIO_RFRECV)) {
|
2020-04-26 16:33:27 +01:00
|
|
|
pinMode( Pin(GPIO_RFRECV), INPUT);
|
|
|
|
mySwitch.enableReceive(Pin(GPIO_RFRECV));
|
2018-10-25 17:17:58 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*********************************************************************************************\
|
|
|
|
* Commands
|
|
|
|
\*********************************************************************************************/
|
|
|
|
|
2019-08-11 14:18:11 +01:00
|
|
|
void CmndRfSend(void)
|
2018-10-25 17:17:58 +01:00
|
|
|
{
|
2019-01-28 13:08:33 +00:00
|
|
|
bool error = false;
|
2018-10-25 17:17:58 +01:00
|
|
|
|
2019-08-11 14:18:11 +01:00
|
|
|
if (XdrvMailbox.data_len) {
|
|
|
|
unsigned long data = 0;
|
|
|
|
unsigned int bits = 24;
|
|
|
|
int protocol = 1;
|
|
|
|
int repeat = 10;
|
|
|
|
int pulse = 350;
|
|
|
|
|
2020-09-23 18:38:24 +01:00
|
|
|
JsonParser parser(XdrvMailbox.data);
|
|
|
|
JsonParserObject root = parser.getRootObject();
|
2020-09-23 07:45:14 +01:00
|
|
|
if (root) {
|
2019-08-11 14:18:11 +01:00
|
|
|
// RFsend {"data":0x501014,"bits":24,"protocol":1,"repeat":10,"pulse":350}
|
|
|
|
char parm_uc[10];
|
2020-09-23 07:45:14 +01:00
|
|
|
data = root.getUInt(PSTR(D_JSON_RF_DATA), data);
|
|
|
|
bits = root.getUInt(PSTR(D_JSON_RF_BITS), bits);
|
|
|
|
protocol = root.getInt(PSTR(D_JSON_RF_PROTOCOL), protocol);
|
|
|
|
repeat = root.getInt(PSTR(D_JSON_RF_REPEAT), repeat);
|
|
|
|
pulse = root.getInt(PSTR(D_JSON_RF_PULSE), pulse);
|
2019-08-11 14:18:11 +01:00
|
|
|
} else {
|
|
|
|
// RFsend data, bits, protocol, repeat, pulse
|
|
|
|
char *p;
|
|
|
|
uint8_t i = 0;
|
|
|
|
for (char *str = strtok_r(XdrvMailbox.data, ", ", &p); str && i < 5; str = strtok_r(nullptr, ", ", &p)) {
|
|
|
|
switch (i++) {
|
|
|
|
case 0:
|
|
|
|
data = strtoul(str, nullptr, 0); // Allow decimal (5246996) and hexadecimal (0x501014) input
|
|
|
|
break;
|
|
|
|
case 1:
|
|
|
|
bits = atoi(str);
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
protocol = atoi(str);
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
repeat = atoi(str);
|
|
|
|
break;
|
|
|
|
case 4:
|
|
|
|
pulse = atoi(str);
|
2018-10-25 17:17:58 +01:00
|
|
|
}
|
|
|
|
}
|
2019-08-11 14:18:11 +01:00
|
|
|
}
|
2018-10-26 11:30:25 +01:00
|
|
|
|
2019-08-11 14:18:11 +01:00
|
|
|
if (!protocol) { protocol = 1; }
|
|
|
|
mySwitch.setProtocol(protocol);
|
|
|
|
if (!pulse) { pulse = 350; } // Default pulse length for protocol 1
|
|
|
|
mySwitch.setPulseLength(pulse);
|
|
|
|
if (!repeat) { repeat = 10; } // Default at init
|
|
|
|
mySwitch.setRepeatTransmit(repeat);
|
|
|
|
if (!bits) { bits = 24; } // Default 24 bits
|
|
|
|
if (data) {
|
|
|
|
mySwitch.send(data, bits);
|
|
|
|
ResponseCmndDone();
|
2018-10-26 11:30:25 +01:00
|
|
|
} else {
|
2018-10-25 17:17:58 +01:00
|
|
|
error = true;
|
|
|
|
}
|
2019-08-11 14:18:11 +01:00
|
|
|
} else {
|
|
|
|
error = true;
|
|
|
|
}
|
|
|
|
if (error) {
|
|
|
|
Response_P(PSTR("{\"" D_CMND_RFSEND "\":\"" D_JSON_NO " " D_JSON_RF_DATA ", " D_JSON_RF_BITS ", " D_JSON_RF_PROTOCOL ", " D_JSON_RF_REPEAT " " D_JSON_OR " " D_JSON_RF_PULSE "\"}"));
|
2018-10-25 17:17:58 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*********************************************************************************************\
|
|
|
|
* Interface
|
|
|
|
\*********************************************************************************************/
|
|
|
|
|
2019-01-28 13:08:33 +00:00
|
|
|
bool Xdrv17(uint8_t function)
|
2018-10-25 17:17:58 +01:00
|
|
|
{
|
2019-01-28 13:08:33 +00:00
|
|
|
bool result = false;
|
2018-10-25 17:17:58 +01:00
|
|
|
|
2020-04-27 11:54:07 +01:00
|
|
|
if (PinUsed(GPIO_RFSEND) || PinUsed(GPIO_RFRECV)) {
|
2018-10-25 17:17:58 +01:00
|
|
|
switch (function) {
|
|
|
|
case FUNC_EVERY_50_MSECOND:
|
2020-04-27 11:54:07 +01:00
|
|
|
if (PinUsed(GPIO_RFRECV)) {
|
2018-10-25 17:17:58 +01:00
|
|
|
RfReceiveCheck();
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
case FUNC_COMMAND:
|
2020-04-27 11:54:07 +01:00
|
|
|
if (PinUsed(GPIO_RFSEND)) {
|
2019-08-11 14:18:11 +01:00
|
|
|
result = DecodeCommand(kRfSendCommands, RfSendCommand);
|
2018-10-25 17:17:58 +01:00
|
|
|
}
|
|
|
|
break;
|
2019-08-11 14:18:11 +01:00
|
|
|
case FUNC_INIT:
|
|
|
|
RfInit();
|
|
|
|
break;
|
2018-10-25 17:17:58 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // USE_RC_SWITCH
|