ui_mem_fb.c 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209
  1. /*
  2. * Copyright (c) 2019 Actions Semi Co., Inc.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief mem slab manager.
  9. */
  10. #define LOG_MODULE_CUSTOMER
  11. #include <os_common_api.h>
  12. #include <ui_mem.h>
  13. #include <ui_math.h>
  14. #ifdef CONFIG_UI_MEMORY_DEBUG
  15. # include <mem_guard.h>
  16. #endif
  17. LOG_MODULE_DECLARE(ui_mem, LOG_LEVEL_INF);
  18. #if defined(CONFIG_UI_MEM_NUMBER_BLOCKS) && CONFIG_UI_MEM_NUMBER_BLOCKS > 0
  19. #if CONFIG_UI_MEM_NUMBER_BLOCKS >= 256
  20. # error "CONFIG_UI_MEM_NUMBER_BLOCKS >= 256"
  21. #endif
  22. /*
  23. * memory alignment
  24. * 1) must align to psram cache line size (32 bytes)
  25. * 2) Verisilicon vg_lite buffer memory requires 64 bytes aligned
  26. */
  27. #ifdef CONFIG_VG_LITE
  28. # define UI_MEM_BLOCK_ALIGN 64
  29. #else
  30. # define UI_MEM_BLOCK_ALIGN 32
  31. #endif
  32. /* align to psram cache line size (32 bytes) */
  33. #ifdef CONFIG_UI_MEMORY_DEBUG
  34. # define UI_MEM_BLOCK_SIZE UI_ROUND_UP(CONFIG_UI_MEM_BLOCK_SIZE + MEM_STAKE_SIZE, UI_MEM_BLOCK_ALIGN)
  35. #else
  36. # define UI_MEM_BLOCK_SIZE UI_ROUND_UP(CONFIG_UI_MEM_BLOCK_SIZE, UI_MEM_BLOCK_ALIGN)
  37. #endif
  38. #define UI_MEM_SIZE (CONFIG_UI_MEM_NUMBER_BLOCKS * UI_MEM_BLOCK_SIZE)
  39. static uint8_t __aligned(UI_MEM_BLOCK_ALIGN) ui_mem_base[UI_MEM_SIZE] __in_section_unique(UI_PSRAM_REGION);
  40. /* store the allocated continuous block counts (max 255) from the index */
  41. static uint8_t alloc_count[CONFIG_UI_MEM_NUMBER_BLOCKS];
  42. #ifndef CONFIG_SIMULATOR
  43. static struct k_spinlock alloc_spinlock;
  44. #else
  45. static OS_MUTEX_DEFINE(alloc_mutex);
  46. #endif
  47. static void *ui_mem_fb_alloc_internal(size_t size)
  48. {
  49. uint8_t nbr_blks = 1;
  50. uint8_t free_blks = 0;
  51. uint8_t first_blk = 0;
  52. #ifndef CONFIG_SIMULATOR
  53. k_spinlock_key_t key;
  54. #endif
  55. if (size == 0) {
  56. return NULL;
  57. }
  58. if (size > UI_MEM_BLOCK_SIZE) {
  59. nbr_blks = (size + UI_MEM_BLOCK_SIZE - 1) / UI_MEM_BLOCK_SIZE;
  60. if (nbr_blks > CONFIG_UI_MEM_NUMBER_BLOCKS) {
  61. return NULL;
  62. }
  63. }
  64. #ifndef CONFIG_SIMULATOR
  65. key = k_spin_lock(&alloc_spinlock);
  66. #else
  67. os_mutex_lock(&alloc_mutex, OS_FOREVER);
  68. #endif
  69. for (int i = 0; i < CONFIG_UI_MEM_NUMBER_BLOCKS;) {
  70. if (alloc_count[i] == 0) {
  71. if (free_blks == 0) {
  72. first_blk = i;
  73. }
  74. if (++free_blks == nbr_blks) {
  75. alloc_count[first_blk] = nbr_blks;
  76. break;
  77. }
  78. i++;
  79. } else {
  80. free_blks = 0;
  81. i += alloc_count[i];
  82. if (i + nbr_blks > CONFIG_UI_MEM_NUMBER_BLOCKS) {
  83. break;
  84. }
  85. }
  86. }
  87. #ifndef CONFIG_SIMULATOR
  88. k_spin_unlock(&alloc_spinlock, key);
  89. #else
  90. os_mutex_unlock(&alloc_mutex);
  91. #endif
  92. if (free_blks == nbr_blks) {
  93. return ui_mem_base + first_blk * UI_MEM_BLOCK_SIZE;
  94. }
  95. return NULL;
  96. }
  97. void *ui_mem_fb_alloc(size_t size)
  98. {
  99. void *ptr = NULL;
  100. int try = 30;
  101. do {
  102. ptr = ui_mem_fb_alloc_internal(size);
  103. if (ptr)
  104. break;
  105. os_sleep(2);
  106. } while (--try > 0);
  107. return ptr;
  108. }
  109. void * ui_mem_fb_aligned_alloc(size_t align, size_t size)
  110. {
  111. return (align <= UI_MEM_BLOCK_ALIGN) ? ui_mem_fb_alloc(size) : NULL;
  112. }
  113. void ui_mem_fb_free(void *ptr)
  114. {
  115. uint8_t *ptr8 = ptr;
  116. uint8_t blkidx;
  117. #ifndef CONFIG_SIMULATOR
  118. k_spinlock_key_t key;
  119. #endif
  120. if (ptr8 < ui_mem_base || ptr8 >= ui_mem_base + UI_MEM_SIZE) {
  121. return;
  122. }
  123. blkidx = (uint32_t)(ptr8 - ui_mem_base) / UI_MEM_BLOCK_SIZE;
  124. #ifndef CONFIG_SIMULATOR
  125. key = k_spin_lock(&alloc_spinlock);
  126. __ASSERT(alloc_count[blkidx] > 0, "double-free for memory at %p", ptr8);
  127. #else
  128. os_mutex_lock(&alloc_mutex, OS_FOREVER);
  129. #endif
  130. alloc_count[blkidx] = 0;
  131. #ifndef CONFIG_SIMULATOR
  132. k_spin_unlock(&alloc_spinlock, key);
  133. #else
  134. os_mutex_unlock(&alloc_mutex);
  135. #endif
  136. }
  137. size_t ui_mem_fb_get_size(void)
  138. {
  139. return UI_MEM_SIZE;
  140. }
  141. void ui_mem_fb_dump(void)
  142. {
  143. os_printk("FB heap at %p, block size %lu, count %u\n", ui_mem_base,
  144. UI_MEM_BLOCK_SIZE, CONFIG_UI_MEM_NUMBER_BLOCKS);
  145. for (int i = 0; i < CONFIG_UI_MEM_NUMBER_BLOCKS;) {
  146. if (alloc_count[i] > 0) {
  147. void *ptr = ui_mem_base + i * UI_MEM_BLOCK_SIZE;
  148. os_printk("ptr %p, %u blocks\n", ptr, alloc_count[i]);
  149. i += alloc_count[i];
  150. } else {
  151. i += 1;
  152. }
  153. }
  154. }
  155. bool ui_mem_is_fb(const void * ptr)
  156. {
  157. const uint8_t *mem_end = ui_mem_base + UI_MEM_SIZE;
  158. const uint8_t *ptr8 = ptr;
  159. return (ptr8 >= ui_mem_base && ptr8 < mem_end) ? true : false;
  160. }
  161. #endif /* CONFIG_UI_MEM_NUMBER_BLOCKS > 0 */
  162. size_t ui_mem_fb_get_num(void)
  163. {
  164. return CONFIG_UI_MEM_NUMBER_BLOCKS;
  165. }