【ESP32-S3】运行报错集合
- 报错
- assert failed: block_locate_free tlsf_control_functions.h:618 (block_size(block) >= *size)
- 参考
报错
assert failed: block_locate_free tlsf_control_functions.h:618 (block_size(block) >= *size)
成功启动HTTP服务器后,只要在浏览器中访问在网址就会立即报错,具体如下
10:22:58.316->startHttpServer服务器启动成功10:22:58.316->========startHttpServer结束==========10:22:58.316->Http Ready! Use'http://192.168.1.55'to connect10:22:59.224->10:22:59.303->assertfailed:block_locate_free tlsf_control_functions.h:618(block_size(block)>=*size)10:22:59.303->10:22:59.303->10:22:59.303->Backtrace:0x4037673d:0x3fcb0ef00x4037e89d:0x3fcb0f100x403856d6:0x3fcb0f300x40384889:0x3fcb10700x40384281:0x3fcb10900x403843e4:0x3fcb10b00x4037787f:0x3fcb10d00x40377899:0x3fcb11000x403774fc:0x3fcb11200x42025e45:0x3fcb11700x420270a3:0x3fcb11900x4202e4a5:0x3fcb11b00x4202e931:0x3fcb11e00x4202e986:0x3fcb12100x4202c139:0x3fcb12300x4202c2ce:0x3fcb12500x42024c40:0x3fcb12700x4037f4da:0x3fcb12a010:22:59.312->ESP32-S3内置内存(典型配置):
- SRAM: 512KB(或320KB,取决于型号)
├── IRAM(指令RAM):128-192KB(用于代码执行)
├── DRAM(数据RAM):192-320KB(用于堆、栈、静态数据)
└── RTC快速内存:8KB(深度睡眠保留)
这里报错是DRAM侧报错,这里的内存不足了。
原因:代码通过 TLSF 算法申请内存时,TLSF 系统在 “空闲内存块” 列表中找到的某块空闲内存(block),其实际尺寸(block_size(block))小于当前代码要申请的内存尺寸(*size)—— 这违反了 TLSF “分配前需确保空闲块足够大” 的基础规则,因此触发assert(断言)保护。
怀疑:比如嵌入式设备(如运行 HTTP 服务的硬件)剩余内存不足,或代码中存在内存泄漏、申请内存尺寸不合理(过大)等问题。
诊断工具:
// 查看具体内存分布voidshowMemoryDetails(){Serial.println("\n===== 内存详细分布 =====");// DRAM(数据RAM)堆内存multi_heap_info_t heap_info;heap_caps_get_info(&heap_info,MALLOC_CAP_INTERNAL|MALLOC_CAP_8BIT);Serial.printf("DRAM堆内存: 总量=%d, 已用=%d, 空闲=%d, 最大块=%d\n",heap_info.total_free_bytes+heap_info.total_allocated_bytes,heap_info.total_allocated_bytes,heap_info.total_free_bytes,heap_info.largest_free_block);// 各个内存区域Serial.println("\n各个内存区域:");heap_caps_print_heap_info(MALLOC_CAP_INTERNAL);// 内部RAMheap_caps_print_heap_info(MALLOC_CAP_SPIRAM);// PSRAM(如果有)// 更详细的分区Serial.println("\n分区使用情况:");Serial.printf("指令内存(IRAM): %d bytes 可用\n",heap_caps_get_free_size(MALLOC_CAP_EXEC));Serial.printf("32位内存: %d bytes 可用\n",heap_caps_get_free_size(MALLOC_CAP_32BIT));Serial.printf("默认内存: %d bytes 可用\n",heap_caps_get_free_size(MALLOC_CAP_DEFAULT));Serial.printf("内部内存: %d bytes 可用\n",heap_caps_get_free_size(MALLOC_CAP_INTERNAL));// 查看栈使用情况(近似值)Serial.printf("栈使用: 约 %d bytes\n",uxTaskGetStackHighWaterMark(NULL)*4);}// 确定内存不足的具体类型voiddiagnoseMemoryShortage(){// 检查不同类型的内存分配能力size_t test_sizes[]={128,256,512,1024,2048,4096,8192};Serial.println("\n===== 内存分配能力测试 =====");for(size_t size:test_sizes){void*ptr=heap_caps_malloc(size,MALLOC_CAP_DEFAULT);if(ptr){Serial.printf("√ 可分配 %4d bytes (实际使用)\n",size);free(ptr);}else{Serial.printf("× 无法分配 %4d bytes (崩溃点)\n",size);}}// 检查连续分配能力Serial.println("\n===== 连续分配测试 =====");void*blocks[20];for(inti=0;i<20;i++){blocks[i]=malloc(256);// 每次256字节if(!blocks[i]){Serial.printf("第 %d 次分配失败 (碎片化问题)\n",i);break;}}// 清理for(inti=0;i<20;i++){if(blocks[i])free(blocks[i]);}}参考
https://docs.espressif.com/projects/esp-techpedia/zh_CN/latest/esp-friends/advanced-development/debugging/heap-stack-debugging.html