device.h 7.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302
  1. /*
  2. * Copyright (c) 2015 Intel Corporation.
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #ifndef ZEPHYR_INCLUDE_PM_DEVICE_H_
  7. #define ZEPHYR_INCLUDE_PM_DEVICE_H_
  8. #include <kernel.h>
  9. #include <sys/atomic.h>
  10. #ifdef __cplusplus
  11. extern "C" {
  12. #endif
  13. /**
  14. * @brief Device Power Management API
  15. *
  16. * @defgroup device_power_management_api Device Power Management API
  17. * @ingroup power_management_api
  18. * @{
  19. */
  20. struct device;
  21. /** @brief Device power states. */
  22. enum pm_device_state {
  23. /** Device is in active or regular state. */
  24. PM_DEVICE_STATE_ACTIVE,
  25. /**
  26. * Device is in low power state.
  27. *
  28. * @note
  29. * Device context is preserved.
  30. */
  31. PM_DEVICE_STATE_LOW_POWER,
  32. /**
  33. * Device is suspended.
  34. *
  35. * @note
  36. * Device context may be lost.
  37. */
  38. PM_DEVICE_STATE_SUSPENDED,
  39. /**
  40. * Device is turned off (power removed).
  41. *
  42. * @note
  43. * Device context is lost.
  44. */
  45. PM_DEVICE_STATE_OFF
  46. };
  47. /** @brief Device PM flags. */
  48. enum pm_device_flag {
  49. /** Indicate if the device is busy or not. */
  50. PM_DEVICE_FLAG_BUSY,
  51. /**
  52. * Indicates whether or not the device is capable of waking the system
  53. * up.
  54. */
  55. PM_DEVICE_FLAGS_WS_CAPABLE,
  56. /** Indicates if the device is being used as wakeup source. */
  57. PM_DEVICE_FLAGS_WS_ENABLED,
  58. /** Indicates that the device is changing its state */
  59. PM_DEVICE_FLAG_TRANSITIONING,
  60. /** Number of flags (internal use only). */
  61. PM_DEVICE_FLAG_COUNT
  62. };
  63. /** @brief Device PM actions. */
  64. enum pm_device_action {
  65. /** Suspend. */
  66. PM_DEVICE_ACTION_SUSPEND,
  67. /** Resume. */
  68. PM_DEVICE_ACTION_RESUME,
  69. /** Turn off. */
  70. PM_DEVICE_ACTION_TURN_OFF,
  71. /** Force suspend. */
  72. PM_DEVICE_ACTION_FORCE_SUSPEND,
  73. /** Low power. */
  74. PM_DEVICE_ACTION_LOW_POWER,
  75. /** EARLY Suspend. */
  76. PM_DEVICE_ACTION_EARLY_SUSPEND,
  77. /** EARLY Resume. */
  78. PM_DEVICE_ACTION_LATE_RESUME,
  79. };
  80. /**
  81. * @brief Device PM info
  82. */
  83. struct pm_device {
  84. /** Pointer to the device */
  85. const struct device *dev;
  86. /** Lock to synchronize the get/put operations */
  87. struct k_mutex lock;
  88. /* Following are packed fields protected by #lock. */
  89. /** Device pm enable flag */
  90. bool enable : 1;
  91. /* Device PM status flags. */
  92. atomic_t flags;
  93. /** Device usage count */
  94. uint32_t usage;
  95. /** Device power state */
  96. enum pm_device_state state;
  97. /** Work object for asynchronous calls */
  98. struct k_work_delayable work;
  99. /** Event conditional var to listen to the sync request events */
  100. struct k_condvar condvar;
  101. };
  102. /**
  103. * @brief Utility macro to initialize #pm_device.
  104. *
  105. * @note DT_PROP_OR is used to retrieve the wakeup_source property because
  106. * it may not be defined on all devices.
  107. *
  108. * @param obj Name of the #pm_device structure being initialized.
  109. * @param node_id Devicetree node for the initialized device (can be invalid).
  110. */
  111. #define Z_PM_DEVICE_INIT(obj, node_id) \
  112. { \
  113. .usage = 0U, \
  114. .lock = Z_MUTEX_INITIALIZER(obj.lock), \
  115. .condvar = Z_CONDVAR_INITIALIZER(obj.condvar), \
  116. .state = PM_DEVICE_STATE_ACTIVE, \
  117. .flags = ATOMIC_INIT(COND_CODE_1( \
  118. DT_NODE_EXISTS(node_id), \
  119. (DT_PROP_OR(node_id, wakeup_source, 0)),\
  120. (0)) << PM_DEVICE_FLAGS_WS_CAPABLE), \
  121. }
  122. /**
  123. * @brief Device power management control function callback.
  124. *
  125. * @param dev Device instance.
  126. * @param action Requested action.
  127. *
  128. * @retval 0 If successful.
  129. * @retval -ENOTSUP If the requested action is not supported.
  130. * @retval Errno Other negative errno on failure.
  131. */
  132. typedef int (*pm_device_control_callback_t)(const struct device *dev,
  133. enum pm_device_action action);
  134. /**
  135. * @brief Get name of device PM state
  136. *
  137. * @param state State id which name should be returned
  138. */
  139. const char *pm_device_state_str(enum pm_device_state state);
  140. /**
  141. * @brief Set the power state of a device.
  142. *
  143. * This function calls the device PM control callback so that the device does
  144. * the necessary operations to put the device into the given state.
  145. *
  146. * @note Some devices may not support all device power states.
  147. *
  148. * @param dev Device instance.
  149. * @param state Device power state to be set.
  150. *
  151. * @retval 0 If successful.
  152. * @retval -ENOTSUP If requested state is not supported.
  153. * @retval -EALREADY If device is already at the requested state.
  154. * @retval -EBUSY If device is changing its state.
  155. * @retval Errno Other negative errno on failure.
  156. */
  157. int pm_device_state_set(const struct device *dev,
  158. enum pm_device_state state);
  159. /**
  160. * @brief Obtain the power state of a device.
  161. *
  162. * @param dev Device instance.
  163. * @param state Pointer where device power state will be stored.
  164. *
  165. * @retval 0 If successful.
  166. * @retval -ENOSYS If device does not implement power management.
  167. */
  168. int pm_device_state_get(const struct device *dev,
  169. enum pm_device_state *state);
  170. #ifdef CONFIG_PM_DEVICE
  171. /**
  172. * @brief Indicate that the device is in the middle of a transaction
  173. *
  174. * Called by a device driver to indicate that it is in the middle of a
  175. * transaction.
  176. *
  177. * @param dev Pointer to device structure of the driver instance.
  178. */
  179. void pm_device_busy_set(const struct device *dev);
  180. /**
  181. * @brief Indicate that the device has completed its transaction
  182. *
  183. * Called by a device driver to indicate the end of a transaction.
  184. *
  185. * @param dev Pointer to device structure of the driver instance.
  186. */
  187. void pm_device_busy_clear(const struct device *dev);
  188. /**
  189. * @brief Check if any device is in the middle of a transaction
  190. *
  191. * Called by an application to see if any device is in the middle
  192. * of a critical transaction that cannot be interrupted.
  193. *
  194. * @retval false if no device is busy
  195. * @retval true if any device is busy
  196. */
  197. bool pm_device_is_any_busy(void);
  198. /**
  199. * @brief Check if a specific device is in the middle of a transaction
  200. *
  201. * Called by an application to see if a particular device is in the
  202. * middle of a critical transaction that cannot be interrupted.
  203. *
  204. * @param dev Pointer to device structure of the specific device driver
  205. * the caller is interested in.
  206. * @retval false if the device is not busy
  207. * @retval true if the device is busy
  208. */
  209. bool pm_device_is_busy(const struct device *dev);
  210. #else
  211. static inline void pm_device_busy_set(const struct device *dev) {}
  212. static inline void pm_device_busy_clear(const struct device *dev) {}
  213. static inline bool pm_device_is_any_busy(void) { return false; }
  214. static inline bool pm_device_is_busy(const struct device *dev) { return false; }
  215. #endif
  216. __deprecated static inline void device_busy_set(const struct device *dev)
  217. {
  218. pm_device_busy_set(dev);
  219. }
  220. __deprecated static inline void device_busy_clear(const struct device *dev)
  221. {
  222. pm_device_busy_clear(dev);
  223. }
  224. __deprecated static inline int device_any_busy_check(void)
  225. {
  226. return pm_device_is_any_busy() ? -EBUSY : 0;
  227. }
  228. __deprecated static inline int device_busy_check(const struct device *dev)
  229. {
  230. return pm_device_is_busy(dev) ? -EBUSY : 0;
  231. }
  232. /** Alias for legacy use of device_pm_control_nop */
  233. #define device_pm_control_nop __DEPRECATED_MACRO NULL
  234. /**
  235. * @brief Enable a power management wakeup source
  236. *
  237. * Enable a wakeup source. This will keep the current device active when the
  238. * system is suspended, allowing it to be used to wake up the system.
  239. *
  240. * @param dev device object to enable.
  241. * @param enable @c true to enable or @c false to disable
  242. *
  243. * @retval true if the wakeup source was successfully enabled.
  244. * @retval false if the wakeup source was not successfully enabled.
  245. */
  246. bool pm_device_wakeup_enable(struct device *dev, bool enable);
  247. /**
  248. * @brief Check if a power management wakeup source is enabled
  249. *
  250. * Checks if a wake up source is enabled.
  251. *
  252. * @param dev device object to check.
  253. *
  254. * @retval true if the wakeup source is enabled.
  255. * @retval false if the wakeup source is not enabled.
  256. */
  257. bool pm_device_wakeup_is_enabled(const struct device *dev);
  258. /**
  259. * @brief Check if a device is wake up capable
  260. *
  261. * @param dev device object to check.
  262. *
  263. * @retval true if the device is wake up capable.
  264. * @retval false if the device is not wake up capable.
  265. */
  266. bool pm_device_wakeup_is_capable(const struct device *dev);
  267. /** @} */
  268. #ifdef __cplusplus
  269. }
  270. #endif
  271. #endif