de_common.c 5.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234
  1. /*
  2. * Copyright (c) 2020 Actions Technology Co., Ltd
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #include <assert.h>
  7. #include <errno.h>
  8. #include <string.h>
  9. #include <sys/atomic.h>
  10. #include "de_common.h"
  11. struct de_command_pool {
  12. sys_slist_t list;
  13. struct k_sem sem;
  14. struct k_spinlock lock;
  15. };
  16. struct de_instance {
  17. uint16_t flags;
  18. uint16_t sequence;
  19. bool waiting;
  20. atomic_t alloc_num;
  21. struct k_sem sem;
  22. struct k_spinlock lock;
  23. struct de_command_pool *pool;
  24. display_engine_instance_callback_t callback;
  25. void *user_data;
  26. };
  27. static struct de_command_entry *de_command_pool_alloc(struct de_command_pool *pool);
  28. static void de_command_pool_free(struct de_command_pool *pool, struct de_command_entry *entry);
  29. static struct de_command_entry __in_section_unique(ram.noinit.drv_de.cmd)
  30. de_entries[CONFIG_DISPLAY_ENGINE_DRAW_COMMAND_NUM + CONFIG_DISPLAY_ENGINE_POST_COMMAND_NUM];
  31. static struct de_command_pool de_draw_pool;
  32. #if CONFIG_DISPLAY_ENGINE_POST_COMMAND_NUM > 0
  33. static struct de_command_pool de_post_pool;
  34. #endif
  35. static struct de_instance de_instances[CONFIG_DISPLAY_ENGINE_INSTANCE_NUM];
  36. int de_alloc_instance(uint32_t flags)
  37. {
  38. for (int i = 0; i < ARRAY_SIZE(de_instances); i++) {
  39. if (de_instances[i].pool == NULL) {
  40. de_instances[i].flags = (uint16_t)flags;
  41. #if CONFIG_DISPLAY_ENGINE_POST_COMMAND_NUM > 0
  42. if (flags & DISPLAY_ENGINE_FLAG_POST) {
  43. de_instances[i].pool = &de_post_pool;
  44. } else
  45. #endif
  46. {
  47. de_instances[i].pool = &de_draw_pool;
  48. }
  49. k_sem_init(&de_instances[i].sem, 0, 1);
  50. memset(&de_instances[i].lock, 0, sizeof(de_instances[i].lock));
  51. return i;
  52. }
  53. }
  54. return -ENOBUFS;
  55. }
  56. int de_free_instance(int inst)
  57. {
  58. assert(inst >= 0 && inst < ARRAY_SIZE(de_instances));
  59. assert(de_instances[inst].pool != NULL);
  60. memset(&de_instances[inst], 0, sizeof(de_instances[inst]));
  61. return 0;
  62. }
  63. bool de_instance_has_flag(int inst, uint32_t flag)
  64. {
  65. assert(inst >= 0 && inst < ARRAY_SIZE(de_instances));
  66. assert(de_instances[inst].pool != NULL);
  67. return (de_instances[inst].flags & flag);
  68. }
  69. __de_func struct de_command_entry *de_instance_alloc_entry(int inst)
  70. {
  71. struct de_command_entry *entry;
  72. assert(inst >= 0 && inst < ARRAY_SIZE(de_instances));
  73. assert(de_instances[inst].pool != NULL);
  74. if (inst < 0 || inst >= ARRAY_SIZE(de_instances) ||
  75. de_instances[inst].pool == NULL) {
  76. return NULL;
  77. }
  78. entry = de_command_pool_alloc(de_instances[inst].pool);
  79. assert (entry != NULL);
  80. if (entry) {
  81. entry->inst = (uint8_t)inst;
  82. entry->seq = de_instances[inst].sequence++;
  83. atomic_inc(&de_instances[inst].alloc_num);
  84. }
  85. return entry;
  86. }
  87. __de_func int de_instance_free_entry(struct de_command_entry *entry)
  88. {
  89. int inst = entry->inst;
  90. bool waiting;
  91. assert(inst >= 0 && inst < ARRAY_SIZE(de_instances));
  92. assert(de_instances[inst].pool != NULL);
  93. atomic_dec(&de_instances[inst].alloc_num);
  94. k_spinlock_key_t key = k_spin_lock(&de_instances[inst].lock);
  95. waiting = de_instances[inst].waiting;
  96. k_spin_unlock(&de_instances[inst].lock, key);
  97. if (waiting) {
  98. k_sem_give(&de_instances[inst].sem);
  99. }
  100. de_command_pool_free(de_instances[inst].pool, entry);
  101. return 0;
  102. }
  103. __de_func int de_instance_poll(int inst, int timeout)
  104. {
  105. k_spinlock_key_t key;
  106. assert(inst >= 0 && inst < ARRAY_SIZE(de_instances));
  107. assert(de_instances[inst].pool != NULL);
  108. wait_finish:
  109. key = k_spin_lock(&de_instances[inst].lock);
  110. de_instances[inst].waiting = (atomic_get(&de_instances[inst].alloc_num) > 0);
  111. k_spin_unlock(&de_instances[inst].lock, key);
  112. if (de_instances[inst].waiting) {
  113. k_sem_take(&de_instances[inst].sem, (timeout >= 0) ? K_MSEC(timeout) : K_FOREVER);
  114. de_instances[inst].waiting = 0;
  115. goto wait_finish;
  116. }
  117. return atomic_get(&de_instances[inst].alloc_num) ? -ETIMEDOUT : 0;
  118. }
  119. int de_instance_register_callback(int inst, display_engine_instance_callback_t callback, void *user_data)
  120. {
  121. assert(inst >= 0 && inst < ARRAY_SIZE(de_instances));
  122. assert(de_instances[inst].pool != NULL);
  123. de_instances[inst].user_data = user_data;
  124. de_instances[inst].callback = callback;
  125. return 0;
  126. }
  127. __de_func int de_instance_notify(struct de_command_entry *entry, int status)
  128. {
  129. int inst = entry->inst;
  130. assert(inst >= 0 && inst < ARRAY_SIZE(de_instances));
  131. assert(de_instances[inst].pool != NULL);
  132. if (de_instances[inst].callback) {
  133. de_instances[inst].callback(status, entry->seq, de_instances[inst].user_data);
  134. }
  135. return 0;
  136. }
  137. static void de_command_pool_init(struct de_command_pool *pool,
  138. struct de_command_entry *entries, int num_entries)
  139. {
  140. sys_slist_init(&pool->list);
  141. for (int i = num_entries - 1; i >= 0; i--) {
  142. sys_slist_append(&pool->list, &entries[i].node);
  143. }
  144. k_sem_init(&pool->sem, num_entries, num_entries);
  145. }
  146. void de_command_pools_init(void)
  147. {
  148. de_command_pool_init(&de_draw_pool, de_entries, CONFIG_DISPLAY_ENGINE_DRAW_COMMAND_NUM);
  149. #if CONFIG_DISPLAY_ENGINE_POST_COMMAND_NUM > 0
  150. de_command_pool_init(&de_post_pool, &de_entries[CONFIG_DISPLAY_ENGINE_DRAW_COMMAND_NUM], CONFIG_DISPLAY_ENGINE_POST_COMMAND_NUM);
  151. #endif
  152. }
  153. __de_func static struct de_command_entry *de_command_pool_alloc(struct de_command_pool *pool)
  154. {
  155. sys_snode_t *node = NULL;
  156. k_spinlock_key_t key;
  157. get_free_entry:
  158. key = k_spin_lock(&pool->lock);
  159. node = sys_slist_peek_head(&pool->list);
  160. if (node)
  161. sys_slist_remove(&pool->list, NULL, node);
  162. k_spin_unlock(&pool->lock, key);
  163. if (node == NULL && !k_is_in_isr()) {
  164. k_sem_take(&pool->sem, K_FOREVER);
  165. goto get_free_entry;
  166. }
  167. assert (node != NULL);
  168. return CONTAINER_OF(node, struct de_command_entry, node);
  169. }
  170. __de_func static void de_command_pool_free(struct de_command_pool *pool, struct de_command_entry *entry)
  171. {
  172. bool empty;
  173. k_spinlock_key_t key = k_spin_lock(&pool->lock);
  174. empty = sys_slist_is_empty(&pool->list);
  175. sys_slist_append(&pool->list, &entry->node);
  176. k_spin_unlock(&pool->lock, key);
  177. if (empty) {
  178. k_sem_give(&pool->sem);
  179. }
  180. }