onoff.h 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645
  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_ONOFF_H_
  8. #define ZEPHYR_INCLUDE_SYS_ONOFF_H_
  9. #include <kernel.h>
  10. #include <zephyr/types.h>
  11. #include <sys/notify.h>
  12. #ifdef __cplusplus
  13. extern "C" {
  14. #endif
  15. /**
  16. * @defgroup resource_mgmt_onoff_apis On-Off Service APIs
  17. * @ingroup kernel_apis
  18. * @{
  19. */
  20. /**
  21. * @brief Flag indicating an error state.
  22. *
  23. * Error states are cleared using onoff_reset().
  24. */
  25. #define ONOFF_FLAG_ERROR BIT(0)
  26. /** @internal */
  27. #define ONOFF_FLAG_ONOFF BIT(1)
  28. /** @internal */
  29. #define ONOFF_FLAG_TRANSITION BIT(2)
  30. /**
  31. * @brief Mask used to isolate bits defining the service state.
  32. *
  33. * Mask a value with this then test for ONOFF_FLAG_ERROR to determine
  34. * whether the machine has an unfixed error, or compare against
  35. * ONOFF_STATE_ON, ONOFF_STATE_OFF, ONOFF_STATE_TO_ON,
  36. * ONOFF_STATE_TO_OFF, or ONOFF_STATE_RESETTING.
  37. */
  38. #define ONOFF_STATE_MASK (ONOFF_FLAG_ERROR \
  39. | ONOFF_FLAG_ONOFF \
  40. | ONOFF_FLAG_TRANSITION)
  41. /**
  42. * @brief Value exposed by ONOFF_STATE_MASK when service is off.
  43. */
  44. #define ONOFF_STATE_OFF 0U
  45. /**
  46. * @brief Value exposed by ONOFF_STATE_MASK when service is on.
  47. */
  48. #define ONOFF_STATE_ON ONOFF_FLAG_ONOFF
  49. /**
  50. * @brief Value exposed by ONOFF_STATE_MASK when the service is in an
  51. * error state (and not in the process of resetting its state).
  52. */
  53. #define ONOFF_STATE_ERROR ONOFF_FLAG_ERROR
  54. /**
  55. * @brief Value exposed by ONOFF_STATE_MASK when service is
  56. * transitioning to on.
  57. */
  58. #define ONOFF_STATE_TO_ON (ONOFF_FLAG_TRANSITION | ONOFF_STATE_ON)
  59. /**
  60. * @brief Value exposed by ONOFF_STATE_MASK when service is
  61. * transitioning to off.
  62. */
  63. #define ONOFF_STATE_TO_OFF (ONOFF_FLAG_TRANSITION | ONOFF_STATE_OFF)
  64. /**
  65. * @brief Value exposed by ONOFF_STATE_MASK when service is in the
  66. * process of resetting.
  67. */
  68. #define ONOFF_STATE_RESETTING (ONOFF_FLAG_TRANSITION | ONOFF_STATE_ERROR)
  69. /* Forward declarations */
  70. struct onoff_manager;
  71. struct onoff_monitor;
  72. /**
  73. * @brief Signature used to notify an on-off manager that a transition
  74. * has completed.
  75. *
  76. * Functions of this type are passed to service-specific transition
  77. * functions to be used to report the completion of the operation.
  78. * The functions may be invoked from any context.
  79. *
  80. * @param mgr the manager for which transition was requested.
  81. *
  82. * @param res the result of the transition. This shall be
  83. * non-negative on success, or a negative error code. If an error is
  84. * indicated the service shall enter an error state.
  85. */
  86. typedef void (*onoff_notify_fn)(struct onoff_manager *mgr,
  87. int res);
  88. /**
  89. * @brief Signature used by service implementations to effect a
  90. * transition.
  91. *
  92. * Service definitions use two required function pointers of this type
  93. * to be notified that a transition is required, and a third optional
  94. * one to reset the service when it is in an error state.
  95. *
  96. * The start function will be called only from the off state.
  97. *
  98. * The stop function will be called only from the on state.
  99. *
  100. * The reset function (where supported) will be called only when
  101. * onoff_has_error() returns true.
  102. *
  103. * @note All transitions functions must be isr-ok.
  104. *
  105. * @param mgr the manager for which transition was requested.
  106. *
  107. * @param notify the function to be invoked when the transition has
  108. * completed. If the transition is synchronous, notify shall be
  109. * invoked by the implementation before the transition function
  110. * returns. Otherwise the implementation shall capture this parameter
  111. * and invoke it when the transition completes.
  112. */
  113. typedef void (*onoff_transition_fn)(struct onoff_manager *mgr,
  114. onoff_notify_fn notify);
  115. /** @brief On-off service transition functions. */
  116. struct onoff_transitions {
  117. /* Function to invoke to transition the service to on. */
  118. onoff_transition_fn start;
  119. /* Function to invoke to transition the service to off. */
  120. onoff_transition_fn stop;
  121. /* Function to force the service state to reset, where
  122. * supported.
  123. */
  124. onoff_transition_fn reset;
  125. };
  126. /**
  127. * @brief State associated with an on-off manager.
  128. *
  129. * No fields in this structure are intended for use by service
  130. * providers or clients. The state is to be initialized once, using
  131. * onoff_manager_init(), when the service provider is initialized. In
  132. * case of error it may be reset through the onoff_reset() API.
  133. */
  134. struct onoff_manager {
  135. /* List of clients waiting for request or reset completion
  136. * notifications.
  137. */
  138. sys_slist_t clients;
  139. /* List of monitors to be notified of state changes including
  140. * errors and transition completion.
  141. */
  142. sys_slist_t monitors;
  143. /* Transition functions. */
  144. const struct onoff_transitions *transitions;
  145. /* Mutex protection for other fields. */
  146. struct k_spinlock lock;
  147. /* The result of the last transition. */
  148. int last_res;
  149. /* Flags identifying the service state. */
  150. uint16_t flags;
  151. /* Number of active clients for the service. */
  152. uint16_t refs;
  153. };
  154. /** @brief Initializer for a onoff_transitions object.
  155. *
  156. * @param _start a function used to transition from off to on state.
  157. *
  158. * @param _stop a function used to transition from on to off state.
  159. *
  160. * @param _reset a function used to clear errors and force the service
  161. * to an off state. Can be null.
  162. */
  163. #define ONOFF_TRANSITIONS_INITIALIZER(_start, _stop, _reset) { \
  164. .start = _start, \
  165. .stop = _stop, \
  166. .reset = _reset, \
  167. }
  168. /** @internal */
  169. #define ONOFF_MANAGER_INITIALIZER(_transitions) { \
  170. .transitions = _transitions, \
  171. }
  172. /**
  173. * @brief Initialize an on-off service to off state.
  174. *
  175. * This function must be invoked exactly once per service instance, by
  176. * the infrastructure that provides the service, and before any other
  177. * on-off service API is invoked on the service.
  178. *
  179. * This function should never be invoked by clients of an on-off
  180. * service.
  181. *
  182. * @param mgr the manager definition object to be initialized.
  183. *
  184. * @param transitions pointer to a structure providing transition
  185. * functions. The referenced object must persist as long as the
  186. * manager can be referenced.
  187. *
  188. * @retval 0 on success
  189. * @retval -EINVAL if start, stop, or flags are invalid
  190. */
  191. int onoff_manager_init(struct onoff_manager *mgr,
  192. const struct onoff_transitions *transitions);
  193. /* Forward declaration */
  194. struct onoff_client;
  195. /**
  196. * @brief Signature used to notify an on-off service client of the
  197. * completion of an operation.
  198. *
  199. * These functions may be invoked from any context including
  200. * pre-kernel, ISR, or cooperative or pre-emptible threads.
  201. * Compatible functions must be isr-ok and not sleep.
  202. *
  203. * @param mgr the manager for which the operation was initiated. This may be
  204. * null if the on-off service uses synchronous transitions.
  205. *
  206. * @param cli the client structure passed to the function that
  207. * initiated the operation.
  208. *
  209. * @param state the state of the machine at the time of completion,
  210. * restricted by ONOFF_STATE_MASK. ONOFF_FLAG_ERROR must be checked
  211. * independently of whether res is negative as a machine error may
  212. * indicate that all future operations except onoff_reset() will fail.
  213. *
  214. * @param res the result of the operation. Expected values are
  215. * service-specific, but the value shall be non-negative if the
  216. * operation succeeded, and negative if the operation failed. If res
  217. * is negative ONOFF_FLAG_ERROR will be set in state, but if res is
  218. * non-negative ONOFF_FLAG_ERROR may still be set in state.
  219. */
  220. typedef void (*onoff_client_callback)(struct onoff_manager *mgr,
  221. struct onoff_client *cli,
  222. uint32_t state,
  223. int res);
  224. /**
  225. * @brief State associated with a client of an on-off service.
  226. *
  227. * Objects of this type are allocated by a client, which is
  228. * responsible for zero-initializing the node field and invoking the
  229. * approprite sys_notify init function to configure notification.
  230. *
  231. * Control of the object content transfers to the service provider
  232. * when a pointer to the object is passed to any on-off manager
  233. * function. While the service provider controls the object the
  234. * client must not change any object fields. Control reverts to the
  235. * client concurrent with release of the owned sys_notify structure,
  236. * or when indicated by an onoff_cancel() return value.
  237. *
  238. * After control has reverted to the client the notify field must be
  239. * reinitialized for the next operation.
  240. */
  241. struct onoff_client {
  242. /** @internal
  243. *
  244. * Links the client into the set of waiting service users.
  245. * Applications must ensure this field is zero-initialized
  246. * before use.
  247. */
  248. sys_snode_t node;
  249. /** @brief Notification configuration. */
  250. struct sys_notify notify;
  251. };
  252. /**
  253. * @brief Identify region of sys_notify flags available for
  254. * containing services.
  255. *
  256. * Bits of the flags field of the sys_notify structure contained
  257. * within the queued_operation structure at and above this position
  258. * may be used by extensions to the onoff_client structure.
  259. *
  260. * These bits are intended for use by containing service
  261. * implementations to record client-specific information and are
  262. * subject to other conditions of use specified on the sys_notify API.
  263. */
  264. #define ONOFF_CLIENT_EXTENSION_POS SYS_NOTIFY_EXTENSION_POS
  265. /**
  266. * @brief Test whether an on-off service has recorded an error.
  267. *
  268. * This function can be used to determine whether the service has
  269. * recorded an error. Errors may be cleared by invoking
  270. * onoff_reset().
  271. *
  272. * This is an unlocked convenience function suitable for use only when
  273. * it is known that no other process might invoke an operation that
  274. * transitions the service between an error and non-error state.
  275. *
  276. * @return true if and only if the service has an uncleared error.
  277. */
  278. static inline bool onoff_has_error(const struct onoff_manager *mgr)
  279. {
  280. return (mgr->flags & ONOFF_FLAG_ERROR) != 0;
  281. }
  282. /**
  283. * @brief Request a reservation to use an on-off service.
  284. *
  285. * The return value indicates the success or failure of an attempt to
  286. * initiate an operation to request the resource be made available.
  287. * If initiation of the operation succeeds the result of the request
  288. * operation is provided through the configured client notification
  289. * method, possibly before this call returns.
  290. *
  291. * Note that the call to this function may succeed in a case where the
  292. * actual request fails. Always check the operation completion
  293. * result.
  294. *
  295. * @param mgr the manager that will be used.
  296. *
  297. * @param cli a non-null pointer to client state providing
  298. * instructions on synchronous expectations and how to notify the
  299. * client when the request completes. Behavior is undefined if client
  300. * passes a pointer object associated with an incomplete service
  301. * operation.
  302. *
  303. * @retval non-negative the observed state of the machine at the time
  304. * the request was processed, if successful.
  305. * @retval -EIO if service has recorded an an error.
  306. * @retval -EINVAL if the parameters are invalid.
  307. * @retval -EAGAIN if the reference count would overflow.
  308. */
  309. int onoff_request(struct onoff_manager *mgr,
  310. struct onoff_client *cli);
  311. /**
  312. * @brief Release a reserved use of an on-off service.
  313. *
  314. * This synchronously releases the caller's previous request. If the
  315. * last request is released the manager will initiate a transition to
  316. * off, which can be observed by registering an onoff_monitor.
  317. *
  318. * @note Behavior is undefined if this is not paired with a preceding
  319. * onoff_request() call that completed successfully.
  320. *
  321. * @param mgr the manager for which a request was successful.
  322. *
  323. * @retval non-negative the observed state (ONOFF_STATE_ON) of the
  324. * machine at the time of the release, if the release succeeds.
  325. * @retval -EIO if service has recorded an an error.
  326. * @retval -ENOTSUP if the machine is not in a state that permits
  327. * release.
  328. */
  329. int onoff_release(struct onoff_manager *mgr);
  330. /**
  331. * @brief Attempt to cancel an in-progress client operation.
  332. *
  333. * It may be that a client has initiated an operation but needs to
  334. * shut down before the operation has completed. For example, when a
  335. * request was made and the need is no longer present.
  336. *
  337. * Cancelling is supported only for onoff_request() and onoff_reset()
  338. * operations, and is a synchronous operation. Be aware that any
  339. * transition that was initiated on behalf of the client will continue
  340. * to progress to completion: it is only notification of transition
  341. * completion that may be eliminated. If there are no active requests
  342. * when a transition to on completes the manager will initiate a
  343. * transition to off.
  344. *
  345. * Client notification does not occur for cancelled operations.
  346. *
  347. * @param mgr the manager for which an operation is to be cancelled.
  348. *
  349. * @param cli a pointer to the same client state that was provided
  350. * when the operation to be cancelled was issued.
  351. *
  352. * @retval non-negative the observed state of the machine at the time
  353. * of the cancellation, if the cancellation succeeds. On successful
  354. * cancellation ownership of @c *cli reverts to the client.
  355. * @retval -EINVAL if the parameters are invalid.
  356. * @retval -EALREADY if cli was not a record of an uncompleted
  357. * notification at the time the cancellation was processed. This
  358. * likely indicates that the operation and client notification had
  359. * already completed.
  360. */
  361. int onoff_cancel(struct onoff_manager *mgr,
  362. struct onoff_client *cli);
  363. /**
  364. * @brief Helper function to safely cancel a request.
  365. *
  366. * Some applications may want to issue requests on an asynchronous
  367. * event (such as connection to a USB bus) and to release on a paired
  368. * event (such as loss of connection to a USB bus). Applications
  369. * cannot precisely determine that an in-progress request is still
  370. * pending without using onoff_monitor and carefully avoiding race
  371. * conditions.
  372. *
  373. * This function is a helper that attempts to cancel the operation and
  374. * issues a release if cancellation fails because the request was
  375. * completed. This synchronously ensures that ownership of the client
  376. * data reverts to the client so is available for a future request.
  377. *
  378. * @param mgr the manager for which an operation is to be cancelled.
  379. *
  380. * @param cli a pointer to the same client state that was provided
  381. * when onoff_request() was invoked. Behavior is undefined if this is
  382. * a pointer to client data associated with an onoff_reset() request.
  383. *
  384. * @retval ONOFF_STATE_TO_ON if the cancellation occurred before the
  385. * transition completed.
  386. *
  387. * @retval ONOFF_STATE_ON if the cancellation occurred after the
  388. * transition completed.
  389. *
  390. * @retval -EINVAL if the parameters are invalid.
  391. *
  392. * @retval negative other errors produced by onoff_release().
  393. */
  394. static inline int onoff_cancel_or_release(struct onoff_manager *mgr,
  395. struct onoff_client *cli)
  396. {
  397. int rv = onoff_cancel(mgr, cli);
  398. if (rv == -EALREADY) {
  399. rv = onoff_release(mgr);
  400. }
  401. return rv;
  402. }
  403. /**
  404. * @brief Clear errors on an on-off service and reset it to its off
  405. * state.
  406. *
  407. * A service can only be reset when it is in an error state as
  408. * indicated by onoff_has_error().
  409. *
  410. * The return value indicates the success or failure of an attempt to
  411. * initiate an operation to reset the resource. If initiation of the
  412. * operation succeeds the result of the reset operation itself is
  413. * provided through the configured client notification method,
  414. * possibly before this call returns. Multiple clients may request a
  415. * reset; all are notified when it is complete.
  416. *
  417. * Note that the call to this function may succeed in a case where the
  418. * actual reset fails. Always check the operation completion result.
  419. *
  420. * @note Due to the conditions on state transition all incomplete
  421. * asynchronous operations will have been informed of the error when
  422. * it occurred. There need be no concern about dangling requests left
  423. * after a reset completes.
  424. *
  425. * @param mgr the manager to be reset.
  426. *
  427. * @param cli pointer to client state, including instructions on how
  428. * to notify the client when reset completes. Behavior is undefined
  429. * if cli references an object associated with an incomplete service
  430. * operation.
  431. *
  432. * @retval non-negative the observed state of the machine at the time
  433. * of the reset, if the reset succeeds.
  434. * @retval -ENOTSUP if reset is not supported by the service.
  435. * @retval -EINVAL if the parameters are invalid.
  436. * @retval -EALREADY if the service does not have a recorded error.
  437. */
  438. int onoff_reset(struct onoff_manager *mgr,
  439. struct onoff_client *cli);
  440. /**
  441. * @brief Signature used to notify a monitor of an onoff service of
  442. * errors or completion of a state transition.
  443. *
  444. * This is similar to onoff_client_callback but provides information
  445. * about all transitions, not just ones associated with a specific
  446. * client. Monitor callbacks are invoked before any completion
  447. * notifications associated with the state change are made.
  448. *
  449. * These functions may be invoked from any context including
  450. * pre-kernel, ISR, or cooperative or pre-emptible threads.
  451. * Compatible functions must be isr-ok and not sleep.
  452. *
  453. * The callback is permitted to unregister itself from the manager,
  454. * but must not register or unregister any other monitors.
  455. *
  456. * @param mgr the manager for which a transition has completed.
  457. *
  458. * @param mon the monitor instance through which this notification
  459. * arrived.
  460. *
  461. * @param state the state of the machine at the time of completion,
  462. * restricted by ONOFF_STATE_MASK. All valid states may be observed.
  463. *
  464. * @param res the result of the operation. Expected values are
  465. * service- and state-specific, but the value shall be non-negative if
  466. * the operation succeeded, and negative if the operation failed.
  467. */
  468. typedef void (*onoff_monitor_callback)(struct onoff_manager *mgr,
  469. struct onoff_monitor *mon,
  470. uint32_t state,
  471. int res);
  472. /**
  473. * @brief Registration state for notifications of onoff service
  474. * transitions.
  475. *
  476. * Any given onoff_monitor structure can be associated with at most
  477. * one onoff_manager instance.
  478. */
  479. struct onoff_monitor {
  480. /* Links the client into the set of waiting service users.
  481. *
  482. * This must be zero-initialized.
  483. */
  484. sys_snode_t node;
  485. /** @brief Callback to be invoked on state change.
  486. *
  487. * This must not be null.
  488. */
  489. onoff_monitor_callback callback;
  490. };
  491. /**
  492. * @brief Add a monitor of state changes for a manager.
  493. *
  494. * @param mgr the manager for which a state changes are to be monitored.
  495. *
  496. * @param mon a linkable node providing a non-null callback to be
  497. * invoked on state changes.
  498. *
  499. * @return non-negative on successful addition, or a negative error
  500. * code.
  501. */
  502. int onoff_monitor_register(struct onoff_manager *mgr,
  503. struct onoff_monitor *mon);
  504. /**
  505. * @brief Remove a monitor of state changes from a manager.
  506. *
  507. * @param mgr the manager for which a state changes are to be monitored.
  508. *
  509. * @param mon a linkable node providing the callback to be invoked on
  510. * state changes.
  511. *
  512. * @return non-negative on successful removal, or a negative error
  513. * code.
  514. */
  515. int onoff_monitor_unregister(struct onoff_manager *mgr,
  516. struct onoff_monitor *mon);
  517. /**
  518. * @brief State used when a driver uses the on-off service API for synchronous
  519. * operations.
  520. *
  521. * This is useful when a subsystem API uses the on-off API to support
  522. * asynchronous operations but the transitions required by a
  523. * particular driver are isr-ok and not sleep. It serves as a
  524. * substitute for #onoff_manager, with locking and persisted state
  525. * updates supported by onoff_sync_lock() and onoff_sync_finalize().
  526. */
  527. struct onoff_sync_service {
  528. /* Mutex protection for other fields. */
  529. struct k_spinlock lock;
  530. /* Negative is error, non-negative is reference count. */
  531. int32_t count;
  532. };
  533. /**
  534. * @brief Lock a synchronous onoff service and provide its state.
  535. *
  536. * @note If an error state is returned it is the caller's responsibility to
  537. * decide whether to preserve it (finalize with the same error state) or clear
  538. * the error (finalize with a non-error result).
  539. *
  540. * @param srv pointer to the synchronous service state.
  541. *
  542. * @param keyp pointer to where the lock key should be stored
  543. *
  544. * @return negative if the service is in an error state, otherwise the
  545. * number of active requests at the time the lock was taken. The lock
  546. * is held on return regardless of whether a negative state is
  547. * returned.
  548. */
  549. int onoff_sync_lock(struct onoff_sync_service *srv,
  550. k_spinlock_key_t *keyp);
  551. /**
  552. * @brief Process the completion of a transition in a synchronous
  553. * service and release lock.
  554. *
  555. * This function updates the service state on the @p res and @p on parameters
  556. * then releases the lock. If @p cli is not null it finalizes the client
  557. * notification using @p res.
  558. *
  559. * If the service was in an error state when locked, and @p res is non-negative
  560. * when finalized, the count is reset to zero before completing finalization.
  561. *
  562. * @param srv pointer to the synchronous service state
  563. *
  564. * @param key the key returned by the preceding invocation of onoff_sync_lock().
  565. *
  566. * @param cli pointer to the onoff client through which completion
  567. * information is returned. If a null pointer is passed only the
  568. * state of the service is updated. For compatibility with the
  569. * behavior of callbacks used with the manager API @p cli must be null
  570. * when @p on is false (the manager does not support callbacks when
  571. * turning off devices).
  572. *
  573. * @param res the result of the transition. A negative value places the service
  574. * into an error state. A non-negative value increments or decrements the
  575. * reference count as specified by @p on.
  576. *
  577. * @param on Only when @p res is non-negative, the service reference count will
  578. * be incremented if@p on is @c true, and decremented if @p on is @c false.
  579. *
  580. * @return negative if the service is left or put into an error state, otherwise
  581. * the number of active requests at the time the lock was released.
  582. */
  583. int onoff_sync_finalize(struct onoff_sync_service *srv,
  584. k_spinlock_key_t key,
  585. struct onoff_client *cli,
  586. int res,
  587. bool on);
  588. /** @} */
  589. #ifdef __cplusplus
  590. }
  591. #endif
  592. #endif /* ZEPHYR_INCLUDE_SYS_ONOFF_H_ */