luat_malloc_air101.c 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465
  1. // 这个文件包含 系统heap和lua heap的默认实现
  2. #include <stdlib.h>
  3. #include <string.h>//add for memset
  4. #include "bget.h"
  5. #include "luat_malloc.h"
  6. #include "FreeRTOS.h"
  7. #define LUAT_LOG_TAG "heap"
  8. #include "luat_log.h"
  9. #include "wm_mem.h"
  10. void* __wrap_malloc(size_t len);
  11. void __wrap_free(void* ptr);
  12. void* __wrap_calloc(size_t itemCount, size_t itemSize);
  13. void* __wrap_zalloc(size_t size);
  14. void* __wrap_realloc(void*ptr, size_t len);
  15. #ifdef LUAT_USE_WLAN
  16. #define LUAT_HEAP_MIN_SIZE (96*1024)
  17. #undef LUAT_HEAP_SIZE
  18. #define LUAT_HEAP_SIZE LUAT_HEAP_MIN_SIZE
  19. #else
  20. #define LUAT_HEAP_MIN_SIZE (128*1024)
  21. #endif
  22. #ifndef LUAT_HEAP_SIZE
  23. #ifdef LUAT_USE_NIMBLE
  24. #define LUAT_HEAP_SIZE (128)*1024
  25. #else
  26. #define LUAT_HEAP_SIZE (128+48)*1024
  27. #endif
  28. #endif
  29. #if (LUAT_HEAP_SIZE < LUAT_HEAP_MIN_SIZE)
  30. #undef LUAT_HEAP_SIZE
  31. #define LUAT_HEAP_SIZE LUAT_HEAP_MIN_SIZE
  32. #endif
  33. #if (LUAT_HEAP_SIZE > LUAT_HEAP_MIN_SIZE)
  34. __attribute__((aligned(8))) static uint64_t heap_ext[(LUAT_HEAP_SIZE - LUAT_HEAP_MIN_SIZE) / 8];
  35. #endif
  36. #ifdef LUAT_USE_TLSF
  37. #include "tlsf.h"
  38. tlsf_t luavm_tlsf;
  39. pool_t luavm_tlsf_ext;
  40. #endif
  41. #ifdef LUAT_USE_PROFILER
  42. #include "luat_profiler.h"
  43. extern int luat_profiler_memdebug;
  44. extern luat_profiler_mem_t profiler_memregs[];
  45. #endif
  46. void luat_heap_init(void) {
  47. // 毕竟sram还是快很多的, 优先sram吧
  48. #ifndef LUAT_USE_TLSF
  49. bpool((void*)(0x20048000 - LUAT_HEAP_MIN_SIZE), LUAT_HEAP_MIN_SIZE - 64);
  50. #else
  51. luavm_tlsf = tlsf_create_with_pool((void*)(0x20048000 - LUAT_HEAP_MIN_SIZE), LUAT_HEAP_MIN_SIZE - 64);
  52. #endif
  53. #ifdef LUAT_USE_PSRAM
  54. char test[] = {0xAA, 0xBB, 0xCC, 0xDD};
  55. char* psram_ptr = (void*)0x30010000;
  56. LLOGD("check psram ...");
  57. memcpy(psram_ptr, test, 4);
  58. if (memcmp(psram_ptr, test, 4)) {
  59. LLOGE("psram is enable, but can't access!!");
  60. }
  61. else {
  62. LLOGD("psram is ok");
  63. memset(psram_ptr, 0, 1024);
  64. // 存在psram, 加入到内存次, 就不使用系统额外的内存了.
  65. #ifdef LUAT_USE_TLSF
  66. luavm_tlsf_ext = tlsf_add_pool(luavm_tlsf, psram_ptr, 4*1024*1024);
  67. #else
  68. bpool(psram_ptr, 4*1024*1024); // 如果是8M内存, 改成 8也可以.
  69. #endif
  70. luat_main();
  71. return;
  72. }
  73. #else
  74. // 暂时还是放在这里吧, 考虑改到0x20028000之前
  75. #if (LUAT_HEAP_SIZE > LUAT_HEAP_MIN_SIZE)
  76. #ifndef LUAT_USE_TLSF
  77. bpool((void*)heap_ext, LUAT_HEAP_SIZE - LUAT_HEAP_MIN_SIZE);
  78. #else
  79. luavm_tlsf_ext = tlsf_add_pool(luavm_tlsf, heap_ext, LUAT_HEAP_SIZE - LUAT_HEAP_MIN_SIZE);
  80. #endif
  81. #endif
  82. #endif
  83. }
  84. //------------------------------------------------
  85. // 管理系统内存
  86. #ifdef LUAT_USE_PROFILER_XXX
  87. void* luat_heap_malloc(size_t len) {
  88. void* ptr = __wrap_malloc(len);
  89. printf("luat_heap_malloc %p %d\n", ptr, len);
  90. return ptr;
  91. }
  92. void luat_heap_free(void* ptr) {
  93. if (ptr == NULL)
  94. return;
  95. printf("luat_heap_free %p\n", ptr);
  96. __wrap_free(ptr);
  97. }
  98. void* luat_heap_realloc(void* ptr, size_t len) {
  99. void* nptr = __wrap_realloc(ptr, len);
  100. printf("luat_heap_realloc %p %d %p\n", ptr, len, nptr);
  101. return nptr;
  102. }
  103. void* luat_heap_calloc(size_t count, size_t _size) {
  104. void* ptr = __wrap_calloc(count, _size);
  105. printf("luat_heap_calloc %p\n", ptr);
  106. return ptr;
  107. }
  108. #else
  109. void* luat_heap_malloc(size_t len) {
  110. return malloc(len);
  111. }
  112. void luat_heap_free(void* ptr) {
  113. if (ptr == NULL)
  114. return;
  115. free(ptr);
  116. }
  117. void* luat_heap_realloc(void* ptr, size_t len) {
  118. return realloc(ptr, len);
  119. }
  120. void* luat_heap_calloc(size_t count, size_t _size) {
  121. return calloc(count, _size);
  122. }
  123. #endif
  124. extern unsigned int heap_size_max;
  125. extern unsigned int total_mem_size;
  126. extern size_t __heap_start;
  127. extern size_t __heap_end;
  128. void luat_meminfo_sys(size_t* total, size_t* used, size_t* max_used)
  129. {
  130. #if configUSE_HEAP4
  131. extern void vPortGetHeapStats( HeapStats_t *pxHeapStats );
  132. HeapStats_t stat = {0};
  133. vPortGetHeapStats(&stat);
  134. *total = (size_t)(&__heap_end) - (size_t)(&__heap_start);
  135. *max_used = *total - stat.xMinimumEverFreeBytesRemaining;
  136. *used = (*total) - (stat.xAvailableHeapSpaceInBytes);
  137. #else
  138. *used = heap_size_max - total_mem_size;
  139. *max_used = *used;
  140. *total = heap_size_max;
  141. #endif
  142. }
  143. //------------------------------------------------
  144. //------------------------------------------------
  145. // ---------- 管理 LuaVM所使用的内存----------------
  146. #ifdef LUAT_USE_TLSF
  147. #include "tlsf.h"
  148. // extern tlsf_t luavm_tlsf;
  149. // extern pool_t luavm_tlsf_ext;
  150. void* __attribute__((section (".ram_run"))) luat_heap_alloc(void *ud, void *ptr, size_t osize, size_t nsize) {
  151. if (ptr == NULL && nsize == 0)
  152. return NULL;
  153. #ifdef LUAT_USE_MEMORY_OPTIMIZATION_CODE_MMAP
  154. if (ptr != NULL) {
  155. uint32_t ptrv = (uint32_t)ptr;
  156. if (ptrv >= 0x08000000 && ptrv < 0x08200000) {
  157. //LLOGD("??? %p %d %d", ptr, osize, nsize);
  158. if (nsize == 0)
  159. return NULL;
  160. }
  161. }
  162. #endif
  163. return tlsf_realloc(luavm_tlsf, ptr, nsize);
  164. }
  165. void luat_meminfo_luavm(size_t *total, size_t *used, size_t *max_used) {
  166. *total = 0;
  167. *used = 0;
  168. *max_used = 0;
  169. tlsf_stat(tlsf_get_pool(luavm_tlsf), total, used, max_used);
  170. if (luavm_tlsf_ext) {
  171. tlsf_stat(luavm_tlsf_ext, total, used, max_used);
  172. }
  173. }
  174. #else
  175. void* __attribute__((section (".ram_run"))) luat_heap_alloc(void *ud, void *ptr, size_t osize, size_t nsize) {
  176. (void)ud;
  177. if (ptr == NULL && nsize == 0)
  178. return NULL;
  179. #ifdef LUAT_USE_MEMORY_OPTIMIZATION_CODE_MMAP
  180. if (ptr != NULL) {
  181. uint32_t ptrv = (uint32_t)ptr;
  182. if (ptrv >= 0x08000000 && ptrv < 0x08200000) {
  183. // LLOGD("??? %p %d %d", ptr, osize, nsize);
  184. return NULL;
  185. }
  186. }
  187. #endif
  188. if (0) {
  189. if (ptr) {
  190. if (nsize) {
  191. // 缩放内存块
  192. LLOGD("realloc %p from %d to %d", ptr, osize, nsize);
  193. }
  194. else {
  195. // 释放内存块
  196. LLOGD("free %p ", ptr);
  197. brel(ptr);
  198. return NULL;
  199. }
  200. }
  201. else {
  202. // 申请内存块
  203. ptr = bget(nsize);
  204. LLOGD("malloc %p type=%d size=%d", ptr, osize, nsize);
  205. return ptr;
  206. }
  207. }
  208. if (nsize)
  209. {
  210. void* ptmp = bgetr(ptr, nsize);
  211. if(ptmp == NULL && osize >= nsize)
  212. {
  213. return ptr;
  214. }
  215. return ptmp;
  216. }
  217. brel(ptr);
  218. return NULL;
  219. }
  220. void luat_meminfo_luavm(size_t *total, size_t *used, size_t *max_used) {
  221. long curalloc, totfree, maxfree;
  222. unsigned long nget, nrel;
  223. bstats(&curalloc, &totfree, &maxfree, &nget, &nrel);
  224. *used = curalloc;
  225. *max_used = bstatsmaxget();
  226. *total = curalloc + totfree;
  227. }
  228. #endif
  229. //-----------------------------------------------------------------------------
  230. #include "wm_include.h"
  231. #include "FreeRTOS.h"
  232. #include "stdio.h"
  233. #ifdef LUAT_USE_PROFILER
  234. void* __wrap_malloc(size_t len) {
  235. #ifdef LUAT_USE_PROFILER
  236. void* ptr = pvPortMalloc(len);
  237. if (luat_profiler_memdebug) {
  238. // printf("malloc %d %p\n", len, ptr);
  239. if (ptr == NULL)
  240. return NULL;
  241. for (size_t i = 0; i < LUAT_PROFILER_MEMDEBUG_ADDR_COUNT; i++)
  242. {
  243. if (profiler_memregs[i].addr == 0) {
  244. profiler_memregs[i].addr = (uint32)ptr;
  245. profiler_memregs[i].len = len;
  246. break;
  247. }
  248. }
  249. }
  250. return ptr;
  251. #else
  252. return pvPortMalloc(len);
  253. #endif
  254. }
  255. void __wrap_free(void* ptr) {
  256. if (ptr == NULL)
  257. return;
  258. #ifdef LUAT_USE_PROFILER
  259. // if (luat_profiler_memdebug)
  260. // printf("free %p\n", ptr);
  261. #endif
  262. u32 addr = (u32)ptr;
  263. if (addr >= 0x20000000 && addr <= 0x40000000) {
  264. // printf("free %p\n", ptr);
  265. vPortFree(ptr);
  266. #ifdef LUAT_USE_PROFILER
  267. for (size_t i = 0; i < LUAT_PROFILER_MEMDEBUG_ADDR_COUNT; i++)
  268. {
  269. if (profiler_memregs[i].addr == addr) {
  270. profiler_memregs[i].len = 0;
  271. profiler_memregs[i].addr = 0;
  272. break;
  273. }
  274. }
  275. #endif
  276. }
  277. }
  278. void *pvPortRealloc( void *pv, size_t xWantedSize );
  279. void* __wrap_realloc(void*ptr, size_t len) {
  280. #ifdef LUAT_USE_PROFILER
  281. void* newptr = pvPortRealloc(ptr, len);
  282. if (luat_profiler_memdebug && newptr) {
  283. // printf("realloc %p %d %p\n", ptr, len, newptr);
  284. uint32_t addr = (uint32_t)ptr;
  285. uint32_t naddr = (uint32_t)newptr;
  286. if (ptr == newptr) {
  287. // 相同内存地址, 只是扩容了
  288. for (size_t i = 0; i < LUAT_PROFILER_MEMDEBUG_ADDR_COUNT; i++)
  289. {
  290. if (profiler_memregs[i].addr == addr) {
  291. profiler_memregs[i].len = len;
  292. addr = 0;
  293. break;
  294. }
  295. }
  296. if (0 != addr) {
  297. for (size_t i = 0; i < LUAT_PROFILER_MEMDEBUG_ADDR_COUNT; i++)
  298. {
  299. if (profiler_memregs[i].addr == 0) {
  300. profiler_memregs[i].addr = addr;
  301. profiler_memregs[i].len = len;
  302. break;
  303. }
  304. }
  305. }
  306. }
  307. else {
  308. if (ptr == NULL) {
  309. for (size_t i = 0; i < LUAT_PROFILER_MEMDEBUG_ADDR_COUNT; i++)
  310. {
  311. if (profiler_memregs[i].addr == 0) {
  312. profiler_memregs[i].addr = naddr;
  313. profiler_memregs[i].len = len;
  314. break;
  315. }
  316. }
  317. }
  318. else {
  319. for (size_t i = 0; i < LUAT_PROFILER_MEMDEBUG_ADDR_COUNT; i++)
  320. {
  321. if (profiler_memregs[i].addr == addr) {
  322. profiler_memregs[i].addr = 0;
  323. profiler_memregs[i].len = 0;
  324. break;
  325. }
  326. }
  327. for (size_t i = 0; i < LUAT_PROFILER_MEMDEBUG_ADDR_COUNT; i++)
  328. {
  329. if (profiler_memregs[i].addr == 0) {
  330. profiler_memregs[i].addr = naddr;
  331. profiler_memregs[i].len = len;
  332. break;
  333. }
  334. }
  335. }
  336. }
  337. }
  338. return newptr;
  339. #endif
  340. return pvPortRealloc(ptr, len);
  341. }
  342. void* __wrap_calloc(size_t itemCount, size_t itemSize) {
  343. void* ptr = pvPortMalloc(itemCount * itemSize);
  344. #ifdef LUAT_USE_PROFILER
  345. if (luat_profiler_memdebug) {
  346. // printf("calloc %p %d\n", ptr, itemCount * itemSize);
  347. if (ptr) {
  348. for (size_t i = 0; i < LUAT_PROFILER_MEMDEBUG_ADDR_COUNT; i++)
  349. {
  350. if (profiler_memregs[i].addr == 0) {
  351. profiler_memregs[i].addr = (uint32_t)ptr;
  352. profiler_memregs[i].len = itemCount * itemSize;
  353. break;
  354. }
  355. }
  356. }
  357. }
  358. #endif
  359. if (ptr == NULL)
  360. return NULL;
  361. memset(ptr, 0, itemCount * itemSize);
  362. return ptr;
  363. }
  364. void* __wrap_zalloc(size_t size) {
  365. void* ptr = pvPortMalloc(size);
  366. #ifdef LUAT_USE_PROFILER
  367. if (luat_profiler_memdebug) {
  368. // printf("zalloc %p %d\n", ptr, size);
  369. if (ptr) {
  370. for (size_t i = 0; i < LUAT_PROFILER_MEMDEBUG_ADDR_COUNT; i++)
  371. {
  372. if (profiler_memregs[i].addr == 0) {
  373. profiler_memregs[i].addr = (uint32_t)ptr;
  374. profiler_memregs[i].len = size;
  375. break;
  376. }
  377. }
  378. }
  379. }
  380. #endif
  381. if (ptr == NULL)
  382. return NULL;
  383. memset(ptr, 0, size);
  384. return ptr;
  385. }
  386. #else
  387. // 标准实现
  388. void* __wrap_malloc(size_t len) {
  389. return pvPortMalloc(len);
  390. }
  391. void __wrap_free(void* ptr) {
  392. if (ptr == NULL)
  393. return;
  394. u32 addr = (u32)ptr;
  395. if (addr >= 0x20000000 && addr <= 0x40000000) {
  396. vPortFree(ptr);
  397. }
  398. }
  399. void *pvPortRealloc( void *pv, size_t xWantedSize );
  400. void* __wrap_realloc(void*ptr, size_t len) {
  401. return pvPortRealloc(ptr, len);
  402. }
  403. void* __wrap_calloc(size_t itemCount, size_t itemSize) {
  404. void* ptr = pvPortMalloc(itemCount * itemSize);
  405. if (ptr == NULL)
  406. return NULL;
  407. memset(ptr, 0, itemCount * itemSize);
  408. return ptr;
  409. }
  410. void* __wrap_zalloc(size_t size) {
  411. void* ptr = pvPortMalloc(size);
  412. if (ptr == NULL)
  413. return NULL;
  414. memset(ptr, 0, size);
  415. return ptr;
  416. }
  417. #endif