ipm.h 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234
  1. /**
  2. * @file
  3. *
  4. * @brief Generic low-level inter-processor mailbox communication API.
  5. */
  6. /*
  7. * Copyright (c) 2015 Intel Corporation
  8. *
  9. * SPDX-License-Identifier: Apache-2.0
  10. */
  11. #ifndef ZEPHYR_INCLUDE_DRIVERS_IPM_H_
  12. #define ZEPHYR_INCLUDE_DRIVERS_IPM_H_
  13. /**
  14. * @brief IPM Interface
  15. * @defgroup ipm_interface IPM Interface
  16. * @ingroup io_interfaces
  17. * @{
  18. */
  19. #include <kernel.h>
  20. #include <device.h>
  21. #ifdef __cplusplus
  22. extern "C" {
  23. #endif
  24. /**
  25. * @typedef ipm_callback_t
  26. * @brief Callback API for incoming IPM messages
  27. *
  28. * These callbacks execute in interrupt context. Therefore, use only
  29. * interrupt-safe APIS. Registration of callbacks is done via
  30. * @a ipm_register_callback
  31. *
  32. * @param ipmdev Driver instance
  33. * @param user_data Pointer to some private data provided at registration
  34. * time.
  35. * @param id Message type identifier.
  36. * @param data Message data pointer. The correct amount of data to read out
  37. * must be inferred using the message id/upper level protocol.
  38. */
  39. typedef void (*ipm_callback_t)(const struct device *ipmdev, void *user_data,
  40. uint32_t id, volatile void *data);
  41. /**
  42. * @typedef ipm_send_t
  43. * @brief Callback API to send IPM messages
  44. *
  45. * See @a ipm_send() for argument definitions.
  46. */
  47. typedef int (*ipm_send_t)(const struct device *ipmdev, int wait, uint32_t id,
  48. const void *data, int size);
  49. /**
  50. * @typedef ipm_max_data_size_get_t
  51. * @brief Callback API to get maximum data size
  52. *
  53. * See @a ipm_max_data_size_get() for argument definitions.
  54. */
  55. typedef int (*ipm_max_data_size_get_t)(const struct device *ipmdev);
  56. /**
  57. * @typedef ipm_max_id_val_get_t
  58. * @brief Callback API to get the ID's maximum value
  59. *
  60. * See @a ipm_max_id_val_get() for argument definitions.
  61. */
  62. typedef uint32_t (*ipm_max_id_val_get_t)(const struct device *ipmdev);
  63. /**
  64. * @typedef ipm_register_callback_t
  65. * @brief Callback API upon registration
  66. *
  67. * See @a ipm_register_callback() for argument definitions.
  68. */
  69. typedef void (*ipm_register_callback_t)(const struct device *port,
  70. ipm_callback_t cb,
  71. void *user_data);
  72. /**
  73. * @typedef ipm_set_enabled_t
  74. * @brief Callback API upon enablement of interrupts
  75. *
  76. * See @a ipm_set_enabled() for argument definitions.
  77. */
  78. typedef int (*ipm_set_enabled_t)(const struct device *ipmdev, int enable);
  79. __subsystem struct ipm_driver_api {
  80. ipm_send_t send;
  81. ipm_register_callback_t register_callback;
  82. ipm_max_data_size_get_t max_data_size_get;
  83. ipm_max_id_val_get_t max_id_val_get;
  84. ipm_set_enabled_t set_enabled;
  85. };
  86. /**
  87. * @brief Try to send a message over the IPM device.
  88. *
  89. * A message is considered consumed once the remote interrupt handler
  90. * finishes. If there is deferred processing on the remote side,
  91. * or if outgoing messages must be queued and wait on an
  92. * event/semaphore, a high-level driver can implement that.
  93. *
  94. * There are constraints on how much data can be sent or the maximum value
  95. * of id. Use the @a ipm_max_data_size_get and @a ipm_max_id_val_get routines
  96. * to determine them.
  97. *
  98. * The @a size parameter is used only on the sending side to determine
  99. * the amount of data to put in the message registers. It is not passed along
  100. * to the receiving side. The upper-level protocol dictates the amount of
  101. * data read back.
  102. *
  103. * @param ipmdev Driver instance
  104. * @param wait If nonzero, busy-wait for remote to consume the message. The
  105. * message is considered consumed once the remote interrupt handler
  106. * finishes. If there is deferred processing on the remote side,
  107. * or you would like to queue outgoing messages and wait on an
  108. * event/semaphore, you can implement that in a high-level driver
  109. * @param id Message identifier. Values are constrained by
  110. * @a ipm_max_data_size_get since many boards only allow for a
  111. * subset of bits in a 32-bit register to store the ID.
  112. * @param data Pointer to the data sent in the message.
  113. * @param size Size of the data.
  114. *
  115. * @retval -EBUSY If the remote hasn't yet read the last data sent.
  116. * @retval -EMSGSIZE If the supplied data size is unsupported by the driver.
  117. * @retval -EINVAL If there was a bad parameter, such as: too-large id value.
  118. * or the device isn't an outbound IPM channel.
  119. * @retval 0 On success.
  120. */
  121. __syscall int ipm_send(const struct device *ipmdev, int wait, uint32_t id,
  122. const void *data, int size);
  123. static inline int z_impl_ipm_send(const struct device *ipmdev, int wait,
  124. uint32_t id,
  125. const void *data, int size)
  126. {
  127. const struct ipm_driver_api *api =
  128. (const struct ipm_driver_api *)ipmdev->api;
  129. return api->send(ipmdev, wait, id, data, size);
  130. }
  131. /**
  132. * @brief Register a callback function for incoming messages.
  133. *
  134. * @param ipmdev Driver instance pointer.
  135. * @param cb Callback function to execute on incoming message interrupts.
  136. * @param user_data Application-specific data pointer which will be passed
  137. * to the callback function when executed.
  138. */
  139. static inline void ipm_register_callback(const struct device *ipmdev,
  140. ipm_callback_t cb, void *user_data)
  141. {
  142. const struct ipm_driver_api *api =
  143. (const struct ipm_driver_api *)ipmdev->api;
  144. api->register_callback(ipmdev, cb, user_data);
  145. }
  146. /**
  147. * @brief Return the maximum number of bytes possible in an outbound message.
  148. *
  149. * IPM implementations vary on the amount of data that can be sent in a
  150. * single message since the data payload is typically stored in registers.
  151. *
  152. * @param ipmdev Driver instance pointer.
  153. *
  154. * @return Maximum possible size of a message in bytes.
  155. */
  156. __syscall int ipm_max_data_size_get(const struct device *ipmdev);
  157. static inline int z_impl_ipm_max_data_size_get(const struct device *ipmdev)
  158. {
  159. const struct ipm_driver_api *api =
  160. (const struct ipm_driver_api *)ipmdev->api;
  161. return api->max_data_size_get(ipmdev);
  162. }
  163. /**
  164. * @brief Return the maximum id value possible in an outbound message.
  165. *
  166. * Many IPM implementations store the message's ID in a register with
  167. * some bits reserved for other uses.
  168. *
  169. * @param ipmdev Driver instance pointer.
  170. *
  171. * @return Maximum possible value of a message ID.
  172. */
  173. __syscall uint32_t ipm_max_id_val_get(const struct device *ipmdev);
  174. static inline uint32_t z_impl_ipm_max_id_val_get(const struct device *ipmdev)
  175. {
  176. const struct ipm_driver_api *api =
  177. (const struct ipm_driver_api *)ipmdev->api;
  178. return api->max_id_val_get(ipmdev);
  179. }
  180. /**
  181. * @brief Enable interrupts and callbacks for inbound channels.
  182. *
  183. * @param ipmdev Driver instance pointer.
  184. * @param enable Set to 0 to disable and to nonzero to enable.
  185. *
  186. * @retval 0 On success.
  187. * @retval -EINVAL If it isn't an inbound channel.
  188. */
  189. __syscall int ipm_set_enabled(const struct device *ipmdev, int enable);
  190. static inline int z_impl_ipm_set_enabled(const struct device *ipmdev,
  191. int enable)
  192. {
  193. const struct ipm_driver_api *api =
  194. (const struct ipm_driver_api *)ipmdev->api;
  195. return api->set_enabled(ipmdev, enable);
  196. }
  197. #ifdef __cplusplus
  198. }
  199. #endif
  200. /**
  201. * @}
  202. */
  203. #include <syscalls/ipm.h>
  204. #endif /* ZEPHYR_INCLUDE_DRIVERS_IPM_H_ */