pcie.h 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295
  1. /*
  2. * Copyright (c) 2019 Intel Corporation
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #ifndef ZEPHYR_INCLUDE_DRIVERS_PCIE_PCIE_H_
  7. #define ZEPHYR_INCLUDE_DRIVERS_PCIE_PCIE_H_
  8. #include <stddef.h>
  9. #include <dt-bindings/pcie/pcie.h>
  10. #include <zephyr/types.h>
  11. #ifdef __cplusplus
  12. extern "C" {
  13. #endif
  14. /**
  15. * @typedef pcie_bdf_t
  16. * @brief A unique PCI(e) endpoint (bus, device, function).
  17. *
  18. * A PCI(e) endpoint is uniquely identified topologically using a
  19. * (bus, device, function) tuple. The internal structure is documented
  20. * in include/dt-bindings/pcie/pcie.h: see PCIE_BDF() and friends, since
  21. * these tuples are referenced from devicetree.
  22. */
  23. typedef uint32_t pcie_bdf_t;
  24. /**
  25. * @typedef pcie_id_t
  26. * @brief A unique PCI(e) identifier (vendor ID, device ID).
  27. *
  28. * The PCIE_CONF_ID register for each endpoint is a (vendor ID, device ID)
  29. * pair, which is meant to tell the system what the PCI(e) endpoint is. Again,
  30. * look to PCIE_ID_* macros in include/dt-bindings/pcie/pcie.h for more.
  31. */
  32. typedef uint32_t pcie_id_t;
  33. struct pcie_mbar {
  34. uintptr_t phys_addr;
  35. size_t size;
  36. };
  37. /*
  38. * These functions are arch-, board-, or SoC-specific.
  39. */
  40. /**
  41. * @brief Look up the BDF based on PCI(e) vendor & device ID
  42. *
  43. * This function is used to look up the BDF for a device given its
  44. * vendor and device ID.
  45. *
  46. * @param id PCI(e) vendor & device ID encoded using PCIE_ID()
  47. * @return The BDF for the device, or PCIE_BDF_NONE if it was not found
  48. */
  49. extern pcie_bdf_t pcie_bdf_lookup(pcie_id_t id);
  50. /**
  51. * @brief Read a 32-bit word from an endpoint's configuration space.
  52. *
  53. * This function is exported by the arch/SoC/board code.
  54. *
  55. * @param bdf PCI(e) endpoint
  56. * @param reg the configuration word index (not address)
  57. * @return the word read (0xFFFFFFFFU if nonexistent endpoint or word)
  58. */
  59. extern uint32_t pcie_conf_read(pcie_bdf_t bdf, unsigned int reg);
  60. /**
  61. * @brief Write a 32-bit word to an endpoint's configuration space.
  62. *
  63. * This function is exported by the arch/SoC/board code.
  64. *
  65. * @param bdf PCI(e) endpoint
  66. * @param reg the configuration word index (not address)
  67. * @param data the value to write
  68. */
  69. extern void pcie_conf_write(pcie_bdf_t bdf, unsigned int reg, uint32_t data);
  70. /**
  71. * @brief Probe for the presence of a PCI(e) endpoint.
  72. *
  73. * @param bdf the endpoint to probe
  74. * @param id the endpoint ID to expect, or PCIE_ID_NONE for "any device"
  75. * @return true if the device is present, false otherwise
  76. */
  77. extern bool pcie_probe(pcie_bdf_t bdf, pcie_id_t id);
  78. /**
  79. * @brief Get the MBAR at a specific BAR index
  80. * @param bdf the PCI(e) endpoint
  81. * @param bar_index 0-based BAR index
  82. * @param mbar Pointer to struct pcie_mbar
  83. * @return true if the mbar was found and is valid, false otherwise
  84. */
  85. extern bool pcie_get_mbar(pcie_bdf_t bdf,
  86. unsigned int bar_index,
  87. struct pcie_mbar *mbar);
  88. /**
  89. * @brief Probe the nth MMIO address assigned to an endpoint.
  90. * @param bdf the PCI(e) endpoint
  91. * @param index (0-based) index
  92. * @param mbar Pointer to struct pcie_mbar
  93. * @return true if the mbar was found and is valid, false otherwise
  94. *
  95. * A PCI(e) endpoint has 0 or more memory-mapped regions. This function
  96. * allows the caller to enumerate them by calling with index=0..n.
  97. * Value of n has to be below 6, as there is a maximum of 6 BARs. The indices
  98. * are order-preserving with respect to the endpoint BARs: e.g., index 0
  99. * will return the lowest-numbered memory BAR on the endpoint.
  100. */
  101. extern bool pcie_probe_mbar(pcie_bdf_t bdf,
  102. unsigned int index,
  103. struct pcie_mbar *mbar);
  104. /**
  105. * @brief Set or reset bits in the endpoint command/status register.
  106. *
  107. * @param bdf the PCI(e) endpoint
  108. * @param bits the powerset of bits of interest
  109. * @param on use true to set bits, false to reset them
  110. */
  111. extern void pcie_set_cmd(pcie_bdf_t bdf, uint32_t bits, bool on);
  112. /**
  113. * @brief Allocate an IRQ for an endpoint.
  114. *
  115. * This function first checks the IRQ register and if it contains a valid
  116. * value this is returned. If the register does not contain a valid value
  117. * allocation of a new one is attempted.
  118. *
  119. * @param bdf the PCI(e) endpoint
  120. * @return the IRQ number, or PCIE_CONF_INTR_IRQ_NONE if allocation failed.
  121. */
  122. extern unsigned int pcie_alloc_irq(pcie_bdf_t bdf);
  123. /**
  124. * @brief Return the IRQ assigned by the firmware/board to an endpoint.
  125. *
  126. * @param bdf the PCI(e) endpoint
  127. * @return the IRQ number, or PCIE_CONF_INTR_IRQ_NONE if unknown.
  128. */
  129. extern unsigned int pcie_get_irq(pcie_bdf_t bdf);
  130. /**
  131. * @brief Enable the PCI(e) endpoint to generate the specified IRQ.
  132. *
  133. * @param bdf the PCI(e) endpoint
  134. * @param irq the IRQ to generate
  135. *
  136. * If MSI is enabled and the endpoint supports it, the endpoint will
  137. * be configured to generate the specified IRQ via MSI. Otherwise, it
  138. * is assumed that the IRQ has been routed by the boot firmware
  139. * to the specified IRQ, and the IRQ is enabled (at the I/O APIC, or
  140. * wherever appropriate).
  141. */
  142. extern void pcie_irq_enable(pcie_bdf_t bdf, unsigned int irq);
  143. /**
  144. * @brief Find a PCI(e) capability in an endpoint's configuration space.
  145. *
  146. * @param bdf the PCI endpoint to examine
  147. * @param cap_id the capability ID of interest
  148. * @return the index of the configuration word, or 0 if no capability.
  149. */
  150. extern uint32_t pcie_get_cap(pcie_bdf_t bdf, uint32_t cap_id);
  151. /**
  152. * @brief Find an Extended PCI(e) capability in an endpoint's configuration space.
  153. *
  154. * @param bdf the PCI endpoint to examine
  155. * @param cap_id the capability ID of interest
  156. * @return the index of the configuration word, or 0 if no capability.
  157. */
  158. extern uint32_t pcie_get_ext_cap(pcie_bdf_t bdf, uint32_t cap_id);
  159. /*
  160. * Configuration word 13 contains the head of the capabilities list.
  161. */
  162. #define PCIE_CONF_CAPPTR 13U /* capabilities pointer */
  163. #define PCIE_CONF_CAPPTR_FIRST(w) (((w) >> 2) & 0x3FU)
  164. /*
  165. * The first word of every capability contains a capability identifier,
  166. * and a link to the next capability (or 0) in configuration space.
  167. */
  168. #define PCIE_CONF_CAP_ID(w) ((w) & 0xFFU)
  169. #define PCIE_CONF_CAP_NEXT(w) (((w) >> 10) & 0x3FU)
  170. /*
  171. * The extended PCI Express capabilies lies at the end of the PCI configuration space
  172. */
  173. #define PCIE_CONF_EXT_CAPPTR 64U
  174. /*
  175. * The first word of every capability contains an extended capability identifier,
  176. * and a link to the next capability (or 0) in the extended configuration space.
  177. */
  178. #define PCIE_CONF_EXT_CAP_ID(w) ((w) & 0xFFFFU)
  179. #define PCIE_CONF_EXT_CAP_VER(w) (((w) >> 16) & 0xFU)
  180. #define PCIE_CONF_EXT_CAP_NEXT(w) (((w) >> 20) & 0xFFFU)
  181. /*
  182. * Configuration word 0 aligns directly with pcie_id_t.
  183. */
  184. #define PCIE_CONF_ID 0U
  185. /*
  186. * Configuration word 1 contains command and status bits.
  187. */
  188. #define PCIE_CONF_CMDSTAT 1U /* command/status register */
  189. #define PCIE_CONF_CMDSTAT_IO 0x00000001U /* I/O access enable */
  190. #define PCIE_CONF_CMDSTAT_MEM 0x00000002U /* mem access enable */
  191. #define PCIE_CONF_CMDSTAT_MASTER 0x00000004U /* bus master enable */
  192. #define PCIE_CONF_CMDSTAT_CAPS 0x00100000U /* capabilities list */
  193. /*
  194. * Configuration word 2 has additional function identification that
  195. * we only care about for debug output (PCIe shell commands).
  196. */
  197. #define PCIE_CONF_CLASSREV 2U /* class/revision register */
  198. #define PCIE_CONF_CLASSREV_CLASS(w) (((w) >> 24) & 0xFFU)
  199. #define PCIE_CONF_CLASSREV_SUBCLASS(w) (((w) >> 16) & 0xFFU)
  200. #define PCIE_CONF_CLASSREV_PROGIF(w) (((w) >> 8) & 0xFFU)
  201. #define PCIE_CONF_CLASSREV_REV(w) ((w) & 0xFFU)
  202. /*
  203. * The only part of configuration word 3 that is of interest to us is
  204. * the header type, as we use it to distinguish functional endpoints
  205. * from bridges (which are, for our purposes, transparent).
  206. */
  207. #define PCIE_CONF_TYPE 3U
  208. #define PCIE_CONF_TYPE_BRIDGE(w) (((w) & 0x007F0000U) != 0U)
  209. /*
  210. * Words 4-9 are BARs are I/O or memory decoders. Memory decoders may
  211. * be 64-bit decoders, in which case the next configuration word holds
  212. * the high-order bits (and is, thus, not a BAR itself).
  213. */
  214. #define PCIE_CONF_BAR0 4U
  215. #define PCIE_CONF_BAR1 5U
  216. #define PCIE_CONF_BAR2 6U
  217. #define PCIE_CONF_BAR3 7U
  218. #define PCIE_CONF_BAR4 8U
  219. #define PCIE_CONF_BAR5 9U
  220. #define PCIE_CONF_BAR_IO(w) (((w) & 0x00000001U) == 0x00000001U)
  221. #define PCIE_CONF_BAR_MEM(w) (((w) & 0x00000001U) != 0x00000001U)
  222. #define PCIE_CONF_BAR_64(w) (((w) & 0x00000006U) == 0x00000004U)
  223. #define PCIE_CONF_BAR_ADDR(w) ((w) & ~0xfUL)
  224. #define PCIE_CONF_BAR_FLAGS(w) ((w) & 0xfUL)
  225. #define PCIE_CONF_BAR_NONE 0U
  226. #define PCIE_CONF_BAR_INVAL 0xFFFFFFF0U
  227. #define PCIE_CONF_BAR_INVAL64 0xFFFFFFFFFFFFFFF0UL
  228. #define PCIE_CONF_BAR_INVAL_FLAGS(w) \
  229. ((((w) & 0x00000006U) == 0x00000006U) || \
  230. (((w) & 0x00000006U) == 0x00000002U))
  231. /*
  232. * Word 15 contains information related to interrupts.
  233. *
  234. * We're only interested in the low byte, which is [supposed to be] set by
  235. * the firmware to indicate which wire IRQ the device interrupt is routed to.
  236. */
  237. #define PCIE_CONF_INTR 15U
  238. #define PCIE_CONF_INTR_IRQ(w) ((w) & 0xFFU)
  239. #define PCIE_CONF_INTR_IRQ_NONE 0xFFU /* no interrupt routed */
  240. #define PCIE_MAX_BUS (0xFFFFFFFF & PCIE_BDF_BUS_MASK)
  241. #define PCIE_MAX_DEV (0xFFFFFFFF & PCIE_BDF_DEV_MASK)
  242. #define PCIE_MAX_FUNC (0xFFFFFFFF & PCIE_BDF_FUNC_MASK)
  243. #ifdef __cplusplus
  244. }
  245. #endif
  246. #endif /* ZEPHYR_INCLUDE_DRIVERS_PCIE_PCIE_H_ */