1083 lines
		
	
	
		
			36 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1083 lines
		
	
	
		
			36 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #define WLED_DEFINE_GLOBAL_VARS //only in one source file, wled.cpp!
 | |
| #include "wled.h"
 | |
| #include "wled_ethernet.h"
 | |
| #include <Arduino.h>
 | |
| 
 | |
| #if defined(ARDUINO_ARCH_ESP32) && defined(WLED_DISABLE_BROWNOUT_DET)
 | |
| #include "soc/soc.h"
 | |
| #include "soc/rtc_cntl_reg.h"
 | |
| #endif
 | |
| 
 | |
| extern "C" void usePWMFixedNMI();
 | |
| 
 | |
| /*
 | |
|  * Main WLED class implementation. Mostly initialization and connection logic
 | |
|  */
 | |
| 
 | |
| WLED::WLED()
 | |
| {
 | |
| }
 | |
| 
 | |
| // turns all LEDs off and restarts ESP
 | |
| void WLED::reset()
 | |
| {
 | |
|   briT = 0;
 | |
|   #ifdef WLED_ENABLE_WEBSOCKETS
 | |
|   ws.closeAll(1012);
 | |
|   #endif
 | |
|   unsigned long dly = millis();
 | |
|   while (millis() - dly < 450) {
 | |
|     yield();        // enough time to send response to client
 | |
|   }
 | |
|   applyBri();
 | |
|   DEBUG_PRINTLN(F("WLED RESET"));
 | |
|   ESP.restart();
 | |
| }
 | |
| 
 | |
| void WLED::loop()
 | |
