notify.h 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359
  1. /*
  2. * Copyright (c) 2019 Peter Bigot Consulting, LLC
  3. * Copyright (c) 2020 Nordic Semiconductor ASA
  4. *
  5. * SPDX-License-Identifier: Apache-2.0
  6. */
  7. #ifndef ZEPHYR_INCLUDE_SYS_NOTIFY_H_
  8. #define ZEPHYR_INCLUDE_SYS_NOTIFY_H_
  9. #include <kernel.h>
  10. #include <zephyr/types.h>
  11. #ifdef __cplusplus
  12. extern "C" {
  13. #endif
  14. struct sys_notify;
  15. /*
  16. * Flag value that overwrites the method field when the operation has
  17. * completed.
  18. */
  19. #define SYS_NOTIFY_METHOD_COMPLETED 0
  20. /*
  21. * Indicates that no notification will be provided.
  22. *
  23. * Callers must check for completions using
  24. * sys_notify_fetch_result().
  25. *
  26. * See sys_notify_init_spinwait().
  27. */
  28. #define SYS_NOTIFY_METHOD_SPINWAIT 1
  29. /*
  30. * Select notification through @ref k_poll signal
  31. *
  32. * See sys_notify_init_signal().
  33. */
  34. #define SYS_NOTIFY_METHOD_SIGNAL 2
  35. /*
  36. * Select notification through a user-provided callback.
  37. *
  38. * See sys_notify_init_callback().
  39. */
  40. #define SYS_NOTIFY_METHOD_CALLBACK 3
  41. #define SYS_NOTIFY_METHOD_MASK 0x03U
  42. #define SYS_NOTIFY_METHOD_POS 0
  43. /**
  44. * @brief Identify the region of sys_notify flags available for
  45. * containing services.
  46. *
  47. * Bits of the flags field of the sys_notify structure at and above
  48. * this position may be used by extensions to the sys_notify
  49. * structure.
  50. *
  51. * These bits are intended for use by containing service
  52. * implementations to record client-specific information. The bits
  53. * are cleared by sys_notify_validate(). Use of these does not
  54. * imply that the flags field becomes public API.
  55. */
  56. #define SYS_NOTIFY_EXTENSION_POS 2
  57. /*
  58. * Mask isolating the bits of sys_notify::flags that are available
  59. * for extension.
  60. */
  61. #define SYS_NOTIFY_EXTENSION_MASK (~BIT_MASK(SYS_NOTIFY_EXTENSION_POS))
  62. /**
  63. * @defgroup sys_notify_apis Asynchronous Notification APIs
  64. * @ingroup kernel_apis
  65. * @{
  66. */
  67. /**
  68. * @brief Generic signature used to notify of result completion by
  69. * callback.
  70. *
  71. * Functions with this role may be invoked from any context including
  72. * pre-kernel, ISR, or cooperative or pre-emptible threads.
  73. * Compatible functions must be isr-ok and not sleep.
  74. *
  75. * Parameters that should generally be passed to such functions include:
  76. *
  77. * * a pointer to a specific client request structure, i.e. the one
  78. * that contains the sys_notify structure.
  79. * * the result of the operation, either as passed to
  80. * sys_notify_finalize() or extracted afterwards using
  81. * sys_notify_fetch_result(). Expected values are
  82. * service-specific, but the value shall be non-negative if the
  83. * operation succeeded, and negative if the operation failed.
  84. */
  85. typedef void (*sys_notify_generic_callback)();
  86. /**
  87. * @brief State associated with notification for an asynchronous
  88. * operation.
  89. *
  90. * Objects of this type are allocated by a client, which must use an
  91. * initialization function (e.g. sys_notify_init_signal()) to
  92. * configure them. Generally the structure is a member of a
  93. * service-specific client structure, such as onoff_client.
  94. *
  95. * Control of the containing object transfers to the service provider
  96. * when a pointer to the object is passed to a service function that
  97. * is documented to take control of the object, such as
  98. * onoff_service_request(). While the service provider controls the
  99. * object the client must not change any object fields. Control
  100. * reverts to the client:
  101. * * if the call to the service API returns an error;
  102. * * when operation completion is posted. This may occur before the
  103. * call to the service API returns.
  104. *
  105. * Operation completion is technically posted when the flags field is
  106. * updated so that sys_notify_fetch_result() returns success. This
  107. * will happen before the signal is posted or callback is invoked.
  108. * Note that although the manager will no longer reference the
  109. * sys_notify object past this point, the containing object may have
  110. * state that will be referenced within the callback. Where callbacks
  111. * are used control of the containing object does not revert to the
  112. * client until the callback has been invoked. (Re-use within the
  113. * callback is explicitly permitted.)
  114. *
  115. * After control has reverted to the client the notify object must be
  116. * reinitialized for the next operation.
  117. *
  118. * The content of this structure is not public API to clients: all
  119. * configuration and inspection should be done with functions like
  120. * sys_notify_init_callback() and sys_notify_fetch_result().
  121. * However, services that use this structure may access certain
  122. * fields directly.
  123. */
  124. struct sys_notify {
  125. union method {
  126. /* Pointer to signal used to notify client.
  127. *
  128. * The signal value corresponds to the res parameter
  129. * of sys_notify_callback.
  130. */
  131. struct k_poll_signal *signal;
  132. /* Generic callback function for callback notification. */
  133. sys_notify_generic_callback callback;
  134. } method;
  135. /*
  136. * Flags recording information about the operation.
  137. *
  138. * Bits below SYS_NOTIFY_EXTENSION_POS are initialized by
  139. * async notify API init functions like
  140. * sys_notify_init_callback(), and must not by modified by
  141. * extensions or client code.
  142. *
  143. * Bits at and above SYS_NOTIFY_EXTENSION_POS are available
  144. * for use by service extensions while the containing object
  145. * is managed by the service. They are not for client use,
  146. * are zeroed by the async notify API init functions, and will
  147. * be zeroed by sys_notify_finalize().
  148. */
  149. uint32_t volatile flags;
  150. /*
  151. * The result of the operation.
  152. *
  153. * This is the value that was (or would be) passed to the
  154. * async infrastructure. This field is the sole record of
  155. * success or failure for spin-wait synchronous operations.
  156. */
  157. int volatile result;
  158. };
  159. /** @internal */
  160. static inline uint32_t sys_notify_get_method(const struct sys_notify *notify)
  161. {
  162. uint32_t method = notify->flags >> SYS_NOTIFY_METHOD_POS;
  163. return method & SYS_NOTIFY_METHOD_MASK;
  164. }
  165. /**
  166. * @brief Validate and initialize the notify structure.
  167. *
  168. * This should be invoked at the start of any service-specific
  169. * configuration validation. It ensures that the basic asynchronous
  170. * notification configuration is consistent, and clears the result.
  171. *
  172. * Note that this function does not validate extension bits (zeroed by
  173. * async notify API init functions like sys_notify_init_callback()).
  174. * It may fail to recognize that an uninitialized structure has been
  175. * passed because only method bits of flags are tested against method
  176. * settings. To reduce the chance of accepting an uninititalized
  177. * operation service validation of structures that contain an
  178. * sys_notify instance should confirm that the extension bits are
  179. * set or cleared as expected.
  180. *
  181. * @retval 0 on successful validation and reinitialization
  182. * @retval -EINVAL if the configuration is not valid.
  183. */
  184. int sys_notify_validate(struct sys_notify *notify);
  185. /**
  186. * @brief Record and signal the operation completion.
  187. *
  188. * @param notify pointer to the notification state structure.
  189. *
  190. * @param res the result of the operation. Expected values are
  191. * service-specific, but the value shall be non-negative if the
  192. * operation succeeded, and negative if the operation failed.
  193. *
  194. * @return If the notification is to be done by callback this returns
  195. * the generic version of the function to be invoked. The caller must
  196. * immediately invoke that function with whatever arguments are
  197. * expected by the callback. If notification is by spin-wait or
  198. * signal, the notification has been completed by the point this
  199. * function returns, and a null pointer is returned.
  200. */
  201. sys_notify_generic_callback sys_notify_finalize(struct sys_notify *notify,
  202. int res);
  203. /**
  204. * @brief Check for and read the result of an asynchronous operation.
  205. *
  206. * @param notify pointer to the object used to specify asynchronous
  207. * function behavior and store completion information.
  208. *
  209. * @param result pointer to storage for the result of the operation.
  210. * The result is stored only if the operation has completed.
  211. *
  212. * @retval 0 if the operation has completed.
  213. * @retval -EAGAIN if the operation has not completed.
  214. */
  215. static inline int sys_notify_fetch_result(const struct sys_notify *notify,
  216. int *result)
  217. {
  218. __ASSERT_NO_MSG(notify != NULL);
  219. __ASSERT_NO_MSG(result != NULL);
  220. int rv = -EAGAIN;
  221. if (sys_notify_get_method(notify) == SYS_NOTIFY_METHOD_COMPLETED) {
  222. rv = 0;
  223. *result = notify->result;
  224. }
  225. return rv;
  226. }
  227. /**
  228. * @brief Initialize a notify object for spin-wait notification.
  229. *
  230. * Clients that use this initialization receive no asynchronous
  231. * notification, and instead must periodically check for completion
  232. * using sys_notify_fetch_result().
  233. *
  234. * On completion of the operation the client object must be
  235. * reinitialized before it can be re-used.
  236. *
  237. * @param notify pointer to the notification configuration object.
  238. */
  239. static inline void sys_notify_init_spinwait(struct sys_notify *notify)
  240. {
  241. __ASSERT_NO_MSG(notify != NULL);
  242. *notify = (struct sys_notify){
  243. .flags = SYS_NOTIFY_METHOD_SPINWAIT,
  244. };
  245. }
  246. /**
  247. * @brief Initialize a notify object for (k_poll) signal notification.
  248. *
  249. * Clients that use this initialization will be notified of the
  250. * completion of operations through the provided signal.
  251. *
  252. * On completion of the operation the client object must be
  253. * reinitialized before it can be re-used.
  254. *
  255. * @note
  256. * This capability is available only when @kconfig{CONFIG_POLL} is
  257. * selected.
  258. *
  259. * @param notify pointer to the notification configuration object.
  260. *
  261. * @param sigp pointer to the signal to use for notification. The
  262. * value must not be null. The signal must be reset before the client
  263. * object is passed to the on-off service API.
  264. */
  265. static inline void sys_notify_init_signal(struct sys_notify *notify,
  266. struct k_poll_signal *sigp)
  267. {
  268. __ASSERT_NO_MSG(notify != NULL);
  269. __ASSERT_NO_MSG(sigp != NULL);
  270. *notify = (struct sys_notify){
  271. .method = {
  272. .signal = sigp,
  273. },
  274. .flags = SYS_NOTIFY_METHOD_SIGNAL,
  275. };
  276. }
  277. /**
  278. * @brief Initialize a notify object for callback notification.
  279. *
  280. * Clients that use this initialization will be notified of the
  281. * completion of operations through the provided callback. Note that
  282. * callbacks may be invoked from various contexts depending on the
  283. * specific service; see @ref sys_notify_generic_callback.
  284. *
  285. * On completion of the operation the client object must be
  286. * reinitialized before it can be re-used.
  287. *
  288. * @param notify pointer to the notification configuration object.
  289. *
  290. * @param handler a function pointer to use for notification.
  291. */
  292. static inline void sys_notify_init_callback(struct sys_notify *notify,
  293. sys_notify_generic_callback handler)
  294. {
  295. __ASSERT_NO_MSG(notify != NULL);
  296. __ASSERT_NO_MSG(handler != NULL);
  297. *notify = (struct sys_notify){
  298. .method = {
  299. .callback = handler,
  300. },
  301. .flags = SYS_NOTIFY_METHOD_CALLBACK,
  302. };
  303. }
  304. /**
  305. * @brief Detect whether a particular notification uses a callback.
  306. *
  307. * The generic handler does not capture the signature expected by the
  308. * callback, and the translation to a service-specific callback must
  309. * be provided by the service. This check allows abstracted services
  310. * to reject callback notification requests when the service doesn't
  311. * provide a translation function.
  312. *
  313. * @return true if and only if a callback is to be used for notification.
  314. */
  315. static inline bool sys_notify_uses_callback(const struct sys_notify *notify)
  316. {
  317. __ASSERT_NO_MSG(notify != NULL);
  318. return sys_notify_get_method(notify) == SYS_NOTIFY_METHOD_CALLBACK;
  319. }
  320. /** @} */
  321. #ifdef __cplusplus
  322. }
  323. #endif
  324. #endif /* ZEPHYR_INCLUDE_SYS_NOTIFY_H_ */