ui_mem.c 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381
  1. /*
  2. * Copyright (c) 2019 Actions Semiconductor Co., Ltd
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #define LOG_MODULE_CUSTOMER
  7. #include <os_common_api.h>
  8. #include <assert.h>
  9. #include <string.h>
  10. #include <ui_mem.h>
  11. #ifdef CONFIG_UI_MEMORY_DEBUG
  12. # include <mem_guard.h>
  13. #endif
  14. #ifdef CONFIG_ACTIONS_PRINTK_DMA
  15. # include <sys/printk.h>
  16. #endif
  17. LOG_MODULE_REGISTER(ui_mem, LOG_LEVEL_INF);
  18. #define FORCE_CALLER_STRING 1
  19. #if FORCE_CALLER_STRING
  20. # define PRI_CALLER "%s"
  21. #else
  22. # define PRI_CALLER "%p"
  23. #endif
  24. #if CONFIG_UI_MEM_NUMBER_BLOCKS > 0
  25. extern void * ui_mem_fb_alloc(size_t size);
  26. extern void * ui_mem_fb_aligned_alloc(size_t align, size_t size);
  27. extern void ui_mem_fb_free(void *ptr);
  28. extern size_t ui_mem_fb_get_size(void);
  29. extern void ui_mem_fb_dump(void);
  30. extern bool ui_mem_is_fb(const void * ptr);
  31. #endif /* CONFIG_UI_MEM_NUMBER_BLOCKS */
  32. #if CONFIG_UI_GUI_MEM_POOL_SIZE > 0
  33. extern int ui_mem_gui_init(void);
  34. extern void * ui_mem_gui_alloc(size_t size);
  35. extern void * ui_mem_gui_aligned_alloc(size_t align, size_t size);
  36. extern void * ui_mem_gui_realloc(void *ptr, size_t size);
  37. extern void ui_mem_gui_free(void *ptr);
  38. extern size_t ui_mem_gui_get_size(void);
  39. extern void ui_mem_gui_dump(void);
  40. extern bool ui_mem_is_gui(const void * ptr);
  41. #endif /* CONFIG_UI_GUI_MEM_POOL_SIZE */
  42. #if CONFIG_UI_RES_MEM_POOL_SIZE > 0
  43. extern int ui_mem_res_init(void);
  44. extern void * ui_mem_res_alloc(size_t size);
  45. extern void * ui_mem_res_aligned_alloc(size_t align, size_t size);
  46. extern void * ui_mem_res_realloc(void * ptr, size_t size);
  47. extern void ui_mem_res_free(void *ptr);
  48. extern size_t ui_mem_res_get_size(void);
  49. extern void ui_mem_res_dump(void);
  50. extern bool ui_mem_is_res(const void * ptr);
  51. #endif /* CONFIG_UI_RES_MEM_POOL_SIZE */
  52. struct ui_mem_func {
  53. const char * name;
  54. int (* init_fn)(void);
  55. void * (* alloc_fn)(size_t size);
  56. void * (* aligned_alloc_fn)(size_t align, size_t size);
  57. void * (* realloc_fn)(void * ptr, size_t size);
  58. void (* free_fn)(void * ptr);
  59. void (* dump_fn)(void);
  60. bool (* is_type_fn)(const void * ptr);
  61. size_t (* get_size)(void);
  62. };
  63. static const struct ui_mem_func mem_table[NUM_UI_MEM_TYPES] = {
  64. [MEM_FB] = {
  65. .name = "FB",
  66. #if CONFIG_UI_MEM_NUMBER_BLOCKS > 0
  67. .alloc_fn = ui_mem_fb_alloc,
  68. .aligned_alloc_fn = ui_mem_fb_aligned_alloc,
  69. .free_fn = ui_mem_fb_free,
  70. .dump_fn = ui_mem_fb_dump,
  71. .is_type_fn = ui_mem_is_fb,
  72. .get_size = ui_mem_fb_get_size,
  73. #endif /* CONFIG_UI_MEM_NUMBER_BLOCKS > 0 */
  74. },
  75. [MEM_GUI] = {
  76. .name = "GUI",
  77. #if CONFIG_UI_GUI_MEM_POOL_SIZE > 0
  78. .init_fn = ui_mem_gui_init,
  79. .alloc_fn = ui_mem_gui_alloc,
  80. .aligned_alloc_fn = ui_mem_gui_aligned_alloc,
  81. .realloc_fn = ui_mem_gui_realloc,
  82. .free_fn = ui_mem_gui_free,
  83. .dump_fn = ui_mem_gui_dump,
  84. .is_type_fn = ui_mem_is_gui,
  85. .get_size = ui_mem_gui_get_size,
  86. #endif /* CONFIG_UI_GUI_MEM_POOL_SIZE > 0 */
  87. },
  88. [MEM_RES] = {
  89. .name = "RES",
  90. #if CONFIG_UI_RES_MEM_POOL_SIZE > 0
  91. .init_fn = ui_mem_res_init,
  92. .alloc_fn = ui_mem_res_alloc,
  93. .aligned_alloc_fn = ui_mem_res_aligned_alloc,
  94. .realloc_fn = ui_mem_res_realloc,
  95. .free_fn = ui_mem_res_free,
  96. .dump_fn = ui_mem_res_dump,
  97. .is_type_fn = ui_mem_is_res,
  98. .get_size = ui_mem_res_get_size,
  99. #elif CONFIG_UI_GUI_MEM_POOL_SIZE > 0
  100. .init_fn = NULL,
  101. .alloc_fn = ui_mem_gui_alloc,
  102. .aligned_alloc_fn = ui_mem_gui_aligned_alloc,
  103. .realloc_fn = ui_mem_gui_realloc,
  104. .free_fn = ui_mem_gui_free,
  105. .dump_fn = NULL,
  106. .is_type_fn = NULL, /* mixed res mem should be considered as gui mem */
  107. .get_size = ui_mem_gui_get_size,
  108. #endif /* CONFIG_UI_RES_MEM_POOL_SIZE > 0 */
  109. },
  110. };
  111. #ifdef CONFIG_UI_MEMORY_DEBUG
  112. static struct mem_guard_head mem_guard[NUM_UI_MEM_TYPES];
  113. #endif
  114. int ui_mem_init(void)
  115. {
  116. int i;
  117. for (i = 0; i < NUM_UI_MEM_TYPES; i++) {
  118. if (mem_table[i].init_fn) {
  119. mem_table[i].init_fn();
  120. }
  121. #ifdef CONFIG_UI_MEMORY_DEBUG
  122. if(mem_table[i].get_size != NULL) {
  123. mem_guard_init(&mem_guard[i], mem_table[i].get_size());
  124. }
  125. #endif
  126. }
  127. return 0;
  128. }
  129. void * ui_mem_alloc(uint8_t type, size_t size, const void * caller)
  130. {
  131. void * ptr = NULL;
  132. if (size == 0)
  133. return NULL;
  134. if (type >= NUM_UI_MEM_TYPES || mem_table[type].alloc_fn == NULL) {
  135. SYS_LOG_ERR("mem %d alloc %u unsupported (caller " PRI_CALLER ")\n",
  136. type, (uint32_t)size, (char *)caller);
  137. return NULL;
  138. }
  139. #ifdef CONFIG_UI_MEMORY_DEBUG
  140. ptr = mem_guard_malloc(&mem_guard[type], mem_table[type].alloc_fn, size, caller);
  141. #else
  142. ptr = mem_table[type].alloc_fn(size);
  143. #endif /* CONFIG_UI_MEMORY_DEBUG */
  144. if (ptr == NULL) {
  145. SYS_LOG_ERR("UI-%s alloc %u failed (caller " PRI_CALLER ")\n",
  146. mem_table[type].name, (uint32_t)size, (char *)caller);
  147. }
  148. return ptr;
  149. }
  150. void * ui_mem_aligned_alloc(uint8_t type, size_t align, size_t size, const void * caller)
  151. {
  152. void * ptr = NULL;
  153. if (size == 0)
  154. return NULL;
  155. if (type >= NUM_UI_MEM_TYPES || mem_table[type].aligned_alloc_fn == NULL) {
  156. SYS_LOG_ERR("mem %d aligned_alloc %u-%u unsupported (caller " PRI_CALLER ")\n",
  157. type, (uint32_t)align, (uint32_t)size, (char *)caller);
  158. return NULL;
  159. }
  160. assert((align & (align - 1)) == 0);
  161. #ifdef CONFIG_UI_MEMORY_DEBUG
  162. ptr = mem_guard_aligned_alloc(&mem_guard[type], mem_table[type].aligned_alloc_fn, align, size, caller);
  163. #else
  164. ptr = mem_table[type].aligned_alloc_fn(align, size);
  165. #endif /* CONFIG_UI_MEMORY_DEBUG */
  166. if (ptr == NULL) {
  167. SYS_LOG_ERR("UI-%s aligned_alloc %u-%u failed (caller " PRI_CALLER ")\n",
  168. mem_table[type].name, (uint32_t)align, (uint32_t)size, (char *)caller);
  169. }
  170. return ptr;
  171. }
  172. void * ui_mem_realloc(uint8_t type, void * ptr, size_t size, const void * caller)
  173. {
  174. if (type >= NUM_UI_MEM_TYPES) {
  175. SYS_LOG_ERR("mem %d realloc %u unsupported (caller " PRI_CALLER ")\n",
  176. type, (uint32_t)size, (char *)caller);
  177. return NULL;
  178. }
  179. if (mem_table[type].realloc_fn == NULL) {
  180. void * ptr_new = ui_mem_alloc(type, size, caller);
  181. if (ptr_new) {
  182. memcpy(ptr_new, ptr, size);
  183. }
  184. ui_mem_free(type, ptr);
  185. ptr = ptr_new;
  186. } else {
  187. #ifdef CONFIG_UI_MEMORY_DEBUG
  188. ptr = mem_guard_realloc(&mem_guard[type], mem_table[type].realloc_fn, ptr, size, caller);
  189. #else
  190. ptr = mem_table[type].realloc_fn(ptr, size);
  191. #endif /* CONFIG_UI_MEMORY_DEBUG */
  192. }
  193. if (ptr == NULL) {
  194. SYS_LOG_ERR("UI-%s realloc %u failed (caller " PRI_CALLER ")\n",
  195. mem_table[type].name, (uint32_t)size, (char *)caller);
  196. }
  197. return ptr;
  198. }
  199. void * ui_mem_calloc(uint8_t type, size_t nmemb, size_t size, const void * caller)
  200. {
  201. void * ptr = NULL;
  202. size *= nmemb;
  203. ptr = ui_mem_alloc(type, size, caller);
  204. if (ptr) {
  205. memset(ptr, 0, size);
  206. }
  207. return ptr;
  208. }
  209. void ui_mem_free(uint8_t type, void * ptr)
  210. {
  211. if (ptr == NULL) {
  212. return;
  213. }
  214. if (type >= NUM_UI_MEM_TYPES) {
  215. SYS_LOG_ERR("mem %d free unsupported\n", type);
  216. return;
  217. }
  218. if (mem_table[type].free_fn) {
  219. #ifdef CONFIG_UI_MEMORY_DEBUG
  220. mem_guard_free(&mem_guard[type], mem_table[type].free_fn, ptr);
  221. #else
  222. mem_table[type].free_fn(ptr);
  223. #endif /* CONFIG_UI_MEMORY_DEBUG */
  224. }
  225. }
  226. void ui_mem_free2(void * ptr)
  227. {
  228. int i;
  229. if (ptr == NULL) {
  230. return;
  231. }
  232. for (i = 0; i < NUM_UI_MEM_TYPES; i++) {
  233. if (mem_table[i].is_type_fn && mem_table[i].is_type_fn(ptr)) {
  234. if (mem_table[i].free_fn) {
  235. #ifdef CONFIG_UI_MEMORY_DEBUG
  236. mem_guard_free(&mem_guard[i], mem_table[i].free_fn, ptr);
  237. #else
  238. mem_table[i].free_fn(ptr);
  239. #endif /* CONFIG_UI_MEMORY_DEBUG */
  240. }
  241. return;
  242. }
  243. }
  244. SYS_LOG_ERR("unknown ptr %p to free", ptr);
  245. }
  246. bool ui_mem_is_type(uint8_t type, const void * ptr)
  247. {
  248. bool is_type = false;
  249. if (type < NUM_UI_MEM_TYPES && mem_table[type].is_type_fn) {
  250. is_type = mem_table[type].is_type_fn(ptr);
  251. }
  252. return is_type;
  253. }
  254. void ui_mem_dump(uint8_t type)
  255. {
  256. if (type >= NUM_UI_MEM_TYPES) {
  257. return;
  258. }
  259. #ifdef CONFIG_ACTIONS_PRINTK_DMA
  260. printk_dma_switch(0);
  261. #endif /* CONFIG_ACTIONS_PRINTK_DMA */
  262. if (mem_table[type].dump_fn) {
  263. os_printk("UI-%s dump:\n", mem_table[type].name);
  264. #ifdef CONFIG_UI_MEMORY_DEBUG
  265. mem_guard_dump(&mem_guard[type]);
  266. #endif /* CONFIG_UI_MEMORY_DEBUG */
  267. mem_table[type].dump_fn();
  268. }
  269. os_printk("\n");
  270. #ifdef CONFIG_ACTIONS_PRINTK_DMA
  271. printk_dma_switch(1);
  272. #endif /* CONFIG_ACTIONS_PRINTK_DMA */
  273. }
  274. void ui_mem_dump_all(void)
  275. {
  276. int i;
  277. for (i = 0; i < NUM_UI_MEM_TYPES; i++) {
  278. ui_mem_dump(i);
  279. }
  280. }
  281. void ui_mem_safe_check(uint16_t view_id)
  282. {
  283. #ifdef CONFIG_UI_MEMORY_DEBUG
  284. int i;
  285. os_printk("view_id %d mem check:\n", view_id);
  286. for (i = 0; i < NUM_UI_MEM_TYPES; i++) {
  287. /* no need to check MEM_FB */
  288. if (i == MEM_FB) continue;
  289. os_printk("checking %s:\n", mem_table[i].name);
  290. mem_guard_leak_check(&mem_guard[i], view_id);
  291. os_printk("check end\n");
  292. }
  293. #endif /* CONFIG_UI_MEMORY_DEBUG */
  294. }
  295. #ifdef CONFIG_UI_MEM_VDB_SHARE_SURFACE_BUFFER
  296. #ifndef CONFIG_SIMULATOR
  297. __in_section_unique(sram.noinit.sufacebuffer) __aligned(64)
  298. #endif
  299. static uint8_t globle_share_surface_buffer[CONFIG_UI_MEM_BLOCK_SIZE];
  300. void *ui_mem_get_share_surface_buffer(void)
  301. {
  302. return &globle_share_surface_buffer;
  303. }
  304. int ui_mem_get_share_surface_buffer_size(void)
  305. {
  306. return CONFIG_UI_MEM_BLOCK_SIZE;
  307. }
  308. #else
  309. void *ui_mem_get_share_surface_buffer(void) { return NULL; }
  310. int ui_mem_get_share_surface_buffer_size(void) { return 0; }
  311. #endif /* CONFIG_UI_MEM_VDB_SHARE_SURFACE_BUFFER */