| {
 | |
|   static uint32_t      lastHeap = UINT32_MAX;
 | |
|   static unsigned long heapTime = 0;
 | |
| #ifdef WLED_DEBUG
 | |
|   static unsigned long lastRun = 0;
 | |
|   unsigned long        loopMillis = millis();
 | |
|   size_t               loopDelay = loopMillis - lastRun;
 | |
|   if (lastRun == 0) loopDelay=0; // startup - don't have valid data from last run.
 | |
|   if (loopDelay > 2) DEBUG_PRINTF_P(PSTR("Loop delayed more than %ums.\n"), loopDelay);
 | |
|   static unsigned long maxLoopMillis = 0;
 | |
|   static size_t        avgLoopMillis = 0;
 | |
|   static unsigned long maxUsermodMillis = 0;
 | |
|   static size_t        avgUsermodMillis = 0;
 | |
|   static unsigned long maxStripMillis = 0;
 | |
|   static size_t        avgStripMillis = 0;
 | |
|   unsigned long        stripMillis;
 | |
| #endif
 | |
| 
 | |
|   handleTime();
 | |
|   #ifndef WLED_DISABLE_INFRARED
 | |
|   handleIR();        // 2nd call to function needed for ESP32 to return valid results -- should be good for ESP8266, too
 | |
|   #endif
 | |
|   handleConnection();
 | |
|   #ifdef WLED_ENABLE_ADALIGHT
 | |
|   handleSerial();
 | |
|   #endif
 | |
|   handleImprovWifiScan();
 | |
|   handleNotifications();
 | |
|   handleTransitions();
 | |
|   #ifdef WLED_ENABLE_DMX
 | |
|   handleDMXOutput();
 | |
|   #endif
 | |
|   #ifdef WLED_ENABLE_DMX_INPUT
 | |
|   dmxInput.update();
 | |
|   #endif
 | |
| 
 | |
|   #ifdef WLED_DEBUG
 | |
|   unsigned long usermodMillis = millis();
 | |
|   #endif
 | |
|   userLoop();
 | |
|   UsermodManager::loop();
 | |
|   #ifdef WLED_DEBUG
 | |
|   usermodMillis = millis() - usermodMillis;
 | |
|   avgUsermodMillis += usermodMillis;
 | |
|   if (usermodMillis > maxUsermodMillis) maxUsermodMillis = usermodMillis;
 | |
|   #endif
 | |
| 
 | |
|   yield();
 | |
|   handleIO();
 | |
|   #ifndef WLED_DISABLE_INFRARED
 | |
|   handleIR();
 | |
|   #endif
 | |
|   #ifndef WLED_DISABLE_ESPNOW
 | |
|   handleRemote();
 | |
|   #endif
 | |
|   #ifndef WLED_DISABLE_ALEXA
 | |
|   handleAlexa();
 | |
|   #endif
 | |
| 
 | |
|   if (doCloseFile) {
 | |
|     closeFile();
 | |
|     yield();
 | |
|   }
 | |
| 
 | |
|   #ifdef WLED_DEBUG
 | |
|   stripMillis = millis();
 | |
|   #endif
 | |
|   if (!realtimeMode || realtimeOverride || (realtimeMode && useMainSegmentOnly))  // block stuff if WARLS/Adalight is enabled
 | |
|   {
 | |
|     if (apActive) dnsServer.processNextRequest();
 | |
|     #ifndef WLED_DISABLE_OTA
 | |
|     if (WLED_CONNECTED && aOtaEnabled && !otaLock && correctPIN) ArduinoOTA.handle();
 | |
|     #endif
 | |
|     handleNightlight();
 | |
|     handlePlaylist();
 | |
|     yield();
 | |
| 
 | |
|     #ifndef WLED_DISABLE_HUESYNC
 | |
|     handleHue();
 | |
|     yield();
 | |
|     #endif
 | |
| 
 | |
|     handlePresets();
 | |
|     yield();
 | |
| 
 | |
|     if (!offMode || strip.isOffRefreshRequired() || strip.needsUpdate())
 | |
|       strip.service();
 | |
|     #ifdef ESP8266
 | |
|     else if (!noWifiSleep)
 | |
|       delay(1); //required to make sure ESP enters modem sleep (see #1184)
 | |
|     #endif
 | |
|   }
 | |
|   #ifdef WLED_DEBUG
 | |
|   stripMillis = millis() - stripMillis;
 | |
|   avgStripMillis += stripMillis;
 | |
|   if (stripMillis > maxStripMillis) maxStripMillis = stripMillis;
 | |
|   #endif
 | |
| 
 | |
|   yield();
 | |
| #ifdef ESP8266
 | |
|   MDNS.update();
 | |
| #endif
 | |
| 
 | |
|   //millis() rolls over every 50 days
 | |
|   if (lastMqttReconnectAttempt > millis()) {
 | |
|     rolloverMillis++;
 | |
|     lastMqttReconnectAttempt = 0;
 | |
|     ntpLastSyncTime = NTP_NEVER;  // force new NTP query
 | |
|     strip.restartRuntime();
 | |
|   }
 | |
|   if (millis() - lastMqttReconnectAttempt > 30000 || lastMqttReconnectAttempt == 0) { // lastMqttReconnectAttempt==0 forces immediate broadcast
 | |
|     lastMqttReconnectAttempt = millis();
 | |
|     #ifndef WLED_DISABLE_MQTT
 | |
|     initMqtt();
 | |
|     #endif
 | |
|     yield();
 | |
|     // refresh WLED nodes list
 | |
|     refreshNodeList();
 | |
|     if (nodeBroadcastEnabled) sendSysInfoUDP();
 | |
|     yield();
 | |
|   }
 | |
| 
 | |
|   // 15min PIN time-out
 | |
|   if (strlen(settingsPIN)>0 && correctPIN && millis() - lastEditTime > PIN_TIMEOUT) {
 | |
|     correctPIN = false;
 | |
|     createEditHandler(false);
 | |
|   }
 | |
| 
 | |
|   // reconnect WiFi to clear stale allocations if heap gets too low
 | |
|   if (millis() - heapTime > 15000) {
 | |
|     uint32_t heap = ESP.getFreeHeap();
 | |
|     if (heap < MIN_HEAP_SIZE && lastHeap < MIN_HEAP_SIZE) {
 | |
|       DEBUG_PRINTF_P(PSTR("Heap too low! %u\n"), heap);
 | |
|       forceReconnect = true;
 | |
|       strip.resetSegments(); // remove all but one segments from memory
 | |
|     } else if (heap < MIN_HEAP_SIZE) {
 | |
|       DEBUG_PRINTLN(F("Heap low, purging segments."));
 | |
|       strip.purgeSegments();
 | |
|     }
 | |
|     lastHeap = heap;
 | |
|     heapTime = millis();
 | |
|   }
 | |
| 
 | |
|   //LED settings have been saved, re-init busses
 | |
|   //This code block causes severe FPS drop on ESP32 with the original "if (busConfigs[0] != nullptr)" conditional. Investigate!
 | |
|   if (doInitBusses) {
 | |
|     doInitBusses = false;
 | |
|     DEBUG_PRINTLN(F("Re-init busses."));
 | |
|     bool aligned = strip.checkSegmentAlignment(); //see if old segments match old bus(ses)
 | |
|     BusManager::removeAll();
 | |
|     unsigned mem = 0;
 | |
|     // determine if it is sensible to use parallel I2S outputs on ESP32 (i.e. more than 5 outputs = 1 I2S + 4 RMT)
 | |
|     bool useParallel = false;
 | |
|     #if defined(ARDUINO_ARCH_ESP32) && !defined(ARDUINO_ARCH_ESP32S2) && !defined(ARDUINO_ARCH_ESP32S3) && !defined(ARDUINO_ARCH_ESP32C3)
 | |
|     unsigned digitalCount = 0;
 | |
|     unsigned maxLedsOnBus = 0;
 | |
|     unsigned maxChannels = 0;
 | |
|     for (unsigned i = 0; i < WLED_MAX_BUSSES+WLED_MIN_VIRTUAL_BUSSES; i++) {
 | |
|       if (busConfigs[i] == nullptr) break;
 | |
|       if (!Bus::isDigital(busConfigs[i]->type)) continue;
 | |
|       if (!Bus::is2Pin(busConfigs[i]->type)) {
 | |
|         digitalCount++;
 | |
|         unsigned channels = Bus::getNumberOfChannels(busConfigs[i]->type);
 | |
|         if (busConfigs[i]->count > maxLedsOnBus) maxLedsOnBus = busConfigs[i]->count;
 | |
|         if (channels > maxChannels) maxChannels  = channels;
 | |
|       }
 | |
|     }
 | |
|     DEBUG_PRINTF_P(PSTR("Maximum LEDs on a bus: %u\nDigital buses: %u\n"), maxLedsOnBus, digitalCount);
 | |
|     // we may remove 300 LEDs per bus limit when NeoPixelBus is updated beyond 2.9.0
 | |
|     if (maxLedsOnBus <= 300 && digitalCount > 5) {
 | |
|       DEBUG_PRINTF_P(PSTR("Switching to parallel I2S."));
 | |
|       useParallel = true;
 | |
|       BusManager::useParallelOutput();
 | |
|       mem = BusManager::memUsage(maxChannels, maxLedsOnBus, 8); // use alternate memory calculation (hse to be used *after* useParallelOutput())
 | |
|     }
 | |
|     #endif
 | |
|     // create buses/outputs
 | |
|     for (unsigned i = 0; i < WLED_MAX_BUSSES+WLED_MIN_VIRTUAL_BUSSES; i++) {
 | |
|       if (busConfigs[i] == nullptr || (!useParallel && i > 10)) break;
 | |
|       if (useParallel && i < 8) {
 | |
|         // if for some unexplained reason the above pre-calculation was wrong, update
 | |
|         unsigned memT = BusManager::memUsage(*busConfigs[i]); // includes x8 memory allocation for parallel I2S
 | |
|         if (memT > mem) mem = memT; // if we have unequal LED count use the largest
 | |
|       } else
 | |
|         mem += BusManager::memUsage(*busConfigs[i]); // includes global buffer
 | |
|       if (mem <= MAX_LED_MEMORY) BusManager::add(*busConfigs[i]);
 | |
|       delete busConfigs[i];
 | |
|       busConfigs[i] = nullptr;
 | |
|     }
 | |
|     strip.finalizeInit(); // also loads default ledmap if present
 | |
|     BusManager::setBrightness(bri); // fix re-initialised bus' brightness #4005
 | |
|     if (aligned) strip.makeAutoSegments();
 | |
|     else strip.fixInvalidSegments();
 | |
|     BusManager::setBrightness(bri); // fix re-initialised bus' brightness
 | |
|     doSerializeConfig = true;
 | |
|   }
 | |
|   if (loadLedmap >= 0) {
 | |
|     strip.deserializeMap(loadLedmap);
 | |
|     loadLedmap = -1;
 | |
|   }
 | |
|   yield();
 | |
|   if (doSerializeConfig) serializeConfig();
 | |
| 
 | |
|   yield();
 | |
|   handleWs();
 | |
| #if defined(STATUSLED)
 | |
|   handleStatusLED();
 | |
| #endif
 | |
| 
 | |
|   toki.resetTick();
 | |
| 
 | |
| #if WLED_WATCHDOG_TIMEOUT > 0
 | |
|   // we finished our mainloop, reset the watchdog timer
 | |
|   static unsigned long lastWDTFeed = 0;
 | |
|   if (!strip.isUpdating() || millis() - lastWDTFeed > (WLED_WATCHDOG_TIMEOUT*500)) {
 | |
|   #ifdef ARDUINO_ARCH_ESP32
 | |
|     esp_task_wdt_reset();
 | |
|   #else
 | |
|     ESP.wdtFeed();
 | |
|   #endif
 | |
|     lastWDTFeed = millis();
 | |
|   }
 | |
| #endif
 | |
| 
 | |
|   if (doReboot && (!doInitBusses || !doSerializeConfig)) // if busses have to be inited & saved, wait until next iteration
 | |
|     reset();
 | |
| 
 | |
| // DEBUG serial logging (every 30s)
 | |
| #ifdef WLED_DEBUG
 | |
|   loopMillis = millis() - loopMillis;
 | |
|   if (loopMillis > 30) {
 | |
|     DEBUG_PRINTF_P(PSTR("Loop took %lums.\n"), loopMillis);
 | |
|     DEBUG_PRINTF_P(PSTR("Usermods took %lums.\n"), usermodMillis);
 | |
|     DEBUG_PRINTF_P(PSTR("Strip took %lums.\n"), stripMillis);
 | |
|   }
 | |
|   avgLoopMillis += loopMillis;
 | |
|   if (loopMillis > maxLoopMillis) maxLoopMillis = loopMillis;
 | |
|   if (millis() - debugTime > 29999) {
 | |
|     DEBUG_PRINTLN(F("---DEBUG INFO---"));
 | |
|     DEBUG_PRINTF_P(PSTR("Runtime: %lu\n"),  millis());
 | |
|     DEBUG_PRINTF_P(PSTR("Unix time: %u,%03u\n"), toki.getTime().sec, toki.getTime().ms);
 | |
|     DEBUG_PRINTF_P(PSTR("Free heap: %u\n"), ESP.getFreeHeap());
 | |
|     #if defined(ARDUINO_ARCH_ESP32)
 | |
|     if (psramFound()) {
 | |
|       DEBUG_PRINTF_P(PSTR("PSRAM: %dkB/%dkB\n"), ESP.getFreePsram()/1024, ESP.getPsramSize()/1024);
 | |
|       if (!psramSafe) DEBUG_PRINTLN(F("Not using PSRAM."));
 | |
|     }
 | |
|     DEBUG_PRINTF_P(PSTR("TX power: %d/%d\n"), WiFi.getTxPower(), txPower);
 | |
|     #endif
 | |
|     DEBUG_PRINTF_P(PSTR("Wifi state: %d\n"), WiFi.status());
 | |
|     #ifndef WLED_DISABLE_ESPNOW
 | |
|     DEBUG_PRINTF_P(PSTR("ESP-NOW state: %u\n"), statusESPNow);
 | |
|     #endif
 | |
| 
 | |
|     if (WiFi.status() != lastWifiState) {
 | |
|       wifiStateChangedTime = millis();
 | |
|     }
 | |
|     lastWifiState = WiFi.status();
 | |
|     DEBUG_PRINTF_P(PSTR("State time: %lu\n"),        wifiStateChangedTime);
 | |
|     DEBUG_PRINTF_P(PSTR("NTP last sync: %lu\n"),     ntpLastSyncTime);
 | |
|     DEBUG_PRINTF_P(PSTR("Client IP: %u.%u.%u.%u\n"), Network.localIP()[0], Network.localIP()[1], Network.localIP()[2], Network.localIP()[3]);
 | |
|     if (loops > 0) { // avoid division by zero
 | |
|       DEBUG_PRINTF_P(PSTR("Loops/sec: %u\n"),         loops / 30);
 | |
|       DEBUG_PRINTF_P(PSTR("Loop time[ms]: %u/%lu\n"), avgLoopMillis/loops,    maxLoopMillis);
 | |
|       DEBUG_PRINTF_P(PSTR("UM time[ms]: %u/%lu\n"),   avgUsermodMillis/loops, maxUsermodMillis);
 | |
|       DEBUG_PRINTF_P(PSTR("Strip time[ms]:%u/%lu\n"), avgStripMillis/loops,   maxStripMillis);
 | |
|     }
 | |
|     strip.printSize();
 | |
|     server.printStatus(DEBUGOUT);
 | |
|     loops = 0;
 | |
|     maxLoopMillis = 0;
 | |
|     maxUsermodMillis = 0;
 | |
|     maxStripMillis = 0;
 | |
|     avgLoopMillis = 0;
 | |
|     avgUsermodMillis = 0;
 | |
|     avgStripMillis = 0;
 | |
|     debugTime = millis();
 | |
|   }
 | |
|   loops++;
 | |
|   lastRun = millis();
 | |
| #endif        // WLED_DEBUG
 | |
| }
 | |
