2017-10-10 14:40:02 +01:00
/*
2017-10-18 17:22:34 +01:00
sonoff . h - Master header file for Sonoff - Tasmota
2017-10-10 14:40:02 +01:00
2019-03-13 17:00:15 +00:00
Copyright ( C ) 2019 Theo Arends
2017-10-10 14:40:02 +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/>.
*/
2017-10-18 17:22:34 +01:00
# ifndef _SONOFF_H_
# define _SONOFF_H_
2018-12-15 14:55:51 +00:00
/*********************************************************************************************\
* Performance ROM ( PROGMEM ) vs RAM ( RODATA )
\ * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
//#define XFUNC_PTR_IN_ROM // Enable for keeping tables in ROM (PROGMEM) which seem to have access issues on some flash types
/*********************************************************************************************\
* Default sensor states
\ * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
# define CODE_IMAGE 0
2019-07-05 12:44:24 +01:00
# define USE_LIGHT // Enable light control
2018-10-26 16:13:17 +01:00
# define USE_ENERGY_SENSOR // Use energy sensors (+14k code)
# define USE_HLW8012 // Use energy sensor for Sonoff Pow and WolfBlitz
# define USE_CSE7766 // Use energy sensor for Sonoff S31 and Pow R2
2017-10-10 14:40:02 +01:00
/*********************************************************************************************\
* Power Type
\ * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
typedef unsigned long power_t ; // Power (Relay) type
2019-07-05 12:44:24 +01:00
const uint32_t POWER_MASK = 0xffffffffUL ; // Power (Relay) full mask
2017-10-10 14:40:02 +01:00
/*********************************************************************************************\
2019-07-05 12:44:24 +01:00
* Constants
2017-10-10 14:40:02 +01:00
\ * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2017-10-12 10:29:40 +01:00
// Changes to the following MAX_ defines will impact settings layout
2019-07-05 12:44:24 +01:00
const uint8_t MAX_SWITCHES = 8 ; // Max number of switches
const uint8_t MAX_RELAYS = 8 ; // Max number of relays
const uint8_t MAX_INTERLOCKS = 4 ; // Max number of interlock groups (MAX_RELAYS / 2)
const uint8_t MAX_LEDS = 4 ; // Max number of leds
const uint8_t MAX_KEYS = 4 ; // Max number of keys or buttons
const uint8_t MAX_PWMS = 5 ; // Max number of PWM channels
const uint8_t MAX_COUNTERS = 4 ; // Max number of counter sensors
const uint8_t MAX_TIMERS = 16 ; // Max number of Timers
const uint8_t MAX_PULSETIMERS = 8 ; // Max number of supported pulse timers
const uint8_t MAX_FRIENDLYNAMES = 4 ; // Max number of Friendly names
const uint8_t MAX_DOMOTICZ_IDX = 4 ; // Max number of Domoticz device, key and switch indices
const uint8_t MAX_DOMOTICZ_SNS_IDX = 12 ; // Max number of Domoticz sensors indices
const uint8_t MAX_KNX_GA = 10 ; // Max number of KNX Group Addresses to read that can be set
const uint8_t MAX_KNX_CB = 10 ; // Max number of KNX Group Addresses to write that can be set
const uint8_t MAX_XNRG_DRIVERS = 32 ; // Max number of allowed energy drivers
const uint8_t MAX_XDSP_DRIVERS = 32 ; // Max number of allowed display drivers
const uint8_t MAX_XDRV_DRIVERS = 96 ; // Max number of allowed driver drivers
const uint8_t MAX_XSNS_DRIVERS = 96 ; // Max number of allowed sensor drivers
const uint8_t MAX_RULE_MEMS = 5 ; // Max number of saved vars
const uint8_t MAX_RULE_SETS = 3 ; // Max number of rule sets of size 512 characters
const uint16_t MAX_RULE_SIZE = 512 ; // Max number of characters in rules
const uint8_t MAX_FAN_SPEED = 4 ; // Max number of iFan02 fan speeds (0 .. 3)
const char MQTT_TOKEN_PREFIX [ ] PROGMEM = " %prefix% " ; // To be substituted by mqtt_prefix[x]
const char MQTT_TOKEN_TOPIC [ ] PROGMEM = " %topic% " ; // To be substituted by mqtt_topic, mqtt_grptopic, mqtt_buttontopic, mqtt_switchtopic
const char WIFI_HOSTNAME [ ] = " %s-%04d " ; // Expands to <MQTT_TOPIC>-<last 4 decimal chars of MAC address>
const uint8_t CONFIG_FILE_SIGN = 0xA5 ; // Configuration file signature
const uint8_t CONFIG_FILE_XOR = 0x5A ; // Configuration file xor (0 = No Xor)
const uint32_t HLW_PREF_PULSE = 12530 ; // was 4975us = 201Hz = 1000W
const uint32_t HLW_UREF_PULSE = 1950 ; // was 1666us = 600Hz = 220V
const uint32_t HLW_IREF_PULSE = 3500 ; // was 1666us = 600Hz = 4.545A
const uint8_t MQTT_RETRY_SECS = 10 ; // Minimum seconds to retry MQTT connection
const uint32_t GLOBAL_VALUES_VALID = 300 ; // Max number of seconds to keep last received values
const power_t APP_POWER = 0 ; // Default saved power state Off
const uint16_t WS2812_MAX_LEDS = 512 ; // Max number of LEDs
const uint32_t PWM_RANGE = 1023 ; // 255..1023 needs to be devisible by 256
//const uint16_t PWM_FREQ = 1000; // 100..1000 Hz led refresh
//const uint16_t PWM_FREQ = 910; // 100..1000 Hz led refresh (iTead value)
const uint16_t PWM_FREQ = 880 ; // 100..1000 Hz led refresh (BN-SZ01 value)
const uint16_t PWM_MAX = 4000 ; // [PWM_MAX] Maximum frequency - Default: 4000
const uint16_t PWM_MIN = 100 ; // [PWM_MIN] Minimum frequency - Default: 100
// For Dimmers use double of your mains AC frequecy (100 for 50Hz and 120 for 60Hz)
// For Controlling Servos use 50 and also set PWM_FREQ as 50 (DO NOT USE THESE VALUES FOR DIMMERS)
//#define PWM_LIGHTSCHEME0_IGNORE_SLEEP // Do not change sleep value for LightAnimate() scheme 0
2018-08-23 15:05:51 +01:00
2019-07-05 12:44:24 +01:00
const uint16_t MAX_POWER_HOLD = 10 ; // Time in SECONDS to allow max agreed power
const uint16_t MAX_POWER_WINDOW = 30 ; // Time in SECONDS to disable allow max agreed power
const uint16_t SAFE_POWER_HOLD = 10 ; // Time in SECONDS to allow max unit safe power
const uint16_t SAFE_POWER_WINDOW = 30 ; // Time in MINUTES to disable allow max unit safe power
const uint8_t MAX_POWER_RETRY = 5 ; // Retry count allowing agreed power limit overflow
2018-12-15 14:55:51 +00:00
2019-07-05 12:44:24 +01:00
const uint8_t STATES = 20 ; // Number of states per second using 50 mSec interval
const uint8_t IMMINENT_RESET_FACTOR = 10 ; // Factor to extent button hold time for imminent Reset to default 40 seconds using KEY_HOLD_TIME of 40
const uint32_t BOOT_LOOP_TIME = 10 ; // Number of seconds to stop detecting boot loops
const uint16_t SYSLOG_TIMER = 600 ; // Seconds to restore syslog_level
const uint16_t SERIALLOG_TIMER = 600 ; // Seconds to disable SerialLog
const uint8_t OTA_ATTEMPTS = 5 ; // Number of times to try fetching the new firmware
2017-10-10 14:40:02 +01:00
2019-07-05 12:44:24 +01:00
const uint16_t INPUT_BUFFER_SIZE = 520 ; // Max number of characters in (serial and http) command buffer
const uint16_t CMDSZ = 24 ; // Max number of characters in command
const uint16_t TOPSZ = 100 ; // Max number of characters in topic string
const uint16_t LOGSZ = 520 ; // Max number of characters in log
const uint16_t MIN_MESSZ = 893 ; // Min number of characters in MQTT message
2017-10-10 14:40:02 +01:00
2019-07-05 12:44:24 +01:00
const uint8_t SENSOR_MAX_MISS = 5 ; // Max number of missed sensor reads before deciding it's offline
2017-10-10 14:40:02 +01:00
2019-07-05 12:44:24 +01:00
const uint8_t MAX_BACKLOG = 30 ; // Max number of commands in backlog
const uint32_t MIN_BACKLOG_DELAY = 2 ; // Minimal backlog delay in 0.1 seconds
2017-10-10 14:40:02 +01:00
2019-07-05 12:44:24 +01:00
const uint32_t SOFT_BAUDRATE = 9600 ; // Default software serial baudrate
const uint32_t APP_BAUDRATE = 115200 ; // Default serial baudrate
const uint32_t SERIAL_POLLING = 100 ; // Serial receive polling in ms
const uint8_t MAX_STATUS = 11 ; // Max number of status lines
2017-10-10 14:40:02 +01:00
2019-07-05 12:44:24 +01:00
const uint32_t DRIVER_BOOT_DELAY = 1 ; // Number of milliseconds to retard driver cycles during boot-up time to reduce overall CPU load whilst Wifi is connecting
const uint32_t LOOP_SLEEP_DELAY = 50 ; // Lowest number of milliseconds to go through the main loop using delay when needed
2017-10-10 14:40:02 +01:00
2019-07-05 12:44:24 +01:00
/*********************************************************************************************\
* Defines
\ * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2017-10-10 14:40:02 +01:00
2019-07-05 12:44:24 +01:00
# define MAX_RULE_TIMERS 8 // Max number of rule timers (4 bytes / timer)
# define MAX_RULE_VARS 5 // Max number of rule variables (10 bytes / variable)
2018-12-15 14:55:51 +00:00
2018-07-14 14:35:56 +01:00
# define NO_EXTRA_4K_HEAP // Allocate 4k heap for WPS in ESP8166/Arduino core v2.4.2 (was always allocated in previous versions)
2017-11-17 16:52:31 +00:00
/*
// Removed from esp8266 core since 20171105
# define min(a,b) ((a)<(b)?(a):(b))
# define max(a,b) ((a)>(b)?(a):(b))
*/
2018-07-14 11:10:13 +01:00
# define tmin(a,b) ((a)<(b)?(a):(b))
# define tmax(a,b) ((a)>(b)?(a):(b))
2017-11-17 16:52:31 +00:00
2018-05-17 14:36:45 +01:00
# define STR_HELPER(x) #x
2018-12-15 14:55:51 +00:00
# ifndef STR
2018-05-17 14:36:45 +01:00
# define STR(x) STR_HELPER(x)
2018-12-15 14:55:51 +00:00
# endif
2018-05-17 14:36:45 +01:00
2018-03-17 14:22:29 +00:00
//enum ws2812NeopixelbusFeature { NEO_RGB, NEO_GRB, NEO_BRG, NEO_RBG, NEO_3LED, NEO_RGBW, NEO_GRBW }; // Doesn't work
# define NEO_RGB 0 // Neopixel RGB leds
# define NEO_GRB 1 // Neopixel GRB leds
# define NEO_BRG 2 // Neopixel BRG leds
# define NEO_RBG 3 // Neopixel RBG leds
# define NEO_3LED 4 // Placeholder to test for 4 led types
# define NEO_RGBW 5 // Neopixel RGBW leds
# define NEO_GRBW 6 // Neopixel GRBW leds
2019-03-13 17:00:15 +00:00
# define LT_SM16716 16 // Lights that use SM16716 will have this bit set in light_type
# define RGB_REMAP_RGBW 0
# define RGB_REMAP_RBGW 6
# define RGB_REMAP_GRBW 24
# define RGB_REMAP_GBRW 30
# define RGB_REMAP_BRGW 48
# define RGB_REMAP_BGRW 54
2018-03-17 14:22:29 +00:00
# define MQTT_PUBSUBCLIENT 1 // Mqtt PubSubClient library
2018-10-26 16:13:17 +01:00
# define MQTT_TASMOTAMQTT 2 // Mqtt TasmotaMqtt library based on esp-mqtt-arduino - soon obsolete
# define MQTT_ESPMQTTARDUINO 3 // Mqtt esp-mqtt-arduino library by Ingo Randolf - obsolete but define is present for debugging purposes
# define MQTT_ARDUINOMQTT 4 // Mqtt arduino-mqtt library by Joel Gaehwiler (https://github.com/256dpi/arduino-mqtt)
2018-03-17 14:22:29 +00:00
2018-04-23 10:37:46 +01:00
// Sunrise and Sunset DawnType
# define DAWN_NORMAL -0.8333
# define DAWN_CIVIL -6.0
# define DAWN_NAUTIC -12.0
# define DAWN_ASTRONOMIC -18.0
2018-05-31 06:46:09 +01:00
// Sensor and Commands definition for KNX Driver
2018-05-17 10:55:40 +01:00
# define KNX_TEMPERATURE 17
# define KNX_HUMIDITY 18
# define KNX_ENERGY_VOLTAGE 19
# define KNX_ENERGY_CURRENT 20
# define KNX_ENERGY_POWER 21
2018-05-17 05:42:51 +01:00
# define KNX_ENERGY_POWERFACTOR 22
2018-05-17 10:55:40 +01:00
# define KNX_ENERGY_DAILY 23
# define KNX_ENERGY_START 24
# define KNX_ENERGY_TOTAL 25
2018-05-30 19:19:17 +01:00
# define KNX_SLOT1 26
# define KNX_SLOT2 27
# define KNX_SLOT3 28
# define KNX_SLOT4 29
# define KNX_SLOT5 30
# define KNX_MAX_device_param 30
2018-05-31 06:46:09 +01:00
# define MAX_KNXTX_CMNDS 5
2018-05-17 05:42:51 +01:00
2017-10-10 14:40:02 +01:00
/*********************************************************************************************\
* Enumeration
\ * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2017-10-18 17:22:34 +01:00
enum WeekInMonthOptions { Last , First , Second , Third , Fourth } ;
enum DayOfTheWeekOptions { Sun = 1 , Mon , Tue , Wed , Thu , Fri , Sat } ;
enum MonthNamesOptions { Jan = 1 , Feb , Mar , Apr , May , Jun , Jul , Aug , Sep , Oct , Nov , Dec } ;
enum HemisphereOptions { North , South } ;
2018-10-30 16:29:01 +00:00
enum GetDateAndTimeOptions { DT_LOCAL , DT_UTC , DT_RESTART , DT_ENERGY } ;
2017-11-08 10:57:16 +00:00
2017-10-18 17:22:34 +01:00
enum LoggingLevels { LOG_LEVEL_NONE , LOG_LEVEL_ERROR , LOG_LEVEL_INFO , LOG_LEVEL_DEBUG , LOG_LEVEL_DEBUG_MORE , LOG_LEVEL_ALL } ;
2017-11-08 10:57:16 +00:00
2019-03-13 17:00:15 +00:00
enum WifiConfigOptions { WIFI_RESTART , WIFI_SMARTCONFIG , WIFI_MANAGER , WIFI_WPSCONFIG , WIFI_RETRY , WIFI_WAIT , WIFI_SERIAL , WIFI_MANAGER_RESET_ONLY , MAX_WIFI_OPTION } ;
2017-11-08 10:57:16 +00:00
2017-11-22 14:34:07 +00:00
enum SwitchModeOptions { TOGGLE , FOLLOW , FOLLOW_INV , PUSHBUTTON , PUSHBUTTON_INV , PUSHBUTTONHOLD , PUSHBUTTONHOLD_INV , PUSHBUTTON_TOGGLE , MAX_SWITCH_OPTION } ;
2017-11-08 10:57:16 +00:00
2017-10-18 17:22:34 +01:00
enum LedStateOptions { LED_OFF , LED_POWER , LED_MQTTSUB , LED_POWER_MQTTSUB , LED_MQTTPUB , LED_POWER_MQTTPUB , LED_MQTT , LED_POWER_MQTT , MAX_LED_OPTION } ;
2017-11-08 10:57:16 +00:00
2017-10-18 17:22:34 +01:00
enum EmulationOptions { EMUL_NONE , EMUL_WEMO , EMUL_HUE , EMUL_MAX } ;
2017-11-08 10:57:16 +00:00
2018-01-18 15:19:28 +00:00
enum TopicOptions { CMND , STAT , TELE , nu1 , RESULT_OR_CMND , RESULT_OR_STAT , RESULT_OR_TELE } ;
2018-01-24 16:31:20 +00:00
enum ExecuteCommandPowerOptions { POWER_OFF , POWER_ON , POWER_TOGGLE , POWER_BLINK , POWER_BLINK_STOP , power_nu1 , POWER_OFF_NO_STATE , POWER_ON_NO_STATE , power_nu2 , POWER_SHOW_STATE } ;
enum PowerOnStateOptions { POWER_ALL_OFF , POWER_ALL_ON , POWER_ALL_SAVED_TOGGLE , POWER_ALL_SAVED , POWER_ALL_ALWAYS_ON , POWER_ALL_OFF_PULSETIME_ON } ;
2018-08-27 13:53:09 +01:00
enum ButtonStates { PRESSED , NOT_PRESSED } ;
enum Shortcuts { SC_CLEAR , SC_DEFAULT , SC_USER } ;
2017-11-08 10:57:16 +00:00
2019-07-05 12:44:24 +01:00
enum SettingsParmaIndex { P_HOLD_TIME , P_MAX_POWER_RETRY , P_TUYA_DIMMER_ID , P_MDNS_DELAYED_START , P_BOOT_LOOP_OFFSET , P_RGB_REMAP , P_IR_UNKNOW_THRESHOLD , P_CSE7766_INVALID_POWER , P_HOLD_IGNORE , P_MAX_PARAM8 } ; // Max is PARAM8_SIZE (18) - SetOption32 until SetOption49
2017-11-08 10:57:16 +00:00
2018-01-05 11:26:19 +00:00
enum DomoticzSensors { DZ_TEMP , DZ_TEMP_HUM , DZ_TEMP_HUM_BARO , DZ_POWER_ENERGY , DZ_ILLUMINANCE , DZ_COUNT , DZ_VOLTAGE , DZ_CURRENT , DZ_AIRQUALITY , DZ_MAX_SENSORS } ;
2018-03-10 16:58:54 +00:00
enum Ws2812ClockIndex { WS_SECOND , WS_MINUTE , WS_HOUR , WS_MARKER } ;
enum Ws2812Color { WS_RED , WS_GREEN , WS_BLUE } ;
2018-12-15 14:55:51 +00:00
2019-07-05 12:44:24 +01:00
enum LightSubtypes { LST_NONE , LST_SINGLE , LST_COLDWARM , LST_RGB , LST_RGBW , LST_RGBWC , LST_MAX = 5 } ; // Do not insert new fields
2018-12-15 14:55:51 +00:00
enum LightTypes { LT_BASIC , LT_PWM1 , LT_PWM2 , LT_PWM3 , LT_PWM4 , LT_PWM5 , LT_PWM6 , LT_PWM7 ,
LT_NU8 , LT_SERIAL1 , LT_SERIAL2 , LT_WS2812 , LT_RGBW , LT_RGBWC , LT_NU14 , LT_NU15 } ; // Do not insert new fields
enum LightSchemes { LS_POWER , LS_WAKEUP , LS_CYCLEUP , LS_CYCLEDN , LS_RANDOM , LS_MAX } ;
2017-11-08 10:57:16 +00:00
2018-10-26 16:13:17 +01:00
enum XsnsFunctions { FUNC_SETTINGS_OVERRIDE , FUNC_MODULE_INIT , FUNC_PRE_INIT , FUNC_INIT ,
FUNC_LOOP , FUNC_EVERY_50_MSECOND , FUNC_EVERY_100_MSECOND , FUNC_EVERY_200_MSECOND , FUNC_EVERY_250_MSECOND , FUNC_EVERY_SECOND ,
2019-07-05 12:44:24 +01:00
FUNC_SAVE_AT_MIDNIGHT , FUNC_SAVE_BEFORE_RESTART ,
FUNC_PREP_BEFORE_TELEPERIOD , FUNC_JSON_APPEND , FUNC_WEB_SENSOR , FUNC_COMMAND , FUNC_COMMAND_SENSOR , FUNC_COMMAND_DRIVER ,
2018-10-26 16:13:17 +01:00
FUNC_MQTT_SUBSCRIBE , FUNC_MQTT_INIT , FUNC_MQTT_DATA ,
FUNC_SET_POWER , FUNC_SET_DEVICE_POWER , FUNC_SHOW_SENSOR ,
2019-07-05 12:44:24 +01:00
FUNC_ENERGY_EVERY_SECOND ,
2018-10-26 16:13:17 +01:00
FUNC_RULES_PROCESS , FUNC_SERIAL , FUNC_FREE_MEM , FUNC_BUTTON_PRESSED ,
2018-12-22 17:42:40 +00:00
FUNC_WEB_ADD_BUTTON , FUNC_WEB_ADD_MAIN_BUTTON , FUNC_WEB_ADD_HANDLER , FUNC_SET_CHANNELS } ;
2017-10-18 17:22:34 +01:00
2018-05-28 14:52:42 +01:00
enum CommandSource { SRC_IGNORE , SRC_MQTT , SRC_RESTART , SRC_BUTTON , SRC_SWITCH , SRC_BACKLOG , SRC_SERIAL , SRC_WEBGUI , SRC_WEBCOMMAND , SRC_WEBCONSOLE , SRC_PULSETIMER ,
2019-07-05 12:44:24 +01:00
SRC_TIMER , SRC_RULE , SRC_MAXPOWER , SRC_MAXENERGY , SRC_OVERTEMP , SRC_LIGHT , SRC_KNX , SRC_DISPLAY , SRC_WEMO , SRC_HUE , SRC_RETRY , SRC_MAX } ;
const char kCommandSource [ ] PROGMEM = " I|MQTT|Restart|Button|Switch|Backlog|Serial|WebGui|WebCommand|WebConsole|PulseTimer|Timer|Rule|MaxPower|MaxEnergy|Overtemp|Light|Knx|Display|Wemo|Hue|Retry " ;
2018-05-28 14:52:42 +01:00
2018-12-15 14:55:51 +00:00
const uint8_t kDefaultRfCode [ 9 ] PROGMEM = { 0x21 , 0x16 , 0x01 , 0x0E , 0x03 , 0x48 , 0x2E , 0x1A , 0x00 } ;
const uint8_t kIFan02Speed [ MAX_FAN_SPEED ] [ 3 ] = { { 6 , 6 , 6 } , { 7 , 6 , 6 } , { 7 , 7 , 6 } , { 7 , 6 , 7 } } ; // Do not use PROGMEM as it fails
2018-01-20 11:12:39 +00:00
/*********************************************************************************************\
* Extern global variables
\ * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
extern uint8_t light_device ; // Light device number
2019-03-13 17:00:15 +00:00
extern uint8_t light_power ; // Light power
extern uint8_t rotary_changed ; // Rotary switch changed
2018-01-20 11:12:39 +00:00
2018-05-30 19:19:17 +01:00
# endif // _SONOFF_H_