p4wq.h 6.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210
  1. /*
  2. * Copyright (c) 2020 Intel Corporation
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #ifndef ZEPHYR_INCLUDE_SYS_P4WQ_H_
  7. #define ZEPHYR_INCLUDE_SYS_P4WQ_H_
  8. #include <kernel.h>
  9. /* Zephyr Pooled Parallel Preemptible Priority-based Work Queues */
  10. struct k_p4wq_work;
  11. /**
  12. * P4 Queue handler callback
  13. */
  14. typedef void (*k_p4wq_handler_t)(struct k_p4wq_work *work);
  15. /**
  16. * @brief P4 Queue Work Item
  17. *
  18. * User-populated struct representing a single work item. The
  19. * priority and deadline fields are interpreted as thread scheduling
  20. * priorities, exactly as per k_thread_priority_set() and
  21. * k_thread_deadline_set().
  22. */
  23. struct k_p4wq_work {
  24. /* Filled out by submitting code */
  25. int32_t priority;
  26. int32_t deadline;
  27. k_p4wq_handler_t handler;
  28. bool sync;
  29. struct k_sem done_sem;
  30. /* reserved for implementation */
  31. union {
  32. struct rbnode rbnode;
  33. sys_dlist_t dlnode;
  34. };
  35. struct k_thread *thread;
  36. struct k_p4wq *queue;
  37. };
  38. #define K_P4WQ_QUEUE_PER_THREAD BIT(0)
  39. #define K_P4WQ_DELAYED_START BIT(1)
  40. #define K_P4WQ_USER_CPU_MASK BIT(2)
  41. /**
  42. * @brief P4 Queue
  43. *
  44. * Kernel pooled parallel preemptible priority-based work queue
  45. */
  46. struct k_p4wq {
  47. struct k_spinlock lock;
  48. /* Pending threads waiting for work items
  49. *
  50. * FIXME: a waitq isn't really the right data structure here.
  51. * Wait queues are priority-sorted, but we don't want that
  52. * sorting overhead since we're effectively doing it ourselves
  53. * by directly mutating the priority when a thread is
  54. * unpended. We just want "blocked threads on a list", but
  55. * there's no clean scheduler API for that.
  56. */
  57. _wait_q_t waitq;
  58. /* Work items waiting for processing */
  59. struct rbtree queue;
  60. /* Work items in progress */
  61. sys_dlist_t active;
  62. /* K_P4WQ_* flags above */
  63. uint32_t flags;
  64. };
  65. struct k_p4wq_initparam {
  66. uint32_t num;
  67. uintptr_t stack_size;
  68. struct k_p4wq *queue;
  69. struct k_thread *threads;
  70. struct z_thread_stack_element *stacks;
  71. uint32_t flags;
  72. };
  73. /**
  74. * @brief Statically initialize a P4 Work Queue
  75. *
  76. * Statically defines a struct k_p4wq object with the specified number
  77. * of threads which will be initialized at boot and ready for use on
  78. * entry to main().
  79. *
  80. * @param name Symbol name of the struct k_p4wq that will be defined
  81. * @param n_threads Number of threads in the work queue pool
  82. * @param stack_sz Requested stack size of each thread, in bytes
  83. */
  84. #define K_P4WQ_DEFINE(name, n_threads, stack_sz) \
  85. static K_THREAD_STACK_ARRAY_DEFINE(_p4stacks_##name, \
  86. n_threads, stack_sz); \
  87. static struct k_thread _p4threads_##name[n_threads]; \
  88. static struct k_p4wq name; \
  89. static const STRUCT_SECTION_ITERABLE(k_p4wq_initparam, \
  90. _init_##name) = { \
  91. .num = n_threads, \
  92. .stack_size = stack_sz, \
  93. .threads = _p4threads_##name, \
  94. .stacks = &(_p4stacks_##name[0][0]), \
  95. .queue = &name, \
  96. .flags = 0, \
  97. }
  98. /**
  99. * @brief Statically initialize an array of P4 Work Queues
  100. *
  101. * Statically defines an array of struct k_p4wq objects with the specified
  102. * number of threads which will be initialized at boot and ready for use on
  103. * entry to main().
  104. *
  105. * @param name Symbol name of the struct k_p4wq array that will be defined
  106. * @param n_threads Number of threads and work queues
  107. * @param stack_sz Requested stack size of each thread, in bytes
  108. * @param flg Flags
  109. */
  110. #define K_P4WQ_ARRAY_DEFINE(name, n_threads, stack_sz, flg) \
  111. static K_THREAD_STACK_ARRAY_DEFINE(_p4stacks_##name, \
  112. n_threads, stack_sz); \
  113. static struct k_thread _p4threads_##name[n_threads]; \
  114. static struct k_p4wq name[n_threads]; \
  115. static const STRUCT_SECTION_ITERABLE(k_p4wq_initparam, \
  116. _init_##name) = { \
  117. .num = n_threads, \
  118. .stack_size = stack_sz, \
  119. .threads = _p4threads_##name, \
  120. .stacks = &(_p4stacks_##name[0][0]), \
  121. .queue = name, \
  122. .flags = K_P4WQ_QUEUE_PER_THREAD | flg, \
  123. }
  124. /**
  125. * @brief Initialize P4 Queue
  126. *
  127. * Initializes a P4 Queue object. These objects must be initialized
  128. * via this function (or statically using K_P4WQ_DEFINE) before any
  129. * other API calls are made on it.
  130. *
  131. * @param queue P4 Queue to initialize
  132. */
  133. void k_p4wq_init(struct k_p4wq *queue);
  134. /**
  135. * @brief Dynamically add a thread object to a P4 Queue pool
  136. *
  137. * Adds a thread to the pool managed by a P4 queue. The thread object
  138. * must not be in use. If k_thread_create() has previously been
  139. * called on it, it must be aborted before being given to the queue.
  140. *
  141. * @param queue P4 Queue to which to add the thread
  142. * @param thread Uninitialized/aborted thread object to add
  143. * @param stack Thread stack memory
  144. * @param stack_size Thread stack size
  145. */
  146. void k_p4wq_add_thread(struct k_p4wq *queue, struct k_thread *thread,
  147. k_thread_stack_t *stack,
  148. size_t stack_size);
  149. /**
  150. * @brief Submit work item to a P4 queue
  151. *
  152. * Submits the specified work item to the queue. The caller must have
  153. * already initialized the relevant fields of the struct. The queue
  154. * will execute the handler when CPU time is available and when no
  155. * higher-priority work items are available. The handler may be
  156. * invoked on any CPU.
  157. *
  158. * The caller must not mutate the struct while it is stored in the
  159. * queue. The memory should remain unchanged until k_p4wq_cancel() is
  160. * called or until the entry to the handler function.
  161. *
  162. * @note This call is a scheduling point, so if the submitted item (or
  163. * any other ready thread) has a higher priority than the current
  164. * thread and the current thread has a preemptible priority then the
  165. * caller will yield.
  166. *
  167. * @param queue P4 Queue to which to submit
  168. * @param item P4 work item to be submitted
  169. */
  170. void k_p4wq_submit(struct k_p4wq *queue, struct k_p4wq_work *item);
  171. /**
  172. * @brief Cancel submitted P4 work item
  173. *
  174. * Cancels a previously-submitted work item and removes it from the
  175. * queue. Returns true if the item was found in the queue and
  176. * removed. If the function returns false, either the item was never
  177. * submitted, has already been executed, or is still running.
  178. *
  179. * @return true if the item was successfully removed, otherwise false
  180. */
  181. bool k_p4wq_cancel(struct k_p4wq *queue, struct k_p4wq_work *item);
  182. /**
  183. * @brief Regain ownership of the work item, wait for completion if it's synchronous
  184. */
  185. int k_p4wq_wait(struct k_p4wq_work *work, k_timeout_t timeout);
  186. void k_p4wq_enable_static_thread(struct k_p4wq *queue, struct k_thread *thread,
  187. uint32_t cpu_mask);
  188. #endif /* ZEPHYR_INCLUDE_SYS_P4WQ_H_ */