| 
 | |
| #if WLED_WATCHDOG_TIMEOUT > 0
 | |
| void WLED::enableWatchdog() {
 | |
|   #ifdef ARDUINO_ARCH_ESP32
 | |
|   esp_err_t watchdog = esp_task_wdt_init(WLED_WATCHDOG_TIMEOUT, true);
 | |
|   DEBUG_PRINT(F("Watchdog enabled: "));
 | |
|   if (watchdog == ESP_OK) {
 | |
|     DEBUG_PRINTLN(F("OK"));
 | |
|   } else {
 | |
|     DEBUG_PRINTLN(watchdog);
 | |
|     return;
 | |
|   }
 | |
|   esp_task_wdt_add(NULL);
 | |
|   #else
 | |
|   ESP.wdtEnable(WLED_WATCHDOG_TIMEOUT * 1000);
 | |
|   #endif
 | |
| }
 | |
| 
 | |
| void WLED::disableWatchdog() {
 | |
|   DEBUG_PRINTLN(F("Watchdog: disabled"));
 | |
|   #ifdef ARDUINO_ARCH_ESP32
 | |
|   esp_task_wdt_delete(NULL);
 | |
|   #else
 | |
|   ESP.wdtDisable();
 | |
|   #endif
 | |
| }
 | |
| #endif
 | |
| 
 | |
| void WLED::setup()
 | |
