/* ESPurna Copyright (C) 2016-2019 by Xose PĂ©rez Copyright (C) 2019-2021 by Maxim Prokhorov 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 . */ #include "espurna.h" #include "main.h" std::vector _loop_callbacks; std::vector _reload_callbacks; bool _reload_config = false; unsigned long _loop_delay = 0; constexpr unsigned long LoopDelayMin { 10ul }; constexpr unsigned long LoopDelayMax { 300ul }; // ----------------------------------------------------------------------------- // GENERAL CALLBACKS // ----------------------------------------------------------------------------- void espurnaRegisterLoop(LoopCallback callback) { _loop_callbacks.push_back(callback); } void espurnaRegisterReload(LoopCallback callback) { _reload_callbacks.push_back(callback); } void espurnaReload() { _reload_config = true; } void _espurnaReload() { for (const auto& callback : _reload_callbacks) { callback(); } } unsigned long espurnaLoopDelay() { return _loop_delay; } void espurnaLoopDelay(unsigned long loop_delay) { _loop_delay = loop_delay; } constexpr unsigned long _loopDelay() { return LOOP_DELAY_TIME; } // ----------------------------------------------------------------------------- // BOOTING // ----------------------------------------------------------------------------- void setup() { // ------------------------------------------------------------------------- // Basic modules, will always run // ------------------------------------------------------------------------- // Cache initial free heap value systemInitialFreeHeap(); // Init logging module #if DEBUG_SUPPORT debugSetup(); #endif // Init GPIO functions gpioSetup(); // Init RTCMEM rtcmemSetup(); // Init EEPROM eepromSetup(); // Init persistance settingsSetup(); // Configure logger and crash recorder #if DEBUG_SUPPORT debugConfigureBoot(); crashSetup(); #endif // Init Serial, SPIFFS and system check systemSetup(); // Init terminal features #if TERMINAL_SUPPORT terminalSetup(); #endif // Hostname & board name initialization if (getSetting("hostname").length() == 0) { setDefaultHostname(); } setBoardName(); // Show welcome message and system configuration info(true); wifiSetup(); otaSetup(); #if TELNET_SUPPORT telnetSetup(); #endif // ------------------------------------------------------------------------- // Check if system is stable // ------------------------------------------------------------------------- #if SYSTEM_CHECK_ENABLED if (!systemCheck()) return; #endif // ------------------------------------------------------------------------- // Next modules will be only loaded if system is flagged as stable // ------------------------------------------------------------------------- // Init webserver required before any module that uses API #if WEB_SUPPORT webSetup(); wsSetup(); #if DEBUG_WEB_SUPPORT debugWebSetup(); #endif #if OTA_WEB_SUPPORT otaWebSetup(); #endif #endif // Multiple modules depend on the generic 'API' services #if API_SUPPORT || TERMINAL_WEB_API_SUPPORT || PROMETHEUS_SUPPORT apiCommonSetup(); #endif #if API_SUPPORT apiSetup(); #endif // Run terminal command and send back the result #if TERMINAL_WEB_API_SUPPORT terminalWebApiSetup(); #endif // Special HTTP metrics endpoint #if PROMETHEUS_SUPPORT prometheusSetup(); #endif // Hardware GPIO expander, needs to be available for modules down below #if MCP23S08_SUPPORT MCP23S08Setup(); #endif // lightSetup must be called before relaySetup #if LIGHT_PROVIDER != LIGHT_PROVIDER_NONE lightSetup(); #endif // rpnSetup must be called before relaySetup #if RPN_RULES_SUPPORT rpnSetup(); #endif #if RELAY_SUPPORT relaySetup(); #endif #if BUTTON_SUPPORT buttonSetup(); #endif #if ENCODER_SUPPORT && (LIGHT_PROVIDER != LIGHT_PROVIDER_NONE) encoderSetup(); #endif #if LED_SUPPORT ledSetup(); #endif #if MQTT_SUPPORT mqttSetup(); #endif #if MDNS_SERVER_SUPPORT mdnsServerSetup(); #endif #if MDNS_CLIENT_SUPPORT mdnsClientSetup(); #endif #if LLMNR_SUPPORT llmnrSetup(); #endif #if NETBIOS_SUPPORT netbiosSetup(); #endif #if SSDP_SUPPORT ssdpSetup(); #endif #if NTP_SUPPORT ntpSetup(); #endif #if I2C_SUPPORT i2cSetup(); #endif #if RFB_SUPPORT rfbSetup(); #endif #if ALEXA_SUPPORT alexaSetup(); #endif #if NOFUSS_SUPPORT nofussSetup(); #endif #if SENSOR_SUPPORT sensorSetup(); #endif #if INFLUXDB_SUPPORT idbSetup(); #endif #if THINGSPEAK_SUPPORT tspkSetup(); #endif #if RFM69_SUPPORT rfm69Setup(); #endif #if IR_SUPPORT irSetup(); #endif #if DOMOTICZ_SUPPORT domoticzSetup(); #endif #if HOMEASSISTANT_SUPPORT haSetup(); #endif #if SCHEDULER_SUPPORT schSetup(); #endif #if UART_MQTT_SUPPORT uartmqttSetup(); #endif #ifdef FOXEL_LIGHTFOX_DUAL lightfoxSetup(); #endif #if THERMOSTAT_SUPPORT thermostatSetup(); #endif #if THERMOSTAT_DISPLAY_SUPPORT displaySetup(); #endif #if TUYA_SUPPORT tuya::setup(); #endif #if KINGART_CURTAIN_SUPPORT kingartCurtainSetup(); #endif #if FAN_SUPPORT fanSetup(); #endif #if GARLAND_SUPPORT garlandSetup(); #endif #if USE_EXTRA extraSetup(); #endif // Update `cfg` version migrate(); // Set up delay() after loop callbacks are finished // Note: should be after settingsSetup() unsigned long loop_delay { getSetting("loopDelay", _loopDelay()) }; _loop_delay = ((LoopDelayMin < loop_delay) && (loop_delay <= LoopDelayMax)) ? loop_delay : LoopDelayMin; if (_loop_delay != loop_delay) { setSetting("loopDelay", _loop_delay); } } void loop() { // Reload config before running any callbacks if (_reload_config) { _espurnaReload(); _reload_config = false; } for (auto* callback : _loop_callbacks) { callback(); } if (_loop_delay) { delay(_loop_delay); } }