rpmsg_multi_instance.h 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212
  1. /*
  2. * Copyright (c) 2021 Nordic Semiconductor ASA
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #ifndef ZEPHYR_INCLUDE_RPMSG_MULTIPLE_INSTANCE_H_
  7. #define ZEPHYR_INCLUDE_RPMSG_MULTIPLE_INSTANCE_H_
  8. #include <openamp/open_amp.h>
  9. #include <metal/sys.h>
  10. #include <metal/device.h>
  11. #include <metal/alloc.h>
  12. #ifdef __cplusplus
  13. extern "C" {
  14. #endif
  15. /**
  16. * @brief RPMsg multiple instance API
  17. * @defgroup rpmsg_multiple_instance_api RPMsg multiple instance APIs
  18. * @{
  19. */
  20. #define VDEV_START_ADDR CONFIG_RPMSG_MULTI_INSTANCE_SHM_BASE_ADDRESS
  21. #define VDEV_SIZE CONFIG_RPMSG_MULTI_INSTANCE_SHM_SIZE
  22. #define SHM_START_ADDR VDEV_START_ADDR
  23. #define SHM_SIZE VDEV_SIZE
  24. #define VRING_ALIGNMENT (4) /**< Alignment of vring buffer. */
  25. #define VDEV_STATUS_SIZE (0x4) /**< Size of status region. */
  26. /** @brief Event callback structure.
  27. *
  28. * It is registered during endpoint registration.
  29. * This structure is packed into the endpoint configuration.
  30. */
  31. struct rpmsg_mi_cb {
  32. /** @brief Bind was successful.
  33. *
  34. * @param priv Private user data.
  35. */
  36. void (*bound)(void *priv);
  37. /** @brief New packet arrived.
  38. *
  39. * @param data Pointer to data buffer.
  40. * @param len Length of @a data.
  41. * @param priv Private user data.
  42. */
  43. void (*received)(const void *data, size_t len, void *priv);
  44. };
  45. /** @brief Endpoint instance. */
  46. struct rpmsg_mi_ept {
  47. /** Name of endpoint. */
  48. const char *name;
  49. /** RPMsg endpoint. */
  50. struct rpmsg_endpoint ep;
  51. /** Event callback structure. */
  52. struct rpmsg_mi_cb *cb;
  53. /** Private user data. */
  54. void *priv;
  55. /** Endpoint was bound. */
  56. volatile bool bound;
  57. /** Linked list node. */
  58. sys_snode_t node;
  59. };
  60. /** @brief Endpoint configuration. */
  61. struct rpmsg_mi_ept_cfg {
  62. /** Name of endpoint. */
  63. const char *name;
  64. /** Event callback structure. */
  65. struct rpmsg_mi_cb *cb;
  66. /** Private user data. */
  67. void *priv;
  68. };
  69. /** @brief Struct describing the context of the RPMsg instance. */
  70. struct rpmsg_mi_ctx {
  71. const char *name;
  72. struct k_work_q ipm_work_q;
  73. struct k_work ipm_work;
  74. const struct device *ipm_tx_handle;
  75. const struct device *ipm_rx_handle;
  76. unsigned int ipm_tx_id;
  77. uintptr_t shm_status_reg_addr;
  78. struct metal_io_region *shm_io;
  79. struct metal_device shm_device;
  80. metal_phys_addr_t shm_physmap[1];
  81. struct rpmsg_virtio_device rvdev;
  82. struct rpmsg_virtio_shm_pool shpool;
  83. struct rpmsg_device *rdev;
  84. struct virtqueue *vq[2];
  85. struct virtio_vring_info rvrings[2];
  86. struct virtio_device vdev;
  87. uintptr_t vring_tx_addr;
  88. uintptr_t vring_rx_addr;
  89. sys_slist_t endpoints;
  90. };
  91. struct rpmsg_mi_ctx_shm_cfg {
  92. /** Physical address shared memory region. */
  93. uintptr_t addr;
  94. /** Size shared memory region. */
  95. size_t size;
  96. /** Internal counter. */
  97. unsigned int instance;
  98. };
  99. /** @brief Configuration of the RPMsg instance. */
  100. struct rpmsg_mi_ctx_cfg {
  101. /** Name of instance. */
  102. const char *name;
  103. /** Stack area for k_work_q. */
  104. k_thread_stack_t *ipm_stack_area;
  105. /** Size of stack area. */
  106. size_t ipm_stack_size;
  107. /** Priority of work_q. */
  108. int ipm_work_q_prio;
  109. /** Name of work_q thread. */
  110. const char *ipm_thread_name;
  111. /** Name of the TX IPM channel. */
  112. const char *ipm_tx_name;
  113. /** Name of the RX IPM channel. */
  114. const char *ipm_rx_name;
  115. /** IPM message identifier. */
  116. unsigned int ipm_tx_id;
  117. /** SHM struct. */
  118. struct rpmsg_mi_ctx_shm_cfg *shm;
  119. };
  120. /** @brief Initialization of RPMsg instance.
  121. *
  122. * Each instance has an automatically allocated area of shared memory.
  123. *
  124. * @param ctx Pointer to the RPMsg instance.
  125. * @param cfg Pointer to the configuration structure.
  126. * @retval 0 if the operation was successful.
  127. * -EINVAL when the incorrect parameters have been passed.
  128. * -EIO when the configuration is not correct.
  129. * -ENODEV failed to get TX or RX IPM handle.
  130. * -ENOMEM when there is not enough memory to register virqueue.
  131. * < 0 on other negative errno code, reported by rpmsg.
  132. */
  133. int rpmsg_mi_ctx_init(struct rpmsg_mi_ctx *ctx, const struct rpmsg_mi_ctx_cfg *cfg);
  134. /** @brief Register IPC endpoint.
  135. *
  136. * Registers IPC endpoint to enable communication with a remote device.
  137. *
  138. * @param ctx Pointer to the RPMsg instance.
  139. * @param ept Pointer to endpoint object.
  140. * @param cfg Pointer to the endpoint configuration.
  141. *
  142. * @retval -EINVAL One of the parameters is incorrect.
  143. * @retval other errno code reported by rpmsg.
  144. */
  145. int rpmsg_mi_ept_register(struct rpmsg_mi_ctx *ctx,
  146. struct rpmsg_mi_ept *ept,
  147. struct rpmsg_mi_ept_cfg *cfg);
  148. /** @brief Send data using given IPC endpoint.
  149. *
  150. * Note: It is not possible to send a message of zero length.
  151. *
  152. * @param ept Endpoint object.
  153. * @param data Pointer to the buffer to send through RPMsg.
  154. * @param len Number of bytes to send.
  155. *
  156. * @retval Number of bytes it has sent or negative error value on failure.
  157. */
  158. int rpmsg_mi_send(struct rpmsg_mi_ept *ept, const void *data, size_t len);
  159. /**
  160. * @}
  161. */
  162. #ifdef __cplusplus
  163. }
  164. #endif
  165. #endif /* ZEPHYR_INCLUDE_RPMSG_MULTIPLE_INNSTANCE_H_ */