| {
 | |
|   #if defined(ARDUINO_ARCH_ESP32) && defined(WLED_DISABLE_BROWNOUT_DET)
 | |
|   WRITE_PERI_REG(RTC_CNTL_BROWN_OUT_REG, 0); //disable brownout detection
 | |
|   #endif
 | |
| 
 | |
|   #ifdef ARDUINO_ARCH_ESP32
 | |
|   pinMode(hardwareRX, INPUT_PULLDOWN); delay(1);        // suppress noise in case RX pin is floating (at low noise energy) - see issue #3128
 | |
|   #endif
 | |
|   #ifdef WLED_BOOTUPDELAY
 | |
|   delay(WLED_BOOTUPDELAY); // delay to let voltage stabilize, helps with boot issues on some setups
 | |
|   #endif
 | |
|   Serial.begin(115200);
 | |
|   #if !ARDUINO_USB_CDC_ON_BOOT
 | |
|   Serial.setTimeout(50);  // this causes troubles on new MCUs that have a "virtual" USB Serial (HWCDC)
 | |
|   #else
 | |
|   #endif
 | |
|   #if defined(WLED_DEBUG) && defined(ARDUINO_ARCH_ESP32) && (defined(CONFIG_IDF_TARGET_ESP32S2) || defined(CONFIG_IDF_TARGET_ESP32C3) || ARDUINO_USB_CDC_ON_BOOT)
 | |
|   delay(2500);  // allow CDC USB serial to initialise
 | |
|   #endif
 | |
|   #if !defined(WLED_DEBUG) && defined(ARDUINO_ARCH_ESP32) && !defined(WLED_DEBUG_HOST) && ARDUINO_USB_CDC_ON_BOOT
 | |
|   Serial.setDebugOutput(false); // switch off kernel messages when using USBCDC
 | |
|   #endif
 | |
|   DEBUG_PRINTLN();
 | |
|   DEBUG_PRINTF_P(PSTR("---WLED %s %u INIT---\n"), versionString, VERSION);
 | |
|   DEBUG_PRINTLN();
 | |
| #ifdef ARDUINO_ARCH_ESP32
 | |
|   DEBUG_PRINTF_P(PSTR("esp32 %s\n"), ESP.getSdkVersion());
 | |
|   #if defined(ESP_ARDUINO_VERSION)
 | |
|     DEBUG_PRINTF_P(PSTR("arduino-esp32 v%d.%d.%d\n"), int(ESP_ARDUINO_VERSION_MAJOR), int(ESP_ARDUINO_VERSION_MINOR), int(ESP_ARDUINO_VERSION_PATCH));  // available since v2.0.0
 | |
|   #else
 | |
|     DEBUG_PRINTLN(F("arduino-esp32 v1.0.x\n"));  // we can't say in more detail.
 | |
|   #endif
 | |
| 
 | |
|   DEBUG_PRINTF_P(PSTR("CPU:   %s rev.%d, %d core(s), %d MHz.\n"), ESP.getChipModel(), (int)ESP.getChipRevision(), ESP.getChipCores(), ESP.getCpuFreqMHz());
 | |
|   DEBUG_PRINTF_P(PSTR("FLASH: %d MB, Mode %d "), (ESP.getFlashChipSize()/1024)/1024, (int)ESP.getFlashChipMode());
 | |
|   #ifdef WLED_DEBUG
 | |
|   switch (ESP.getFlashChipMode()) {
 | |
|     // missing: Octal modes
 | |
|     case FM_QIO:  DEBUG_PRINT(F("(QIO)")); break;
 | |
|     case FM_QOUT: DEBUG_PRINT(F("(QOUT)"));break;
 | |
|     case FM_DIO:  DEBUG_PRINT(F("(DIO)")); break;
 | |
|     case FM_DOUT: DEBUG_PRINT(F("(DOUT)"));break;
 | |
|     #if defined(CONFIG_IDF_TARGET_ESP32S3) && CONFIG_ESPTOOLPY_FLASHMODE_OPI
 | |
|     case FM_FAST_READ: DEBUG_PRINT(F("(OPI)")); break;
 | |
|     #else
 | |
|     case FM_FAST_READ: DEBUG_PRINT(F("(fast_read)")); break;
 | |
|     #endif
 | |
|     case FM_SLOW_READ: DEBUG_PRINT(F("(slow_read)")); break;
 | |
|     default: break;
 | |
|   }
 | |
|   #endif
 | |
|   DEBUG_PRINTF_P(PSTR(", speed %u MHz.\n"), ESP.getFlashChipSpeed()/1000000);
 | |
| 
 | |
| #else
 | |
|   DEBUG_PRINTF_P(PSTR("esp8266 @ %u MHz.\nCore: %s\n"), ESP.getCpuFreqMHz(), ESP.getCoreVersion());
 | |
|   DEBUG_PRINTF_P(PSTR("FLASH: %u MB\n"), (ESP.getFlashChipSize()/1024)/1024);
 | |
| #endif
 | |
|   DEBUG_PRINTF_P(PSTR("heap %u\n"), ESP.getFreeHeap());
 | |
| 
 | |
| #if defined(ARDUINO_ARCH_ESP32)
 | |
|   // BOARD_HAS_PSRAM also means that a compiler flag "-mfix-esp32-psram-cache-issue" was used and so PSRAM is safe to use on rev.1 ESP32
 | |
|   #if !defined(BOARD_HAS_PSRAM) && !(defined(CONFIG_IDF_TARGET_ESP32S2) || defined(CONFIG_IDF_TARGET_ESP32S3) || defined(CONFIG_IDF_TARGET_ESP32C3))
 | |
|   if (psramFound() && ESP.getChipRevision() < 3) psramSafe = false;
 | |
|   if (!psramSafe) DEBUG_PRINTLN(F("Not using PSRAM."));
 | |
|   #endif
 | |
|   pDoc = new PSRAMDynamicJsonDocument((psramSafe && psramFound() ? 2 : 1)*JSON_BUFFER_SIZE);
 | |
|   DEBUG_PRINTF_P(PSTR("JSON buffer allocated: %u\n"), (psramSafe && psramFound() ? 2 : 1)*JSON_BUFFER_SIZE);
 | |
|   // if the above fails requestJsonBufferLock() will always return false preventing crashes
 | |
|   if (psramFound()) {
 | |
|     DEBUG_PRINTF_P(PSTR("PSRAM: %dkB/%dkB\n"), ESP.getFreePsram()/1024, ESP.getPsramSize()/1024);
 | |
|   }
 | |
|   DEBUG_PRINTF_P(PSTR("TX power: %d/%d\n"), WiFi.getTxPower(), txPower);
 | |
| #endif
 | |
| 
 | |
| #ifdef ESP8266
 | |
|   usePWMFixedNMI(); // link the NMI fix
 | |
| #endif
 | |
| 
 | |
| #if defined(WLED_DEBUG) && !defined(WLED_DEBUG_HOST)
 | |
|   PinManager::allocatePin(hardwareTX, true, PinOwner::DebugOut); // TX (GPIO1 on ESP32) reserved for debug output
 | |
| #endif
 | |
| #ifdef WLED_ENABLE_DMX //reserve GPIO2 as hardcoded DMX pin
 | |
|   PinManager::allocatePin(2, true, PinOwner::DMX);
 | |
| #endif
 | |
| 
 | |
|   DEBUG_PRINTLN(F("Registering usermods ..."));
 | |
|   registerUsermods();
 | |
| 
 | |
|   DEBUG_PRINTF_P(PSTR("heap %u\n"), ESP.getFreeHeap());
 | |
| 
 | |
|   bool fsinit = false;
 | |
|   DEBUGFS_PRINTLN(F("Mount FS"));
 | |
| #ifdef ARDUINO_ARCH_ESP32
 | |
|   fsinit = WLED_FS.begin(true);
 | |
| #else
 | |
|   fsinit = WLED_FS.begin();
 | |
| #endif
 | |
|   if (!fsinit) {
 | |
|     DEBUGFS_PRINTLN(F("FS failed!"));
 | |
|     errorFlag = ERR_FS_BEGIN;
 | |
|   }
 | |
| #ifdef WLED_ADD_EEPROM_SUPPORT
 | |
|   else deEEP();
 | |
| #else
 | |
|   initPresetsFile();
 | |
| #endif
 | |
|   updateFSInfo();
 | |
| 
 | |
|   // generate module IDs must be done before AP setup
 | |
|   escapedMac = WiFi.macAddress();
 | |
|   escapedMac.replace(":", "");
 | |
|   escapedMac.toLowerCase();
 | |
| 
 | |
|   WLED_SET_AP_SSID(); // otherwise it is empty on first boot until config is saved
 | |
|   multiWiFi.push_back(WiFiConfig(CLIENT_SSID,CLIENT_PASS)); // initialise vector with default WiFi
 | |
| 
 | |
|   DEBUG_PRINTLN(F("Reading config"));
 | |
|   deserializeConfigFromFS();
 | |
|   DEBUG_PRINTF_P(PSTR("heap %u\n"), ESP.getFreeHeap());
 | |
| 
 | |
| #if defined(STATUSLED) && STATUSLED>=0
 | |
|   if (!PinManager::isPinAllocated(STATUSLED)) {
 | |
|     // NOTE: Special case: The status LED should *NOT* be allocated.
 | |
|     //       See comments in handleStatusLed().
 | |
|     pinMode(STATUSLED, OUTPUT);
 | |
|   }
 | |
| #endif
 | |
| 
 | |
|   DEBUG_PRINTLN(F("Initializing strip"));
 | |
|   beginStrip();
 | |
|   DEBUG_PRINTF_P(PSTR("heap %u\n"), ESP.getFreeHeap());
 | |
| 
 | |
|   DEBUG_PRINTLN(F("Usermods setup"));
 | |
|   userSetup();
 | |
|   UsermodManager::setup();
 | |
|   DEBUG_PRINTF_P(PSTR("heap %u\n"), ESP.getFreeHeap());
 | |
| 
 | |
|   if (strcmp(multiWiFi[0].clientSSID, DEFAULT_CLIENT_SSID) == 0)
 | |
|     showWelcomePage = true;
 | |
|   WiFi.persistent(false);
 | |
|   WiFi.onEvent(WiFiEvent);
 | |
|   WiFi.mode(WIFI_STA); // enable scanning
 | |
|   findWiFi(true);      // start scanning for available WiFi-s
 | |
| 
 | |
|   // all GPIOs are allocated at this point
 | |
|   serialCanRX = !PinManager::isPinAllocated(hardwareRX); // Serial RX pin (GPIO 3 on ESP32 and ESP8266)
 | |
|   serialCanTX = !PinManager::isPinAllocated(hardwareTX) || PinManager::getPinOwner(hardwareTX) == PinOwner::DebugOut; // Serial TX pin (GPIO 1 on ESP32 and ESP8266)
 | |
| 
 | |
|   #ifdef WLED_ENABLE_ADALIGHT
 | |
|   //Serial RX (Adalight, Improv, Serial JSON) only possible if GPIO3 unused
 | |
|   //Serial TX (Debug, Improv, Serial JSON) only possible if GPIO1 unused
 | |
|   if (serialCanRX && serialCanTX) {
 | |
|     Serial.println(F("Ada"));
 | |
|   }
 | |
|   #endif
 | |
| 
 | |
|   // fill in unique mdns default
 | |
|   if (strcmp(cmDNS, "x") == 0) sprintf_P(cmDNS, PSTR("wled-%*s"), 6, escapedMac.c_str() + 6);
 | |
| #ifndef WLED_DISABLE_MQTT
 | |
|   if (mqttDeviceTopic[0] == 0) sprintf_P(mqttDeviceTopic, PSTR("wled/%*s"), 6, escapedMac.c_str() + 6);
 | |
|   if (mqttClientID[0] == 0)    sprintf_P(mqttClientID, PSTR("WLED-%*s"), 6, escapedMac.c_str() + 6);
 | |
| #endif
 | |
| 
 | |
| #ifndef WLED_DISABLE_OTA
 | |
|   if (aOtaEnabled) {
 | |
|     ArduinoOTA.onStart([]() {
 | |
|       #ifdef ESP8266
 | |
|       wifi_set_sleep_type(NONE_SLEEP_T);
 | |
|       #endif
 | |
|       #if WLED_WATCHDOG_TIMEOUT > 0
 | |
|       WLED::instance().disableWatchdog();
 | |
|       #endif
 | |
|       DEBUG_PRINTLN(F("Start ArduinoOTA"));
 | |
|     });
 | |
|     ArduinoOTA.onError([](ota_error_t error) {
 | |
|       #if WLED_WATCHDOG_TIMEOUT > 0
 | |
|       // reenable watchdog on failed update
 | |
|       WLED::instance().enableWatchdog();
 | |
|       #endif
 | |
|     });
 | |
|     if (strlen(cmDNS) > 0)
 | |
|       ArduinoOTA.setHostname(cmDNS);
 | |
|   }
 | |
| #endif
 | |
| #ifdef WLED_ENABLE_DMX
 | |
|   initDMXOutput();
 | |
| #endif
 | |
| #ifdef WLED_ENABLE_DMX_INPUT
 | |
|   dmxInput.init(dmxInputReceivePin, dmxInputTransmitPin, dmxInputEnablePin, dmxInputPort);
 | |
| #endif
 | |
| 
 | |
| #ifdef WLED_ENABLE_ADALIGHT
 | |
|   if (serialCanRX && Serial.available() > 0 && Serial.peek() == 'I') handleImprovPacket();
 | |
| #endif
 | |
| 
 | |
|   // HTTP server page init
 | |
|   DEBUG_PRINTLN(F("initServer"));
 | |
|   initServer();
 | |
|   DEBUG_PRINTF_P(PSTR("heap %u\n"), ESP.getFreeHeap());
 | |
| 
 | |
| #ifndef WLED_DISABLE_INFRARED
 | |
|   // init IR
 | |
|   DEBUG_PRINTLN(F("initIR"));
 | |
|   initIR();
 | |
|   DEBUG_PRINTF_P(PSTR("heap %u\n"), ESP.getFreeHeap());
 | |
| #endif
 | |
| 
 | |
|   // Seed FastLED random functions with an esp random value, which already works properly at this point.
 | |
|   const uint32_t seed32 = hw_random();
 | |
|   random16_set_seed((uint16_t)seed32);
 | |
| 
 | |
|   #if WLED_WATCHDOG_TIMEOUT > 0
 | |
|   enableWatchdog();
 | |
|   #endif
 | |
| 
 | |
|   #if defined(ARDUINO_ARCH_ESP32) && defined(WLED_DISABLE_BROWNOUT_DET)
 | |
|   WRITE_PERI_REG(RTC_CNTL_BROWN_OUT_REG, 1); //enable brownout detector
 | |
|   #endif
 | |
| }
 | |
