Improvements to heap-memory and PSRAM handling (#4791)
* Improved heap and PSRAM handling - Segment `allocateData()` uses more elaborate DRAM checking to reduce fragmentation and allow for larger setups to run on low heap - Segment data allocation fails if minimum contiguous block size runs low to keep the UI working - Increased `MAX_SEGMENT_DATA` to account for better segment data handling - Memory allocation functions try to keep enough DRAM for segment data - Added constant `PSRAM_THRESHOLD` to improve PSARM usage - Increase MIN_HEAP_SIZE to reduce risk of breaking UI due to low memory for JSON response - ESP32 makes use of IRAM (no 8bit access) for pixeluffers, freeing up to 50kB of RAM - Fix to properly get available heap on all platforms: added function `getFreeHeapSize()` - Bugfix for effects that divide by SEGLEN: don't run FX in service() if segment is not active -Syntax fix in AR: calloc() uses (numelements, size) as arguments * Added new functions for allocation and heap checking - added `allocate_buffer()` function that can be used to allocate large buffers: takes parameters to set preferred ram location, including 32bit accessible RAM on ESP32. Returns null if heap runs low or switches to PSRAM - getFreeHeapSize() and getContiguousFreeHeap() helper functions for all platforms to correctly report free useable heap - updated some constants - updated segment data allocation to free the data if it is large - replaced "psramsafe" variable with it's #ifdef: BOARD_HAS_PSRAM and made accomodating changes - added some compile-time checks to handle invalid env. definitions - updated all allocation functions and some of the logic behind them - added use of fast RTC-Memory where available - increased MIN_HEAP_SIZE for all systems (improved stability in tests) - updated memory calculation in web-UI to account for required segment buffer - added UI alerts if buffer allocation fails - made getUsedSegmentData() non-private (used in buffer alloc function) - changed MAX_SEGMENT_DATA - added more detailed memory log to DEBUG output - added debug output to buffer alloc function
This commit is contained in:
		| @@ -224,8 +224,8 @@ void FFTcode(void * parameter) | |||||||
|   DEBUGSR_PRINT("FFT started on core: "); DEBUGSR_PRINTLN(xPortGetCoreID()); |   DEBUGSR_PRINT("FFT started on core: "); DEBUGSR_PRINTLN(xPortGetCoreID()); | ||||||
|  |  | ||||||
|   // allocate FFT buffers on first call |   // allocate FFT buffers on first call | ||||||
|   if (vReal == nullptr) vReal = (float*) calloc(sizeof(float), samplesFFT); |   if (vReal == nullptr) vReal = (float*) calloc(samplesFFT, sizeof(float)); | ||||||
|   if (vImag == nullptr) vImag = (float*) calloc(sizeof(float), samplesFFT); |   if (vImag == nullptr) vImag = (float*) calloc(samplesFFT, sizeof(float)); | ||||||
|   if ((vReal == nullptr) || (vImag == nullptr)) { |   if ((vReal == nullptr) || (vImag == nullptr)) { | ||||||
|     // something went wrong |     // something went wrong | ||||||
|     if (vReal) free(vReal); vReal = nullptr; |     if (vReal) free(vReal); vReal = nullptr; | ||||||
|   | |||||||
							
								
								
									
										33
									
								
								wled00/FX.h
									
									
									
									
									
								
							
							
						
						
									
										33
									
								
								wled00/FX.h
									
									
									
									
									
								
							| @@ -88,23 +88,26 @@ extern byte realtimeMode;           // used in getMappedPixelIndex() | |||||||
| #endif | #endif | ||||||
| #define FPS_CALC_SHIFT 7 // bit shift for fixed point math | #define FPS_CALC_SHIFT 7 // bit shift for fixed point math | ||||||
|  |  | ||||||
| /* each segment uses 82 bytes of SRAM memory, so if you're application fails because of | // heap memory limit for effects data, pixel buffers try to reserve it if PSRAM is available | ||||||
|   insufficient memory, decreasing MAX_NUM_SEGMENTS may help */ |  | ||||||
| #ifdef ESP8266 | #ifdef ESP8266 | ||||||
|   #define MAX_NUM_SEGMENTS  16 |   #define MAX_NUM_SEGMENTS  16 | ||||||
|   /* How much data bytes all segments combined may allocate */ |   /* How much data bytes all segments combined may allocate */ | ||||||
|   #define MAX_SEGMENT_DATA  5120 |   #define MAX_SEGMENT_DATA  (6*1024) // 6k by default | ||||||
| #elif defined(CONFIG_IDF_TARGET_ESP32S2) | #elif defined(CONFIG_IDF_TARGET_ESP32S2) | ||||||
|   #define MAX_NUM_SEGMENTS  20 |   #define MAX_NUM_SEGMENTS  32 | ||||||
|   #define MAX_SEGMENT_DATA  (MAX_NUM_SEGMENTS*512)  // 10k by default (S2 is short on free RAM) |   #define MAX_SEGMENT_DATA  (20*1024) // 20k by default (S2 is short on free RAM), limit does not apply if PSRAM is available | ||||||
| #else | #else | ||||||
|   #define MAX_NUM_SEGMENTS  32  // warning: going beyond 32 may consume too much RAM for stable operation |   #ifdef BOARD_HAS_PSRAM | ||||||
|   #define MAX_SEGMENT_DATA  (MAX_NUM_SEGMENTS*1280) // 40k by default |     #define MAX_NUM_SEGMENTS  64 | ||||||
|  |   #else | ||||||
|  |     #define MAX_NUM_SEGMENTS  32 | ||||||
|  |   #endif | ||||||
|  |   #define MAX_SEGMENT_DATA  (64*1024) // 64k by default, limit does not apply if PSRAM is available | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
| /* How much data bytes each segment should max allocate to leave enough space for other segments, | /* How much data bytes each segment should max allocate to leave enough space for other segments, | ||||||
|   assuming each segment uses the same amount of data. 256 for ESP8266, 640 for ESP32. */ |   assuming each segment uses the same amount of data. 256 for ESP8266, 640 for ESP32. */ | ||||||
| #define FAIR_DATA_PER_SEG (MAX_SEGMENT_DATA / WS2812FX::getMaxSegments()) | #define FAIR_DATA_PER_SEG (MAX_SEGMENT_DATA / MAX_NUM_SEGMENTS) | ||||||
|  |  | ||||||
| #define MIN_SHOW_DELAY   (_frametime < 16 ? 8 : 15) | #define MIN_SHOW_DELAY   (_frametime < 16 ? 8 : 15) | ||||||
|  |  | ||||||
| @@ -533,7 +536,6 @@ class Segment { | |||||||
|  |  | ||||||
|   protected: |   protected: | ||||||
|  |  | ||||||
|     inline static unsigned getUsedSegmentData()            { return Segment::_usedSegmentData; } |  | ||||||
|     inline static void     addUsedSegmentData(int len)     { Segment::_usedSegmentData += len; } |     inline static void     addUsedSegmentData(int len)     { Segment::_usedSegmentData += len; } | ||||||
|  |  | ||||||
|     inline uint32_t *getPixels() const                              { return pixels; } |     inline uint32_t *getPixels() const                              { return pixels; } | ||||||
| @@ -600,8 +602,8 @@ class Segment { | |||||||
|     , _t(nullptr) |     , _t(nullptr) | ||||||
|     { |     { | ||||||
|       DEBUGFX_PRINTF_P(PSTR("-- Creating segment: %p [%d,%d:%d,%d]\n"), this, (int)start, (int)stop, (int)startY, (int)stopY); |       DEBUGFX_PRINTF_P(PSTR("-- Creating segment: %p [%d,%d:%d,%d]\n"), this, (int)start, (int)stop, (int)startY, (int)stopY); | ||||||
|       // allocate render buffer (always entire segment) |       // allocate render buffer (always entire segment), prefer PSRAM if DRAM is running low. Note: impact on FPS with PSRAM buffer is low (<2% with QSPI PSRAM) | ||||||
|       pixels = static_cast<uint32_t*>(d_calloc(sizeof(uint32_t), length())); // error handling is also done in isActive() |       pixels = static_cast<uint32_t*>(allocate_buffer(length() * sizeof(uint32_t), BFRALLOC_PREFER_PSRAM | BFRALLOC_NOBYTEACCESS | BFRALLOC_CLEAR)); | ||||||
|       if (!pixels) { |       if (!pixels) { | ||||||
|         DEBUGFX_PRINTLN(F("!!! Not enough RAM for pixel buffer !!!")); |         DEBUGFX_PRINTLN(F("!!! Not enough RAM for pixel buffer !!!")); | ||||||
|         extern byte errorFlag; |         extern byte errorFlag; | ||||||
| @@ -623,7 +625,7 @@ class Segment { | |||||||
|       #endif |       #endif | ||||||
|       clearName(); |       clearName(); | ||||||
|       deallocateData(); |       deallocateData(); | ||||||
|       d_free(pixels); |       p_free(pixels); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     Segment& operator= (const Segment &orig); // copy assignment |     Segment& operator= (const Segment &orig); // copy assignment | ||||||
| @@ -646,7 +648,7 @@ class Segment { | |||||||
|     inline uint16_t groupLength()          const { return grouping + spacing; } |     inline uint16_t groupLength()          const { return grouping + spacing; } | ||||||
|     inline uint8_t  getLightCapabilities() const { return _capabilities; } |     inline uint8_t  getLightCapabilities() const { return _capabilities; } | ||||||
|     inline void     deactivate()                 { setGeometry(0,0); } |     inline void     deactivate()                 { setGeometry(0,0); } | ||||||
|     inline Segment &clearName()                  { d_free(name); name = nullptr; return *this; } |     inline Segment &clearName()                  { p_free(name); name = nullptr; return *this; } | ||||||
|     inline Segment &setName(const String &name)  { return setName(name.c_str()); } |     inline Segment &setName(const String &name)  { return setName(name.c_str()); } | ||||||
|  |  | ||||||
|     inline static unsigned vLength()                       { return Segment::_vLength; } |     inline static unsigned vLength()                       { return Segment::_vLength; } | ||||||
| @@ -672,6 +674,7 @@ class Segment { | |||||||
|     inline uint16_t dataSize() const { return _dataLen; } |     inline uint16_t dataSize() const { return _dataLen; } | ||||||
|     bool allocateData(size_t len);  // allocates effect data buffer in heap and clears it |     bool allocateData(size_t len);  // allocates effect data buffer in heap and clears it | ||||||
|     void deallocateData();          // deallocates (frees) effect data buffer from heap |     void deallocateData();          // deallocates (frees) effect data buffer from heap | ||||||
|  |     inline static unsigned getUsedSegmentData()            { return Segment::_usedSegmentData; } | ||||||
|     /** |     /** | ||||||
|       * Flags that before the next effect is calculated, |       * Flags that before the next effect is calculated, | ||||||
|       * the internal segment state should be reset. |       * the internal segment state should be reset. | ||||||
| @@ -868,8 +871,8 @@ class WS2812FX { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     ~WS2812FX() { |     ~WS2812FX() { | ||||||
|       d_free(_pixels); |       p_free(_pixels); | ||||||
|       d_free(_pixelCCT); // just in case |       p_free(_pixelCCT); // just in case | ||||||
|       d_free(customMappingTable); |       d_free(customMappingTable); | ||||||
|       _mode.clear(); |       _mode.clear(); | ||||||
|       _modeData.clear(); |       _modeData.clear(); | ||||||
|   | |||||||
							
								
								
									
										79
									
								
								wled00/FX_fcn.cpp
									
									
									
									
									
										
										
										Normal file → Executable file
									
								
							
							
						
						
									
										79
									
								
								wled00/FX_fcn.cpp
									
									
									
									
									
										
										
										Normal file → Executable file
									
								
							| @@ -68,10 +68,10 @@ Segment::Segment(const Segment &orig) { | |||||||
|   if (!stop) return;  // nothing to do if segment is inactive/invalid |   if (!stop) return;  // nothing to do if segment is inactive/invalid | ||||||
|   if (orig.pixels) { |   if (orig.pixels) { | ||||||
|     // allocate pixel buffer: prefer IRAM/PSRAM |     // allocate pixel buffer: prefer IRAM/PSRAM | ||||||
|     pixels = static_cast<uint32_t*>(d_malloc(sizeof(uint32_t) * orig.length())); |     pixels = static_cast<uint32_t*>(allocate_buffer(orig.length() * sizeof(uint32_t), BFRALLOC_PREFER_PSRAM | BFRALLOC_NOBYTEACCESS)); | ||||||
|     if (pixels) { |     if (pixels) { | ||||||
|       memcpy(pixels, orig.pixels, sizeof(uint32_t) * orig.length()); |       memcpy(pixels, orig.pixels, sizeof(uint32_t) * orig.length()); | ||||||
|       if (orig.name) { name = static_cast<char*>(d_malloc(strlen(orig.name)+1)); if (name) strcpy(name, orig.name); } |       if (orig.name) { name = static_cast<char*>(allocate_buffer(strlen(orig.name)+1, BFRALLOC_PREFER_PSRAM)); if (name) strcpy(name, orig.name); } | ||||||
|       if (orig.data) { if (allocateData(orig._dataLen)) memcpy(data, orig.data, orig._dataLen); } |       if (orig.data) { if (allocateData(orig._dataLen)) memcpy(data, orig.data, orig._dataLen); } | ||||||
|     } else { |     } else { | ||||||
|       DEBUGFX_PRINTLN(F("!!! Not enough RAM for pixel buffer !!!")); |       DEBUGFX_PRINTLN(F("!!! Not enough RAM for pixel buffer !!!")); | ||||||
| @@ -97,10 +97,10 @@ Segment& Segment::operator= (const Segment &orig) { | |||||||
|   //DEBUG_PRINTF_P(PSTR("-- Copying segment: %p -> %p\n"), &orig, this); |   //DEBUG_PRINTF_P(PSTR("-- Copying segment: %p -> %p\n"), &orig, this); | ||||||
|   if (this != &orig) { |   if (this != &orig) { | ||||||
|     // clean destination |     // clean destination | ||||||
|     if (name) { d_free(name); name = nullptr; } |     if (name) { p_free(name); name = nullptr; } | ||||||
|     if (_t) stopTransition(); // also erases _t |     if (_t) stopTransition(); // also erases _t | ||||||
|     deallocateData(); |     deallocateData(); | ||||||
|     d_free(pixels); |     p_free(pixels); | ||||||
|     // copy source |     // copy source | ||||||
|     memcpy((void*)this, (void*)&orig, sizeof(Segment)); |     memcpy((void*)this, (void*)&orig, sizeof(Segment)); | ||||||
|     // erase pointers to allocated data |     // erase pointers to allocated data | ||||||
| @@ -111,10 +111,10 @@ Segment& Segment::operator= (const Segment &orig) { | |||||||
|     // copy source data |     // copy source data | ||||||
|     if (orig.pixels) { |     if (orig.pixels) { | ||||||
|       // allocate pixel buffer: prefer IRAM/PSRAM |       // allocate pixel buffer: prefer IRAM/PSRAM | ||||||
|       pixels = static_cast<uint32_t*>(d_malloc(sizeof(uint32_t) * orig.length())); |       pixels = static_cast<uint32_t*>(allocate_buffer(orig.length() * sizeof(uint32_t), BFRALLOC_PREFER_PSRAM | BFRALLOC_NOBYTEACCESS)); | ||||||
|       if (pixels) { |       if (pixels) { | ||||||
|         memcpy(pixels, orig.pixels, sizeof(uint32_t) * orig.length()); |         memcpy(pixels, orig.pixels, sizeof(uint32_t) * orig.length()); | ||||||
|         if (orig.name) { name = static_cast<char*>(d_malloc(strlen(orig.name)+1)); if (name) strcpy(name, orig.name); } |         if (orig.name) { name = static_cast<char*>(allocate_buffer(strlen(orig.name)+1, BFRALLOC_PREFER_PSRAM)); if (name) strcpy(name, orig.name); } | ||||||
|         if (orig.data) { if (allocateData(orig._dataLen)) memcpy(data, orig.data, orig._dataLen); } |         if (orig.data) { if (allocateData(orig._dataLen)) memcpy(data, orig.data, orig._dataLen); } | ||||||
|       } else { |       } else { | ||||||
|         DEBUG_PRINTLN(F("!!! Not enough RAM for pixel buffer !!!")); |         DEBUG_PRINTLN(F("!!! Not enough RAM for pixel buffer !!!")); | ||||||
| @@ -130,10 +130,10 @@ Segment& Segment::operator= (const Segment &orig) { | |||||||
| Segment& Segment::operator= (Segment &&orig) noexcept { | Segment& Segment::operator= (Segment &&orig) noexcept { | ||||||
|   //DEBUG_PRINTF_P(PSTR("-- Moving segment: %p -> %p\n"), &orig, this); |   //DEBUG_PRINTF_P(PSTR("-- Moving segment: %p -> %p\n"), &orig, this); | ||||||
|   if (this != &orig) { |   if (this != &orig) { | ||||||
|     if (name) { d_free(name); name = nullptr; } // free old name |     if (name) { p_free(name); name = nullptr; } // free old name | ||||||
|     if (_t) stopTransition(); // also erases _t |     if (_t) stopTransition(); // also erases _t | ||||||
|     deallocateData(); // free old runtime data |     deallocateData(); // free old runtime data | ||||||
|     d_free(pixels);   // free old pixel buffer |     p_free(pixels);   // free old pixel buffer | ||||||
|     // move source data |     // move source data | ||||||
|     memcpy((void*)this, (void*)&orig, sizeof(Segment)); |     memcpy((void*)this, (void*)&orig, sizeof(Segment)); | ||||||
|     orig.name = nullptr; |     orig.name = nullptr; | ||||||
| @@ -150,32 +150,35 @@ bool Segment::allocateData(size_t len) { | |||||||
|   if (len == 0) return false;    // nothing to do |   if (len == 0) return false;    // nothing to do | ||||||
|   if (data && _dataLen >= len) { // already allocated enough (reduce fragmentation) |   if (data && _dataLen >= len) { // already allocated enough (reduce fragmentation) | ||||||
|     if (call == 0) { |     if (call == 0) { | ||||||
|  |       if (_dataLen < FAIR_DATA_PER_SEG) { // segment data is small | ||||||
|         //DEBUG_PRINTF_P(PSTR("--   Clearing data (%d): %p\n"), len, this); |         //DEBUG_PRINTF_P(PSTR("--   Clearing data (%d): %p\n"), len, this); | ||||||
|         memset(data, 0, len);  // erase buffer if called during effect initialisation |         memset(data, 0, len);  // erase buffer if called during effect initialisation | ||||||
|  |         return true; // no need to reallocate | ||||||
|       } |       } | ||||||
|  |     } | ||||||
|  |     else | ||||||
|       return true; |       return true; | ||||||
|   } |   } | ||||||
|   //DEBUG_PRINTF_P(PSTR("--   Allocating data (%d): %p\n"), len, this); |   //DEBUG_PRINTF_P(PSTR("--   Allocating data (%d): %p\n"), len, this); | ||||||
|  |   // limit to MAX_SEGMENT_DATA if there is no PSRAM, otherwise prefer functionality over speed | ||||||
|  |   #ifndef BOARD_HAS_PSRAM | ||||||
|   if (Segment::getUsedSegmentData() + len - _dataLen > MAX_SEGMENT_DATA) { |   if (Segment::getUsedSegmentData() + len - _dataLen > MAX_SEGMENT_DATA) { | ||||||
|     // not enough memory |     // not enough memory | ||||||
|     DEBUG_PRINTF_P(PSTR("!!! Not enough RAM: %d/%d !!!\n"), len, Segment::getUsedSegmentData()); |     DEBUG_PRINTF_P(PSTR("SegmentData limit reached: %d/%d\n"), len, Segment::getUsedSegmentData()); | ||||||
|     errorFlag = ERR_NORAM; |     errorFlag = ERR_NORAM; | ||||||
|     return false; |     return false; | ||||||
|   } |   } | ||||||
|   // prefer DRAM over SPI RAM on ESP32 since it is slow |   #endif | ||||||
|   if (data) { |  | ||||||
|     data = (byte*)d_realloc_malloc(data, len); // realloc with malloc fallback |  | ||||||
|     if (!data) { |  | ||||||
|       data = nullptr; |  | ||||||
|       Segment::addUsedSegmentData(-_dataLen); // subtract original buffer size |  | ||||||
|       _dataLen = 0;   // reset data length |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
|   else data = (byte*)d_malloc(len); |  | ||||||
|  |  | ||||||
|   if (data) { |   if (data) { | ||||||
|     memset(data, 0, len);  // erase buffer |     d_free(data); // free data and try to allocate again (segment buffer may be blocking contiguous heap) | ||||||
|     Segment::addUsedSegmentData(len - _dataLen); |     Segment::addUsedSegmentData(-_dataLen); // subtract buffer size | ||||||
|  |   } | ||||||
|  |  | ||||||
|  |   data = static_cast<byte*>(allocate_buffer(len, BFRALLOC_PREFER_DRAM | BFRALLOC_CLEAR)); // prefer DRAM over PSRAM for speed | ||||||
|  |  | ||||||
|  |   if (data) { | ||||||
|  |     Segment::addUsedSegmentData(len); | ||||||
|     _dataLen = len; |     _dataLen = len; | ||||||
|     //DEBUG_PRINTF_P(PSTR("---  Allocated data (%p): %d/%d -> %p\n"), this, len, Segment::getUsedSegmentData(), data); |     //DEBUG_PRINTF_P(PSTR("---  Allocated data (%p): %d/%d -> %p\n"), this, len, Segment::getUsedSegmentData(), data); | ||||||
|     return true; |     return true; | ||||||
| @@ -209,7 +212,11 @@ void Segment::deallocateData() { | |||||||
| void Segment::resetIfRequired() { | void Segment::resetIfRequired() { | ||||||
|   if (!reset || !isActive()) return; |   if (!reset || !isActive()) return; | ||||||
|   //DEBUG_PRINTF_P(PSTR("-- Segment reset: %p\n"), this); |   //DEBUG_PRINTF_P(PSTR("-- Segment reset: %p\n"), this); | ||||||
|   if (data && _dataLen > 0) memset(data, 0, _dataLen);  // prevent heap fragmentation (just erase buffer instead of deallocateData()) |   if (data && _dataLen > 0) { | ||||||
|  |     if (_dataLen > FAIR_DATA_PER_SEG) deallocateData(); // do not keep large allocations | ||||||
|  |     else memset(data, 0, _dataLen);  // can prevent heap fragmentation | ||||||
|  |     DEBUG_PRINTF_P(PSTR("-- Segment %p reset, data cleared\n"), this); | ||||||
|  |   } | ||||||
|   if (pixels) for (size_t i = 0; i < length(); i++) pixels[i] = BLACK; // clear pixel buffer |   if (pixels) for (size_t i = 0; i < length(); i++) pixels[i] = BLACK; // clear pixel buffer | ||||||
|   next_time = 0; step = 0; call = 0; aux0 = 0; aux1 = 0; |   next_time = 0; step = 0; call = 0; aux0 = 0; aux1 = 0; | ||||||
|   reset = false; |   reset = false; | ||||||
| @@ -466,7 +473,7 @@ void Segment::setGeometry(uint16_t i1, uint16_t i2, uint8_t grp, uint8_t spc, ui | |||||||
|   if (length() != oldLength) { |   if (length() != oldLength) { | ||||||
|     // allocate render buffer (always entire segment), prefer IRAM/PSRAM. Note: impact on FPS with PSRAM buffer is low (<2% with QSPI PSRAM) on S2/S3 |     // allocate render buffer (always entire segment), prefer IRAM/PSRAM. Note: impact on FPS with PSRAM buffer is low (<2% with QSPI PSRAM) on S2/S3 | ||||||
|     p_free(pixels); |     p_free(pixels); | ||||||
|     pixels = static_cast<uint32_t*>(d_malloc(sizeof(uint32_t) * length())); |     pixels = static_cast<uint32_t*>(allocate_buffer(length() * sizeof(uint32_t), BFRALLOC_PREFER_PSRAM | BFRALLOC_NOBYTEACCESS)); | ||||||
|     if (!pixels) { |     if (!pixels) { | ||||||
|       DEBUGFX_PRINTLN(F("!!! Not enough RAM for pixel buffer !!!")); |       DEBUGFX_PRINTLN(F("!!! Not enough RAM for pixel buffer !!!")); | ||||||
|       deallocateData(); |       deallocateData(); | ||||||
| @@ -581,8 +588,8 @@ Segment &Segment::setName(const char *newName) { | |||||||
|   if (newName) { |   if (newName) { | ||||||
|     const int newLen = min(strlen(newName), (size_t)WLED_MAX_SEGNAME_LEN); |     const int newLen = min(strlen(newName), (size_t)WLED_MAX_SEGNAME_LEN); | ||||||
|     if (newLen) { |     if (newLen) { | ||||||
|       if (name) d_free(name); // free old name |       if (name) p_free(name); // free old name | ||||||
|       name = static_cast<char*>(d_malloc(newLen+1)); |       name = static_cast<char*>(allocate_buffer(newLen+1, BFRALLOC_PREFER_PSRAM)); | ||||||
|       if (mode == FX_MODE_2DSCROLLTEXT) startTransition(strip.getTransition(), true); // if the name changes in scrolling text mode, we need to copy the segment for blending |       if (mode == FX_MODE_2DSCROLLTEXT) startTransition(strip.getTransition(), true); // if the name changes in scrolling text mode, we need to copy the segment for blending | ||||||
|       if (name) strlcpy(name, newName, newLen+1); |       if (name) strlcpy(name, newName, newLen+1); | ||||||
|       return *this; |       return *this; | ||||||
| @@ -1177,7 +1184,10 @@ void WS2812FX::finalizeInit() { | |||||||
|     mem += bus.memUsage(Bus::isDigital(bus.type) && !Bus::is2Pin(bus.type) ? digitalCount++ : 0); // includes global buffer |     mem += bus.memUsage(Bus::isDigital(bus.type) && !Bus::is2Pin(bus.type) ? digitalCount++ : 0); // includes global buffer | ||||||
|     if (mem <= MAX_LED_MEMORY) { |     if (mem <= MAX_LED_MEMORY) { | ||||||
|       if (BusManager::add(bus) == -1) break; |       if (BusManager::add(bus) == -1) break; | ||||||
|     } else DEBUG_PRINTF_P(PSTR("Out of LED memory! Bus %d (%d) #%u not created."), (int)bus.type, (int)bus.count, digitalCount); |     } else { | ||||||
|  |       errorFlag = ERR_NORAM_PX; // alert UI | ||||||
|  |       DEBUG_PRINTF_P(PSTR("Out of LED memory! Bus %d (%d) #%u not created."), (int)bus.type, (int)bus.count, digitalCount); | ||||||
|  |     } | ||||||
|   } |   } | ||||||
|   busConfigs.clear(); |   busConfigs.clear(); | ||||||
|   busConfigs.shrink_to_fit(); |   busConfigs.shrink_to_fit(); | ||||||
| @@ -1209,10 +1219,11 @@ void WS2812FX::finalizeInit() { | |||||||
|   deserializeMap();     // (re)load default ledmap (will also setUpMatrix() if ledmap does not exist) |   deserializeMap();     // (re)load default ledmap (will also setUpMatrix() if ledmap does not exist) | ||||||
|  |  | ||||||
|   // allocate frame buffer after matrix has been set up (gaps!) |   // allocate frame buffer after matrix has been set up (gaps!) | ||||||
|   d_free(_pixels); // using realloc on large buffers can cause additional fragmentation instead of reducing it |   p_free(_pixels); // using realloc on large buffers can cause additional fragmentation instead of reducing it | ||||||
|   _pixels = static_cast<uint32_t*>(d_malloc(getLengthTotal() * sizeof(uint32_t))); |   // use PSRAM if available: there is no measurable perfomance impact between PSRAM and DRAM on S2/S3 with QSPI PSRAM for this buffer | ||||||
|  |   _pixels = static_cast<uint32_t*>(allocate_buffer(getLengthTotal() * sizeof(uint32_t), BFRALLOC_ENFORCE_PSRAM | BFRALLOC_NOBYTEACCESS | BFRALLOC_CLEAR)); | ||||||
|   DEBUG_PRINTF_P(PSTR("strip buffer size: %uB\n"), getLengthTotal() * sizeof(uint32_t)); |   DEBUG_PRINTF_P(PSTR("strip buffer size: %uB\n"), getLengthTotal() * sizeof(uint32_t)); | ||||||
|   DEBUG_PRINTF_P(PSTR("Heap after strip init: %uB\n"), ESP.getFreeHeap()); |   DEBUG_PRINTF_P(PSTR("Heap after strip init: %uB\n"), getFreeHeapSize()); | ||||||
| } | } | ||||||
|  |  | ||||||
| void WS2812FX::service() { | void WS2812FX::service() { | ||||||
| @@ -1552,7 +1563,11 @@ void WS2812FX::blendSegment(const Segment &topSegment) const { | |||||||
| } | } | ||||||
|  |  | ||||||
| void WS2812FX::show() { | void WS2812FX::show() { | ||||||
|   if (!_pixels) return; // no pixels allocated, nothing to show |   if (!_pixels) { | ||||||
|  |     DEBUGFX_PRINTLN(F("Error: no _pixels!")); | ||||||
|  |     errorFlag = ERR_NORAM; | ||||||
|  |     return; // no pixels allocated, nothing to show | ||||||
|  |   } | ||||||
|  |  | ||||||
|   unsigned long showNow = millis(); |   unsigned long showNow = millis(); | ||||||
|   size_t diff = showNow - _lastShow; |   size_t diff = showNow - _lastShow; | ||||||
| @@ -1562,7 +1577,7 @@ void WS2812FX::show() { | |||||||
|   // we need to keep track of each pixel's CCT when blending segments (if CCT is present) |   // we need to keep track of each pixel's CCT when blending segments (if CCT is present) | ||||||
|   // and then set appropriate CCT from that pixel during paint (see below). |   // and then set appropriate CCT from that pixel during paint (see below). | ||||||
|   if ((hasCCTBus() || correctWB) && !cctFromRgb) |   if ((hasCCTBus() || correctWB) && !cctFromRgb) | ||||||
|     _pixelCCT = static_cast<uint8_t*>(d_malloc(totalLen * sizeof(uint8_t))); // allocate CCT buffer if necessary |     _pixelCCT = static_cast<uint8_t*>(allocate_buffer(totalLen * sizeof(uint8_t), BFRALLOC_PREFER_PSRAM)); // allocate CCT buffer if necessary, prefer PSRAM | ||||||
|   if (_pixelCCT) memset(_pixelCCT, 127, totalLen); // set neutral (50:50) CCT |   if (_pixelCCT) memset(_pixelCCT, 127, totalLen); // set neutral (50:50) CCT | ||||||
|  |  | ||||||
|   if (realtimeMode == REALTIME_MODE_INACTIVE || useMainSegmentOnly || realtimeOverride > REALTIME_OVERRIDE_NONE) { |   if (realtimeMode == REALTIME_MODE_INACTIVE || useMainSegmentOnly || realtimeOverride > REALTIME_OVERRIDE_NONE) { | ||||||
| @@ -1596,7 +1611,7 @@ void WS2812FX::show() { | |||||||
|   } |   } | ||||||
|   Bus::setCCT(oldCCT);  // restore old CCT for ABL adjustments |   Bus::setCCT(oldCCT);  // restore old CCT for ABL adjustments | ||||||
|  |  | ||||||
|   d_free(_pixelCCT); |   p_free(_pixelCCT); | ||||||
|   _pixelCCT = nullptr; |   _pixelCCT = nullptr; | ||||||
|  |  | ||||||
|   // some buses send asynchronously and this method will return before |   // some buses send asynchronously and this method will return before | ||||||
|   | |||||||
| @@ -39,35 +39,29 @@ uint32_t colorBalanceFromKelvin(uint16_t kelvin, uint32_t rgb); | |||||||
| uint8_t realtimeBroadcast(uint8_t type, IPAddress client, uint16_t length, const byte *buffer, uint8_t bri=255, bool isRGBW=false); | uint8_t realtimeBroadcast(uint8_t type, IPAddress client, uint16_t length, const byte *buffer, uint8_t bri=255, bool isRGBW=false); | ||||||
|  |  | ||||||
| //util.cpp | //util.cpp | ||||||
| // PSRAM allocation wrappers | // memory allocation wrappers | ||||||
| #if !defined(ESP8266) && !defined(CONFIG_IDF_TARGET_ESP32C3) |  | ||||||
| extern "C" { | extern "C" { | ||||||
|   void *p_malloc(size_t);           // prefer PSRAM over DRAM |   // prefer DRAM over PSRAM (if available) in d_ alloc functions | ||||||
|   void *p_calloc(size_t, size_t);   // prefer PSRAM over DRAM |   void *d_malloc(size_t); | ||||||
|   void *p_realloc(void *, size_t);  // prefer PSRAM over DRAM |   void *d_calloc(size_t, size_t); | ||||||
|   void *p_realloc_malloc(void *ptr, size_t size); // realloc with malloc fallback, prefer PSRAM over DRAM |   void *d_realloc_malloc(void *ptr, size_t size); | ||||||
|   inline void p_free(void *ptr) { heap_caps_free(ptr); } |   #ifndef ESP8266 | ||||||
|   void *d_malloc(size_t);           // prefer DRAM over PSRAM |  | ||||||
|   void *d_calloc(size_t, size_t);   // prefer DRAM over PSRAM |  | ||||||
|   void *d_realloc(void *, size_t);  // prefer DRAM over PSRAM |  | ||||||
|   void *d_realloc_malloc(void *ptr, size_t size); // realloc with malloc fallback, prefer DRAM over PSRAM |  | ||||||
|   inline void d_free(void *ptr) { heap_caps_free(ptr); } |   inline void d_free(void *ptr) { heap_caps_free(ptr); } | ||||||
| } |  | ||||||
|   #else |   #else | ||||||
| extern "C" { |   inline void d_free(void *ptr) { free(ptr); } | ||||||
|   void *realloc_malloc(void *ptr, size_t size); |  | ||||||
| } |  | ||||||
| #define p_malloc malloc |  | ||||||
| #define p_calloc calloc |  | ||||||
| #define p_realloc realloc |  | ||||||
| #define p_realloc_malloc realloc_malloc |  | ||||||
| #define p_free free |  | ||||||
| #define d_malloc malloc |  | ||||||
| #define d_calloc calloc |  | ||||||
| #define d_realloc realloc |  | ||||||
| #define d_realloc_malloc realloc_malloc |  | ||||||
| #define d_free free |  | ||||||
|   #endif |   #endif | ||||||
|  |   #if defined(BOARD_HAS_PSRAM) | ||||||
|  |   // prefer PSRAM over DRAM in p_ alloc functions | ||||||
|  |   void *p_malloc(size_t); | ||||||
|  |   void *p_calloc(size_t, size_t); | ||||||
|  |   void *p_realloc_malloc(void *ptr, size_t size); | ||||||
|  |   inline void p_free(void *ptr) { heap_caps_free(ptr); } | ||||||
|  |   #else | ||||||
|  |   #define p_malloc d_malloc | ||||||
|  |   #define p_calloc d_calloc | ||||||
|  |   #define p_free d_free | ||||||
|  |   #endif | ||||||
|  | } | ||||||
|  |  | ||||||
| //color mangling macros | //color mangling macros | ||||||
| #define RGBW32(r,g,b,w) (uint32_t((byte(w) << 24) | (byte(r) << 16) | (byte(g) << 8) | (byte(b)))) | #define RGBW32(r,g,b,w) (uint32_t((byte(w) << 24) | (byte(r) << 16) | (byte(g) << 8) | (byte(b)))) | ||||||
| @@ -902,7 +896,7 @@ void BusManager::esp32RMTInvertIdle() { | |||||||
|     else if (lvl == RMT_IDLE_LEVEL_LOW) lvl = RMT_IDLE_LEVEL_HIGH; |     else if (lvl == RMT_IDLE_LEVEL_LOW) lvl = RMT_IDLE_LEVEL_HIGH; | ||||||
|     else continue; |     else continue; | ||||||
|     rmt_set_idle_level(ch, idle_out, lvl); |     rmt_set_idle_level(ch, idle_out, lvl); | ||||||
|     u++ |     u++; | ||||||
|   } |   } | ||||||
| } | } | ||||||
| #endif | #endif | ||||||
|   | |||||||
| @@ -201,7 +201,7 @@ bool deserializeConfig(JsonObject doc, bool fromFS) { | |||||||
|   } |   } | ||||||
|   #endif |   #endif | ||||||
|  |  | ||||||
|   DEBUG_PRINTF_P(PSTR("Heap before buses: %d\n"), ESP.getFreeHeap()); |   DEBUG_PRINTF_P(PSTR("Heap before buses: %d\n"), getFreeHeapSize()); | ||||||
|   JsonArray ins = hw_led["ins"]; |   JsonArray ins = hw_led["ins"]; | ||||||
|   if (!ins.isNull()) { |   if (!ins.isNull()) { | ||||||
|     int s = 0;  // bus iterator |     int s = 0;  // bus iterator | ||||||
|   | |||||||
| @@ -546,8 +546,21 @@ static_assert(WLED_MAX_BUSSES <= 32, "WLED_MAX_BUSSES exceeds hard limit"); | |||||||
|   #endif |   #endif | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
| // minimum heap size required to process web requests | // minimum heap size required to process web requests: try to keep free heap above this value | ||||||
| #define MIN_HEAP_SIZE 8192 | #ifdef ESP8266 | ||||||
|  |   #define MIN_HEAP_SIZE (9*1024) | ||||||
|  | #else | ||||||
|  |   #define MIN_HEAP_SIZE (15*1024) // WLED allocation functions (util.cpp) try to keep this much contiguous heap free for other tasks | ||||||
|  | #endif | ||||||
|  | // threshold for PSRAM use: if heap is running low, requests to allocate_buffer(prefer DRAM) above PSRAM_THRESHOLD may be put in PSRAM | ||||||
|  | // if heap is depleted, PSRAM will be used regardless of threshold | ||||||
|  | #if defined(CONFIG_IDF_TARGET_ESP32S3) | ||||||
|  |   #define PSRAM_THRESHOLD (12*1024) // S3 has plenty of DRAM | ||||||
|  | #elif defined(CONFIG_IDF_TARGET_ESP32) | ||||||
|  |   #define PSRAM_THRESHOLD (5*1024) | ||||||
|  | #else | ||||||
|  |   #define PSRAM_THRESHOLD (2*1024) // S2 does not have a lot of RAM. C3 and ESP8266 do not support PSRAM: the value is not used | ||||||
|  | #endif | ||||||
|  |  | ||||||
| // Web server limits | // Web server limits | ||||||
| #ifdef ESP8266 | #ifdef ESP8266 | ||||||
|   | |||||||
| @@ -195,7 +195,6 @@ | |||||||
| 			if (isAna(t)) return 5;	// analog | 			if (isAna(t)) return 5;	// analog | ||||||
| 			let len = parseInt(d.getElementsByName("LC"+n)[0].value); | 			let len = parseInt(d.getElementsByName("LC"+n)[0].value); | ||||||
| 			len += parseInt(d.getElementsByName("SL"+n)[0].value); // skipped LEDs are allocated too | 			len += parseInt(d.getElementsByName("SL"+n)[0].value); // skipped LEDs are allocated too | ||||||
| 			let dbl = 0; |  | ||||||
| 			let ch = 3*hasRGB(t) + hasW(t) + hasCCT(t); | 			let ch = 3*hasRGB(t) + hasW(t) + hasCCT(t); | ||||||
| 			let mul = 1; | 			let mul = 1; | ||||||
| 			if (isDig(t)) { | 			if (isDig(t)) { | ||||||
| @@ -207,7 +206,7 @@ | |||||||
| 					mul = 2; | 					mul = 2; | ||||||
| 				} | 				} | ||||||
| 			} | 			} | ||||||
| 			return len * ch * mul + dbl; | 			return len * ch * mul + len * 4; // add 4 bytes per LED for segment buffer (TODO: how to account for global buffer?) | ||||||
| 		} | 		} | ||||||
|  |  | ||||||
| 		function UI(change=false) | 		function UI(change=false) | ||||||
|   | |||||||
| @@ -434,35 +434,44 @@ inline uint8_t hw_random8() { return HW_RND_REGISTER; }; | |||||||
| inline uint8_t hw_random8(uint32_t upperlimit) { return (hw_random8() * upperlimit) >> 8; }; // input range 0-255 | inline uint8_t hw_random8(uint32_t upperlimit) { return (hw_random8() * upperlimit) >> 8; }; // input range 0-255 | ||||||
| inline uint8_t hw_random8(uint32_t lowerlimit, uint32_t upperlimit) { uint32_t range = upperlimit - lowerlimit; return lowerlimit + hw_random8(range); }; // input range 0-255 | inline uint8_t hw_random8(uint32_t lowerlimit, uint32_t upperlimit) { uint32_t range = upperlimit - lowerlimit; return lowerlimit + hw_random8(range); }; // input range 0-255 | ||||||
|  |  | ||||||
| // PSRAM allocation wrappers | // memory allocation wrappers (util.cpp) | ||||||
| #if !defined(ESP8266) && !defined(CONFIG_IDF_TARGET_ESP32C3) |  | ||||||
| extern "C" { | extern "C" { | ||||||
|   void *p_malloc(size_t);           // prefer PSRAM over DRAM |   // prefer DRAM in d_xalloc functions, PSRAM as fallback | ||||||
|   void *p_calloc(size_t, size_t);   // prefer PSRAM over DRAM |   void *d_malloc(size_t); | ||||||
|   void *p_realloc(void *, size_t);  // prefer PSRAM over DRAM |   void *d_calloc(size_t, size_t); | ||||||
|   void *p_realloc_malloc(void *ptr, size_t size); // realloc with malloc fallback, prefer PSRAM over DRAM |   void *d_realloc_malloc(void *ptr, size_t size); | ||||||
|   inline void p_free(void *ptr) { heap_caps_free(ptr); } |   #ifndef ESP8266 | ||||||
|   void *d_malloc(size_t);           // prefer DRAM over PSRAM |  | ||||||
|   void *d_calloc(size_t, size_t);   // prefer DRAM over PSRAM |  | ||||||
|   void *d_realloc(void *, size_t);  // prefer DRAM over PSRAM |  | ||||||
|   void *d_realloc_malloc(void *ptr, size_t size); // realloc with malloc fallback, prefer DRAM over PSRAM |  | ||||||
|   inline void d_free(void *ptr) { heap_caps_free(ptr); } |   inline void d_free(void *ptr) { heap_caps_free(ptr); } | ||||||
| } |  | ||||||
|   #else |   #else | ||||||
| extern "C" { |   inline void d_free(void *ptr) { free(ptr); } | ||||||
|   void *realloc_malloc(void *ptr, size_t size); |  | ||||||
| } |  | ||||||
| #define p_malloc malloc |  | ||||||
| #define p_calloc calloc |  | ||||||
| #define p_realloc realloc |  | ||||||
| #define p_realloc_malloc realloc_malloc |  | ||||||
| #define p_free free |  | ||||||
| #define d_malloc malloc |  | ||||||
| #define d_calloc calloc |  | ||||||
| #define d_realloc realloc |  | ||||||
| #define d_realloc_malloc realloc_malloc |  | ||||||
| #define d_free free |  | ||||||
|   #endif |   #endif | ||||||
|  |   #if defined(BOARD_HAS_PSRAM) | ||||||
|  |   // prefer PSRAM in p_xalloc functions, DRAM as fallback | ||||||
|  |   void *p_malloc(size_t); | ||||||
|  |   void *p_calloc(size_t, size_t); | ||||||
|  |   void *p_realloc_malloc(void *ptr, size_t size); | ||||||
|  |   inline void p_free(void *ptr) { heap_caps_free(ptr); } | ||||||
|  |   #else | ||||||
|  |   #define p_malloc d_malloc | ||||||
|  |   #define p_calloc d_calloc | ||||||
|  |   #define p_realloc_malloc d_realloc_malloc | ||||||
|  |   #define p_free d_free | ||||||
|  |   #endif | ||||||
|  | } | ||||||
|  | #ifndef ESP8266 | ||||||
|  | inline size_t getFreeHeapSize() { return heap_caps_get_free_size(MALLOC_CAP_INTERNAL | MALLOC_CAP_8BIT); } // returns free heap (ESP.getFreeHeap() can include other memory types) | ||||||
|  | inline size_t getContiguousFreeHeap() { return heap_caps_get_largest_free_block(MALLOC_CAP_INTERNAL | MALLOC_CAP_8BIT); } // returns largest contiguous free block | ||||||
|  | #else | ||||||
|  | inline size_t getFreeHeapSize() { return ESP.getFreeHeap(); } // returns free heap | ||||||
|  | inline size_t getContiguousFreeHeap() { return ESP.getMaxFreeBlockSize(); } // returns largest contiguous free block | ||||||
|  | #endif | ||||||
|  | #define BFRALLOC_NOBYTEACCESS    (1 << 0) // ESP32 has 32bit accessible DRAM (usually ~50kB free) that must not be byte-accessed | ||||||
|  | #define BFRALLOC_PREFER_DRAM     (1 << 1) // prefer DRAM over PSRAM | ||||||
|  | #define BFRALLOC_ENFORCE_DRAM    (1 << 2) // use DRAM only, no PSRAM | ||||||
|  | #define BFRALLOC_PREFER_PSRAM    (1 << 3) // prefer PSRAM over DRAM | ||||||
|  | #define BFRALLOC_ENFORCE_PSRAM   (1 << 4) // use PSRAM if available, otherwise uses DRAM | ||||||
|  | #define BFRALLOC_CLEAR           (1 << 5) // clear allocated buffer after allocation | ||||||
|  | void *allocate_buffer(size_t size, uint32_t type); | ||||||
|  |  | ||||||
| void handleBootLoop();   // detect and handle bootloops | void handleBootLoop();   // detect and handle bootloops | ||||||
| #ifndef ESP8266 | #ifndef ESP8266 | ||||||
|   | |||||||
| @@ -422,8 +422,8 @@ bool handleFileRead(AsyncWebServerRequest* request, String path){ | |||||||
|   DEBUGFS_PRINT(F("WS FileRead: ")); DEBUGFS_PRINTLN(path); |   DEBUGFS_PRINT(F("WS FileRead: ")); DEBUGFS_PRINTLN(path); | ||||||
|   if(path.endsWith("/")) path += "index.htm"; |   if(path.endsWith("/")) path += "index.htm"; | ||||||
|   if(path.indexOf(F("sec")) > -1) return false; |   if(path.indexOf(F("sec")) > -1) return false; | ||||||
|   #ifdef ARDUINO_ARCH_ESP32 |   #ifdef BOARD_HAS_PSRAM | ||||||
|   if (psramSafe && psramFound() && path.endsWith(FPSTR(getPresetsFileName()))) { |   if (path.endsWith(FPSTR(getPresetsFileName()))) { | ||||||
|     size_t psize; |     size_t psize; | ||||||
|     const uint8_t *presets = getPresetCache(psize); |     const uint8_t *presets = getPresetCache(psize); | ||||||
|     if (presets) { |     if (presets) { | ||||||
|   | |||||||
| @@ -812,7 +812,7 @@ void serializeInfo(JsonObject root) | |||||||
|   root[F("clock")] = ESP.getCpuFreqMHz(); |   root[F("clock")] = ESP.getCpuFreqMHz(); | ||||||
|   root[F("flash")] = (ESP.getFlashChipSize()/1024)/1024; |   root[F("flash")] = (ESP.getFlashChipSize()/1024)/1024; | ||||||
|   #ifdef WLED_DEBUG |   #ifdef WLED_DEBUG | ||||||
|   root[F("maxalloc")] = ESP.getMaxAllocHeap(); |   root[F("maxalloc")] = getContiguousFreeHeap(); | ||||||
|   root[F("resetReason0")] = (int)rtc_get_reset_reason(0); |   root[F("resetReason0")] = (int)rtc_get_reset_reason(0); | ||||||
|   root[F("resetReason1")] = (int)rtc_get_reset_reason(1); |   root[F("resetReason1")] = (int)rtc_get_reset_reason(1); | ||||||
|   #endif |   #endif | ||||||
| @@ -823,15 +823,15 @@ void serializeInfo(JsonObject root) | |||||||
|   root[F("clock")] = ESP.getCpuFreqMHz(); |   root[F("clock")] = ESP.getCpuFreqMHz(); | ||||||
|   root[F("flash")] = (ESP.getFlashChipSize()/1024)/1024; |   root[F("flash")] = (ESP.getFlashChipSize()/1024)/1024; | ||||||
|   #ifdef WLED_DEBUG |   #ifdef WLED_DEBUG | ||||||
|   root[F("maxalloc")] = ESP.getMaxFreeBlockSize(); |   root[F("maxalloc")] = getContiguousFreeHeap(); | ||||||
|   root[F("resetReason")] = (int)ESP.getResetInfoPtr()->reason; |   root[F("resetReason")] = (int)ESP.getResetInfoPtr()->reason; | ||||||
|   #endif |   #endif | ||||||
|   root[F("lwip")] = LWIP_VERSION_MAJOR; |   root[F("lwip")] = LWIP_VERSION_MAJOR; | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|   root[F("freeheap")] = ESP.getFreeHeap(); |   root[F("freeheap")] = getFreeHeapSize(); | ||||||
|   #if defined(ARDUINO_ARCH_ESP32) |   #if defined(BOARD_HAS_PSRAM) | ||||||
|   if (psramFound()) root[F("psram")] = ESP.getFreePsram(); |   root[F("psram")] = ESP.getFreePsram(); | ||||||
|   #endif |   #endif | ||||||
|   root[F("uptime")] = millis()/1000 + rolloverMillis*4294967; |   root[F("uptime")] = millis()/1000 + rolloverMillis*4294967; | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										226
									
								
								wled00/util.cpp
									
									
									
									
									
								
							
							
						
						
									
										226
									
								
								wled00/util.cpp
									
									
									
									
									
								
							| @@ -629,92 +629,186 @@ int32_t hw_random(int32_t lowerlimit, int32_t upperlimit) { | |||||||
|   return hw_random(diff) + lowerlimit; |   return hw_random(diff) + lowerlimit; | ||||||
| } | } | ||||||
|  |  | ||||||
| #if !defined(ESP8266) && !defined(CONFIG_IDF_TARGET_ESP32C3) // ESP8266 does not support PSRAM, ESP32-C3 does not have PSRAM | // PSRAM compile time checks to provide info for misconfigured env | ||||||
| // p_x prefer PSRAM, d_x prefer DRAM | #if defined(BOARD_HAS_PSRAM) | ||||||
| void *p_malloc(size_t size) { |   #if defined(IDF_TARGET_ESP32C3) || defined(ESP8266) | ||||||
|   int caps1 = MALLOC_CAP_SPIRAM  | MALLOC_CAP_8BIT; |     #error "ESP32-C3 and ESP8266 with PSRAM is not supported, please remove BOARD_HAS_PSRAM definition" | ||||||
|   int caps2 = MALLOC_CAP_DEFAULT | MALLOC_CAP_8BIT; |   #else | ||||||
|   if (psramSafe) { |     // BOARD_HAS_PSRAM also means that compiler flag "-mfix-esp32-psram-cache-issue" has to be used | ||||||
|     if (heap_caps_get_free_size(caps2) > 3*MIN_HEAP_SIZE && size < 512) std::swap(caps1, caps2);  // use DRAM for small alloactions & when heap is plenty |     #warning "BOARD_HAS_PSRAM defined, make sure to use -mfix-esp32-psram-cache-issue to prevent issues on rev.1 ESP32 boards \ | ||||||
|     return heap_caps_malloc_prefer(size, 2, caps1, caps2); // otherwise prefer PSRAM if it exists |               see https://docs.espressif.com/projects/esp-idf/en/stable/esp32/api-guides/external-ram.html#esp32-rev-v1-0" | ||||||
|   } |   #endif | ||||||
|   return heap_caps_malloc(size, caps2); | #else | ||||||
| } |   #if !defined(IDF_TARGET_ESP32C3) && !defined(ESP8266) | ||||||
|  |     #pragma message("BOARD_HAS_PSRAM not defined, not using PSRAM.") | ||||||
|  |   #endif | ||||||
|  | #endif | ||||||
|  |  | ||||||
| void *p_realloc(void *ptr, size_t size) { | // memory allocation functions with minimum free heap size check | ||||||
|   int caps1 = MALLOC_CAP_SPIRAM  | MALLOC_CAP_8BIT; | #ifdef ESP8266 | ||||||
|   int caps2 = MALLOC_CAP_DEFAULT | MALLOC_CAP_8BIT; | static void *validateFreeHeap(void *buffer) { | ||||||
|   if (psramSafe) { |   // make sure there is enough free heap left if buffer was allocated in DRAM region, free it if not | ||||||
|     if (heap_caps_get_free_size(caps2) > 3*MIN_HEAP_SIZE && size < 512) std::swap(caps1, caps2);  // use DRAM for small alloactions & when heap is plenty |   if (getContiguousFreeHeap() < MIN_HEAP_SIZE) { | ||||||
|     return heap_caps_realloc_prefer(ptr, size, 2, caps1, caps2); // otherwise prefer PSRAM if it exists |     free(buffer); | ||||||
|  |     return nullptr; | ||||||
|   } |   } | ||||||
|   return heap_caps_realloc(ptr, size, caps2); |   return buffer; | ||||||
| } |  | ||||||
|  |  | ||||||
| // realloc with malloc fallback, original buffer is freed if realloc fails but not copied! |  | ||||||
| void *p_realloc_malloc(void *ptr, size_t size) { |  | ||||||
|   void *newbuf = p_realloc(ptr, size); // try realloc first |  | ||||||
|   if (newbuf) return newbuf; // realloc successful |  | ||||||
|   p_free(ptr); // free old buffer if realloc failed |  | ||||||
|   return p_malloc(size); // fallback to malloc |  | ||||||
| } |  | ||||||
|  |  | ||||||
| void *p_calloc(size_t count, size_t size) { |  | ||||||
|   int caps1 = MALLOC_CAP_SPIRAM  | MALLOC_CAP_8BIT; |  | ||||||
|   int caps2 = MALLOC_CAP_DEFAULT | MALLOC_CAP_8BIT; |  | ||||||
|   if (psramSafe) { |  | ||||||
|     if (heap_caps_get_free_size(caps2) > 3*MIN_HEAP_SIZE && size < 512) std::swap(caps1, caps2);  // use DRAM for small alloactions & when heap is plenty |  | ||||||
|     return heap_caps_calloc_prefer(count, size, 2, caps1, caps2); // otherwise prefer PSRAM if it exists |  | ||||||
|   } |  | ||||||
|   return heap_caps_calloc(count, size, caps2); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| void *d_malloc(size_t size) { | void *d_malloc(size_t size) { | ||||||
|   int caps1 = MALLOC_CAP_DEFAULT | MALLOC_CAP_8BIT; |   // note: using "if (getContiguousFreeHeap() > MIN_HEAP_SIZE + size)" did perform worse in tests with regards to keeping heap healthy and UI working | ||||||
|   int caps2 = MALLOC_CAP_SPIRAM  | MALLOC_CAP_8BIT; |   void *buffer = malloc(size); | ||||||
|   if (psramSafe) { |   return validateFreeHeap(buffer); | ||||||
|     if (heap_caps_get_largest_free_block(caps1) < 3*MIN_HEAP_SIZE && size > MIN_HEAP_SIZE) std::swap(caps1, caps2);  // prefer PSRAM for large alloactions & when DRAM is low |  | ||||||
|     return heap_caps_malloc_prefer(size, 2, caps1, caps2); // otherwise prefer DRAM |  | ||||||
|   } |  | ||||||
|   return heap_caps_malloc(size, caps1); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| void *d_realloc(void *ptr, size_t size) { | void *d_calloc(size_t count, size_t size) { | ||||||
|   int caps1 = MALLOC_CAP_DEFAULT | MALLOC_CAP_8BIT; |   void *buffer = calloc(count, size); | ||||||
|   int caps2 = MALLOC_CAP_SPIRAM  | MALLOC_CAP_8BIT; |   return validateFreeHeap(buffer); | ||||||
|   if (psramSafe) { |  | ||||||
|     if (heap_caps_get_largest_free_block(caps1) < 3*MIN_HEAP_SIZE && size > MIN_HEAP_SIZE) std::swap(caps1, caps2);  // prefer PSRAM for large alloactions & when DRAM is low |  | ||||||
|     return heap_caps_realloc_prefer(ptr, size, 2, caps1, caps2); // otherwise prefer DRAM |  | ||||||
| } | } | ||||||
|   return heap_caps_realloc(ptr, size, caps1); |  | ||||||
|  | // realloc with malloc fallback, note: on ESPS8266 there is no safe way to ensure MIN_HEAP_SIZE during realloc()s, free buffer and allocate new one | ||||||
|  | void *d_realloc_malloc(void *ptr, size_t size) { | ||||||
|  |   //void *buffer = realloc(ptr, size); | ||||||
|  |   //buffer = validateFreeHeap(buffer); | ||||||
|  |   //if (buffer) return buffer; // realloc successful | ||||||
|  |   //d_free(ptr); // free old buffer if realloc failed (or min heap was exceeded) | ||||||
|  |   //return d_malloc(size); // fallback to malloc | ||||||
|  |   free(ptr); | ||||||
|  |   return d_malloc(size); | ||||||
|  | } | ||||||
|  | #else | ||||||
|  | static void *validateFreeHeap(void *buffer) { | ||||||
|  |   // make sure there is enough free heap left if buffer was allocated in DRAM region, free it if not | ||||||
|  |   // TODO: between allocate and free, heap can run low (async web access), only IDF V5 allows for a pre-allocation-check of all free blocks | ||||||
|  |   if ((uintptr_t)buffer > SOC_DRAM_LOW && (uintptr_t)buffer < SOC_DRAM_HIGH && getContiguousFreeHeap() < MIN_HEAP_SIZE) { | ||||||
|  |     free(buffer); | ||||||
|  |     return nullptr; | ||||||
|  |   } | ||||||
|  |   return buffer; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void *d_malloc(size_t size) { | ||||||
|  |   void *buffer; | ||||||
|  |   #if defined(CONFIG_IDF_TARGET_ESP32C3) || defined(CONFIG_IDF_TARGET_ESP32S2) || defined(CONFIG_IDF_TARGET_ESP32S3) | ||||||
|  |   // the newer ESP32 variants have byte-accessible fast RTC memory that can be used as heap, access speed is on-par with DRAM | ||||||
|  |   // the system does prefer normal DRAM until full, since free RTC memory is ~7.5k only, its below the minimum heap threshold and needs to be allocated explicitly | ||||||
|  |   // use RTC RAM for small allocations to improve fragmentation or if DRAM is running low | ||||||
|  |   if (size < 256 || getContiguousFreeHeap() < 2*MIN_HEAP_SIZE + size) | ||||||
|  |     buffer = heap_caps_malloc_prefer(size, 2, MALLOC_CAP_RTCRAM, MALLOC_CAP_INTERNAL | MALLOC_CAP_8BIT); | ||||||
|  |   else | ||||||
|  |   #endif | ||||||
|  |   buffer = heap_caps_malloc(size, MALLOC_CAP_INTERNAL | MALLOC_CAP_8BIT); // allocate in any available heap memory | ||||||
|  |   buffer = validateFreeHeap(buffer); // make sure there is enough free heap left | ||||||
|  |   #ifdef BOARD_HAS_PSRAM | ||||||
|  |   if (!buffer) | ||||||
|  |     return heap_caps_malloc(size, MALLOC_CAP_SPIRAM | MALLOC_CAP_8BIT); // DRAM failed, use PSRAM if available | ||||||
|  |   #endif | ||||||
|  |   return buffer; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | void *d_calloc(size_t count, size_t size) { | ||||||
|  |   void *buffer = d_malloc(count * size); | ||||||
|  |   if (buffer) memset(buffer, 0, count * size); // clear allocated buffer | ||||||
|  |   return buffer; | ||||||
| } | } | ||||||
|  |  | ||||||
| // realloc with malloc fallback, original buffer is freed if realloc fails but not copied! | // realloc with malloc fallback, original buffer is freed if realloc fails but not copied! | ||||||
| void *d_realloc_malloc(void *ptr, size_t size) { | void *d_realloc_malloc(void *ptr, size_t size) { | ||||||
|   void *newbuf = d_realloc(ptr, size); // try realloc first |   void *buffer = heap_caps_realloc(ptr, size, MALLOC_CAP_INTERNAL | MALLOC_CAP_8BIT); | ||||||
|   if (newbuf) return newbuf; // realloc successful |   buffer = validateFreeHeap(buffer); | ||||||
|   d_free(ptr); // free old buffer if realloc failed |   if (buffer) return buffer; // realloc successful | ||||||
|  |   d_free(ptr); // free old buffer if realloc failed (or min heap was exceeded) | ||||||
|   return d_malloc(size); // fallback to malloc |   return d_malloc(size); // fallback to malloc | ||||||
| } | } | ||||||
|  |  | ||||||
| void *d_calloc(size_t count, size_t size) { | #ifdef BOARD_HAS_PSRAM | ||||||
|   int caps1 = MALLOC_CAP_DEFAULT | MALLOC_CAP_8BIT; | // p_xalloc: prefer PSRAM, use DRAM as fallback | ||||||
|   int caps2 = MALLOC_CAP_SPIRAM  | MALLOC_CAP_8BIT; | void *p_malloc(size_t size) { | ||||||
|   if (psramSafe) { |   void *buffer = heap_caps_malloc_prefer(size, 2, MALLOC_CAP_SPIRAM | MALLOC_CAP_8BIT, MALLOC_CAP_INTERNAL | MALLOC_CAP_8BIT); | ||||||
|     if (size > MIN_HEAP_SIZE) std::swap(caps1, caps2);  // prefer PSRAM for large alloactions |   return validateFreeHeap(buffer); | ||||||
|     return heap_caps_calloc_prefer(count, size, 2, caps1, caps2); // otherwise prefer DRAM |  | ||||||
| } | } | ||||||
|   return heap_caps_calloc(count, size, caps1); |  | ||||||
|  | void *p_calloc(size_t count, size_t size) { | ||||||
|  |   void *buffer = p_malloc(count * size); | ||||||
|  |   if (buffer) memset(buffer, 0, count * size); // clear allocated buffer | ||||||
|  |   return buffer; | ||||||
| } | } | ||||||
| #else // ESP8266 & ESP32-C3 |  | ||||||
| // realloc with malloc fallback, original buffer is freed if realloc fails but not copied! | // realloc with malloc fallback, original buffer is freed if realloc fails but not copied! | ||||||
| void *realloc_malloc(void *ptr, size_t size) { | void *p_realloc_malloc(void *ptr, size_t size) { | ||||||
|   void *newbuf = realloc(ptr, size); // try realloc first |   void *buffer = heap_caps_realloc(ptr, size, MALLOC_CAP_SPIRAM | MALLOC_CAP_8BIT); | ||||||
|   if (newbuf) return newbuf; // realloc successful |   if (buffer) return buffer; // realloc successful | ||||||
|   free(ptr); // free old buffer if realloc failed |   p_free(ptr); // free old buffer if realloc failed | ||||||
|   return malloc(size); // fallback to malloc |   return p_malloc(size); // fallback to malloc | ||||||
| } | } | ||||||
| #endif | #endif | ||||||
|  | #endif | ||||||
|  |  | ||||||
|  | // allocation function for buffers like pixel-buffers and segment data | ||||||
|  | // optimises the use of memory types to balance speed and heap availability, always favours DRAM if possible | ||||||
|  | // if multiple conflicting types are defined, the lowest bits of "type" take priority (see fcn_declare.h for types) | ||||||
|  | void *allocate_buffer(size_t size, uint32_t type) { | ||||||
|  |   void *buffer = nullptr; | ||||||
|  |   #ifdef CONFIG_IDF_TARGET_ESP32 | ||||||
|  |   // only classic ESP32 has "32bit accessible only" aka IRAM type. Using it frees up normal DRAM for other purposes | ||||||
|  |   // this memory region is used for IRAM_ATTR functions, whatever is left is unused and can be used for pixel buffers | ||||||
|  |   // prefer this type over PSRAM as it is slightly faster, except for _pixels where it is on-par as PSRAM-caching does a good job for mostly sequential access | ||||||
|  |   if (type & BFRALLOC_NOBYTEACCESS) { | ||||||
|  |     // prefer 32bit region, then PSRAM, fallback to any heap. Note: if adding "INTERNAL"-flag this wont work | ||||||
|  |     buffer = heap_caps_malloc_prefer(size, 3, MALLOC_CAP_32BIT, MALLOC_CAP_SPIRAM, MALLOC_CAP_8BIT); | ||||||
|  |     buffer = validateFreeHeap(buffer); | ||||||
|  |   } | ||||||
|  |   else | ||||||
|  |   #endif | ||||||
|  |   #if !defined(BOARD_HAS_PSRAM) | ||||||
|  |   buffer = d_malloc(size); | ||||||
|  |   #else | ||||||
|  |   if (type & BFRALLOC_PREFER_DRAM) { | ||||||
|  |     if (getContiguousFreeHeap() < 3*(MIN_HEAP_SIZE/2) + size && size > PSRAM_THRESHOLD) | ||||||
|  |       buffer = p_malloc(size); // prefer PSRAM for large allocations & when DRAM is low | ||||||
|  |     else | ||||||
|  |       buffer = d_malloc(size); // allocate in DRAM if enough free heap is available, PSRAM as fallback | ||||||
|  |   } | ||||||
|  |   else if (type & BFRALLOC_ENFORCE_DRAM) | ||||||
|  |     buffer = heap_caps_malloc(size, MALLOC_CAP_INTERNAL | MALLOC_CAP_8BIT); // use DRAM only, otherwise return nullptr | ||||||
|  |   else if (type & BFRALLOC_PREFER_PSRAM) { | ||||||
|  |     // if DRAM is plenty, prefer it over PSRAM for speed, reserve enough DRAM for segment data: if MAX_SEGMENT_DATA is exceeded, always uses PSRAM | ||||||
|  |     if (getContiguousFreeHeap() > 4*MIN_HEAP_SIZE + size + ((uint32_t)(MAX_SEGMENT_DATA - Segment::getUsedSegmentData()))) | ||||||
|  |       buffer = d_malloc(size); | ||||||
|  |     else | ||||||
|  |       buffer = p_malloc(size); // prefer PSRAM | ||||||
|  |   } | ||||||
|  |   else if (type & BFRALLOC_ENFORCE_PSRAM) | ||||||
|  |     buffer = heap_caps_malloc(size, MALLOC_CAP_SPIRAM | MALLOC_CAP_8BIT); // use PSRAM only, otherwise return nullptr | ||||||
|  |   buffer = validateFreeHeap(buffer); | ||||||
|  |   #endif | ||||||
|  |   if (buffer && (type & BFRALLOC_CLEAR)) | ||||||
|  |     memset(buffer, 0, size); // clear allocated buffer | ||||||
|  |   /* | ||||||
|  |   #if !defined(ESP8266) && defined(WLED_DEBUG) | ||||||
|  |   if (buffer) { | ||||||
|  |     DEBUG_PRINTF_P(PSTR("*Buffer allocated: size:%d, address:%p"), size, (uintptr_t)buffer); | ||||||
|  |     if ((uintptr_t)buffer > SOC_DRAM_LOW && (uintptr_t)buffer < SOC_DRAM_HIGH) | ||||||
|  |       DEBUG_PRINTLN(F(" in DRAM")); | ||||||
|  |     #ifndef CONFIG_IDF_TARGET_ESP32C3 | ||||||
|  |     else if ((uintptr_t)buffer > SOC_EXTRAM_DATA_LOW && (uintptr_t)buffer < SOC_EXTRAM_DATA_HIGH) | ||||||
|  |       DEBUG_PRINTLN(F(" in PSRAM")); | ||||||
|  |     #endif | ||||||
|  |     #ifdef CONFIG_IDF_TARGET_ESP32 | ||||||
|  |     else if ((uintptr_t)buffer > SOC_IRAM_LOW && (uintptr_t)buffer < SOC_IRAM_HIGH) | ||||||
|  |       DEBUG_PRINTLN(F(" in IRAM"));   // only used on ESP32 (MALLOC_CAP_32BIT) | ||||||
|  |     #else | ||||||
|  |     else if ((uintptr_t)buffer > SOC_RTC_DRAM_LOW && (uintptr_t)buffer < SOC_RTC_DRAM_HIGH) | ||||||
|  |       DEBUG_PRINTLN(F(" in RTCRAM")); // not available on ESP32 | ||||||
|  |     #endif | ||||||
|  |     else | ||||||
|  |       DEBUG_PRINTLN(F(" in ???")); // unknown (check soc.h for other memory regions) | ||||||
|  |   } else | ||||||
|  |     DEBUG_PRINTF_P(PSTR("Buffer allocation failed: size:%d\n"), size); | ||||||
|  |   #endif  | ||||||
|  |   */ | ||||||
|  |   return buffer; | ||||||
|  | } | ||||||
|  |  | ||||||
| // bootloop detection and handling | // bootloop detection and handling | ||||||
| // checks if the ESP reboots multiple times due to a crash or watchdog timeout | // checks if the ESP reboots multiple times due to a crash or watchdog timeout | ||||||
|   | |||||||
| @@ -171,7 +171,7 @@ void WLED::loop() | |||||||
|  |  | ||||||
|   // reconnect WiFi to clear stale allocations if heap gets too low |   // reconnect WiFi to clear stale allocations if heap gets too low | ||||||
|   if (millis() - heapTime > 15000) { |   if (millis() - heapTime > 15000) { | ||||||
|     uint32_t heap = ESP.getFreeHeap(); |     uint32_t heap = getFreeHeapSize(); | ||||||
|     if (heap < MIN_HEAP_SIZE && lastHeap < MIN_HEAP_SIZE) { |     if (heap < MIN_HEAP_SIZE && lastHeap < MIN_HEAP_SIZE) { | ||||||
|       DEBUG_PRINTF_P(PSTR("Heap too low! %u\n"), heap); |       DEBUG_PRINTF_P(PSTR("Heap too low! %u\n"), heap); | ||||||
|       forceReconnect = true; |       forceReconnect = true; | ||||||
| @@ -241,13 +241,37 @@ void WLED::loop() | |||||||
|     DEBUG_PRINTLN(F("---DEBUG INFO---")); |     DEBUG_PRINTLN(F("---DEBUG INFO---")); | ||||||
|     DEBUG_PRINTF_P(PSTR("Runtime: %lu\n"),  millis()); |     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("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 defined(ARDUINO_ARCH_ESP32) | ||||||
|  |     DEBUG_PRINTLN(F("=== Memory Info ===")); | ||||||
|  |     // Internal DRAM (standard 8-bit accessible heap) | ||||||
|  |     size_t dram_free = heap_caps_get_free_size(MALLOC_CAP_8BIT | MALLOC_CAP_INTERNAL); | ||||||
|  |     size_t dram_largest = heap_caps_get_largest_free_block(MALLOC_CAP_8BIT | MALLOC_CAP_INTERNAL); | ||||||
|  |     DEBUG_PRINTF_P(PSTR("DRAM 8-bit:   Free: %7u bytes | Largest block: %7u bytes\n"), dram_free, dram_largest); | ||||||
|  |     #ifdef BOARD_HAS_PSRAM | ||||||
|  |     size_t psram_free = heap_caps_get_free_size(MALLOC_CAP_SPIRAM); | ||||||
|  |     size_t psram_largest = heap_caps_get_largest_free_block(MALLOC_CAP_SPIRAM); | ||||||
|  |     DEBUG_PRINTF_P(PSTR("PSRAM:        Free: %7u bytes | Largest block: %6u bytes\n"), psram_free, psram_largest); | ||||||
|  |     #endif | ||||||
|  |     #if defined(CONFIG_IDF_TARGET_ESP32) | ||||||
|  |     // 32-bit DRAM (not byte accessible, only available on ESP32) | ||||||
|  |     size_t dram32_free = heap_caps_get_free_size(MALLOC_CAP_32BIT | MALLOC_CAP_INTERNAL) - dram_free; // returns all 32bit DRAM, subtract 8bit DRAM | ||||||
|  |     //size_t dram32_largest = heap_caps_get_largest_free_block(MALLOC_CAP_32BIT | MALLOC_CAP_INTERNAL); // returns largest DRAM block -> not useful | ||||||
|  |     DEBUG_PRINTF_P(PSTR("DRAM 32-bit:  Free: %7u bytes | Largest block: N/A\n"), dram32_free); | ||||||
|  |     #else | ||||||
|  |     // Fast RTC Memory (not available on ESP32) | ||||||
|  |     size_t rtcram_free = heap_caps_get_free_size(MALLOC_CAP_RTCRAM); | ||||||
|  |     size_t rtcram_largest = heap_caps_get_largest_free_block(MALLOC_CAP_RTCRAM); | ||||||
|  |     DEBUG_PRINTF_P(PSTR("RTC RAM:      Free: %7u bytes | Largest block: %7u bytes\n"), rtcram_free, rtcram_largest); | ||||||
|  |     #endif | ||||||
|     if (psramFound()) { |     if (psramFound()) { | ||||||
|       DEBUG_PRINTF_P(PSTR("PSRAM: %dkB/%dkB\n"), ESP.getFreePsram()/1024, ESP.getPsramSize()/1024); |       DEBUG_PRINTF_P(PSTR("PSRAM: %dkB/%dkB\n"), ESP.getFreePsram()/1024, ESP.getPsramSize()/1024); | ||||||
|       if (!psramSafe) DEBUG_PRINTLN(F("Not using PSRAM.")); |     #ifndef BOARD_HAS_PSRAM | ||||||
|  |       DEBUG_PRINTLN(F("BOARD_HAS_PSRAM not defined, not using PSRAM.")); | ||||||
|  |     #endif | ||||||
|     } |     } | ||||||
|     DEBUG_PRINTF_P(PSTR("TX power: %d/%d\n"), WiFi.getTxPower(), txPower); |     DEBUG_PRINTF_P(PSTR("TX power: %d/%d\n"), WiFi.getTxPower(), txPower); | ||||||
|  |     #else // ESP8266 | ||||||
|  |     DEBUG_PRINTF_P(PSTR("Free heap/contiguous: %u/%u\n"), getFreeHeapSize(), getContiguousFreeHeap()); | ||||||
|     #endif |     #endif | ||||||
|     DEBUG_PRINTF_P(PSTR("Wifi state: %d\n"), WiFi.status()); |     DEBUG_PRINTF_P(PSTR("Wifi state: %d\n"), WiFi.status()); | ||||||
|     #ifndef WLED_DISABLE_ESPNOW |     #ifndef WLED_DISABLE_ESPNOW | ||||||
| @@ -367,20 +391,16 @@ void WLED::setup() | |||||||
|   DEBUG_PRINTF_P(PSTR("esp8266 @ %u MHz.\nCore: %s\n"), ESP.getCpuFreqMHz(), ESP.getCoreVersion()); |   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); |   DEBUG_PRINTF_P(PSTR("FLASH: %u MB\n"), (ESP.getFlashChipSize()/1024)/1024); | ||||||
| #endif | #endif | ||||||
|   DEBUG_PRINTF_P(PSTR("heap %u\n"), ESP.getFreeHeap()); |   DEBUG_PRINTF_P(PSTR("heap %u\n"), getFreeHeapSize()); | ||||||
|  |  | ||||||
|  | #if defined(BOARD_HAS_PSRAM) | ||||||
|  |   // if JSON buffer allocation fails requestJsonBufferLock() will always return false preventing crashes | ||||||
|  |   pDoc = new PSRAMDynamicJsonDocument(2 * JSON_BUFFER_SIZE); | ||||||
|  |   DEBUG_PRINTF_P(PSTR("JSON buffer size: %ubytes\n"), (2 * JSON_BUFFER_SIZE)); | ||||||
|  |   DEBUG_PRINTF_P(PSTR("PSRAM: %dkB/%dkB\n"), ESP.getFreePsram()/1024, ESP.getPsramSize()/1024); | ||||||
|  | #endif | ||||||
|  |  | ||||||
| #if defined(ARDUINO_ARCH_ESP32) | #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); |   DEBUG_PRINTF_P(PSTR("TX power: %d/%d\n"), WiFi.getTxPower(), txPower); | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
| @@ -395,7 +415,7 @@ void WLED::setup() | |||||||
|   PinManager::allocatePin(2, true, PinOwner::DMX); |   PinManager::allocatePin(2, true, PinOwner::DMX); | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|   DEBUG_PRINTF_P(PSTR("heap %u\n"), ESP.getFreeHeap()); |   DEBUG_PRINTF_P(PSTR("heap %u\n"), getFreeHeapSize()); | ||||||
|  |  | ||||||
|   bool fsinit = false; |   bool fsinit = false; | ||||||
|   DEBUGFS_PRINTLN(F("Mount FS")); |   DEBUGFS_PRINTLN(F("Mount FS")); | ||||||
| @@ -433,7 +453,7 @@ void WLED::setup() | |||||||
|   } |   } | ||||||
|   DEBUG_PRINTLN(F("Reading config")); |   DEBUG_PRINTLN(F("Reading config")); | ||||||
|   bool needsCfgSave = deserializeConfigFromFS(); |   bool needsCfgSave = deserializeConfigFromFS(); | ||||||
|   DEBUG_PRINTF_P(PSTR("heap %u\n"), ESP.getFreeHeap()); |   DEBUG_PRINTF_P(PSTR("heap %u\n"), getFreeHeapSize()); | ||||||
|  |  | ||||||
| #if defined(STATUSLED) && STATUSLED>=0 | #if defined(STATUSLED) && STATUSLED>=0 | ||||||
|   if (!PinManager::isPinAllocated(STATUSLED)) { |   if (!PinManager::isPinAllocated(STATUSLED)) { | ||||||
| @@ -445,12 +465,12 @@ void WLED::setup() | |||||||
|  |  | ||||||
|   DEBUG_PRINTLN(F("Initializing strip")); |   DEBUG_PRINTLN(F("Initializing strip")); | ||||||
|   beginStrip(); |   beginStrip(); | ||||||
|   DEBUG_PRINTF_P(PSTR("heap %u\n"), ESP.getFreeHeap()); |   DEBUG_PRINTF_P(PSTR("heap %u\n"), getFreeHeapSize()); | ||||||
|  |  | ||||||
|   DEBUG_PRINTLN(F("Usermods setup")); |   DEBUG_PRINTLN(F("Usermods setup")); | ||||||
|   userSetup(); |   userSetup(); | ||||||
|   UsermodManager::setup(); |   UsermodManager::setup(); | ||||||
|   DEBUG_PRINTF_P(PSTR("heap %u\n"), ESP.getFreeHeap()); |   DEBUG_PRINTF_P(PSTR("heap %u\n"), getFreeHeapSize()); | ||||||
|  |  | ||||||
|   if (needsCfgSave) serializeConfigToFS(); // usermods required new parameters; need to wait for strip to be initialised #4752 |   if (needsCfgSave) serializeConfigToFS(); // usermods required new parameters; need to wait for strip to be initialised #4752 | ||||||
|  |  | ||||||
| @@ -515,13 +535,13 @@ void WLED::setup() | |||||||
|   // HTTP server page init |   // HTTP server page init | ||||||
|   DEBUG_PRINTLN(F("initServer")); |   DEBUG_PRINTLN(F("initServer")); | ||||||
|   initServer(); |   initServer(); | ||||||
|   DEBUG_PRINTF_P(PSTR("heap %u\n"), ESP.getFreeHeap()); |   DEBUG_PRINTF_P(PSTR("heap %u\n"), getFreeHeapSize()); | ||||||
|  |  | ||||||
| #ifndef WLED_DISABLE_INFRARED | #ifndef WLED_DISABLE_INFRARED | ||||||
|   // init IR |   // init IR | ||||||
|   DEBUG_PRINTLN(F("initIR")); |   DEBUG_PRINTLN(F("initIR")); | ||||||
|   initIR(); |   initIR(); | ||||||
|   DEBUG_PRINTF_P(PSTR("heap %u\n"), ESP.getFreeHeap()); |   DEBUG_PRINTF_P(PSTR("heap %u\n"), getFreeHeapSize()); | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|   // Seed FastLED random functions with an esp random value, which already works properly at this point. |   // Seed FastLED random functions with an esp random value, which already works properly at this point. | ||||||
|   | |||||||
| @@ -167,16 +167,13 @@ | |||||||
| // The following is a construct to enable code to compile without it. | // The following is a construct to enable code to compile without it. | ||||||
| // There is a code that will still not use PSRAM though: | // There is a code that will still not use PSRAM though: | ||||||
| //    AsyncJsonResponse is a derived class that implements DynamicJsonDocument (AsyncJson-v6.h) | //    AsyncJsonResponse is a derived class that implements DynamicJsonDocument (AsyncJson-v6.h) | ||||||
| #if defined(ARDUINO_ARCH_ESP32) | #if defined(BOARD_HAS_PSRAM) | ||||||
| extern bool psramSafe; |  | ||||||
| struct PSRAM_Allocator { | struct PSRAM_Allocator { | ||||||
|   void* allocate(size_t size) { |   void* allocate(size_t size) { | ||||||
|     if (psramSafe && psramFound()) return ps_malloc(size); // use PSRAM if it exists |     return ps_malloc(size); // use PSRAM | ||||||
|     else                           return malloc(size);    // fallback |  | ||||||
|   } |   } | ||||||
|   void* reallocate(void* ptr, size_t new_size) { |   void* reallocate(void* ptr, size_t new_size) { | ||||||
|     if (psramSafe && psramFound()) return ps_realloc(ptr, new_size); // use PSRAM if it exists |     return ps_realloc(ptr, new_size); // use PSRAM | ||||||
|     else                           return realloc(ptr, new_size);    // fallback |  | ||||||
|   } |   } | ||||||
|   void deallocate(void* pointer) { |   void deallocate(void* pointer) { | ||||||
|     free(pointer); |     free(pointer); | ||||||
| @@ -894,8 +891,6 @@ WLED_GLOBAL byte optionType; | |||||||
| WLED_GLOBAL bool configNeedsWrite  _INIT(false);        // flag to initiate saving of config | WLED_GLOBAL bool configNeedsWrite  _INIT(false);        // flag to initiate saving of config | ||||||
| WLED_GLOBAL bool doReboot          _INIT(false);        // flag to initiate reboot from async handlers | WLED_GLOBAL bool doReboot          _INIT(false);        // flag to initiate reboot from async handlers | ||||||
|  |  | ||||||
| WLED_GLOBAL bool psramSafe         _INIT(true);         // is it safe to use PSRAM (on ESP32 rev.1; compiler fix used "-mfix-esp32-psram-cache-issue") |  | ||||||
|  |  | ||||||
| // status led | // status led | ||||||
| #if defined(STATUSLED) | #if defined(STATUSLED) | ||||||
| WLED_GLOBAL unsigned long ledStatusLastMillis _INIT(0); | WLED_GLOBAL unsigned long ledStatusLastMillis _INIT(0); | ||||||
| @@ -969,8 +964,11 @@ WLED_GLOBAL int8_t spi_sclk  _INIT(SPISCLKPIN); | |||||||
|  |  | ||||||
| // global ArduinoJson buffer | // global ArduinoJson buffer | ||||||
| #if defined(ARDUINO_ARCH_ESP32) | #if defined(ARDUINO_ARCH_ESP32) | ||||||
| WLED_GLOBAL JsonDocument *pDoc _INIT(nullptr); |  | ||||||
| WLED_GLOBAL SemaphoreHandle_t jsonBufferLockMutex _INIT(xSemaphoreCreateRecursiveMutex()); | WLED_GLOBAL SemaphoreHandle_t jsonBufferLockMutex _INIT(xSemaphoreCreateRecursiveMutex()); | ||||||
|  | #endif | ||||||
|  | #ifdef BOARD_HAS_PSRAM | ||||||
|  | // if board has PSRAM, use it for JSON document (allocated in setup()) | ||||||
|  | WLED_GLOBAL JsonDocument *pDoc _INIT(nullptr); | ||||||
| #else | #else | ||||||
| WLED_GLOBAL StaticJsonDocument<JSON_BUFFER_SIZE> gDoc; | WLED_GLOBAL StaticJsonDocument<JSON_BUFFER_SIZE> gDoc; | ||||||
| WLED_GLOBAL JsonDocument *pDoc _INIT(&gDoc); | WLED_GLOBAL JsonDocument *pDoc _INIT(&gDoc); | ||||||
|   | |||||||
| @@ -368,7 +368,7 @@ void initServer() | |||||||
|   }); |   }); | ||||||
|  |  | ||||||
|   server.on(F("/freeheap"), HTTP_GET, [](AsyncWebServerRequest *request){ |   server.on(F("/freeheap"), HTTP_GET, [](AsyncWebServerRequest *request){ | ||||||
|     request->send(200, FPSTR(CONTENT_TYPE_PLAIN), (String)ESP.getFreeHeap()); |     request->send(200, FPSTR(CONTENT_TYPE_PLAIN), (String)getFreeHeapSize()); | ||||||
|   }); |   }); | ||||||
|  |  | ||||||
| #ifdef WLED_ENABLE_USERMOD_PAGE | #ifdef WLED_ENABLE_USERMOD_PAGE | ||||||
|   | |||||||
| @@ -124,8 +124,8 @@ void sendDataWs(AsyncWebSocketClient * client) | |||||||
|   DEBUG_PRINTF_P(PSTR("JSON buffer size: %u for WS request (%u).\n"), pDoc->memoryUsage(), len); |   DEBUG_PRINTF_P(PSTR("JSON buffer size: %u for WS request (%u).\n"), pDoc->memoryUsage(), len); | ||||||
|  |  | ||||||
|   // the following may no longer be necessary as heap management has been fixed by @willmmiles in AWS |   // the following may no longer be necessary as heap management has been fixed by @willmmiles in AWS | ||||||
|   size_t heap1 = ESP.getFreeHeap(); |   size_t heap1 = getFreeHeapSize(); | ||||||
|   DEBUG_PRINTF_P(PSTR("heap %u\n"), ESP.getFreeHeap()); |   DEBUG_PRINTF_P(PSTR("heap %u\n"), getFreeHeapSize()); | ||||||
|   #ifdef ESP8266 |   #ifdef ESP8266 | ||||||
|   if (len>heap1) { |   if (len>heap1) { | ||||||
|     DEBUG_PRINTLN(F("Out of memory (WS)!")); |     DEBUG_PRINTLN(F("Out of memory (WS)!")); | ||||||
| @@ -134,8 +134,8 @@ void sendDataWs(AsyncWebSocketClient * client) | |||||||
|   #endif |   #endif | ||||||
|   AsyncWebSocketBuffer buffer(len); |   AsyncWebSocketBuffer buffer(len); | ||||||
|   #ifdef ESP8266 |   #ifdef ESP8266 | ||||||
|   size_t heap2 = ESP.getFreeHeap(); |   size_t heap2 = getFreeHeapSize(); | ||||||
|   DEBUG_PRINTF_P(PSTR("heap %u\n"), ESP.getFreeHeap()); |   DEBUG_PRINTF_P(PSTR("heap %u\n"), getFreeHeapSize()); | ||||||
|   #else |   #else | ||||||
|   size_t heap2 = 0; // ESP32 variants do not have the same issue and will work without checking heap allocation |   size_t heap2 = 0; // ESP32 variants do not have the same issue and will work without checking heap allocation | ||||||
|   #endif |   #endif | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Damian Schneider
					Damian Schneider