2020-05-23 12:09:16 +01:00
/*
2019-10-27 11:04:08 +00:00
tasmota . ino - Tasmota firmware for iTead Sonoff , Wemos and NodeMCU hardware
2017-05-13 12:02:10 +01:00
2021-01-01 12:44:04 +00:00
Copyright ( C ) 2021 Theo Arends
2017-05-13 12:02:10 +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-07-15 14:07:30 +01:00
2020-07-16 17:21:19 +01:00
/*********************************************************************************************\
* Preferred IDE is Visual Studio Code with PlatformIO extension which doesn ' t need prerequisites
*
* Limited support for Arduino IDE needs Prerequisites :
* - Change libraries / PubSubClient / src / PubSubClient . h
* # define MQTT_MAX_PACKET_SIZE 1200
*
* Arduino IDE 1.8 .12 and up parameters for partly support
* - Select IDE Tools - Board : " Generic ESP8266 Module "
* - Select IDE Tools - Flash Mode : " DOUT (compatible) "
* - Select IDE Tools - Flash Size : " 1M (FS:none OTA:~502KB) "
* - Select IDE Tools - LwIP Variant : " v2 Higher Bandwidth (no feature) "
* - Select IDE Tools - VTables : " Flash "
* - Select IDE Tools - Espressif FW : " nonos-sdk-2.2.1+100 (190703) "
\ * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2017-01-28 13:41:01 +00:00
2017-10-18 17:22:34 +01:00
// Location specific includes
2020-11-01 14:24:12 +00:00
# ifndef ESP32_STAGE // ESP32 Stage has no core_version.h file. Disable include via PlatformIO Option
2020-07-17 10:49:56 +01:00
# include <core_version.h> // Arduino_Esp8266 version information (ARDUINO_ESP8266_RELEASE and ARDUINO_ESP8266_RELEASE_2_7_1)
2020-11-01 14:24:12 +00:00
# endif
2020-04-10 17:24:08 +01:00
# include "tasmota_compat.h"
2019-10-27 11:04:08 +00:00
# include "tasmota_version.h" // Tasmota version information
# include "tasmota.h" // Enumeration used in my_user_config.h
2018-10-23 14:51:51 +01:00
# include "my_user_config.h" // Fixed user configurable options
2020-06-10 19:14:46 +01:00
# ifdef USE_TLS
2020-04-10 09:52:22 +01:00
# include <t_bearssl.h> // We need to include before "tasmota_globals.h" to take precedence over the BearSSL version in Arduino
2020-06-10 19:14:46 +01:00
# endif // USE_TLS
2020-04-09 14:32:23 +01:00
# include "tasmota_globals.h" // Function prototypes and global configuration
2018-10-23 14:51:51 +01:00
# include "i18n.h" // Language support configured by my_user_config.h
2019-10-27 11:04:08 +00:00
# include "tasmota_template.h" // Hardware configuration
2017-03-03 11:35:23 +00:00
2018-02-27 13:59:46 +00:00
// Libraries
2018-11-20 14:00:24 +00:00
# include <ESP8266HTTPClient.h> // Ota
2017-01-28 13:41:01 +00:00
# include <ESP8266httpUpdate.h> // Ota
2017-06-04 16:40:27 +01:00
# include <StreamString.h> // Webserver, Updater
2021-01-24 15:35:36 +00:00
# include <ext_printf.h>
2021-01-25 15:02:56 +00:00
# include <SBuffer.hpp>
2020-09-24 07:51:43 +01:00
# include <JsonParser.h>
# include <JsonGenerator.h>
2018-03-21 16:58:39 +00:00
# ifdef USE_ARDUINO_OTA
# include <ArduinoOTA.h> // Arduino OTA
# ifndef USE_DISCOVERY
# define USE_DISCOVERY
# endif
# endif // USE_ARDUINO_OTA
2017-01-28 13:41:01 +00:00
# ifdef USE_DISCOVERY
2018-03-21 16:58:39 +00:00
# include <ESP8266mDNS.h> // MQTT, Webserver, Arduino OTA
2017-01-28 13:41:01 +00:00
# endif // USE_DISCOVERY
2020-09-03 19:31:31 +01:00
//#ifdef USE_I2C
2017-01-28 13:41:01 +00:00
# include <Wire.h> // I2C support library
2020-09-03 19:31:31 +01:00
//#endif // USE_I2C
2017-07-15 14:07:30 +01:00
# ifdef USE_SPI
2021-01-20 16:03:53 +00:00
# include <SPI.h> // SPI support, TFT, SDcard
2017-04-29 13:40:53 +01:00
# endif // USE_SPI
2019-02-17 09:42:55 +00:00
2021-01-20 14:43:26 +00:00
# ifdef USE_UFILESYS
# ifdef ESP8266
# include <LittleFS.h>
# include <SPI.h>
# ifdef USE_SDCARD
# include <SD.h>
2021-04-14 12:41:10 +01:00
# include <SdFat.h>
2021-01-20 14:43:26 +00:00
# endif // USE_SDCARD
# endif // ESP8266
# ifdef ESP32
# include <LITTLEFS.h>
# ifdef USE_SDCARD
# include <SD.h>
# endif // USE_SDCARD
# include "FFat.h"
# include "FS.h"
# endif // ESP32
# endif // USE_UFILESYS
2017-10-18 17:22:34 +01:00
// Structs
2017-02-19 16:49:17 +00:00
# include "settings.h"
2017-01-28 13:41:01 +00:00
2019-12-02 09:44:27 +00:00
/*********************************************************************************************\
* Global variables
\ * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2021-04-12 10:49:53 +01:00
const uint32_t VERSION_MARKER [ ] PROGMEM = { 0x5AA55AA5 , 0xFFFFFFFF , 0xA55AA55A } ;
2018-08-27 15:07:23 +01:00
WiFiUDP PortUdp ; // UDP Syslog and Alexa
2020-10-28 16:32:07 +00:00
struct {
2020-10-29 11:21:24 +00:00
uint32_t global_update ; // Timestamp of last global temperature and humidity update
2020-10-28 16:32:07 +00:00
uint32_t baudrate ; // Current Serial baudrate
uint32_t pulse_timer [ MAX_PULSETIMERS ] ; // Power off timer
uint32_t blink_timer ; // Power cycle timer
2020-11-27 13:43:45 +00:00
uint32_t backlog_timer ; // Timer for next command in backlog
2020-10-28 16:32:07 +00:00
uint32_t loop_load_avg ; // Indicative loop load average
2020-12-18 15:30:37 +00:00
uint32_t log_buffer_pointer ; // Index in log buffer
2020-10-28 16:32:07 +00:00
uint32_t uptime ; // Counting every second until 4294967295 = 130 year
2021-03-23 10:21:38 +00:00
uint32_t zc_time ; // Zero-cross moment (microseconds)
uint32_t zc_offset ; // Zero cross moment offset due to monitoring chip processing (microseconds)
uint32_t zc_code_offset ; // Zero cross moment offset due to executing power code (microseconds)
uint32_t zc_interval ; // Zero cross interval around 8333 (60Hz) or 10000 (50Hz) (microseconds)
2020-12-20 12:22:01 +00:00
GpioOptionABits gpio_optiona ; // GPIO Option_A flags
2021-01-01 16:04:36 +00:00
void * log_buffer_mutex ; // Control access to log buffer
2020-10-28 16:32:07 +00:00
2020-10-28 18:03:39 +00:00
power_t power ; // Current copy of Settings.power
power_t rel_inverted ; // Relay inverted flag (1 = (0 = On, 1 = Off))
power_t last_power ; // Last power set state
power_t blink_power ; // Blink power state
power_t blink_powersave ; // Blink start power save state
power_t blink_mask ; // Blink relay active mask
2020-10-29 11:21:24 +00:00
int serial_in_byte_counter ; // Index in receive buffer
2020-10-28 18:03:39 +00:00
float temperature_celsius ; // Provide a global temperature to be used by some sensors
float humidity ; // Provide a global humidity to be used by some sensors
float pressure_hpa ; // Provide a global pressure to be used by some sensors
2020-10-29 12:58:50 +00:00
uint16_t gpio_pin [ MAX_GPIO_PIN ] ; // GPIO functions indexed by pin number
2020-10-30 11:29:48 +00:00
myio my_module ; // Active copy of Module GPIOs (17 x 16 bits)
2020-10-29 12:37:09 +00:00
uint16_t blink_counter ; // Number of blink cycles
uint16_t seriallog_timer ; // Timer to disable Seriallog
uint16_t syslog_timer ; // Timer to re-enable syslog_level
uint16_t tele_period ; // Tele period timer
2020-10-29 12:58:50 +00:00
int16_t save_data_counter ; // Counter and flag for config save to Flash
RulesBitfield rules_flag ; // Rule state flags (16 bits)
2020-10-29 12:37:09 +00:00
2020-10-30 11:29:48 +00:00
bool serial_local ; // Handle serial locally
bool fallback_topic_flag ; // Use Topic or FallbackTopic
2021-04-07 16:55:33 +01:00
bool backlog_nodelay ; // Execute all backlog commands with no delay
2020-10-30 11:29:48 +00:00
bool backlog_mutex ; // Command backlog pending
bool stop_flash_rotate ; // Allow flash configuration rotation
bool blinkstate ; // LED state
bool pwm_present ; // Any PWM channel configured with SetOption15 0
bool i2c_enabled ; // I2C configured
2021-03-10 21:20:21 +00:00
# ifdef ESP32
bool i2c_enabled_2 ; // I2C configured, second controller on ESP32, Wire1
# endif
2020-10-30 11:29:48 +00:00
bool ntp_force_sync ; // Force NTP sync
bool skip_light_fade ; // Temporarily skip light fading
bool restart_halt ; // Do not restart but stay in wait loop
2020-11-01 11:12:27 +00:00
bool module_changed ; // Indicate module changed since last restart
2021-01-14 16:03:01 +00:00
bool wifi_stay_asleep ; // Allow sleep only incase of ESP32 BLE
2021-02-15 15:51:13 +00:00
bool no_autoexec ; // Disable autoexec
2020-10-30 11:29:48 +00:00
StateBitfield global_state ; // Global states (currently Wifi and Mqtt) (8 bits)
2020-12-31 15:17:30 +00:00
uint8_t spi_enabled ; // SPI configured
uint8_t soft_spi_enabled ; // Software SPI configured
2020-10-29 11:39:44 +00:00
uint8_t blinks ; // Number of LED blinks
uint8_t restart_flag ; // Tasmota restart flag
2020-10-29 11:58:22 +00:00
uint8_t ota_state_flag ; // OTA state flag
2020-10-29 11:39:44 +00:00
uint8_t wifi_state_flag ; // Wifi state flag
2020-10-29 12:58:50 +00:00
uint8_t mqtt_cmnd_blocked ; // Ignore flag for publish command
uint8_t mqtt_cmnd_blocked_reset ; // Count down to reset if needed
uint8_t state_250mS ; // State 250msecond per second flag
uint8_t latching_relay_pulse ; // Latching relay pulse timer
uint8_t active_device ; // Active device in ExecuteCommandPower
2020-10-29 15:16:34 +00:00
uint8_t sleep ; // Current copy of Settings.sleep
2020-10-30 11:29:48 +00:00
uint8_t leds_present ; // Max number of LED supported
uint8_t led_inverted ; // LED inverted flag (1 = (0 = On, 1 = Off))
uint8_t led_power ; // LED power state
uint8_t ledlnk_inverted ; // Link LED inverted flag (1 = (0 = On, 1 = Off))
uint8_t pwm_inverted ; // PWM inverted flag (1 = inverted)
uint8_t energy_driver ; // Energy monitor configured
uint8_t light_driver ; // Light module configured
uint8_t light_type ; // Light types
uint8_t serial_in_byte ; // Received byte
uint8_t devices_present ; // Max number of devices supported
uint8_t masterlog_level ; // Master log level used to override set log level
uint8_t seriallog_level ; // Current copy of Settings.seriallog_level
uint8_t syslog_level ; // Current copy of Settings.syslog_level
2020-12-22 14:26:07 +00:00
uint8_t templog_level ; // Temporary log level to be used by HTTP cm and Telegram
2020-10-30 11:29:48 +00:00
uint8_t module_type ; // Current copy of Settings.module or user template type
uint8_t last_source ; // Last command source
uint8_t shutters_present ; // Number of actual define shutters
2020-12-18 15:49:11 +00:00
// uint8_t prepped_loglevel; // Delayed log level message
2020-10-30 11:29:48 +00:00
# ifndef SUPPORT_IF_STATEMENT
uint8_t backlog_index ; // Command backlog index
uint8_t backlog_pointer ; // Command backlog pointer
String backlog [ MAX_BACKLOG ] ; // Command backlog buffer
# endif
2020-10-29 11:21:24 +00:00
2020-10-30 11:29:48 +00:00
char version [ 16 ] ; // Composed version string like 255.255.255.255
char image_name [ 33 ] ; // Code image and/or commit
char hostname [ 33 ] ; // Composed Wifi hostname
char serial_in_buffer [ INPUT_BUFFER_SIZE ] ; // Receive buffer
char mqtt_client [ 99 ] ; // Composed MQTT Clientname
char mqtt_topic [ TOPSZ ] ; // Composed MQTT topic
char mqtt_data [ MESSZ ] ; // MQTT publish buffer and web page ajax buffer
2020-12-18 15:30:37 +00:00
char log_buffer [ LOG_BUFFER_SIZE ] ; // Web log buffer
2020-10-28 16:32:07 +00:00
} TasmotaGlobal ;
2019-09-09 16:24:27 +01:00
# ifdef SUPPORT_IF_STATEMENT
# include <LinkedList.h>
2019-10-14 15:12:59 +01:00
LinkedList < String > backlog ; // Command backlog implemented with LinkedList
2019-09-09 16:24:27 +01:00
# define BACKLOG_EMPTY (backlog.size() == 0)
# else
2020-10-30 11:29:48 +00:00
# define BACKLOG_EMPTY (TasmotaGlobal.backlog_pointer == TasmotaGlobal.backlog_index)
2019-09-09 16:24:27 +01:00
# endif
2018-11-24 16:16:27 +00:00
2017-07-25 17:05:47 +01:00
/*********************************************************************************************\
2019-12-02 09:44:27 +00:00
* Main
2017-07-25 17:05:47 +01:00
\ * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2018-11-24 16:43:18 +00:00
2020-05-11 15:45:22 +01:00
void setup ( void ) {
2020-04-17 17:27:31 +01:00
# ifdef ESP32
2020-04-18 15:34:29 +01:00
# ifdef DISABLE_ESP32_BROWNOUT
DisableBrownout ( ) ; // Workaround possible weak LDO resulting in brownout detection during Wifi connection
2020-04-17 17:27:31 +01:00
# endif
# endif
2020-12-17 15:41:04 +00:00
RtcPreInit ( ) ;
2021-01-11 16:56:18 +00:00
SettingsInit ( ) ;
2020-12-17 15:41:04 +00:00
2020-10-28 16:32:07 +00:00
memset ( & TasmotaGlobal , 0 , sizeof ( TasmotaGlobal ) ) ;
TasmotaGlobal . baudrate = APP_BAUDRATE ;
2020-12-07 16:03:22 +00:00
TasmotaGlobal . seriallog_timer = SERIALLOG_TIMER ;
2020-10-28 18:03:39 +00:00
TasmotaGlobal . temperature_celsius = NAN ;
2020-10-29 11:39:44 +00:00
TasmotaGlobal . blinks = 201 ;
TasmotaGlobal . wifi_state_flag = WIFI_RESTART ;
2020-10-29 12:37:09 +00:00
TasmotaGlobal . tele_period = 9999 ;
2020-10-29 12:58:50 +00:00
TasmotaGlobal . active_device = 1 ;
2020-10-30 11:29:48 +00:00
TasmotaGlobal . global_state . data = 0xF ; // Init global state (wifi_down, mqtt_down) to solve possible network issues
2020-04-18 15:34:29 +01:00
2018-09-04 15:22:34 +01:00
RtcRebootLoad ( ) ;
2019-10-16 18:00:20 +01:00
if ( ! RtcRebootValid ( ) ) {
RtcReboot . fast_reboot_count = 0 ;
}
2020-07-15 15:09:32 +01:00
# ifdef FIRMWARE_MINIMAL
2020-11-27 15:43:56 +00:00
RtcReboot . fast_reboot_count = 0 ; // Disable fast reboot and quick power cycle detection
2020-07-15 15:09:32 +01:00
# else
2020-11-27 15:43:56 +00:00
if ( ResetReason ( ) = = REASON_DEEP_SLEEP_AWAKE ) {
RtcReboot . fast_reboot_count = 0 ; // Disable fast reboot and quick power cycle detection
} else {
RtcReboot . fast_reboot_count + + ;
}
2020-07-15 15:09:32 +01:00
# endif
2018-09-04 15:22:34 +01:00
RtcRebootSave ( ) ;
2018-08-30 13:27:33 +01:00
2021-01-05 10:49:13 +00:00
if ( RtcSettingsLoad ( 0 ) ) {
2020-12-07 14:33:09 +00:00
uint32_t baudrate = ( RtcSettings . baudrate / 300 ) * 300 ; // Make it a valid baudrate
if ( baudrate ) { TasmotaGlobal . baudrate = baudrate ; }
}
2020-10-28 16:32:07 +00:00
Serial . begin ( TasmotaGlobal . baudrate ) ;
2020-12-07 14:33:09 +00:00
Serial . println ( ) ;
2020-05-11 15:45:22 +01:00
// Serial.setRxBufferSize(INPUT_BUFFER_SIZE); // Default is 256 chars
2020-10-30 11:29:48 +00:00
TasmotaGlobal . seriallog_level = LOG_LEVEL_INFO ; // Allow specific serial messages until config loaded
2017-01-28 13:41:01 +00:00
2021-04-21 10:55:52 +01:00
AddLog ( LOG_LEVEL_INFO , PSTR ( " HDW: %s " ) , GetDeviceHardware ( ) . c_str ( ) ) ;
2021-01-08 14:00:40 +00:00
# ifdef USE_UFILESYS
UfsInit ( ) ; // xdrv_50_filesystem.ino
# endif
2017-10-18 17:22:34 +01:00
SettingsLoad ( ) ;
SettingsDelta ( ) ;
2017-01-28 13:41:01 +00:00
2017-10-18 17:22:34 +01:00
OsWatchInit ( ) ;
2017-02-28 15:01:48 +00:00
2020-12-07 16:03:22 +00:00
TasmotaGlobal . seriallog_level = Settings . seriallog_level ;
TasmotaGlobal . syslog_level = Settings . syslog_level ;
2020-12-07 14:33:09 +00:00
TasmotaGlobal . module_changed = ( Settings . module ! = Settings . last_module ) ;
if ( TasmotaGlobal . module_changed ) {
Settings . baudrate = APP_BAUDRATE / 300 ;
Settings . serial_config = TS_SERIAL_8N1 ;
}
2020-12-07 16:03:22 +00:00
SetSerialBaudrate ( Settings . baudrate * 300 ) ; // Reset serial interface if current baudrate is different from requested baudrate
2020-12-07 14:33:09 +00:00
2020-12-07 16:03:22 +00:00
if ( 1 = = RtcReboot . fast_reboot_count ) { // Allow setting override only when all is well
2019-10-18 14:18:39 +01:00
UpdateQuickPowerCycle ( true ) ;
2018-10-23 14:51:51 +01:00
}
2021-04-22 17:10:26 +01:00
if ( ResetReason ( ) ! = REASON_DEEP_SLEEP_AWAKE ) {
# ifdef ESP8266
Settings . flag4 . network_wifi = 1 ; // Make sure we're in control
# endif
# ifdef ESP32
if ( ! Settings . flag4 . network_ethernet ) {
Settings . flag4 . network_wifi = 1 ; // Make sure we're in control
}
# endif
}
2020-10-30 11:29:48 +00:00
TasmotaGlobal . stop_flash_rotate = Settings . flag . stop_flash_rotate ; // SetOption12 - Switch between dynamic or fixed slot flash save location
2020-10-29 12:58:50 +00:00
TasmotaGlobal . save_data_counter = Settings . save_data ;
2020-10-29 15:16:34 +00:00
TasmotaGlobal . sleep = Settings . sleep ;
2018-11-16 08:36:41 +00:00
# ifndef USE_EMULATION
Settings . flag2 . emulation = 0 ;
2019-05-20 14:09:42 +01:00
# else
# ifndef USE_EMULATION_WEMO
if ( EMUL_WEMO = = Settings . flag2 . emulation ) { Settings . flag2 . emulation = 0 ; }
# endif
# ifndef USE_EMULATION_HUE
if ( EMUL_HUE = = Settings . flag2 . emulation ) { Settings . flag2 . emulation = 0 ; }
# endif
2018-11-16 08:36:41 +00:00
# endif // USE_EMULATION
2017-01-28 13:41:01 +00:00
2020-10-28 16:32:07 +00:00
// AddLogBuffer(LOG_LEVEL_DEBUG, (uint8_t*)&TasmotaGlobal, sizeof(TasmotaGlobal));
2019-11-10 11:05:09 +00:00
if ( Settings . param [ P_BOOT_LOOP_OFFSET ] ) { // SetOption36
2019-01-30 13:34:31 +00:00
// Disable functionality as possible cause of fast restart within BOOT_LOOP_TIME seconds (Exception, WDT or restarts)
if ( RtcReboot . fast_reboot_count > Settings . param [ P_BOOT_LOOP_OFFSET ] ) { // Restart twice
2019-11-03 12:51:22 +00:00
Settings . flag3 . user_esp8285_enable = 0 ; // SetOption51 - Enable ESP8285 user GPIO's - Disable ESP8285 Generic GPIOs interfering with flash SPI
2019-01-30 13:34:31 +00:00
if ( RtcReboot . fast_reboot_count > Settings . param [ P_BOOT_LOOP_OFFSET ] + 1 ) { // Restart 3 times
2019-06-30 15:44:36 +01:00
for ( uint32_t i = 0 ; i < MAX_RULE_SETS ; i + + ) {
2019-01-30 13:34:31 +00:00
if ( bitRead ( Settings . rule_stop , i ) ) {
bitWrite ( Settings . rule_enabled , i , 0 ) ; // Disable rules causing boot loop
}
2018-08-30 13:27:33 +01:00
}
}
2019-01-30 13:34:31 +00:00
if ( RtcReboot . fast_reboot_count > Settings . param [ P_BOOT_LOOP_OFFSET ] + 2 ) { // Restarted 4 times
Settings . rule_enabled = 0 ; // Disable all rules
2021-02-15 15:51:13 +00:00
TasmotaGlobal . no_autoexec = true ;
2018-08-30 13:47:48 +01:00
}
2019-01-30 13:34:31 +00:00
if ( RtcReboot . fast_reboot_count > Settings . param [ P_BOOT_LOOP_OFFSET ] + 3 ) { // Restarted 5 times
2021-02-28 11:50:02 +00:00
for ( uint32_t i = 0 ; i < nitems ( Settings . my_gp . io ) ; i + + ) {
2019-01-30 13:34:31 +00:00
Settings . my_gp . io [ i ] = GPIO_NONE ; // Reset user defined GPIO disabling sensors
}
}
if ( RtcReboot . fast_reboot_count > Settings . param [ P_BOOT_LOOP_OFFSET ] + 4 ) { // Restarted 6 times
2020-06-17 13:06:46 +01:00
Settings . module = Settings . fallback_module ; // Reset module to fallback module
// Settings.last_module = Settings.fallback_module;
2019-01-30 13:34:31 +00:00
}
2021-01-23 15:26:23 +00:00
AddLog ( LOG_LEVEL_INFO , PSTR ( " FRC: " D_LOG_SOME_SETTINGS_RESET " (%d) " ) , RtcReboot . fast_reboot_count ) ;
2018-08-28 09:26:33 +01:00
}
}
2020-12-07 14:33:09 +00:00
snprintf_P ( TasmotaGlobal . version , sizeof ( TasmotaGlobal . version ) , PSTR ( " %d.%d.%d " ) , VERSION > > 24 & 0xff , VERSION > > 16 & 0xff , VERSION > > 8 & 0xff ) ; // Release version 6.3.0
if ( VERSION & 0xff ) { // Development or patched version 6.3.0.10
snprintf_P ( TasmotaGlobal . version , sizeof ( TasmotaGlobal . version ) , PSTR ( " %s.%d " ) , TasmotaGlobal . version , VERSION & 0xff ) ;
}
// Thehackbox inserts "release" or "commit number" before compiling using sed -i -e 's/PSTR("(%s)")/PSTR("(85cff52-%s)")/g' tasmota.ino
2021-01-18 20:48:04 +00:00
snprintf_P ( TasmotaGlobal . image_name , sizeof ( TasmotaGlobal . image_name ) , PSTR ( " (%s) " ) , PSTR ( CODE_IMAGE_STR ) ) ; // Results in (85cff52-tasmota) or (release-tasmota)
2020-12-07 14:33:09 +00:00
2020-10-30 11:29:48 +00:00
Format ( TasmotaGlobal . mqtt_client , SettingsText ( SET_MQTT_CLIENT ) , sizeof ( TasmotaGlobal . mqtt_client ) ) ;
Format ( TasmotaGlobal . mqtt_topic , SettingsText ( SET_MQTT_TOPIC ) , sizeof ( TasmotaGlobal . mqtt_topic ) ) ;
2020-11-04 10:20:17 +00:00
if ( strchr ( SettingsText ( SET_HOSTNAME ) , ' % ' ) ! = nullptr ) {
2019-12-16 14:13:57 +00:00
SettingsUpdateText ( SET_HOSTNAME , WIFI_HOSTNAME ) ;
2020-10-30 11:29:48 +00:00
snprintf_P ( TasmotaGlobal . hostname , sizeof ( TasmotaGlobal . hostname ) - 1 , SettingsText ( SET_HOSTNAME ) , TasmotaGlobal . mqtt_topic , ESP_getChipId ( ) & 0x1FFF ) ;
2017-01-28 13:41:01 +00:00
} else {
2020-10-30 11:29:48 +00:00
snprintf_P ( TasmotaGlobal . hostname , sizeof ( TasmotaGlobal . hostname ) - 1 , SettingsText ( SET_HOSTNAME ) ) ;
2017-01-28 13:41:01 +00:00
}
2018-09-05 14:38:48 +01:00
2021-03-15 14:51:31 +00:00
RtcInit ( ) ;
2018-09-05 14:38:48 +01:00
GpioInit ( ) ;
2021-04-06 14:23:07 +01:00
ButtonInit ( ) ;
SwitchInit ( ) ;
# ifdef ROTARY_V1
RotaryInit ( ) ;
# endif // ROTARY_V1
XdrvCall ( FUNC_PRE_INIT ) ;
XsnsCall ( FUNC_PRE_INIT ) ;
2018-09-05 14:38:48 +01:00
2021-04-06 14:23:07 +01:00
SetPowerOnState ( ) ;
2017-10-18 17:22:34 +01:00
WifiConnect ( ) ;
2017-01-28 13:41:01 +00:00
2021-01-23 15:26:23 +00:00
AddLog ( LOG_LEVEL_INFO , PSTR ( D_PROJECT " %s %s " D_VERSION " %s%s- " ARDUINO_CORE_RELEASE " (%s) " ) ,
2021-01-18 20:48:04 +00:00
PSTR ( PROJECT ) , SettingsText ( SET_DEVICENAME ) , TasmotaGlobal . version , TasmotaGlobal . image_name , GetBuildDateAndTime ( ) . c_str ( ) ) ;
2019-02-08 13:55:45 +00:00
# ifdef FIRMWARE_MINIMAL
2021-01-23 15:26:23 +00:00
AddLog ( LOG_LEVEL_INFO , PSTR ( D_WARNING_MINIMAL_VERSION ) ) ;
2019-02-08 13:55:45 +00:00
# endif // FIRMWARE_MINIMAL
2017-12-01 14:59:53 +00:00
2021-04-12 10:49:53 +01:00
memcpy_P ( TasmotaGlobal . mqtt_data , VERSION_MARKER , 1 ) ; // Dummy for compiler saving VERSION_MARKER
2018-03-21 16:58:39 +00:00
# ifdef USE_ARDUINO_OTA
ArduinoOTAInit ( ) ;
# endif // USE_ARDUINO_OTA
2018-06-04 17:10:38 +01:00
XdrvCall ( FUNC_INIT ) ;
2018-01-24 16:31:20 +00:00
XsnsCall ( FUNC_INIT ) ;
2020-06-16 09:36:16 +01:00
# ifdef USE_SCRIPT
2020-07-02 05:46:04 +01:00
if ( bitRead ( Settings . rule_enabled , 0 ) ) Run_Scripter ( " >BS " , 3 , 0 ) ;
2020-06-16 09:36:16 +01:00
# endif
2020-06-14 11:36:44 +01:00
2020-10-29 12:58:50 +00:00
TasmotaGlobal . rules_flag . system_init = 1 ;
2017-01-28 13:41:01 +00:00
}
2020-05-11 15:45:22 +01:00
void BacklogLoop ( void ) {
2020-11-27 13:43:45 +00:00
if ( TimeReached ( TasmotaGlobal . backlog_timer ) ) {
2020-10-30 11:29:48 +00:00
if ( ! BACKLOG_EMPTY & & ! TasmotaGlobal . backlog_mutex ) {
TasmotaGlobal . backlog_mutex = true ;
2020-10-15 11:32:40 +01:00
bool nodelay = false ;
bool nodelay_detected = false ;
String cmd ;
do {
# ifdef SUPPORT_IF_STATEMENT
cmd = backlog . shift ( ) ;
2019-10-04 13:46:31 +01:00
# else
2020-10-30 11:29:48 +00:00
cmd = TasmotaGlobal . backlog [ TasmotaGlobal . backlog_pointer ] ;
TasmotaGlobal . backlog [ TasmotaGlobal . backlog_pointer ] = ( const char * ) nullptr ; // Force deallocation of the String internal memory
TasmotaGlobal . backlog_pointer + + ;
if ( TasmotaGlobal . backlog_pointer > = MAX_BACKLOG ) { TasmotaGlobal . backlog_pointer = 0 ; }
2019-12-06 15:12:13 +00:00
# endif
2020-10-15 11:32:40 +01:00
nodelay_detected = ! strncasecmp_P ( cmd . c_str ( ) , PSTR ( D_CMND_NODELAY ) , strlen ( D_CMND_NODELAY ) ) ;
if ( nodelay_detected ) { nodelay = true ; }
2020-10-15 14:33:17 +01:00
} while ( ! BACKLOG_EMPTY & & nodelay_detected ) ;
2020-10-28 16:32:07 +00:00
if ( ! nodelay_detected ) {
ExecuteCommand ( ( char * ) cmd . c_str ( ) , SRC_BACKLOG ) ;
}
2021-04-07 16:55:33 +01:00
if ( nodelay | | TasmotaGlobal . backlog_nodelay ) {
2020-11-27 13:43:45 +00:00
TasmotaGlobal . backlog_timer = millis ( ) ; // Reset backlog_timer which has been set by ExecuteCommand (CommandHandler)
2020-10-28 16:32:07 +00:00
}
2020-10-30 11:29:48 +00:00
TasmotaGlobal . backlog_mutex = false ;
2019-10-04 13:46:31 +01:00
}
2021-04-09 13:00:09 +01:00
if ( BACKLOG_EMPTY ) {
2021-04-07 16:55:33 +01:00
TasmotaGlobal . backlog_nodelay = false ;
}
2019-10-04 13:46:31 +01:00
}
}
2020-05-11 15:45:22 +01:00
void SleepDelay ( uint32_t mseconds ) {
2021-04-09 13:00:09 +01:00
if ( ! TasmotaGlobal . backlog_nodelay & & mseconds ) {
2021-01-03 15:30:45 +00:00
uint32_t wait = millis ( ) + mseconds ;
2021-01-04 11:47:00 +00:00
while ( ! TimeReached ( wait ) & & ! Serial . available ( ) ) { // We need to service serial buffer ASAP as otherwise we get uart buffer overrun
2021-01-04 11:31:05 +00:00
delay ( 1 ) ;
2020-05-11 15:45:22 +01:00
}
} else {
delay ( 0 ) ;
}
}
2021-01-10 16:46:30 +00:00
void Scheduler ( void ) {
2018-02-27 13:59:46 +00:00
XdrvCall ( FUNC_LOOP ) ;
2018-12-11 13:24:52 +00:00
XsnsCall ( FUNC_LOOP ) ;
2018-02-27 13:59:46 +00:00
2021-01-10 16:46:30 +00:00
// check LEAmDNS.h
// MDNS.update() needs to be called in main loop
# ifdef ESP8266 // Not needed with esp32 mdns
# ifdef USE_DISCOVERY
# ifdef USE_WEBSERVER
# ifdef WEBSERVER_ADVERTISE
MdnsUpdate ( ) ;
# endif // WEBSERVER_ADVERTISE
# endif // USE_WEBSERVER
# endif // USE_DISCOVERY
# endif // ESP8266
2017-10-18 17:22:34 +01:00
OsWatchLoop ( ) ;
2019-01-07 11:38:47 +00:00
ButtonLoop ( ) ;
2018-12-28 15:35:19 +00:00
SwitchLoop ( ) ;
2020-02-21 15:09:21 +00:00
# ifdef USE_DEVICE_GROUPS
DeviceGroupsLoop ( ) ;
# endif // USE_DEVICE_GROUPS
2019-10-09 16:52:52 +01:00
BacklogLoop ( ) ;
2018-12-28 15:35:19 +00:00
2021-01-03 14:28:52 +00:00
static uint32_t state_50msecond = 0 ; // State 50msecond timer
2018-08-26 14:42:35 +01:00
if ( TimeReached ( state_50msecond ) ) {
SetNextTimeInterval ( state_50msecond , 50 ) ;
2020-07-01 09:22:20 +01:00
# ifdef ROTARY_V1
RotaryHandler ( ) ;
2020-07-01 09:24:46 +01:00
# endif // ROTARY_V1
2018-08-26 16:10:18 +01:00
XdrvCall ( FUNC_EVERY_50_MSECOND ) ;
XsnsCall ( FUNC_EVERY_50_MSECOND ) ;
2018-08-26 14:42:35 +01:00
}
2020-10-28 11:40:52 +00:00
2021-01-03 14:28:52 +00:00
static uint32_t state_100msecond = 0 ; // State 100msecond timer
2018-08-26 14:42:35 +01:00
if ( TimeReached ( state_100msecond ) ) {
SetNextTimeInterval ( state_100msecond , 100 ) ;
Every100mSeconds ( ) ;
2018-08-26 16:10:18 +01:00
XdrvCall ( FUNC_EVERY_100_MSECOND ) ;
XsnsCall ( FUNC_EVERY_100_MSECOND ) ;
2018-08-26 14:42:35 +01:00
}
2020-10-28 11:40:52 +00:00
2021-01-03 14:28:52 +00:00
static uint32_t state_250msecond = 0 ; // State 250msecond timer
2018-08-26 14:42:35 +01:00
if ( TimeReached ( state_250msecond ) ) {
SetNextTimeInterval ( state_250msecond , 250 ) ;
Every250mSeconds ( ) ;
2018-08-26 16:10:18 +01:00
XdrvCall ( FUNC_EVERY_250_MSECOND ) ;
XsnsCall ( FUNC_EVERY_250_MSECOND ) ;
2019-10-18 11:47:22 +01:00
}
2020-10-28 11:40:52 +00:00
2021-01-03 14:28:52 +00:00
static uint32_t state_second = 0 ; // State second timer
2019-10-18 11:47:22 +01:00
if ( TimeReached ( state_second ) ) {
SetNextTimeInterval ( state_second , 1000 ) ;
PerformEverySecond ( ) ;
XdrvCall ( FUNC_EVERY_SECOND ) ;
XsnsCall ( FUNC_EVERY_SECOND ) ;
2018-08-26 14:42:35 +01:00
}
2018-02-27 13:59:46 +00:00
2020-10-30 11:29:48 +00:00
if ( ! TasmotaGlobal . serial_local ) { SerialInput ( ) ; }
2017-01-28 13:41:01 +00:00
2018-03-21 16:58:39 +00:00
# ifdef USE_ARDUINO_OTA
2020-01-03 15:50:56 +00:00
ArduinoOtaLoop ( ) ;
2018-03-21 16:58:39 +00:00
# endif // USE_ARDUINO_OTA
2021-01-10 16:46:30 +00:00
}
void loop ( void ) {
uint32_t my_sleep = millis ( ) ;
Scheduler ( ) ;
2018-03-21 16:58:39 +00:00
2018-11-18 14:02:52 +00:00
uint32_t my_activity = millis ( ) - my_sleep ;
2018-12-01 17:53:42 +00:00
2020-05-11 15:45:22 +01:00
if ( Settings . flag3 . sleep_normal ) { // SetOption60 - Enable normal sleep instead of dynamic sleep
// yield(); // yield == delay(0), delay contains yield, auto yield in loop
2021-01-03 14:28:52 +00:00
SleepDelay ( TasmotaGlobal . sleep ) ; // https://github.com/esp8266/Arduino/issues/2021
2018-11-22 11:25:18 +00:00
} else {
2020-10-29 15:16:34 +00:00
if ( my_activity < ( uint32_t ) TasmotaGlobal . sleep ) {
SleepDelay ( ( uint32_t ) TasmotaGlobal . sleep - my_activity ) ; // Provide time for background tasks like wifi
2018-12-01 16:47:25 +00:00
} else {
2020-10-30 11:29:48 +00:00
if ( TasmotaGlobal . global_state . network_down ) {
2020-05-11 15:45:22 +01:00
SleepDelay ( my_activity / 2 ) ; // If wifi down and my_activity > setoption36 then force loop delay to 1/3 of my_activity period
2018-12-01 16:47:25 +00:00
}
2018-11-22 11:25:18 +00:00
}
}
2018-11-25 16:31:53 +00:00
2020-05-11 15:45:22 +01:00
if ( ! my_activity ) { my_activity + + ; } // We cannot divide by 0
2020-10-29 15:16:34 +00:00
uint32_t loop_delay = TasmotaGlobal . sleep ;
2020-05-11 15:45:22 +01:00
if ( ! loop_delay ) { loop_delay + + ; } // We cannot divide by 0
uint32_t loops_per_second = 1000 / loop_delay ; // We need to keep track of this many loops per second
2018-11-25 16:31:53 +00:00
uint32_t this_cycle_ratio = 100 * my_activity / loop_delay ;
2020-10-28 16:32:07 +00:00
TasmotaGlobal . loop_load_avg = TasmotaGlobal . loop_load_avg - ( TasmotaGlobal . loop_load_avg / loops_per_second ) + ( this_cycle_ratio / loops_per_second ) ; // Take away one loop average away and add the new one
2017-08-08 15:08:08 +01:00
}