| 
 | |
| void WLED::beginStrip()
 | |
| {
 | |
|   // Initialize NeoPixel Strip and button
 | |
|   strip.finalizeInit(); // busses created during deserializeConfig() if config existed
 | |
|   strip.makeAutoSegments();
 | |
|   strip.setBrightness(0);
 | |
|   strip.setShowCallback(handleOverlayDraw);
 | |
| 
 | |
|   if (turnOnAtBoot) {
 | |
|     if (briS > 0) bri = briS;
 | |
|     else if (bri == 0) bri = 128;
 | |
|   } else {
 | |
|     // fix for #3196
 | |
|     if (bootPreset > 0) {
 | |
|       // set all segments black (no transition)
 | |
|       for (unsigned i = 0; i < strip.getSegmentsNum(); i++) {
 | |
|         Segment &seg = strip.getSegment(i);
 | |
|         if (seg.isActive()) seg.colors[0] = BLACK;
 | |
|       }
 | |
|       col[0] = col[1] = col[2] = col[3] = 0;  // needed for colorUpdated()
 | |
|     }
 | |
|     briLast = briS; bri = 0;
 | |
|     strip.fill(BLACK);
 | |
|     strip.show();
 | |
|   }
 | |
|   if (bootPreset > 0) {
 | |
|     applyPreset(bootPreset, CALL_MODE_INIT);
 | |
|   }
 | |
|   colorUpdated(CALL_MODE_INIT); // will not send notification
 | |
| 
 | |
|   // init relay pin
 | |
|   if (rlyPin >= 0) {
 | |
|     pinMode(rlyPin, rlyOpenDrain ? OUTPUT_OPEN_DRAIN : OUTPUT);
 | |
|     digitalWrite(rlyPin, (rlyMde ? bri : !bri));
 | |
|   }
 | |
| }
 | |
| 
 | |
| void WLED::initAP(bool resetAP)
 | |
