| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645 | /* * Copyright (c) 2019 Peter Bigot Consulting, LLC * Copyright (c) 2020 Nordic Semiconductor ASA * * SPDX-License-Identifier: Apache-2.0 */#ifndef ZEPHYR_INCLUDE_SYS_ONOFF_H_#define ZEPHYR_INCLUDE_SYS_ONOFF_H_#include <kernel.h>#include <zephyr/types.h>#include <sys/notify.h>#ifdef __cplusplusextern "C" {#endif/** * @defgroup resource_mgmt_onoff_apis On-Off Service APIs * @ingroup kernel_apis * @{ *//** * @brief Flag indicating an error state. * * Error states are cleared using onoff_reset(). */#define ONOFF_FLAG_ERROR BIT(0)/** @internal */#define ONOFF_FLAG_ONOFF BIT(1)/** @internal */#define ONOFF_FLAG_TRANSITION BIT(2)/** * @brief Mask used to isolate bits defining the service state. * * Mask a value with this then test for ONOFF_FLAG_ERROR to determine * whether the machine has an unfixed error, or compare against * ONOFF_STATE_ON, ONOFF_STATE_OFF, ONOFF_STATE_TO_ON, * ONOFF_STATE_TO_OFF, or ONOFF_STATE_RESETTING. */#define ONOFF_STATE_MASK (ONOFF_FLAG_ERROR   \			  | ONOFF_FLAG_ONOFF \			  | ONOFF_FLAG_TRANSITION)/** * @brief Value exposed by ONOFF_STATE_MASK when service is off. */#define ONOFF_STATE_OFF 0U/** * @brief Value exposed by ONOFF_STATE_MASK when service is on. */#define ONOFF_STATE_ON ONOFF_FLAG_ONOFF/** * @brief Value exposed by ONOFF_STATE_MASK when the service is in an * error state (and not in the process of resetting its state). */#define ONOFF_STATE_ERROR ONOFF_FLAG_ERROR/** * @brief Value exposed by ONOFF_STATE_MASK when service is * transitioning to on. */#define ONOFF_STATE_TO_ON (ONOFF_FLAG_TRANSITION | ONOFF_STATE_ON)/** * @brief Value exposed by ONOFF_STATE_MASK when service is * transitioning to off. */#define ONOFF_STATE_TO_OFF (ONOFF_FLAG_TRANSITION | ONOFF_STATE_OFF)/** * @brief Value exposed by ONOFF_STATE_MASK when service is in the * process of resetting. */#define ONOFF_STATE_RESETTING (ONOFF_FLAG_TRANSITION | ONOFF_STATE_ERROR)/* Forward declarations */struct onoff_manager;struct onoff_monitor;/** * @brief Signature used to notify an on-off manager that a transition * has completed. * * Functions of this type are passed to service-specific transition * functions to be used to report the completion of the operation. * The functions may be invoked from any context. * * @param mgr the manager for which transition was requested. * * @param res the result of the transition.  This shall be * non-negative on success, or a negative error code.  If an error is * indicated the service shall enter an error state. */typedef void (*onoff_notify_fn)(struct onoff_manager *mgr,				int res);/** * @brief Signature used by service implementations to effect a * transition. * * Service definitions use two required function pointers of this type * to be notified that a transition is required, and a third optional * one to reset the service when it is in an error state. * * The start function will be called only from the off state. * * The stop function will be called only from the on state. * * The reset function (where supported) will be called only when * onoff_has_error() returns true. * * @note All transitions functions must be isr-ok. * * @param mgr the manager for which transition was requested. * * @param notify the function to be invoked when the transition has * completed.  If the transition is synchronous, notify shall be * invoked by the implementation before the transition function * returns.  Otherwise the implementation shall capture this parameter * and invoke it when the transition completes. */typedef void (*onoff_transition_fn)(struct onoff_manager *mgr,				    onoff_notify_fn notify);/** @brief On-off service transition functions. */struct onoff_transitions {	/* Function to invoke to transition the service to on. */	onoff_transition_fn start;	/* Function to invoke to transition the service to off. */	onoff_transition_fn stop;	/* Function to force the service state to reset, where	 * supported.	 */	onoff_transition_fn reset;};/** * @brief State associated with an on-off manager. * * No fields in this structure are intended for use by service * providers or clients.  The state is to be initialized once, using * onoff_manager_init(), when the service provider is initialized.  In * case of error it may be reset through the onoff_reset() API. */struct onoff_manager {	/* List of clients waiting for request or reset completion	 * notifications.	 */	sys_slist_t clients;	/* List of monitors to be notified of state changes including	 * errors and transition completion.	 */	sys_slist_t monitors;	/* Transition functions. */	const struct onoff_transitions *transitions;	/* Mutex protection for other fields. */	struct k_spinlock lock;	/* The result of the last transition. */	int last_res;	/* Flags identifying the service state. */	uint16_t flags;	/* Number of active clients for the service. */	uint16_t refs;};/** @brief Initializer for a onoff_transitions object. * * @param _start a function used to transition from off to on state. * * @param _stop a function used to transition from on to off state. * * @param _reset a function used to clear errors and force the service * to an off state. Can be null. */#define ONOFF_TRANSITIONS_INITIALIZER(_start, _stop, _reset) { \		.start = _start,			       \		.stop = _stop,				       \		.reset = _reset,			       \}/** @internal */#define ONOFF_MANAGER_INITIALIZER(_transitions) { \		.transitions = _transitions,	  \}/** * @brief Initialize an on-off service to off state. * * This function must be invoked exactly once per service instance, by * the infrastructure that provides the service, and before any other * on-off service API is invoked on the service. * * This function should never be invoked by clients of an on-off * service. * * @param mgr the manager definition object to be initialized. * * @param transitions pointer to a structure providing transition * functions.  The referenced object must persist as long as the * manager can be referenced. * * @retval 0 on success * @retval -EINVAL if start, stop, or flags are invalid */int onoff_manager_init(struct onoff_manager *mgr,		       const struct onoff_transitions *transitions);/* Forward declaration */struct onoff_client;/** * @brief Signature used to notify an on-off service client of the * completion of an operation. * * These functions may be invoked from any context including * pre-kernel, ISR, or cooperative or pre-emptible threads. * Compatible functions must be isr-ok and not sleep. * * @param mgr the manager for which the operation was initiated.  This may be * null if the on-off service uses synchronous transitions. * * @param cli the client structure passed to the function that * initiated the operation. * * @param state the state of the machine at the time of completion, * restricted by ONOFF_STATE_MASK.  ONOFF_FLAG_ERROR must be checked * independently of whether res is negative as a machine error may * indicate that all future operations except onoff_reset() will fail. * * @param res the result of the operation.  Expected values are * service-specific, but the value shall be non-negative if the * operation succeeded, and negative if the operation failed.  If res * is negative ONOFF_FLAG_ERROR will be set in state, but if res is * non-negative ONOFF_FLAG_ERROR may still be set in state. */typedef void (*onoff_client_callback)(struct onoff_manager *mgr,				      struct onoff_client *cli,				      uint32_t state,				      int res);/** * @brief State associated with a client of an on-off service. * * Objects of this type are allocated by a client, which is * responsible for zero-initializing the node field and invoking the * approprite sys_notify init function to configure notification. * * Control of the object content transfers to the service provider * when a pointer to the object is passed to any on-off manager * function.  While the service provider controls the object the * client must not change any object fields.  Control reverts to the * client concurrent with release of the owned sys_notify structure, * or when indicated by an onoff_cancel() return value. * * After control has reverted to the client the notify field must be * reinitialized for the next operation. */struct onoff_client {	/** @internal	 *	 * Links the client into the set of waiting service users.	 * Applications must ensure this field is zero-initialized	 * before use.	 */	sys_snode_t node;	/** @brief Notification configuration. */	struct sys_notify notify;};/** * @brief Identify region of sys_notify flags available for * containing services. * * Bits of the flags field of the sys_notify structure contained * within the queued_operation structure at and above this position * may be used by extensions to the onoff_client structure. * * These bits are intended for use by containing service * implementations to record client-specific information and are * subject to other conditions of use specified on the sys_notify API. */#define ONOFF_CLIENT_EXTENSION_POS SYS_NOTIFY_EXTENSION_POS/** * @brief Test whether an on-off service has recorded an error. * * This function can be used to determine whether the service has * recorded an error.  Errors may be cleared by invoking * onoff_reset(). * * This is an unlocked convenience function suitable for use only when * it is known that no other process might invoke an operation that * transitions the service between an error and non-error state. * * @return true if and only if the service has an uncleared error. */static inline bool onoff_has_error(const struct onoff_manager *mgr){	return (mgr->flags & ONOFF_FLAG_ERROR) != 0;}/** * @brief Request a reservation to use an on-off service. * * The return value indicates the success or failure of an attempt to * initiate an operation to request the resource be made available. * If initiation of the operation succeeds the result of the request * operation is provided through the configured client notification * method, possibly before this call returns. * * Note that the call to this function may succeed in a case where the * actual request fails.  Always check the operation completion * result. * * @param mgr the manager that will be used. * * @param cli a non-null pointer to client state providing * instructions on synchronous expectations and how to notify the * client when the request completes.  Behavior is undefined if client * passes a pointer object associated with an incomplete service * operation. * * @retval non-negative the observed state of the machine at the time * the request was processed, if successful. * @retval -EIO if service has recorded an an error. * @retval -EINVAL if the parameters are invalid. * @retval -EAGAIN if the reference count would overflow. */int onoff_request(struct onoff_manager *mgr,		  struct onoff_client *cli);/** * @brief Release a reserved use of an on-off service. * * This synchronously releases the caller's previous request.  If the * last request is released the manager will initiate a transition to * off, which can be observed by registering an onoff_monitor. * * @note Behavior is undefined if this is not paired with a preceding * onoff_request() call that completed successfully. * * @param mgr the manager for which a request was successful. * * @retval non-negative the observed state (ONOFF_STATE_ON) of the * machine at the time of the release, if the release succeeds. * @retval -EIO if service has recorded an an error. * @retval -ENOTSUP if the machine is not in a state that permits * release. */int onoff_release(struct onoff_manager *mgr);/** * @brief Attempt to cancel an in-progress client operation. * * It may be that a client has initiated an operation but needs to * shut down before the operation has completed.  For example, when a * request was made and the need is no longer present. * * Cancelling is supported only for onoff_request() and onoff_reset() * operations, and is a synchronous operation.  Be aware that any * transition that was initiated on behalf of the client will continue * to progress to completion: it is only notification of transition * completion that may be eliminated.  If there are no active requests * when a transition to on completes the manager will initiate a * transition to off. * * Client notification does not occur for cancelled operations. * * @param mgr the manager for which an operation is to be cancelled. * * @param cli a pointer to the same client state that was provided * when the operation to be cancelled was issued. * * @retval non-negative the observed state of the machine at the time * of the cancellation, if the cancellation succeeds.  On successful * cancellation ownership of @c *cli reverts to the client. * @retval -EINVAL if the parameters are invalid. * @retval -EALREADY if cli was not a record of an uncompleted * notification at the time the cancellation was processed.  This * likely indicates that the operation and client notification had * already completed. */int onoff_cancel(struct onoff_manager *mgr,		 struct onoff_client *cli);/** * @brief Helper function to safely cancel a request. * * Some applications may want to issue requests on an asynchronous * event (such as connection to a USB bus) and to release on a paired * event (such as loss of connection to a USB bus).  Applications * cannot precisely determine that an in-progress request is still * pending without using onoff_monitor and carefully avoiding race * conditions. * * This function is a helper that attempts to cancel the operation and * issues a release if cancellation fails because the request was * completed.  This synchronously ensures that ownership of the client * data reverts to the client so is available for a future request. * * @param mgr the manager for which an operation is to be cancelled. * * @param cli a pointer to the same client state that was provided * when onoff_request() was invoked.  Behavior is undefined if this is * a pointer to client data associated with an onoff_reset() request. * * @retval ONOFF_STATE_TO_ON if the cancellation occurred before the * transition completed. * * @retval ONOFF_STATE_ON if the cancellation occurred after the * transition completed. * * @retval -EINVAL if the parameters are invalid. * * @retval negative other errors produced by onoff_release(). */static inline int onoff_cancel_or_release(struct onoff_manager *mgr,					  struct onoff_client *cli){	int rv = onoff_cancel(mgr, cli);	if (rv == -EALREADY) {		rv = onoff_release(mgr);	}	return rv;}/** * @brief Clear errors on an on-off service and reset it to its off * state. * * A service can only be reset when it is in an error state as * indicated by onoff_has_error(). * * The return value indicates the success or failure of an attempt to * initiate an operation to reset the resource.  If initiation of the * operation succeeds the result of the reset operation itself is * provided through the configured client notification method, * possibly before this call returns.  Multiple clients may request a * reset; all are notified when it is complete. * * Note that the call to this function may succeed in a case where the * actual reset fails.  Always check the operation completion result. * * @note Due to the conditions on state transition all incomplete * asynchronous operations will have been informed of the error when * it occurred.  There need be no concern about dangling requests left * after a reset completes. * * @param mgr the manager to be reset. * * @param cli pointer to client state, including instructions on how * to notify the client when reset completes.  Behavior is undefined * if cli references an object associated with an incomplete service * operation. * * @retval non-negative the observed state of the machine at the time * of the reset, if the reset succeeds. * @retval -ENOTSUP if reset is not supported by the service. * @retval -EINVAL if the parameters are invalid. * @retval -EALREADY if the service does not have a recorded error. */int onoff_reset(struct onoff_manager *mgr,		struct onoff_client *cli);/** * @brief Signature used to notify a monitor of an onoff service of * errors or completion of a state transition. * * This is similar to onoff_client_callback but provides information * about all transitions, not just ones associated with a specific * client.  Monitor callbacks are invoked before any completion * notifications associated with the state change are made. * * These functions may be invoked from any context including * pre-kernel, ISR, or cooperative or pre-emptible threads. * Compatible functions must be isr-ok and not sleep. * * The callback is permitted to unregister itself from the manager, * but must not register or unregister any other monitors. * * @param mgr the manager for which a transition has completed. * * @param mon the monitor instance through which this notification * arrived. * * @param state the state of the machine at the time of completion, * restricted by ONOFF_STATE_MASK.  All valid states may be observed. * * @param res the result of the operation.  Expected values are * service- and state-specific, but the value shall be non-negative if * the operation succeeded, and negative if the operation failed. */typedef void (*onoff_monitor_callback)(struct onoff_manager *mgr,				       struct onoff_monitor *mon,				       uint32_t state,				       int res);/** * @brief Registration state for notifications of onoff service * transitions. * * Any given onoff_monitor structure can be associated with at most * one onoff_manager instance. */struct onoff_monitor {	/* Links the client into the set of waiting service users.	 *	 * This must be zero-initialized.	 */	sys_snode_t node;	/** @brief Callback to be invoked on state change.	 *	 * This must not be null.	 */	onoff_monitor_callback callback;};/** * @brief Add a monitor of state changes for a manager. * * @param mgr the manager for which a state changes are to be monitored. * * @param mon a linkable node providing a non-null callback to be * invoked on state changes. * * @return non-negative on successful addition, or a negative error * code. */int onoff_monitor_register(struct onoff_manager *mgr,			   struct onoff_monitor *mon);/** * @brief Remove a monitor of state changes from a manager. * * @param mgr the manager for which a state changes are to be monitored. * * @param mon a linkable node providing the callback to be invoked on * state changes. * * @return non-negative on successful removal, or a negative error * code. */int onoff_monitor_unregister(struct onoff_manager *mgr,			     struct onoff_monitor *mon);/** * @brief State used when a driver uses the on-off service API for synchronous * operations. * * This is useful when a subsystem API uses the on-off API to support * asynchronous operations but the transitions required by a * particular driver are isr-ok and not sleep.  It serves as a * substitute for #onoff_manager, with locking and persisted state * updates supported by onoff_sync_lock() and onoff_sync_finalize(). */struct onoff_sync_service {	/* Mutex protection for other fields. */	struct k_spinlock lock;	/* Negative is error, non-negative is reference count. */	int32_t count;};/** * @brief Lock a synchronous onoff service and provide its state. * * @note If an error state is returned it is the caller's responsibility to * decide whether to preserve it (finalize with the same error state) or clear * the error (finalize with a non-error result). * * @param srv pointer to the synchronous service state. * * @param keyp pointer to where the lock key should be stored * * @return negative if the service is in an error state, otherwise the * number of active requests at the time the lock was taken.  The lock * is held on return regardless of whether a negative state is * returned. */int onoff_sync_lock(struct onoff_sync_service *srv,		    k_spinlock_key_t *keyp);/** * @brief Process the completion of a transition in a synchronous * service and release lock. * * This function updates the service state on the @p res and @p on parameters * then releases the lock.  If @p cli is not null it finalizes the client * notification using @p res. * * If the service was in an error state when locked, and @p res is non-negative * when finalized, the count is reset to zero before completing finalization. * * @param srv pointer to the synchronous service state * * @param key the key returned by the preceding invocation of onoff_sync_lock(). * * @param cli pointer to the onoff client through which completion * information is returned.  If a null pointer is passed only the * state of the service is updated.  For compatibility with the * behavior of callbacks used with the manager API @p cli must be null * when @p on is false (the manager does not support callbacks when * turning off devices). * * @param res the result of the transition.  A negative value places the service * into an error state.  A non-negative value increments or decrements the * reference count as specified by @p on. * * @param on Only when @p res is non-negative, the service reference count will * be incremented if@p on is @c true, and decremented if @p on is @c false. * * @return negative if the service is left or put into an error state, otherwise * the number of active requests at the time the lock was released. */int onoff_sync_finalize(struct onoff_sync_service *srv,			 k_spinlock_key_t key,			 struct onoff_client *cli,			 int res,			 bool on);/** @} */#ifdef __cplusplus}#endif#endif /* ZEPHYR_INCLUDE_SYS_ONOFF_H_ */
 |