serveLiveLeds: Use dynamic buffer
There were three problems here:
- AsyncWebServer is going to copy to a heap buffer anyways, so we might
   as well just pass it one it can use
- The buffer size estimate was wrong -- we need 9 bytes per pixel
   ("RRGGBB",), so the buffer could overflow, and it was not
   considering the extra 2D requirements
- On ESP8266, the stack allocation was overflowing the stack, causing
  corruption and crashes.
			
			
This commit is contained in:
		| @@ -1152,10 +1152,10 @@ bool serveLiveLeds(AsyncWebServerRequest* request, uint32_t wsClient) | |||||||
|   } |   } | ||||||
| #endif | #endif | ||||||
|  |  | ||||||
|   char buffer[2048];  // shoud be enough for 256 LEDs [RRGGBB] + all other text (9+25) |   DynamicBuffer buffer(9 + (9*MAX_LIVE_LEDS) + 7 + 5 + 6 + 5 + 6 + 5 + 2);   | ||||||
|   strcpy_P(buffer, PSTR("{\"leds\":[")); |   char* buf = buffer.data();      // assign buffer for oappnd() functions | ||||||
|   obuf = buffer;      // assign buffer for oappnd() functions |   strncpy_P(buffer.data(), PSTR("{\"leds\":["), buffer.size()); | ||||||
|   olen = 9; |   buf += 9; // sizeof(PSTR()) from last line | ||||||
|  |  | ||||||
|   for (size_t i = 0; i < used; i += n) |   for (size_t i = 0; i < used; i += n) | ||||||
|   { |   { | ||||||
| @@ -1170,29 +1170,27 @@ bool serveLiveLeds(AsyncWebServerRequest* request, uint32_t wsClient) | |||||||
|     r = scale8(qadd8(w, r), strip.getBrightness()); //R, add white channel to RGB channels as a simple RGBW -> RGB map |     r = scale8(qadd8(w, r), strip.getBrightness()); //R, add white channel to RGB channels as a simple RGBW -> RGB map | ||||||
|     g = scale8(qadd8(w, g), strip.getBrightness()); //G |     g = scale8(qadd8(w, g), strip.getBrightness()); //G | ||||||
|     b = scale8(qadd8(w, b), strip.getBrightness()); //B |     b = scale8(qadd8(w, b), strip.getBrightness()); //B | ||||||
|     olen += sprintf_P(obuf + olen, PSTR("\"%06X\","), RGBW32(r,g,b,0)); |     buf += sprintf_P(buf, PSTR("\"%06X\","), RGBW32(r,g,b,0)); | ||||||
|   } |   } | ||||||
|   olen -= 1; |   buf--;  // remove last comma | ||||||
|   oappend((const char*)F("],\"n\":")); |   buf += sprintf_P(buf, PSTR("],\"n\":%d"), n); | ||||||
|   oappendi(n); |  | ||||||
| #ifndef WLED_DISABLE_2D | #ifndef WLED_DISABLE_2D | ||||||
|   if (strip.isMatrix) { |   if (strip.isMatrix) { | ||||||
|     oappend((const char*)F(",\"w\":")); |     buf += sprintf_P(buf, PSTR(",\"w\":%d"), Segment::maxWidth/n); | ||||||
|     oappendi(Segment::maxWidth/n); |     buf += sprintf_P(buf, PSTR(",\"h\":%d"), Segment::maxHeight/n); | ||||||
|     oappend((const char*)F(",\"h\":")); |  | ||||||
|     oappendi(Segment::maxHeight/n); |  | ||||||
|   } |   } | ||||||
| #endif | #endif | ||||||
|   oappend("}"); |   (*buf++) = '}'; | ||||||
|  |   (*buf++) = 0; | ||||||
|  |    | ||||||
|   if (request) { |   if (request) { | ||||||
|     request->send(200, FPSTR(CONTENT_TYPE_JSON), buffer); |     request->send(200, FPSTR(CONTENT_TYPE_JSON), toString(std::move(buffer))); | ||||||
|   } |   } | ||||||
|   #ifdef WLED_ENABLE_WEBSOCKETS |   #ifdef WLED_ENABLE_WEBSOCKETS | ||||||
|   else { |   else { | ||||||
|     wsc->text(obuf, olen); |     wsc->text(toString(std::move(buffer))); | ||||||
|   } |   } | ||||||
|   #endif   |   #endif   | ||||||
|   obuf = nullptr; |  | ||||||
|   return true; |   return true; | ||||||
| } | } | ||||||
| #endif | #endif | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user
	 Will Miles
					Will Miles