I am trying to process a favicon for a Minecraft server based upon the esp32-minecraft-server project (https://github.com/nikisalli/esp32-minecraft-server).
I have modified minecraft.cpp (https://github.com/nikisalli/esp32-minecraft-server/blob/main/lib/minecraft/minecraft.cpp) so that instead of using the writeString function in the writeResponse function, I am instead saving a long string containing the favicon data (formatted in base64) into PROGMEM and loading it byte by byte. I thought this might stop the ESP32 from crashing and rebooting which happens when the string is over 5997 bytes (discovered through trial and error).
writeResponse Function (No Edits)
void minecraft::player::writeResponse(){
packet p(S, mtx);
p.writeVarInt(0);
p.writeString("{\"version\": {\"name\": \"1.16.5\",\"protocol\": 754},\"players\": {\"max\": 5,\"online\": 5,\"sample\": [{\"name\": \"L_S___S_S_S__S_L\",\"id\": \"00000000-0000-0000-0000-000000000000\"},{\"name\": \"L_SS__S_S_S_S__L\",\"id\": \"00000000-0000-0000-0000-000000000001\"},{\"name\": \"L_S_S_S_S_SS___L\",\"id\": \"00000000-0000-0000-0000-000000000002\"},{\"name\": \"L_S__SS_S_S_S__L\",\"id\": \"00000000-0000-0000-0000-000000000003\"},{\"name\": \"L_S___S_S_S__S_L\",\"id\": \"00000000-0000-0000-0000-000000000004\"}]},\"description\": {\"text\": \"esp32 server\"}}");
logout("response packet sent");
p.writePacket();
}
writeResponse Function (Modified)
void minecraft::player::writeResponse() {
packet p(S, mtx);
p.writeVarInt(0);
// Define the string in PROGMEM
const char response[] PROGMEM = "{\"version\": {\"name\": \"1.16.5\",\"protocol\": 754},\"players\": {\"max\": 5,\"online\": 5,\"sample\": [{\"name\": \"L_S___S_S_S__S_L\",\"id\": \"00000000-0000-0000-0000-000000000000\"},{\"name\": \"L_SS__S_S_S_S__L\",\"id\": \"00000000-0000-0000-0000-000000000001\"},{\"name\": \"L_S_S_S_S_SS___L\",\"id\": \"00000000-0000-0000-0000-000000000002\"},{\"name\": \"L_S__SS_S_S_S__L\",\"id\": \"00000000-0000-0000-0000-000000000003\"},{\"name\": \"L_S___S_S_S__S_L\",\"id\": \"00000000-0000-0000-0000-000000000004\"}]},\"description\": {\"text\": \"esp32 server\"},\"favicon\":\"\"}";
// Get the length of the string
size_t responseLength = strlen_P(response);
// Write the length of the string
p.writeVarInt(responseLength);
// Read and write the string byte by byte
for (size_t i = 0; i < responseLength; ++i) {
p.write(pgm_read_byte(&response[i]));
}
logout("response packet sent");
p.writePacket();
}
I have discovered the code fails at (*mtx).lock() in the writePacket function. This is the same point the error was occurring when I was using the writeString function with an equally long string.
void packet::writePacket(){
Serial.println("BEFORE LOCK");
(*mtx).lock();
Serial.println("AFTER LOCK");
serverWriteVarInt(index);
S->write(buffer, index);
(*mtx).unlock();
}
Here is the serial output. The error occurs when I load the Minecraft server details in the Multiplayer menu:
* Executing task: C:\Users\Elect\.platformio\penv\Scripts\platformio.exe device monitor
Please build project in debug configuration to get more details about an exception.
See [https://docs.platformio.org/page/projectconf/build_configurations.html][3]
--- Terminal on COM11 | 115200 8-N-1
--- Available filters and text transformations: colorize, debug, default, direct, esp32_exception_decoder, hexlify, log2file, nocontrol, printable, send_on_enter, time
--- More details at
--- Quit: Ctrl+C | Menu: Ctrl+T | Help: Ctrl+T followed by Ctrl+H
e[ 65][E][esConnecting to WiFi...
Connected to the WiFi network
172.20.10.5
[INFO] server started
[INFO] New client connected: 0
[INFO] p0 started task 0 pinned to core 0
[INFO] p0 <- request packet received
[INFO] p0 -> response packet sent
BEFORE LOCK
assert failed: xQueueSemaphoreTake queue.c:1545 (( pxQueue ))
Backtrace: 0x40083669:0x3ffecbd0 0x4008bbd9:0x3ffecbf0 0x40091059:0x3ffecc10 0x4008cbe9:0x3ffecd40 0x40082936:0x3ffecd80 0x4008294f:0x3ffecda0 0x400d3080:0x3ffecdc0 0x400d5136:0x3ffecde0 0x400d54bf:0x3fff0890 0x400d2a81:0x3fff0910
#0 0x40083669:0x3ffecbd0 in panic_abort at /Users/ficeto/Desktop/ESP32/ESP32S2/esp-idf-public/components/esp_system/panic.c:408
#1 0x4008bbd9:0x3ffecbf0 in esp_system_abort at /Users/ficeto/Desktop/ESP32/ESP32S2/esp-idf-public/components/esp_system/esp_system.c:137
#2 0x40091059:0x3ffecc10 in __assert_func at /Users/ficeto/Desktop/ESP32/ESP32S2/esp-idf-public/components/newlib/assert.c:85
#3 0x4008cbe9:0x3ffecd40 in xQueueSemaphoreTake at /Users/ficeto/Desktop/ESP32/ESP32S2/esp-idf-public/components/freertos/queue.c:1549 (discriminator 1)
#4 0x40082936:0x3ffecd80 in pthread_mutex_lock_internal at /Users/ficeto/Desktop/ESP32/ESP32S2/esp-idf-public/components/pthread/pthread.c:620
#5 0x4008294f:0x3ffecda0 in pthread_mutex_lock at /Users/ficeto/Desktop/ESP32/ESP32S2/esp-idf-public/components/pthread/pthread.c:650
#6 0x400d3080:0x3ffecdc0 in packet::writePacket() at c:\users\elect\.platformio\packages\toolchain-xtensa-esp32@8.4.0+2021r2-patch5\xtensa-esp32-elf\include\c++\8.4.0\xtensa-esp32-elf\no-rtti\bits/gthr-default.h:761
(inlined by) std::mutex::lock() at c:\users\elect\.platformio\packages\toolchain-xtensa-esp32@8.4.0+2021r2-patch5\xtensa-esp32-elf\include\c++\8.4.0\bits/std_mutex.h:103
(inlined by) packet::writePacket() at lib/minecraft/minecraft.cpp:28
#7 0x400d5136:0x3ffecde0 in minecraft::player::writeResponse() at lib/minecraft/minecraft.cpp:392 (discriminator 3)
#8 0x400d54bf:0x3fff0890 in minecraft::player::join() at lib/minecraft/minecraft.cpp:670
#9 0x400d2a81:0x3fff0910 in playerHandler(void*) at src/main.cpp:30 (discriminator 15)
ELF file SHA256: 352caedde0dc5df0
Rebooting...
ets Jun 8 2016 00:22:57
rst:0xc (SW_CPU_RESET),boot:0x17 (SPI_FAST_FLASH_BOOT)
configsip: 0, SPIWP:0xee
clk_drv:0x00,q_drv:0x00,d_drv:0x00,cs0_drv:0x00,hd_drv:0x00,wp_drv:0x00
mode:DIO, clock div:2
load:0x3fff0030,len:1184
load:0x40078000,len:13232
load:0x40080400,len:3028
entry 0x400805e4
[ 65][E][esConnecting to WiFi...
Connected to the WiFi network
172.20.10.5
[INFO] server started
I've checked with the following code in writePacket to make sure that mtx has been initalised:
if (mtx != nullptr) {
Serial.println("Mutex pointer value: " + String((uintptr_t)mtx, HEX));
} else {
Serial.println("Mutex pointer is nullptr");
}
And I get Mutex pointer value: 35417578
, indicating it has initialised correctly.
Do you have any suggestions on what I do to allow the base64 favicon data to be processed correctly or is it simply a memory limit? If a memory limit, how do I know what this limit is?
i think issue is in your writeResponse
function:
for loop shold be for (size_t i = 0; i < responseLength; i++)
instead
for (size_t i = 0; i < responseLength; ++i)
also increse buffer
size in packet
class if needed; currently, it is buffer[6000]
.