WLED/wled00/wled.cpp

455 lines
11 KiB
C++
Raw Normal View History

#define WLED_DEFINE_GLOBAL_VARS //only in one source file, wled.cpp!
2020-03-28 13:30:51 +01:00
#include "wled.h"
2020-03-30 11:19:09 +02:00
#include <Arduino.h>
2020-03-30 10:26:41 +02:00
2020-03-30 11:19:09 +02:00
WLED::WLED()
{
2020-03-28 13:30:51 +01:00
}
2020-03-30 12:42:21 +02:00
// turns all LEDs off and restarts ESP
2020-03-28 13:30:51 +01:00
void WLED::reset()
{
2020-03-30 11:19:09 +02:00
briT = 0;
long dly = millis();
while (millis() - dly < 250) {
2020-03-30 12:42:21 +02:00
yield(); // enough time to send response to client
2020-03-30 11:19:09 +02:00
}
setAllLeds();
DEBUG_PRINTLN("MODULE RESET");
ESP.restart();
2020-03-28 13:30:51 +01:00
}
bool oappendi(int i)
{
2020-03-30 11:19:09 +02:00
char s[11];
sprintf(s, "%ld", i);
return oappend(s);
2020-03-28 13:30:51 +01:00
}
2020-03-30 11:19:09 +02:00
bool oappend(const char* txt)
2020-03-28 13:30:51 +01:00
{
2020-03-30 11:19:09 +02:00
uint16_t len = strlen(txt);
if (olen + len >= OMAX)
2020-03-30 12:42:21 +02:00
return false; // buffer full
2020-03-30 11:19:09 +02:00
strcpy(obuf + olen, txt);
olen += len;
return true;
2020-03-28 13:30:51 +01:00
}
void WLED::loop()
{
2020-03-30 12:42:21 +02:00
handleIR(); // 2nd call to function needed for ESP32 to return valid results -- should be good for ESP8266, too
2020-03-30 11:19:09 +02:00
handleConnection();
handleSerial();
handleNotifications();
handleTransitions();
2020-03-28 13:30:51 +01:00
#ifdef WLED_ENABLE_DMX
2020-03-30 11:19:09 +02:00
handleDMX();
2020-03-28 13:30:51 +01:00
#endif
2020-03-30 11:19:09 +02:00
userLoop();
2020-03-28 13:30:51 +01:00
2020-03-30 11:19:09 +02:00
yield();
handleIO();
handleIR();
handleNetworkTime();
handleAlexa();
2020-03-28 13:30:51 +01:00
2020-03-30 11:19:09 +02:00
handleOverlays();
yield();
2020-03-28 13:30:51 +01:00
#ifdef WLED_USE_ANALOG_LEDS
2020-03-30 11:19:09 +02:00
strip.setRgbwPwm();
2020-03-28 13:30:51 +01:00
#endif
2020-03-30 11:19:09 +02:00
if (doReboot)
reset();
2020-03-28 13:30:51 +01:00
2020-03-30 12:42:21 +02:00
if (!realtimeMode) // block stuff if WARLS/Adalight is enabled
2020-03-30 11:19:09 +02:00
{
if (apActive)
dnsServer.processNextRequest();
2020-03-28 13:30:51 +01:00
#ifndef WLED_DISABLE_OTA
2020-03-30 11:19:09 +02:00
if (WLED_CONNECTED && aOtaEnabled)
ArduinoOTA.handle();
2020-03-28 13:30:51 +01:00
#endif
2020-03-30 11:19:09 +02:00
handleNightlight();
yield();
2020-03-28 13:30:51 +01:00
2020-03-30 11:19:09 +02:00
handleHue();
handleBlynk();
2020-03-28 13:30:51 +01:00
yield();
2020-03-30 11:19:09 +02:00
if (!offMode)
strip.service();
}
yield();
2020-03-28 13:30:51 +01:00
#ifdef ESP8266
2020-03-30 11:19:09 +02:00
MDNS.update();
2020-03-28 13:30:51 +01:00
#endif
2020-03-30 11:19:09 +02:00
if (millis() - lastMqttReconnectAttempt > 30000)
initMqtt();
2020-03-28 13:30:51 +01:00
2020-03-30 12:42:21 +02:00
// DEBUG serial logging
2020-03-28 13:30:51 +01:00
#ifdef WLED_DEBUG
2020-03-30 11:19:09 +02:00
if (millis() - debugTime > 9999) {
DEBUG_PRINTLN("---DEBUG INFO---");
DEBUG_PRINT("Runtime: ");
DEBUG_PRINTLN(millis());
DEBUG_PRINT("Unix time: ");
DEBUG_PRINTLN(now());
DEBUG_PRINT("Free heap: ");
DEBUG_PRINTLN(ESP.getFreeHeap());
DEBUG_PRINT("Wifi state: ");
DEBUG_PRINTLN(WiFi.status());
if (WiFi.status() != lastWifiState) {
wifiStateChangedTime = millis();
2020-03-28 13:30:51 +01:00
}
2020-03-30 11:19:09 +02:00
lastWifiState = WiFi.status();
DEBUG_PRINT("State time: ");
DEBUG_PRINTLN(wifiStateChangedTime);
DEBUG_PRINT("NTP last sync: ");
DEBUG_PRINTLN(ntpLastSyncTime);
DEBUG_PRINT("Client IP: ");
DEBUG_PRINTLN(WiFi.localIP());
DEBUG_PRINT("Loops/sec: ");
DEBUG_PRINTLN(loops / 10);
loops = 0;
debugTime = millis();
}
loops++;
2020-03-30 12:42:21 +02:00
#endif // WLED_DEBU
2020-03-28 13:30:51 +01:00
}
2020-03-30 10:43:37 +02:00
void WLED::setup()
2020-03-28 13:30:51 +01:00
{
2020-03-30 11:19:09 +02:00
EEPROM.begin(EEPSIZE);
ledCount = EEPROM.read(229) + ((EEPROM.read(398) << 8) & 0xFF00);
if (ledCount > MAX_LEDS || ledCount == 0)
ledCount = 30;
2020-03-28 13:30:51 +01:00
#ifdef ESP8266
2020-03-30 12:42:21 +02:00
#if LEDPIN == 3
if (ledCount > MAX_LEDS_DMA)
ledCount = MAX_LEDS_DMA; // DMA method uses too much ram
#endif
2020-03-28 13:30:51 +01:00
#endif
2020-03-30 11:19:09 +02:00
Serial.begin(115200);
Serial.setTimeout(50);
DEBUG_PRINTLN();
DEBUG_PRINT("---WLED ");
DEBUG_PRINT(versionString);
DEBUG_PRINT(" ");
DEBUG_PRINT(VERSION);
DEBUG_PRINTLN(" INIT---");
2020-03-28 13:30:51 +01:00
#ifdef ARDUINO_ARCH_ESP32
2020-03-30 11:19:09 +02:00
DEBUG_PRINT("esp32 ");
DEBUG_PRINTLN(ESP.getSdkVersion());
2020-03-28 13:30:51 +01:00
#else
2020-03-30 11:19:09 +02:00
DEBUG_PRINT("esp8266 ");
DEBUG_PRINTLN(ESP.getCoreVersion());
2020-03-28 13:30:51 +01:00
#endif
2020-03-30 11:19:09 +02:00
int heapPreAlloc = ESP.getFreeHeap();
DEBUG_PRINT("heap ");
DEBUG_PRINTLN(ESP.getFreeHeap());
2020-03-28 13:30:51 +01:00
2020-03-30 12:42:21 +02:00
strip.init(EEPROM.read(372), ledCount, EEPROM.read(2204)); // init LEDs quickly
2020-03-30 11:19:09 +02:00
strip.setBrightness(0);
2020-03-28 13:30:51 +01:00
2020-03-30 11:19:09 +02:00
DEBUG_PRINT("LEDs inited. heap usage ~");
DEBUG_PRINTLN(heapPreAlloc - ESP.getFreeHeap());
2020-03-28 13:30:51 +01:00
#ifndef WLED_DISABLE_FILESYSTEM
2020-03-30 12:42:21 +02:00
#ifdef ARDUINO_ARCH_ESP32
SPIFFS.begin(true);
#endif
SPIFFS.begin();
2020-03-28 13:30:51 +01:00
#endif
2020-03-30 11:19:09 +02:00
DEBUG_PRINTLN("Load EEPROM");
loadSettingsFromEEPROM(true);
beginStrip();
userSetup();
if (strcmp(clientSSID, DEFAULT_CLIENT_SSID) == 0)
showWelcomePage = true;
WiFi.persistent(false);
if (macroBoot > 0)
applyMacro(macroBoot);
Serial.println("Ada");
2020-03-30 12:42:21 +02:00
// generate module IDs
2020-03-30 11:19:09 +02:00
escapedMac = WiFi.macAddress();
escapedMac.replace(":", "");
escapedMac.toLowerCase();
2020-03-30 12:42:21 +02:00
if (strcmp(cmDNS, "x") == 0) // fill in unique mdns default
2020-03-30 11:19:09 +02:00
{
strcpy(cmDNS, "wled-");
sprintf(cmDNS + 5, "%*s", 6, escapedMac.c_str() + 6);
}
if (mqttDeviceTopic[0] == 0) {
strcpy(mqttDeviceTopic, "wled/");
sprintf(mqttDeviceTopic + 5, "%*s", 6, escapedMac.c_str() + 6);
}
if (mqttClientID[0] == 0) {
strcpy(mqttClientID, "WLED-");
sprintf(mqttClientID + 5, "%*s", 6, escapedMac.c_str() + 6);
}
strip.service();
2020-03-28 13:30:51 +01:00
#ifndef WLED_DISABLE_OTA
2020-03-30 11:19:09 +02:00
if (aOtaEnabled) {
ArduinoOTA.onStart([]() {
2020-03-28 13:30:51 +01:00
#ifdef ESP8266
2020-03-30 11:19:09 +02:00
wifi_set_sleep_type(NONE_SLEEP_T);
2020-03-28 13:30:51 +01:00
#endif
2020-03-30 11:19:09 +02:00
DEBUG_PRINTLN("Start ArduinoOTA");
});
if (strlen(cmDNS) > 0)
ArduinoOTA.setHostname(cmDNS);
}
2020-03-28 13:30:51 +01:00
#endif
#ifdef WLED_ENABLE_DMX
2020-03-30 12:42:21 +02:00
dmx.init(512); // initialize with bus length
2020-03-28 13:30:51 +01:00
#endif
2020-03-30 12:42:21 +02:00
// HTTP server page init
2020-03-30 11:19:09 +02:00
initServer();
2020-03-28 13:30:51 +01:00
}
void WLED::beginStrip()
{
2020-03-30 11:19:09 +02:00
// Initialize NeoPixel Strip and button
strip.setShowCallback(handleOverlayDraw);
2020-03-28 13:30:51 +01:00
#ifdef BTNPIN
2020-03-30 11:19:09 +02:00
pinMode(BTNPIN, INPUT_PULLUP);
2020-03-28 13:30:51 +01:00
#endif
2020-03-30 11:19:09 +02:00
if (bootPreset > 0)
applyPreset(bootPreset, turnOnAtBoot);
colorUpdated(NOTIFIER_CALL_MODE_INIT);
2020-03-28 13:30:51 +01:00
2020-03-30 12:42:21 +02:00
// init relay pin
2020-03-28 13:30:51 +01:00
#if RLYPIN >= 0
2020-03-30 11:19:09 +02:00
pinMode(RLYPIN, OUTPUT);
2020-03-28 13:30:51 +01:00
#if RLYMDE
2020-03-30 11:19:09 +02:00
digitalWrite(RLYPIN, bri);
2020-03-28 13:30:51 +01:00
#else
2020-03-30 11:19:09 +02:00
digitalWrite(RLYPIN, !bri);
2020-03-28 13:30:51 +01:00
#endif
#endif
2020-03-30 12:42:21 +02:00
// disable button if it is "pressed" unintentionally
2020-03-28 13:30:51 +01:00
#ifdef BTNPIN
2020-03-30 11:19:09 +02:00
if (digitalRead(BTNPIN) == LOW)
2020-03-28 13:30:51 +01:00
buttonEnabled = false;
2020-03-30 11:19:09 +02:00
#else
buttonEnabled = false;
2020-03-28 13:30:51 +01:00
#endif
}
void WLED::initAP(bool resetAP)
{
2020-03-30 11:19:09 +02:00
if (apBehavior == AP_BEHAVIOR_BUTTON_ONLY && !resetAP)
return;
if (!apSSID[0] || resetAP)
strcpy(apSSID, "WLED-AP");
if (resetAP)
strcpy(apPass, DEFAULT_AP_PASS);
DEBUG_PRINT("Opening access point ");
DEBUG_PRINTLN(apSSID);
WiFi.softAPConfig(IPAddress(4, 3, 2, 1), IPAddress(4, 3, 2, 1), IPAddress(255, 255, 255, 0));
WiFi.softAP(apSSID, apPass, apChannel, apHide);
2020-03-30 12:42:21 +02:00
if (!apActive) // start captive portal if AP active
2020-03-30 11:19:09 +02:00
{
DEBUG_PRINTLN("Init AP interfaces");
server.begin();
if (udpPort > 0 && udpPort != ntpLocalPort) {
udpConnected = notifierUdp.begin(udpPort);
2020-03-28 13:30:51 +01:00
}
2020-03-30 11:19:09 +02:00
if (udpRgbPort > 0 && udpRgbPort != ntpLocalPort && udpRgbPort != udpPort) {
udpRgbConnected = rgbUdp.begin(udpRgbPort);
}
dnsServer.setErrorReplyCode(DNSReplyCode::NoError);
dnsServer.start(53, "*", WiFi.softAPIP());
}
apActive = true;
2020-03-28 13:30:51 +01:00
}
void WLED::initConnection()
{
2020-03-30 12:42:21 +02:00
WiFi.disconnect(); // close old connections
2020-03-28 13:30:51 +01:00
#ifdef ESP8266
2020-03-30 11:19:09 +02:00
WiFi.setPhyMode(WIFI_PHY_MODE_11N);
2020-03-28 13:30:51 +01:00
#endif
2020-03-30 11:19:09 +02:00
if (staticIP[0] != 0 && staticGateway[0] != 0) {
WiFi.config(staticIP, staticGateway, staticSubnet, IPAddress(8, 8, 8, 8));
} else {
WiFi.config(0U, 0U, 0U);
}
lastReconnectAttempt = millis();
if (!WLED_WIFI_CONFIGURED) {
DEBUG_PRINT("No connection configured. ");
if (!apActive)
2020-03-30 12:42:21 +02:00
initAP(); // instantly go to ap mode
2020-03-30 11:19:09 +02:00
return;
} else if (!apActive) {
if (apBehavior == AP_BEHAVIOR_ALWAYS) {
initAP();
} else {
DEBUG_PRINTLN("Access point disabled.");
WiFi.softAPdisconnect(true);
2020-03-28 13:30:51 +01:00
}
2020-03-30 11:19:09 +02:00
}
showWelcomePage = false;
2020-03-28 13:30:51 +01:00
2020-03-30 11:19:09 +02:00
DEBUG_PRINT("Connecting to ");
DEBUG_PRINT(clientSSID);
DEBUG_PRINTLN("...");
2020-03-28 13:30:51 +01:00
#ifdef ESP8266
2020-03-30 11:19:09 +02:00
WiFi.hostname(serverDescription);
2020-03-28 13:30:51 +01:00
#endif
2020-03-30 11:19:09 +02:00
WiFi.begin(clientSSID, clientPass);
2020-03-28 13:30:51 +01:00
#ifdef ARDUINO_ARCH_ESP32
2020-03-30 11:19:09 +02:00
WiFi.setSleep(!noWifiSleep);
WiFi.setHostname(serverDescription);
2020-03-28 13:30:51 +01:00
#else
2020-03-30 11:19:09 +02:00
wifi_set_sleep_type((noWifiSleep) ? NONE_SLEEP_T : MODEM_SLEEP_T);
2020-03-28 13:30:51 +01:00
#endif
}
void WLED::initInterfaces()
{
2020-03-30 11:19:09 +02:00
DEBUG_PRINTLN("Init STA interfaces");
2020-03-28 13:30:51 +01:00
2020-03-30 11:19:09 +02:00
if (hueIP[0] == 0) {
hueIP[0] = WiFi.localIP()[0];
hueIP[1] = WiFi.localIP()[1];
hueIP[2] = WiFi.localIP()[2];
}
2020-03-28 13:30:51 +01:00
2020-03-30 12:42:21 +02:00
// init Alexa hue emulation
2020-03-30 11:19:09 +02:00
if (alexaEnabled)
alexaInit();
2020-03-28 13:30:51 +01:00
#ifndef WLED_DISABLE_OTA
2020-03-30 11:19:09 +02:00
if (aOtaEnabled)
ArduinoOTA.begin();
2020-03-28 13:30:51 +01:00
#endif
2020-03-30 11:19:09 +02:00
strip.service();
// Set up mDNS responder:
if (strlen(cmDNS) > 0) {
if (!aOtaEnabled)
MDNS.begin(cmDNS);
DEBUG_PRINTLN("mDNS started");
MDNS.addService("http", "tcp", 80);
MDNS.addService("wled", "tcp", 80);
MDNS.addServiceTxt("wled", "tcp", "mac", escapedMac.c_str());
}
server.begin();
if (udpPort > 0 && udpPort != ntpLocalPort) {
udpConnected = notifierUdp.begin(udpPort);
if (udpConnected && udpRgbPort != udpPort)
udpRgbConnected = rgbUdp.begin(udpRgbPort);
}
if (ntpEnabled)
ntpConnected = ntpUdp.begin(ntpLocalPort);
initBlynk(blynkApiKey);
e131.begin((e131Multicast) ? E131_MULTICAST : E131_UNICAST, e131Universe, E131_MAX_UNIVERSE_COUNT);
reconnectHue();
initMqtt();
interfacesInited = true;
wasConnected = true;
2020-03-28 13:30:51 +01:00
}
byte stacO = 0;
uint32_t lastHeap;
unsigned long heapTime = 0;
void WLED::handleConnection()
{
2020-03-30 11:19:09 +02:00
if (millis() < 2000 && (!WLED_WIFI_CONFIGURED || apBehavior == AP_BEHAVIOR_ALWAYS))
return;
if (lastReconnectAttempt == 0)
initConnection();
2020-03-30 12:42:21 +02:00
// reconnect WiFi to clear stale allocations if heap gets too low
2020-03-30 11:19:09 +02:00
if (millis() - heapTime > 5000) {
uint32_t heap = ESP.getFreeHeap();
if (heap < 9000 && lastHeap < 9000) {
DEBUG_PRINT("Heap too low! ");
DEBUG_PRINTLN(heap);
forceReconnect = true;
2020-03-28 13:30:51 +01:00
}
2020-03-30 11:19:09 +02:00
lastHeap = heap;
heapTime = millis();
}
2020-03-28 13:30:51 +01:00
2020-03-30 11:19:09 +02:00
byte stac = 0;
if (apActive) {
2020-03-28 13:30:51 +01:00
#ifdef ESP8266
2020-03-30 11:19:09 +02:00
stac = wifi_softap_get_station_num();
2020-03-28 13:30:51 +01:00
#else
2020-03-30 11:19:09 +02:00
wifi_sta_list_t stationList;
esp_wifi_ap_get_sta_list(&stationList);
stac = stationList.num;
2020-03-28 13:30:51 +01:00
#endif
2020-03-30 11:19:09 +02:00
if (stac != stacO) {
stacO = stac;
DEBUG_PRINT("Connected AP clients: ");
DEBUG_PRINTLN(stac);
2020-03-30 12:42:21 +02:00
if (!WLED_CONNECTED && WLED_WIFI_CONFIGURED) { // trying to connect, but not connected
2020-03-30 11:19:09 +02:00
if (stac)
2020-03-30 12:42:21 +02:00
WiFi.disconnect(); // disable search so that AP can work
2020-03-30 11:19:09 +02:00
else
2020-03-30 12:42:21 +02:00
initConnection(); // restart search
2020-03-30 11:19:09 +02:00
}
2020-03-28 13:30:51 +01:00
}
2020-03-30 11:19:09 +02:00
}
if (forceReconnect) {
DEBUG_PRINTLN("Forcing reconnect.");
initConnection();
interfacesInited = false;
forceReconnect = false;
wasConnected = false;
return;
}
if (!WLED_CONNECTED) {
if (interfacesInited) {
DEBUG_PRINTLN("Disconnected!");
interfacesInited = false;
initConnection();
2020-03-28 13:30:51 +01:00
}
2020-03-30 11:19:09 +02:00
if (millis() - lastReconnectAttempt > ((stac) ? 300000 : 20000) && WLED_WIFI_CONFIGURED)
initConnection();
if (!apActive && millis() - lastReconnectAttempt > 12000 && (!wasConnected || apBehavior == AP_BEHAVIOR_NO_CONN))
initAP();
2020-03-30 12:42:21 +02:00
} else if (!interfacesInited) { // newly connected
2020-03-30 11:19:09 +02:00
DEBUG_PRINTLN("");
DEBUG_PRINT("Connected! IP address: ");
DEBUG_PRINTLN(WiFi.localIP());
initInterfaces();
userConnected();
2020-03-30 12:42:21 +02:00
// shut down AP
2020-03-30 11:19:09 +02:00
if (apBehavior != AP_BEHAVIOR_ALWAYS && apActive) {
dnsServer.stop();
WiFi.softAPdisconnect(true);
apActive = false;
DEBUG_PRINTLN("Access point disabled.");
2020-03-28 13:30:51 +01:00
}
2020-03-30 11:19:09 +02:00
}
2020-03-28 13:30:51 +01:00
}