mpsc_pbuf.h 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248
  1. /*
  2. * Copyright (c) 2021 Nordic Semiconductor ASA
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #ifndef ZEPHYR_INCLUDE_SYS_MPSC_PBUF_H_
  7. #define ZEPHYR_INCLUDE_SYS_MPSC_PBUF_H_
  8. #include <kernel.h>
  9. #include <sys/mpsc_packet.h>
  10. #include <string.h>
  11. #include <stdint.h>
  12. #include <stdbool.h>
  13. #ifdef __cplusplus
  14. extern "C" {
  15. #endif
  16. /**
  17. * @brief Multi producer, single consumer packet buffer API
  18. * @defgroup mpsc_buf MPSC (Multi producer, single consumer) packet buffer API
  19. * @ingroup kernel_apis
  20. * @{
  21. */
  22. /*
  23. * Multi producer, single consumer packet buffer allows to allocate variable
  24. * length consecutive space for storing a packet. When space is allocated
  25. * it can be filled by the user (except for the first 2 bits) and when packet
  26. * is ready it is committed. It is allowed to allocate another packet before
  27. * committing the previous one.
  28. *
  29. * If buffer is full and packet cannot be allocated then null is returned unless
  30. * overwrite mode is selected. In that mode, oldest entry are dropped (user is
  31. * notified) until allocation succeeds. It can happen that candidate for
  32. * dropping is currently being claimed. In that case, it is ommited and next
  33. * packet is dropped and claimed packet is marked as invalid when freeing.
  34. *
  35. * Reading packets is performed in two steps. First packet is claimed. Claiming
  36. * returns pointer to the packet within the buffer. Packet is freed when no
  37. * longer in use.
  38. */
  39. /**@defgroup MPSC_PBUF_FLAGS MPSC packet buffer flags
  40. * @{
  41. */
  42. /** @brief Flag indicating that buffer size is power of 2.
  43. *
  44. * When buffer size is power of 2 then optimizations are applied.
  45. */
  46. #define MPSC_PBUF_SIZE_POW2 BIT(0)
  47. /** @brief Flag indicating buffer full policy.
  48. *
  49. * If flag is set then when allocating from a full buffer oldest packets are
  50. * dropped. When flag is not set then allocation returns null.
  51. */
  52. #define MPSC_PBUF_MODE_OVERWRITE BIT(1)
  53. /**@} */
  54. /* Forward declaration */
  55. struct mpsc_pbuf_buffer;
  56. /** @brief Callback prototype for getting length of a packet.
  57. *
  58. * @param packet User packet.
  59. *
  60. * @return Size of the packet in 32 bit words.
  61. */
  62. typedef uint32_t (*mpsc_pbuf_get_wlen)(const union mpsc_pbuf_generic *packet);
  63. /** @brief Callback called when packet is dropped.
  64. *
  65. * @param buffer Packet buffer.
  66. *
  67. * @param packet Packet that is being dropped.
  68. */
  69. typedef void (*mpsc_pbuf_notify_drop)(const struct mpsc_pbuf_buffer *buffer,
  70. const union mpsc_pbuf_generic *packet);
  71. /** @brief MPSC packet buffer structure. */
  72. struct mpsc_pbuf_buffer {
  73. /** Temporary write index. */
  74. uint32_t tmp_wr_idx;
  75. /** Write index. */
  76. uint32_t wr_idx;
  77. /** Temporary read index. */
  78. uint32_t tmp_rd_idx;
  79. /** Read index. */
  80. uint32_t rd_idx;
  81. /** Flags. */
  82. uint32_t flags;
  83. /** Lock. */
  84. struct k_spinlock lock;
  85. /** User callback called whenever packet is dropped. */
  86. mpsc_pbuf_notify_drop notify_drop;
  87. /** Callback for getting packet length. */
  88. mpsc_pbuf_get_wlen get_wlen;
  89. /* Buffer. */
  90. uint32_t *buf;
  91. /* Buffer size in 32 bit words. */
  92. uint32_t size;
  93. struct k_sem sem;
  94. };
  95. /** @brief MPSC packet buffer configuration. */
  96. struct mpsc_pbuf_buffer_config {
  97. /* Pointer to a memory used for storing packets. */
  98. uint32_t *buf;
  99. /* Buffer size in 32 bit words. */
  100. uint32_t size;
  101. /* Callbacks. */
  102. mpsc_pbuf_notify_drop notify_drop;
  103. mpsc_pbuf_get_wlen get_wlen;
  104. /* Configuration flags. */
  105. uint32_t flags;
  106. };
  107. /** @brief Initnialize a packet buffer.
  108. *
  109. * @param buffer Buffer.
  110. *
  111. * @param config Configuration.
  112. */
  113. void mpsc_pbuf_init(struct mpsc_pbuf_buffer *buffer,
  114. const struct mpsc_pbuf_buffer_config *config);
  115. /** @brief Allocate a packet.
  116. *
  117. * If a buffer is configured to overwrite mode then if there is no space to
  118. * allocated a new buffer, oldest packets are dropped. Otherwise allocation
  119. * fails and null pointer is returned.
  120. *
  121. * @param buffer Buffer.
  122. *
  123. * @param wlen Number of words to allocate.
  124. *
  125. * @param timeout Timeout. If called from thread context it will pend for given
  126. * timeout if packet cannot be allocated before dropping the oldest or
  127. * returning null.
  128. *
  129. * @return Pointer to the allocated space or null if it cannot be allocated.
  130. */
  131. union mpsc_pbuf_generic *mpsc_pbuf_alloc(struct mpsc_pbuf_buffer *buffer,
  132. size_t wlen, k_timeout_t timeout);
  133. /** @brief Commit a packet.
  134. *
  135. * @param buffer Buffer.
  136. *
  137. * @param packet Pointer to a packet allocated by @ref mpsc_pbuf_alloc.
  138. */
  139. void mpsc_pbuf_commit(struct mpsc_pbuf_buffer *buffer,
  140. union mpsc_pbuf_generic *packet);
  141. /** @brief Put single word packet into a buffer.
  142. *
  143. * Function is optimized for storing a packet which fit into a single word.
  144. * Note that 2 bits of that word is used by the buffer.
  145. *
  146. * @param buffer Buffer.
  147. *
  148. * @param word Packet content consisting of MPSC_PBUF_HDR with valid bit set
  149. * and data on remaining bits.
  150. */
  151. void mpsc_pbuf_put_word(struct mpsc_pbuf_buffer *buffer,
  152. const union mpsc_pbuf_generic word);
  153. /** @brief Put a packet consisting of a word and a pointer.
  154. * *
  155. * Function is optimized for storing packet consisting of a word and a pointer.
  156. * Note that 2 bits of a first word is used by the buffer.
  157. *
  158. * @param buffer Buffer.
  159. *
  160. * @param word First word of a packet consisting of MPSC_PBUF_HDR with valid
  161. * bit set and data on remaining bits.
  162. *
  163. * @param data User data.
  164. */
  165. void mpsc_pbuf_put_word_ext(struct mpsc_pbuf_buffer *buffer,
  166. const union mpsc_pbuf_generic word,
  167. const void *data);
  168. /** @brief Put a packet into a buffer.
  169. *
  170. * Copy data into a buffer.
  171. * Note that 2 bits of a first word is used by the buffer.
  172. *
  173. * @param buffer Buffer.
  174. *
  175. * @param data First word of data must contain MPSC_PBUF_HDR with valid bit set.
  176. *
  177. * @param wlen Packet size in words.
  178. */
  179. void mpsc_pbuf_put_data(struct mpsc_pbuf_buffer *buffer,
  180. const uint32_t *data, size_t wlen);
  181. /** @brief Claim the first pending packet.
  182. *
  183. * @param buffer Buffer.
  184. *
  185. * @return Pointer to the claimed packet or null if none available.
  186. */
  187. const union mpsc_pbuf_generic *mpsc_pbuf_claim(struct mpsc_pbuf_buffer *buffer);
  188. /** @brief Free a packet.
  189. *
  190. * @param buffer Buffer.
  191. *
  192. * @param packet Packet.
  193. */
  194. void mpsc_pbuf_free(struct mpsc_pbuf_buffer *buffer,
  195. union mpsc_pbuf_generic *packet);
  196. /** @brief Check if there are any message pending.
  197. *
  198. * @param buffer Buffer.
  199. *
  200. * @retval true if pending.
  201. * @retval false if no message is pending.
  202. */
  203. bool mpsc_pbuf_is_pending(struct mpsc_pbuf_buffer *buffer);
  204. /**
  205. * @}
  206. */
  207. #ifdef __cplusplus
  208. }
  209. #endif
  210. #endif /* ZEPHYR_INCLUDE_SYS_MPSC_PBUF_H_ */