| {
 | |
|   if (apBehavior == AP_BEHAVIOR_BUTTON_ONLY && !resetAP)
 | |
|     return;
 | |
| 
 | |
|   if (resetAP) {
 | |
|     WLED_SET_AP_SSID();
 | |
|     strcpy_P(apPass, PSTR(WLED_AP_PASS));
 | |
|   }
 | |
|   DEBUG_PRINT(F("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);
 | |
|   #ifdef ARDUINO_ARCH_ESP32
 | |
|   WiFi.setTxPower(wifi_power_t(txPower));
 | |
|   #endif
 | |
| 
 | |
|   if (!apActive) // start captive portal if AP active
 | |
|   {
 | |
|     DEBUG_PRINTLN(F("Init AP interfaces"));
 | |
|     server.begin();
 | |
|     if (udpPort > 0 && udpPort != ntpLocalPort) {
 | |
|       udpConnected = notifierUdp.begin(udpPort);
 | |
|     }
 | |
|     if (udpRgbPort > 0 && udpRgbPort != ntpLocalPort && udpRgbPort != udpPort) {
 | |
|       udpRgbConnected = rgbUdp.begin(udpRgbPort);
 | |
|     }
 | |
|     if (udpPort2 > 0 && udpPort2 != ntpLocalPort && udpPort2 != udpPort && udpPort2 != udpRgbPort) {
 | |
|       udp2Connected = notifier2Udp.begin(udpPort2);
 | |
|     }
 | |
|     e131.begin(false, e131Port, e131Universe, E131_MAX_UNIVERSE_COUNT);
 | |
|     ddp.begin(false, DDP_DEFAULT_PORT);
 | |
| 
 | |
|     dnsServer.setErrorReplyCode(DNSReplyCode::NoError);
 | |
|     dnsServer.start(53, "*", WiFi.softAPIP());
 | |
|   }
 | |
|   apActive = true;
 | |
| }
 | |
| 
 | |
| bool WLED::initEthernet()
 | |
| {
 | |
| #if defined(ARDUINO_ARCH_ESP32) && defined(WLED_USE_ETHERNET)
 | |
| 
 | |
|   static bool successfullyConfiguredEthernet = false;
 | |
| 
 | |
|   if (successfullyConfiguredEthernet) {
 | |
|     // DEBUG_PRINTLN(F("initE: ETH already successfully configured, ignoring"));
 | |
|     return false;
 | |
|   }
 | |
|   if (ethernetType == WLED_ETH_NONE) {
 | |
|     return false;
 | |
|   }
 | |
|   if (ethernetType >= WLED_NUM_ETH_TYPES) {
 | |
|     DEBUG_PRINTF_P(PSTR("initE: Ignoring attempt for invalid ethernetType (%d)\n"), ethernetType);
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   DEBUG_PRINTF_P(PSTR("initE: Attempting ETH config: %d\n"), ethernetType);
 | |
| 
 | |
|   // Ethernet initialization should only succeed once -- else reboot required
 | |
|   ethernet_settings es = ethernetBoards[ethernetType];
 | |
|   managed_pin_type pinsToAllocate[10] = {
 | |
|     // first six pins are non-configurable
 | |
|     esp32_nonconfigurable_ethernet_pins[0],
 | |
|     esp32_nonconfigurable_ethernet_pins[1],
 | |
|     esp32_nonconfigurable_ethernet_pins[2],
 | |
|     esp32_nonconfigurable_ethernet_pins[3],
 | |
|     esp32_nonconfigurable_ethernet_pins[4],
 | |
|     esp32_nonconfigurable_ethernet_pins[5],
 | |
|     { (int8_t)es.eth_mdc,   true },  // [6] = MDC  is output and mandatory
 | |
|     { (int8_t)es.eth_mdio,  true },  // [7] = MDIO is bidirectional and mandatory
 | |
|     { (int8_t)es.eth_power, true },  // [8] = optional pin, not all boards use
 | |
|     { ((int8_t)0xFE),       false }, // [9] = replaced with eth_clk_mode, mandatory
 | |
|   };
 | |
|   // update the clock pin....
 | |
|   if (es.eth_clk_mode == ETH_CLOCK_GPIO0_IN) {
 | |
|     pinsToAllocate[9].pin = 0;
 | |
|     pinsToAllocate[9].isOutput = false;
 | |
|   } else if (es.eth_clk_mode == ETH_CLOCK_GPIO0_OUT) {
 | |
|     pinsToAllocate[9].pin = 0;
 | |
|     pinsToAllocate[9].isOutput = true;
 | |
|   } else if (es.eth_clk_mode == ETH_CLOCK_GPIO16_OUT) {
 | |
|     pinsToAllocate[9].pin = 16;
 | |
|     pinsToAllocate[9].isOutput = true;
 | |
|   } else if (es.eth_clk_mode == ETH_CLOCK_GPIO17_OUT) {
 | |
|     pinsToAllocate[9].pin = 17;
 | |
|     pinsToAllocate[9].isOutput = true;
 | |
|   } else {
 | |
|     DEBUG_PRINTF_P(PSTR("initE: Failing due to invalid eth_clk_mode (%d)\n"), es.eth_clk_mode);
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   if (!PinManager::allocateMultiplePins(pinsToAllocate, 10, PinOwner::Ethernet)) {
 | |
|     DEBUG_PRINTLN(F("initE: Failed to allocate ethernet pins"));
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   /*
 | |
|   For LAN8720 the most correct way is to perform clean reset each time before init
 | |
|   applying LOW to power or nRST pin for at least 100 us (please refer to datasheet, page 59)
 | |
|   ESP_IDF > V4 implements it (150 us, lan87xx_reset_hw(esp_eth_phy_t *phy) function in 
 | |
|   /components/esp_eth/src/esp_eth_phy_lan87xx.c, line 280)
 | |
|   but ESP_IDF < V4 does not. Lets do it:
 | |
|   [not always needed, might be relevant in some EMI situations at startup and for hot resets]
 | |
|   */
 | |
|   #if ESP_IDF_VERSION_MAJOR==3
 | |
|   if(es.eth_power>0 && es.eth_type==ETH_PHY_LAN8720) {
 | |
|     pinMode(es.eth_power, OUTPUT);
 | |
|     digitalWrite(es.eth_power, 0);
 | |
|     delayMicroseconds(150);
 | |
|     digitalWrite(es.eth_power, 1);
 | |
|     delayMicroseconds(10);
 | |
|   }
 | |
|   #endif
 | |
| 
 | |
|   if (!ETH.begin(
 | |
|                 (uint8_t) es.eth_address,
 | |
|                 (int)     es.eth_power,
 | |
|                 (int)     es.eth_mdc,
 | |
|                 (int)     es.eth_mdio,
 | |
|                 (eth_phy_type_t)   es.eth_type,
 | |
|                 (eth_clock_mode_t) es.eth_clk_mode
 | |
|                 )) {
 | |
|     DEBUG_PRINTLN(F("initC: ETH.begin() failed"));
 | |
|     // de-allocate the allocated pins
 | |
|     for (managed_pin_type mpt : pinsToAllocate) {
 | |
|       PinManager::deallocatePin(mpt.pin, PinOwner::Ethernet);
 | |
|     }
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   successfullyConfiguredEthernet = true;
 | |
|   DEBUG_PRINTLN(F("initC: *** Ethernet successfully configured! ***"));
 | |
|   return true;
 | |
| #else
 | |
|   return false; // Ethernet not enabled for build
 | |
| #endif
 | |
| }
 | |
| 
 | |
| // performs asynchronous scan for available networks (which may take couple of seconds to finish)
 | |
| // returns configured WiFi ID with the strongest signal (or default if no configured networks available)
 | |
| int8_t WLED::findWiFi(bool doScan) {
 | |
|   if (multiWiFi.size() <= 1) {
 | |
|     DEBUG_PRINTLN(F("Defaulf WiFi used."));
 | |
|     return 0;
 | |
|   }
 | |
| 
 | |
|   if (doScan) WiFi.scanDelete();  // restart scan
 | |
| 
 | |
|   int status = WiFi.scanComplete(); // complete scan may take as much as several seconds (usually <3s with not very crowded air)
 | |
| 
 | |
|   if (status == WIFI_SCAN_FAILED) {
 | |
|     DEBUG_PRINTLN(F("WiFi scan started."));
 | |
|     WiFi.scanNetworks(true);  // start scanning in asynchronous mode
 | |
|   } else if (status >= 0) {   // status contains number of found networks
 | |
|     DEBUG_PRINT(F("WiFi scan completed: ")); DEBUG_PRINTLN(status);
 | |
|     int rssi = -9999;
 | |
|     unsigned selected = selectedWiFi;
 | |
|     for (int o = 0; o < status; o++) {
 | |
|       DEBUG_PRINT(F(" WiFi available: ")); DEBUG_PRINT(WiFi.SSID(o));
 | |
|       DEBUG_PRINT(F(" RSSI: ")); DEBUG_PRINT(WiFi.RSSI(o)); DEBUG_PRINTLN(F("dB"));
 | |
|       for (unsigned n = 0; n < multiWiFi.size(); n++)
 | |
|         if (!strcmp(WiFi.SSID(o).c_str(), multiWiFi[n].clientSSID)) {
 | |
|           // find the WiFi with the strongest signal (but keep priority of entry if signal difference is not big)
 | |
|           if ((n < selected && WiFi.RSSI(o) > rssi-10) || WiFi.RSSI(o) > rssi) {
 | |
|             rssi = WiFi.RSSI(o);
 | |
|             selected = n;
 | |
|           }
 | |
|           break;
 | |
|         }
 | |
|     }
 | |
|     DEBUG_PRINT(F("Selected: ")); DEBUG_PRINT(multiWiFi[selected].clientSSID);
 | |
|     DEBUG_PRINT(F(" RSSI: ")); DEBUG_PRINT(rssi); DEBUG_PRINTLN(F("dB"));
 | |
|     return selected;
 | |
|   }
 | |
|   //DEBUG_PRINT(F("WiFi scan running."));
 | |
|   return status; // scan is still running or there was an error
 | |
| }
 | |
| 
 | |
| void WLED::initConnection()
 | |
| {
 | |
|   DEBUG_PRINTF_P(PSTR("initConnection() called @ %lus.\n"), millis()/1000);
 | |
|   #ifdef WLED_ENABLE_WEBSOCKETS
 | |
|   ws.onEvent(wsEvent);
 | |
|   #endif
 | |
| 
 | |
| #ifndef WLED_DISABLE_ESPNOW
 | |
|   if (statusESPNow == ESP_NOW_STATE_ON) {
 | |
|     DEBUG_PRINTLN(F("ESP-NOW stopping."));
 | |
|     quickEspNow.stop();
 | |
|     statusESPNow = ESP_NOW_STATE_UNINIT;
 | |
|   }
 | |
| #endif
 | |
| 
 | |
|   WiFi.disconnect(true); // close old connections
 | |
| #ifdef ESP8266
 | |
|   WiFi.setPhyMode(force802_3g ? WIFI_PHY_MODE_11G : WIFI_PHY_MODE_11N);
 | |
| #endif
 | |
| 
 | |
|   if (multiWiFi[selectedWiFi].staticIP != 0U && multiWiFi[selectedWiFi].staticGW != 0U) {
 | |
|     WiFi.config(multiWiFi[selectedWiFi].staticIP, multiWiFi[selectedWiFi].staticGW, multiWiFi[selectedWiFi].staticSN, dnsAddress);
 | |
|   } else {
 | |
|     WiFi.config(IPAddress((uint32_t)0), IPAddress((uint32_t)0), IPAddress((uint32_t)0));
 | |
|   }
 | |
| 
 | |
|   lastReconnectAttempt = millis();
 | |
| 
 | |
|   if (!WLED_WIFI_CONFIGURED) {
 | |
|     DEBUG_PRINTLN(F("No connection configured."));
 | |
|     if (!apActive) initAP();        // instantly go to ap mode
 | |
|     return;
 | |
|   } else if (!apActive) {
 | |
|     if (apBehavior == AP_BEHAVIOR_ALWAYS) {
 | |
|       DEBUG_PRINTLN(F("Access point ALWAYS enabled."));
 | |
|       initAP();
 | |
|     } else {
 | |
|       DEBUG_PRINTLN(F("Access point disabled (init)."));
 | |
|       WiFi.softAPdisconnect(true);
 | |
|       WiFi.mode(WIFI_STA);
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (WLED_WIFI_CONFIGURED) {
 | |
|     showWelcomePage = false;
 | |
|     
 | |
|     DEBUG_PRINTF_P(PSTR("Connecting to %s...\n"), multiWiFi[selectedWiFi].clientSSID);
 | |
| 
 | |
|     // convert the "serverDescription" into a valid DNS hostname (alphanumeric)
 | |
|     char hostname[25];
 | |
|     prepareHostname(hostname);
 | |
|     WiFi.begin(multiWiFi[selectedWiFi].clientSSID, multiWiFi[selectedWiFi].clientPass); // no harm if called multiple times
 | |
| 
 | |
| #ifdef ARDUINO_ARCH_ESP32
 | |
|     WiFi.setTxPower(wifi_power_t(txPower));
 | |
|     WiFi.setSleep(!noWifiSleep);
 | |
|     WiFi.setHostname(hostname);
 | |
| #else
 | |
|     wifi_set_sleep_type((noWifiSleep) ? NONE_SLEEP_T : MODEM_SLEEP_T);
 | |
|     WiFi.hostname(hostname);
 | |
| #endif
 | |
|   }
 | |
| 
 | |
| #ifndef WLED_DISABLE_ESPNOW
 | |
|   if (enableESPNow) {
 | |
|     quickEspNow.onDataSent(espNowSentCB);     // see udp.cpp
 | |
|     quickEspNow.onDataRcvd(espNowReceiveCB);  // see udp.cpp
 | |
|     bool espNowOK;
 | |
|     if (apActive) {
 | |
|       DEBUG_PRINTLN(F("ESP-NOW initing in AP mode."));
 | |
|       #ifdef ESP32
 | |
|       quickEspNow.setWiFiBandwidth(WIFI_IF_AP, WIFI_BW_HT20); // Only needed for ESP32 in case you need coexistence with ESP8266 in the same network
 | |
|       #endif //ESP32
 | |
|       espNowOK = quickEspNow.begin(apChannel, WIFI_IF_AP);  // Same channel must be used for both AP and ESP-NOW
 | |
|     } else {
 | |
|       DEBUG_PRINTLN(F("ESP-NOW initing in STA mode."));
 | |
|       espNowOK = quickEspNow.begin(); // Use no parameters to start ESP-NOW on same channel as WiFi, in STA mode
 | |
|     }
 | |
|     statusESPNow = espNowOK ? ESP_NOW_STATE_ON : ESP_NOW_STATE_ERROR;
 | |
|   }
 | |
| #endif
 | |
| }
 | |
| 
 | |
| void WLED::initInterfaces()
 | |
| {
 | |
|   DEBUG_PRINTLN(F("Init STA interfaces"));
 | |
| 
 | |
| #ifndef WLED_DISABLE_HUESYNC
 | |
|   IPAddress ipAddress = Network.localIP();
 | |
|   if (hueIP[0] == 0) {
 | |
|     hueIP[0] = ipAddress[0];
 | |
|     hueIP[1] = ipAddress[1];
 | |
|     hueIP[2] = ipAddress[2];
 | |
|   }
 | |
| #endif
 | |
| 
 | |
| #ifndef WLED_DISABLE_ALEXA
 | |
|   // init Alexa hue emulation
 | |
|   if (alexaEnabled)
 | |
|     alexaInit();
 | |
| #endif
 | |
| 
 | |
| #ifndef WLED_DISABLE_OTA
 | |
|   if (aOtaEnabled)
 | |
|     ArduinoOTA.begin();
 | |
| #endif
 | |
| 
 | |
|   // Set up mDNS responder:
 | |
|   if (strlen(cmDNS) > 0) {
 | |
|     // "end" must be called before "begin" is called a 2nd time
 | |
|     // see https://github.com/esp8266/Arduino/issues/7213
 | |
|     MDNS.end();
 | |
|     MDNS.begin(cmDNS);
 | |
| 
 | |
|     DEBUG_PRINTLN(F("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 (udpConnected && udpPort2 != udpPort && udpPort2 != udpRgbPort)
 | |
|       udp2Connected = notifier2Udp.begin(udpPort2);
 | |
|   }
 | |
|   if (ntpEnabled)
 | |
|     ntpConnected = ntpUdp.begin(ntpLocalPort);
 | |
| 
 | |
|   e131.begin(e131Multicast, e131Port, e131Universe, E131_MAX_UNIVERSE_COUNT);
 | |
|   ddp.begin(false, DDP_DEFAULT_PORT);
 | |
|   reconnectHue();
 | |
| #ifndef WLED_DISABLE_MQTT
 | |
|   initMqtt();
 | |
| #endif
 | |
|   interfacesInited = true;
 | |
|   wasConnected = true;
 | |
| }
 | |
| 
 | |
| void WLED::handleConnection()
 | |
| {
 | |
|   static bool scanDone = true;
 | |
|   static byte stacO = 0;
 | |
|   const unsigned long now = millis();
 | |
|   const unsigned long nowS = now/1000;
 | |
|   const bool wifiConfigured = WLED_WIFI_CONFIGURED;
 | |
| 
 | |
|   // ignore connection handling if WiFi is configured and scan still running
 | |
|   // or within first 2s if WiFi is not configured or AP is always active
 | |
|   if ((wifiConfigured && multiWiFi.size() > 1 && WiFi.scanComplete() < 0) || (now < 2000 && (!wifiConfigured || apBehavior == AP_BEHAVIOR_ALWAYS)))
 | |
|     return;
 | |
| 
 | |
|   if (lastReconnectAttempt == 0 || forceReconnect) {
 | |
|     DEBUG_PRINTF_P(PSTR("Initial connect or forced reconnect (@ %lus).\n"), nowS);
 | |
|     selectedWiFi = findWiFi(); // find strongest WiFi
 | |
|     initConnection();
 | |
|     interfacesInited = false;
 | |
|     forceReconnect = false;
 | |
|     wasConnected = false;
 | |
|     return;
 | |
|   }
 | |
| 
 | |
|   byte stac = 0;
 | |
|   if (apActive) {
 | |
| #ifdef ESP8266
 | |
|     stac = wifi_softap_get_station_num();
 | |
| #else
 | |
|     wifi_sta_list_t stationList;
 | |
|     esp_wifi_ap_get_sta_list(&stationList);
 | |
|     stac = stationList.num;
 | |
| #endif
 | |
|     if (stac != stacO) {
 | |
|       stacO = stac;
 | |
|       DEBUG_PRINTF_P(PSTR("Connected AP clients: %d\n"), (int)stac);
 | |
|       if (!WLED_CONNECTED && wifiConfigured) {        // trying to connect, but not connected
 | |
|         if (stac)
 | |
|           WiFi.disconnect();        // disable search so that AP can work
 | |
|         else
 | |
|           initConnection();         // restart search
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| 
 | |
|   if (!Network.isConnected()) {
 | |
|     if (interfacesInited) {
 | |
|       if (scanDone && multiWiFi.size() > 1) {
 | |
|         DEBUG_PRINTLN(F("WiFi scan initiated on disconnect."));
 | |
|         findWiFi(true); // reinit scan
 | |
|         scanDone = false;
 | |
|         return;         // try to connect in next iteration
 | |
|       }
 | |
|       DEBUG_PRINTLN(F("Disconnected!"));
 | |
|       selectedWiFi = findWiFi();
 | |
|       initConnection();
 | |
|       interfacesInited = false;
 | |
|       scanDone = true;
 | |
|       return;
 | |
|     }
 | |
|     //send improv failed 6 seconds after second init attempt (24 sec. after provisioning)
 | |
|     if (improvActive > 2 && now - lastReconnectAttempt > 6000) {
 | |
|       sendImprovStateResponse(0x03, true);
 | |
|       improvActive = 2;
 | |
|     }
 | |
|     if (now - lastReconnectAttempt > ((stac) ? 300000 : 18000) && wifiConfigured) {
 | |
|       if (improvActive == 2) improvActive = 3;
 | |
|       DEBUG_PRINTF_P(PSTR("Last reconnect (%lus) too old (@ %lus).\n"), lastReconnectAttempt/1000, nowS);
 | |
|       if (++selectedWiFi >= multiWiFi.size()) selectedWiFi = 0; // we couldn't connect, try with another network from the list
 | |
|       initConnection();
 | |
|     }
 | |
|     if (!apActive && now - lastReconnectAttempt > 12000 && (!wasConnected || apBehavior == AP_BEHAVIOR_NO_CONN)) {
 | |
|       if (!(apBehavior == AP_BEHAVIOR_TEMPORARY && now > WLED_AP_TIMEOUT)) {
 | |
|         DEBUG_PRINTF_P(PSTR("Not connected AP (@ %lus).\n"), nowS);
 | |
|         initAP();  // start AP only within first 5min
 | |
|       }
 | |
|     }
 | |
|     if (apActive && apBehavior == AP_BEHAVIOR_TEMPORARY && now > WLED_AP_TIMEOUT && stac == 0) { // disconnect AP after 5min if no clients connected
 | |
|       // if AP was enabled more than 10min after boot or if client was connected more than 10min after boot do not disconnect AP mode
 | |
|       if (now < 2*WLED_AP_TIMEOUT) {
 | |
|         dnsServer.stop();
 | |
|         WiFi.softAPdisconnect(true);
 | |
|         apActive = false;
 | |
|         DEBUG_PRINTF_P(PSTR("Temporary AP disabled (@ %lus).\n"), nowS);
 | |
|       }
 | |
|     }
 | |
|   } else if (!interfacesInited) { //newly connected
 | |
|     DEBUG_PRINTLN();
 | |
|     DEBUG_PRINT(F("Connected! IP address: "));
 | |
|     DEBUG_PRINTLN(Network.localIP());
 | |
|     if (improvActive) {
 | |
|       if (improvError == 3) sendImprovStateResponse(0x00, true);
 | |
|       sendImprovStateResponse(0x04);
 | |
|       if (improvActive > 1) sendImprovIPRPCResult(ImprovRPCType::Command_Wifi);
 | |
|     }
 | |
|     initInterfaces();
 | |
|     userConnected();
 | |
|     UsermodManager::connected();
 | |
|     lastMqttReconnectAttempt = 0; // force immediate update
 | |
| 
 | |
|     // shut down AP
 | |
|     if (apBehavior != AP_BEHAVIOR_ALWAYS && apActive) {
 | |
|       dnsServer.stop();
 | |
|       WiFi.softAPdisconnect(true);
 | |
|       apActive = false;
 | |
|       DEBUG_PRINTLN(F("Access point disabled (connected)."));
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| // If status LED pin is allocated for other uses, does nothing
 | |
| // else blink at 1Hz when WLED_CONNECTED is false (no WiFi, ?? no Ethernet ??)
 | |
| // else blink at 2Hz when MQTT is enabled but not connected
 | |
| // else turn the status LED off
 | |
| #if defined(STATUSLED)
 | |
| void WLED::handleStatusLED()
 | |
| {
 | |
|   uint32_t c = 0;
 | |
| 
 | |
|   #if STATUSLED>=0
 | |
|   if (PinManager::isPinAllocated(STATUSLED)) {
 | |
|     return; //lower priority if something else uses the same pin
 | |
|   }
 | |
|   #endif
 | |
| 
 | |
|   if (WLED_CONNECTED) {
 | |
|     c = RGBW32(0,255,0,0);
 | |
|     ledStatusType = 2;
 | |
|   } else if (WLED_MQTT_CONNECTED) {
 | |
|     c = RGBW32(0,128,0,0);
 | |
|     ledStatusType = 4;
 | |
|   } else if (apActive) {
 | |
|     c = RGBW32(0,0,255,0);
 | |
|     ledStatusType = 1;
 | |
|   }
 | |
|   if (ledStatusType) {
 | |
|     if (millis() - ledStatusLastMillis >= (1000/ledStatusType)) {
 | |
|       ledStatusLastMillis = millis();
 | |
|       ledStatusState = !ledStatusState;
 | |
|       #if STATUSLED>=0
 | |
|       digitalWrite(STATUSLED, ledStatusState);
 | |
|       #else
 | |
|       BusManager::setStatusPixel(ledStatusState ? c : 0);
 | |
|       #endif
 | |
|     }
 | |
|   } else {
 | |
|     #if STATUSLED>=0
 | |
|       #ifdef STATUSLEDINVERTED
 | |
|       digitalWrite(STATUSLED, HIGH);
 | |
|       #else
 | |
|       digitalWrite(STATUSLED, LOW);
 | |
|       #endif
 | |
|     #else
 | |
|       BusManager::setStatusPixel(0);
 | |
|     #endif
 | |
|   }
 | |
| }
 | |
| #endif
 | 
