123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381 |
- /*
- * Copyright (c) 2019 Actions Semiconductor Co., Ltd
- *
- * SPDX-License-Identifier: Apache-2.0
- */
- #define LOG_MODULE_CUSTOMER
- #include <os_common_api.h>
- #include <assert.h>
- #include <string.h>
- #include <ui_mem.h>
- #ifdef CONFIG_UI_MEMORY_DEBUG
- # include <mem_guard.h>
- #endif
- #ifdef CONFIG_ACTIONS_PRINTK_DMA
- # include <sys/printk.h>
- #endif
- LOG_MODULE_REGISTER(ui_mem, LOG_LEVEL_INF);
- #define FORCE_CALLER_STRING 1
- #if FORCE_CALLER_STRING
- # define PRI_CALLER "%s"
- #else
- # define PRI_CALLER "%p"
- #endif
- #if CONFIG_UI_MEM_NUMBER_BLOCKS > 0
- extern void * ui_mem_fb_alloc(size_t size);
- extern void * ui_mem_fb_aligned_alloc(size_t align, size_t size);
- extern void ui_mem_fb_free(void *ptr);
- extern size_t ui_mem_fb_get_size(void);
- extern void ui_mem_fb_dump(void);
- extern bool ui_mem_is_fb(const void * ptr);
- #endif /* CONFIG_UI_MEM_NUMBER_BLOCKS */
- #if CONFIG_UI_GUI_MEM_POOL_SIZE > 0
- extern int ui_mem_gui_init(void);
- extern void * ui_mem_gui_alloc(size_t size);
- extern void * ui_mem_gui_aligned_alloc(size_t align, size_t size);
- extern void * ui_mem_gui_realloc(void *ptr, size_t size);
- extern void ui_mem_gui_free(void *ptr);
- extern size_t ui_mem_gui_get_size(void);
- extern void ui_mem_gui_dump(void);
- extern bool ui_mem_is_gui(const void * ptr);
- #endif /* CONFIG_UI_GUI_MEM_POOL_SIZE */
- #if CONFIG_UI_RES_MEM_POOL_SIZE > 0
- extern int ui_mem_res_init(void);
- extern void * ui_mem_res_alloc(size_t size);
- extern void * ui_mem_res_aligned_alloc(size_t align, size_t size);
- extern void * ui_mem_res_realloc(void * ptr, size_t size);
- extern void ui_mem_res_free(void *ptr);
- extern size_t ui_mem_res_get_size(void);
- extern void ui_mem_res_dump(void);
- extern bool ui_mem_is_res(const void * ptr);
- #endif /* CONFIG_UI_RES_MEM_POOL_SIZE */
- struct ui_mem_func {
- const char * name;
- int (* init_fn)(void);
- void * (* alloc_fn)(size_t size);
- void * (* aligned_alloc_fn)(size_t align, size_t size);
- void * (* realloc_fn)(void * ptr, size_t size);
- void (* free_fn)(void * ptr);
- void (* dump_fn)(void);
- bool (* is_type_fn)(const void * ptr);
- size_t (* get_size)(void);
- };
- static const struct ui_mem_func mem_table[NUM_UI_MEM_TYPES] = {
- [MEM_FB] = {
- .name = "FB",
- #if CONFIG_UI_MEM_NUMBER_BLOCKS > 0
- .alloc_fn = ui_mem_fb_alloc,
- .aligned_alloc_fn = ui_mem_fb_aligned_alloc,
- .free_fn = ui_mem_fb_free,
- .dump_fn = ui_mem_fb_dump,
- .is_type_fn = ui_mem_is_fb,
- .get_size = ui_mem_fb_get_size,
- #endif /* CONFIG_UI_MEM_NUMBER_BLOCKS > 0 */
- },
- [MEM_GUI] = {
- .name = "GUI",
- #if CONFIG_UI_GUI_MEM_POOL_SIZE > 0
- .init_fn = ui_mem_gui_init,
- .alloc_fn = ui_mem_gui_alloc,
- .aligned_alloc_fn = ui_mem_gui_aligned_alloc,
- .realloc_fn = ui_mem_gui_realloc,
- .free_fn = ui_mem_gui_free,
- .dump_fn = ui_mem_gui_dump,
- .is_type_fn = ui_mem_is_gui,
- .get_size = ui_mem_gui_get_size,
- #endif /* CONFIG_UI_GUI_MEM_POOL_SIZE > 0 */
- },
- [MEM_RES] = {
- .name = "RES",
- #if CONFIG_UI_RES_MEM_POOL_SIZE > 0
- .init_fn = ui_mem_res_init,
- .alloc_fn = ui_mem_res_alloc,
- .aligned_alloc_fn = ui_mem_res_aligned_alloc,
- .realloc_fn = ui_mem_res_realloc,
- .free_fn = ui_mem_res_free,
- .dump_fn = ui_mem_res_dump,
- .is_type_fn = ui_mem_is_res,
- .get_size = ui_mem_res_get_size,
- #elif CONFIG_UI_GUI_MEM_POOL_SIZE > 0
- .init_fn = NULL,
- .alloc_fn = ui_mem_gui_alloc,
- .aligned_alloc_fn = ui_mem_gui_aligned_alloc,
- .realloc_fn = ui_mem_gui_realloc,
- .free_fn = ui_mem_gui_free,
- .dump_fn = NULL,
- .is_type_fn = NULL, /* mixed res mem should be considered as gui mem */
- .get_size = ui_mem_gui_get_size,
- #endif /* CONFIG_UI_RES_MEM_POOL_SIZE > 0 */
- },
- };
- #ifdef CONFIG_UI_MEMORY_DEBUG
- static struct mem_guard_head mem_guard[NUM_UI_MEM_TYPES];
- #endif
- int ui_mem_init(void)
- {
- int i;
- for (i = 0; i < NUM_UI_MEM_TYPES; i++) {
- if (mem_table[i].init_fn) {
- mem_table[i].init_fn();
- }
- #ifdef CONFIG_UI_MEMORY_DEBUG
- if(mem_table[i].get_size != NULL) {
- mem_guard_init(&mem_guard[i], mem_table[i].get_size());
- }
- #endif
- }
- return 0;
- }
- void * ui_mem_alloc(uint8_t type, size_t size, const void * caller)
- {
- void * ptr = NULL;
- if (size == 0)
- return NULL;
- if (type >= NUM_UI_MEM_TYPES || mem_table[type].alloc_fn == NULL) {
- SYS_LOG_ERR("mem %d alloc %u unsupported (caller " PRI_CALLER ")\n",
- type, (uint32_t)size, (char *)caller);
- return NULL;
- }
- #ifdef CONFIG_UI_MEMORY_DEBUG
- ptr = mem_guard_malloc(&mem_guard[type], mem_table[type].alloc_fn, size, caller);
- #else
- ptr = mem_table[type].alloc_fn(size);
- #endif /* CONFIG_UI_MEMORY_DEBUG */
- if (ptr == NULL) {
- SYS_LOG_ERR("UI-%s alloc %u failed (caller " PRI_CALLER ")\n",
- mem_table[type].name, (uint32_t)size, (char *)caller);
- }
- return ptr;
- }
- void * ui_mem_aligned_alloc(uint8_t type, size_t align, size_t size, const void * caller)
- {
- void * ptr = NULL;
- if (size == 0)
- return NULL;
- if (type >= NUM_UI_MEM_TYPES || mem_table[type].aligned_alloc_fn == NULL) {
- SYS_LOG_ERR("mem %d aligned_alloc %u-%u unsupported (caller " PRI_CALLER ")\n",
- type, (uint32_t)align, (uint32_t)size, (char *)caller);
- return NULL;
- }
- assert((align & (align - 1)) == 0);
- #ifdef CONFIG_UI_MEMORY_DEBUG
- ptr = mem_guard_aligned_alloc(&mem_guard[type], mem_table[type].aligned_alloc_fn, align, size, caller);
- #else
- ptr = mem_table[type].aligned_alloc_fn(align, size);
- #endif /* CONFIG_UI_MEMORY_DEBUG */
- if (ptr == NULL) {
- SYS_LOG_ERR("UI-%s aligned_alloc %u-%u failed (caller " PRI_CALLER ")\n",
- mem_table[type].name, (uint32_t)align, (uint32_t)size, (char *)caller);
- }
- return ptr;
- }
- void * ui_mem_realloc(uint8_t type, void * ptr, size_t size, const void * caller)
- {
- if (type >= NUM_UI_MEM_TYPES) {
- SYS_LOG_ERR("mem %d realloc %u unsupported (caller " PRI_CALLER ")\n",
- type, (uint32_t)size, (char *)caller);
- return NULL;
- }
- if (mem_table[type].realloc_fn == NULL) {
- void * ptr_new = ui_mem_alloc(type, size, caller);
- if (ptr_new) {
- memcpy(ptr_new, ptr, size);
- }
- ui_mem_free(type, ptr);
- ptr = ptr_new;
- } else {
- #ifdef CONFIG_UI_MEMORY_DEBUG
- ptr = mem_guard_realloc(&mem_guard[type], mem_table[type].realloc_fn, ptr, size, caller);
- #else
- ptr = mem_table[type].realloc_fn(ptr, size);
- #endif /* CONFIG_UI_MEMORY_DEBUG */
- }
- if (ptr == NULL) {
- SYS_LOG_ERR("UI-%s realloc %u failed (caller " PRI_CALLER ")\n",
- mem_table[type].name, (uint32_t)size, (char *)caller);
- }
- return ptr;
- }
- void * ui_mem_calloc(uint8_t type, size_t nmemb, size_t size, const void * caller)
- {
- void * ptr = NULL;
- size *= nmemb;
- ptr = ui_mem_alloc(type, size, caller);
- if (ptr) {
- memset(ptr, 0, size);
- }
- return ptr;
- }
- void ui_mem_free(uint8_t type, void * ptr)
- {
- if (ptr == NULL) {
- return;
- }
- if (type >= NUM_UI_MEM_TYPES) {
- SYS_LOG_ERR("mem %d free unsupported\n", type);
- return;
- }
- if (mem_table[type].free_fn) {
- #ifdef CONFIG_UI_MEMORY_DEBUG
- mem_guard_free(&mem_guard[type], mem_table[type].free_fn, ptr);
- #else
- mem_table[type].free_fn(ptr);
- #endif /* CONFIG_UI_MEMORY_DEBUG */
- }
- }
- void ui_mem_free2(void * ptr)
- {
- int i;
- if (ptr == NULL) {
- return;
- }
- for (i = 0; i < NUM_UI_MEM_TYPES; i++) {
- if (mem_table[i].is_type_fn && mem_table[i].is_type_fn(ptr)) {
- if (mem_table[i].free_fn) {
- #ifdef CONFIG_UI_MEMORY_DEBUG
- mem_guard_free(&mem_guard[i], mem_table[i].free_fn, ptr);
- #else
- mem_table[i].free_fn(ptr);
- #endif /* CONFIG_UI_MEMORY_DEBUG */
- }
- return;
- }
- }
- SYS_LOG_ERR("unknown ptr %p to free", ptr);
- }
- bool ui_mem_is_type(uint8_t type, const void * ptr)
- {
- bool is_type = false;
- if (type < NUM_UI_MEM_TYPES && mem_table[type].is_type_fn) {
- is_type = mem_table[type].is_type_fn(ptr);
- }
- return is_type;
- }
- void ui_mem_dump(uint8_t type)
- {
- if (type >= NUM_UI_MEM_TYPES) {
- return;
- }
- #ifdef CONFIG_ACTIONS_PRINTK_DMA
- printk_dma_switch(0);
- #endif /* CONFIG_ACTIONS_PRINTK_DMA */
- if (mem_table[type].dump_fn) {
- os_printk("UI-%s dump:\n", mem_table[type].name);
- #ifdef CONFIG_UI_MEMORY_DEBUG
- mem_guard_dump(&mem_guard[type]);
- #endif /* CONFIG_UI_MEMORY_DEBUG */
- mem_table[type].dump_fn();
- }
- os_printk("\n");
- #ifdef CONFIG_ACTIONS_PRINTK_DMA
- printk_dma_switch(1);
- #endif /* CONFIG_ACTIONS_PRINTK_DMA */
- }
- void ui_mem_dump_all(void)
- {
- int i;
- for (i = 0; i < NUM_UI_MEM_TYPES; i++) {
- ui_mem_dump(i);
- }
- }
- void ui_mem_safe_check(uint16_t view_id)
- {
- #ifdef CONFIG_UI_MEMORY_DEBUG
- int i;
- os_printk("view_id %d mem check:\n", view_id);
- for (i = 0; i < NUM_UI_MEM_TYPES; i++) {
- /* no need to check MEM_FB */
- if (i == MEM_FB) continue;
- os_printk("checking %s:\n", mem_table[i].name);
- mem_guard_leak_check(&mem_guard[i], view_id);
- os_printk("check end\n");
- }
- #endif /* CONFIG_UI_MEMORY_DEBUG */
- }
- #ifdef CONFIG_UI_MEM_VDB_SHARE_SURFACE_BUFFER
- #ifndef CONFIG_SIMULATOR
- __in_section_unique(sram.noinit.sufacebuffer) __aligned(64)
- #endif
- static uint8_t globle_share_surface_buffer[CONFIG_UI_MEM_BLOCK_SIZE];
- void *ui_mem_get_share_surface_buffer(void)
- {
- return &globle_share_surface_buffer;
- }
- int ui_mem_get_share_surface_buffer_size(void)
- {
- return CONFIG_UI_MEM_BLOCK_SIZE;
- }
- #else
- void *ui_mem_get_share_surface_buffer(void) { return NULL; }
- int ui_mem_get_share_surface_buffer_size(void) { return 0; }
- #endif /* CONFIG_UI_MEM_VDB_SHARE_SURFACE_BUFFER */
|