pcie_ep.h 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310
  1. /**
  2. * @file
  3. *
  4. * @brief Public APIs for the PCIe EP drivers.
  5. */
  6. /*
  7. * SPDX-License-Identifier: Apache-2.0
  8. *
  9. * Copyright 2020 Broadcom
  10. *
  11. */
  12. #ifndef ZEPHYR_INCLUDE_DRIVERS_PCIE_EP_H_
  13. #define ZEPHYR_INCLUDE_DRIVERS_PCIE_EP_H_
  14. #include <device.h>
  15. #include <init.h>
  16. #include <kernel.h>
  17. #include <stdint.h>
  18. enum pcie_ob_mem_type {
  19. PCIE_OB_ANYMEM, /**< PCIe OB window within any address range */
  20. PCIE_OB_LOWMEM, /**< PCIe OB window within 32-bit address range */
  21. PCIE_OB_HIGHMEM, /**< PCIe OB window above 32-bit address range */
  22. };
  23. enum pci_ep_irq_type {
  24. PCIE_EP_IRQ_LEGACY, /**< Raise Legacy interrupt */
  25. PCIE_EP_IRQ_MSI, /**< Raise MSI interrupt */
  26. PCIE_EP_IRQ_MSIX, /**< Raise MSIX interrupt */
  27. };
  28. enum xfer_direction {
  29. HOST_TO_DEVICE, /**< Read from Host */
  30. DEVICE_TO_HOST, /**< Write to Host */
  31. };
  32. enum pcie_reset {
  33. PCIE_PERST, /**< Cold reset */
  34. PCIE_PERST_INB, /**< Inband hot reset */
  35. PCIE_FLR, /**< Functional Level Reset */
  36. PCIE_RESET_MAX
  37. };
  38. /**
  39. * @typedef pcie_ep_reset_callback_t
  40. * @brief Callback API for PCIe reset interrupts
  41. *
  42. * These callbacks execute in interrupt context. Therefore, use only
  43. * interrupt-safe APIS. Registration of callbacks is done via
  44. * @a pcie_ep_register_reset_cb
  45. *
  46. * @param arg Pointer provided at registration time, later to be
  47. * passed back as argument to callback function
  48. */
  49. typedef void (*pcie_ep_reset_callback_t)(void *arg);
  50. struct pcie_ep_driver_api {
  51. int (*conf_read)(const struct device *dev, uint32_t offset,
  52. uint32_t *data);
  53. void (*conf_write)(const struct device *dev, uint32_t offset,
  54. uint32_t data);
  55. int (*map_addr)(const struct device *dev, uint64_t pcie_addr,
  56. uint64_t *mapped_addr, uint32_t size,
  57. enum pcie_ob_mem_type ob_mem_type);
  58. void (*unmap_addr)(const struct device *dev, uint64_t mapped_addr);
  59. int (*raise_irq)(const struct device *dev,
  60. enum pci_ep_irq_type irq_type,
  61. uint32_t irq_num);
  62. int (*register_reset_cb)(const struct device *dev,
  63. enum pcie_reset reset,
  64. pcie_ep_reset_callback_t cb, void *arg);
  65. int (*dma_xfer)(const struct device *dev, uint64_t mapped_addr,
  66. uintptr_t local_addr, uint32_t size,
  67. enum xfer_direction dir);
  68. };
  69. /**
  70. * @brief Read PCIe EP configuration space
  71. *
  72. * @details This API reads EP's own configuration space
  73. *
  74. * @param dev Pointer to the device structure for the driver instance
  75. * @param offset Offset within configuration space
  76. * @param data Pointer to data read from the offset
  77. *
  78. * @return 0 if successful, negative errno code if failure.
  79. */
  80. static inline int pcie_ep_conf_read(const struct device *dev,
  81. uint32_t offset, uint32_t *data)
  82. {
  83. const struct pcie_ep_driver_api *api =
  84. (const struct pcie_ep_driver_api *)dev->api;
  85. return api->conf_read(dev, offset, data);
  86. }
  87. /**
  88. * @brief Write PCIe EP configuration space
  89. *
  90. * @details This API writes EP's own configuration space
  91. *
  92. * @param dev Pointer to the device structure for the driver instance
  93. * @param offset Offset within configuration space
  94. * @param data Data to be written at the offset
  95. *
  96. * @return N/A
  97. */
  98. static inline void pcie_ep_conf_write(const struct device *dev,
  99. uint32_t offset, uint32_t data)
  100. {
  101. const struct pcie_ep_driver_api *api =
  102. (const struct pcie_ep_driver_api *)dev->api;
  103. api->conf_write(dev, offset, data);
  104. }
  105. /**
  106. * @brief Map a host memory buffer to PCIe outbound region
  107. *
  108. * @details This API maps a host memory buffer to PCIe outbound region,
  109. * It is left to EP driver to manage multiple mappings through
  110. * multiple PCIe outbound regions if supported by SoC
  111. *
  112. * @param dev Pointer to the device structure for the driver instance
  113. * @param pcie_addr Host memory buffer address to be mapped
  114. * @param mapped_addr Mapped PCIe outbound region address
  115. * @param size Host memory buffer size (bytes)
  116. * @param ob_mem_type Hint if lowmem/highmem outbound region has to be used,
  117. * this is useful in cases where bus master cannot generate
  118. * more than 32-bit address; it becomes essential to use
  119. * lowmem outbound region
  120. *
  121. * @return Mapped size : If mapped size is less than requested size,
  122. * then requestor has to call the same API again to map
  123. * the unmapped host buffer after data transfer is done with
  124. * mapped size. This situation may arise because of the
  125. * mapping alignment requirements.
  126. *
  127. * @return Negative errno code if failure.
  128. */
  129. static inline int pcie_ep_map_addr(const struct device *dev,
  130. uint64_t pcie_addr,
  131. uint64_t *mapped_addr, uint32_t size,
  132. enum pcie_ob_mem_type ob_mem_type)
  133. {
  134. const struct pcie_ep_driver_api *api =
  135. (const struct pcie_ep_driver_api *)dev->api;
  136. return api->map_addr(dev, pcie_addr, mapped_addr, size, ob_mem_type);
  137. }
  138. /**
  139. * @brief Remove mapping to PCIe outbound region
  140. *
  141. * @details This API removes mapping to PCIe outbound region.
  142. * Mapped PCIe outbound region address is given as argument
  143. * to figure out the outbound region to be unmapped
  144. *
  145. * @param dev Pointer to the device structure for the driver instance
  146. * @param mapped_addr PCIe outbound region address
  147. *
  148. * @return N/A
  149. */
  150. static inline void pcie_ep_unmap_addr(const struct device *dev,
  151. uint64_t mapped_addr)
  152. {
  153. const struct pcie_ep_driver_api *api =
  154. (const struct pcie_ep_driver_api *)dev->api;
  155. api->unmap_addr(dev, mapped_addr);
  156. }
  157. /**
  158. * @brief Raise interrupt to Host
  159. *
  160. * @details This API raises interrupt to Host
  161. *
  162. * @param dev Pointer to the device structure for the driver instance
  163. * @param irq_type Type of Interrupt be raised (legacy, MSI or MSI-X)
  164. * @param irq_num MSI or MSI-X interrupt number
  165. *
  166. * @return 0 if successful, negative errno code if failure.
  167. */
  168. static inline int pcie_ep_raise_irq(const struct device *dev,
  169. enum pci_ep_irq_type irq_type,
  170. uint32_t irq_num)
  171. {
  172. const struct pcie_ep_driver_api *api =
  173. (const struct pcie_ep_driver_api *)dev->api;
  174. return api->raise_irq(dev, irq_type, irq_num);
  175. }
  176. /**
  177. * @brief Register callback function for reset interrupts
  178. *
  179. * @details If reset interrupts are handled by device, this API can be
  180. * used to register callback function, which will be
  181. * executed part of corresponding PCIe reset handler
  182. *
  183. * @param dev Pointer to the device structure for the driver instance
  184. * @param reset Reset interrupt type
  185. * @param cb Callback function being registered
  186. * @param arg Argument to be passed back to callback function
  187. *
  188. * @return 0 if successful, negative errno code if failure.
  189. */
  190. static inline int pcie_ep_register_reset_cb(const struct device *dev,
  191. enum pcie_reset reset,
  192. pcie_ep_reset_callback_t cb,
  193. void *arg)
  194. {
  195. const struct pcie_ep_driver_api *api =
  196. (const struct pcie_ep_driver_api *)dev->api;
  197. if (api->register_reset_cb) {
  198. return api->register_reset_cb(dev, reset, cb, arg);
  199. }
  200. return -ENOTSUP;
  201. }
  202. /**
  203. * @brief Data transfer between mapped Host memory and device memory with
  204. * "System DMA". The term "System DMA" is used to clarify that we
  205. * are not talking about dedicated "PCIe DMA"; rather the one
  206. * which does not understand PCIe address directly, and
  207. * uses the mapped Host memory.
  208. *
  209. * @details If DMA controller is available in the EP device, this API can be
  210. * used to achieve data transfer between mapped Host memory,
  211. * i.e. outbound memory and EP device's local memory with DMA
  212. *
  213. * @param dev Pointer to the device structure for the driver instance
  214. * @param mapped_addr Mapped Host memory address
  215. * @param local_addr Device memory address
  216. * @param size DMA transfer length (bytes)
  217. * @param dir Direction of DMA transfer
  218. *
  219. * @return 0 if successful, negative errno code if failure.
  220. */
  221. static inline int pcie_ep_dma_xfer(const struct device *dev,
  222. uint64_t mapped_addr,
  223. uintptr_t local_addr, uint32_t size,
  224. const enum xfer_direction dir)
  225. {
  226. const struct pcie_ep_driver_api *api =
  227. (const struct pcie_ep_driver_api *)dev->api;
  228. if (api->dma_xfer) {
  229. return api->dma_xfer(dev, mapped_addr, local_addr, size, dir);
  230. }
  231. return -ENOTSUP;
  232. }
  233. /**
  234. * @brief Data transfer using memcpy
  235. *
  236. * @details Helper API to achieve data transfer with memcpy
  237. * through PCIe outbound memory
  238. *
  239. * @param dev Pointer to the device structure for the driver instance
  240. * @param pcie_addr Host memory buffer address
  241. * @param local_addr Local memory buffer address
  242. * @param size Data transfer size (bytes)
  243. * @param ob_mem_type Hint if lowmem/highmem outbound region has to be used
  244. * (PCIE_OB_LOWMEM / PCIE_OB_HIGHMEM / PCIE_OB_ANYMEM),
  245. * should be PCIE_OB_LOWMEM if bus master cannot generate
  246. * more than 32-bit address
  247. * @param dir Data transfer direction (HOST_TO_DEVICE / DEVICE_TO_HOST)
  248. *
  249. * @return 0 if successful, negative errno code if failure.
  250. */
  251. int pcie_ep_xfer_data_memcpy(const struct device *dev, uint64_t pcie_addr,
  252. uintptr_t *local_addr, uint32_t size,
  253. enum pcie_ob_mem_type ob_mem_type,
  254. enum xfer_direction dir);
  255. /**
  256. * @brief Data transfer using system DMA
  257. *
  258. * @details Helper API to achieve data transfer with system DMA through PCIe
  259. * outbound memory, this API is based off pcie_ep_xfer_data_memcpy,
  260. * here we use "system dma" instead of memcpy
  261. *
  262. * @param dev Pointer to the device structure for the driver instance
  263. * @param pcie_addr Host memory buffer address
  264. * @param local_addr Local memory buffer address
  265. * @param size Data transfer size (bytes)
  266. * @param ob_mem_type Hint if lowmem/highmem outbound region has to be used
  267. * (PCIE_OB_LOWMEM / PCIE_OB_HIGHMEM / PCIE_OB_ANYMEM)
  268. * @param dir Data transfer direction (HOST_TO_DEVICE / DEVICE_TO_HOST)
  269. *
  270. * @return 0 if successful, negative errno code if failure.
  271. */
  272. int pcie_ep_xfer_data_dma(const struct device *dev, uint64_t pcie_addr,
  273. uintptr_t *local_addr, uint32_t size,
  274. enum pcie_ob_mem_type ob_mem_type,
  275. enum xfer_direction dir);
  276. #endif /* ZEPHYR_INCLUDE_DRIVERS_PCIE_EP_H_ */