msi.h 4.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166
  1. /*
  2. * Copyright (c) 2019 Intel Corporation
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. #ifndef ZEPHYR_INCLUDE_DRIVERS_PCIE_MSI_H_
  7. #define ZEPHYR_INCLUDE_DRIVERS_PCIE_MSI_H_
  8. #include <kernel.h>
  9. #include <zephyr/types.h>
  10. #include <stdbool.h>
  11. #include <drivers/pcie/pcie.h>
  12. #ifdef __cplusplus
  13. extern "C" {
  14. #endif
  15. struct msix_vector {
  16. uint32_t msg_addr;
  17. uint32_t msg_up_addr;
  18. uint32_t msg_data;
  19. uint32_t vector_ctrl;
  20. };
  21. struct msi_vector {
  22. pcie_bdf_t bdf;
  23. arch_msi_vector_t arch;
  24. #ifdef CONFIG_PCIE_MSI_X
  25. struct msix_vector *msix_vector;
  26. bool msix;
  27. #endif /* CONFIG_PCIE_MSI_X */
  28. };
  29. typedef struct msi_vector msi_vector_t;
  30. #ifdef CONFIG_PCIE_MSI_MULTI_VECTOR
  31. /**
  32. * @brief Allocate vector(s) for the endpoint MSI message(s)
  33. *
  34. * @param bdf the target PCI endpoint
  35. * @param priority the MSI vectors base interrupt priority
  36. * @param vectors an array for storing allocated MSI vectors
  37. * @param n_vector the size of the MSI vectors array
  38. *
  39. * @return the number of allocated MSI vectors.
  40. */
  41. extern uint8_t pcie_msi_vectors_allocate(pcie_bdf_t bdf,
  42. unsigned int priority,
  43. msi_vector_t *vectors,
  44. uint8_t n_vector);
  45. /**
  46. * @brief Connect the MSI vector to the handler
  47. *
  48. * @param bdf the target PCI endpoint
  49. * @param vector the MSI vector to connect
  50. * @param routine Interrupt service routine
  51. * @param parameter ISR parameter
  52. * @param flags Arch-specific IRQ configuration flag
  53. *
  54. * @return True on success, false otherwise
  55. */
  56. extern bool pcie_msi_vector_connect(pcie_bdf_t bdf,
  57. msi_vector_t *vector,
  58. void (*routine)(const void *parameter),
  59. const void *parameter,
  60. uint32_t flags);
  61. #endif /* CONFIG_PCIE_MSI_MULTI_VECTOR */
  62. /**
  63. * @brief Compute the target address for an MSI posted write.
  64. *
  65. * This function is exported by the arch, board or SoC code.
  66. *
  67. * @param irq The IRQ we wish to trigger via MSI.
  68. * @param vector The vector for which you want the address (or NULL)
  69. * @return A (32-bit) value for the MSI MAP register.
  70. */
  71. extern uint32_t pcie_msi_map(unsigned int irq,
  72. msi_vector_t *vector);
  73. /**
  74. * @brief Compute the data for an MSI posted write.
  75. *
  76. * This function is exported by the arch, board or SoC code.
  77. *
  78. * @param irq The IRQ we wish to trigger via MSI.
  79. * @param vector The vector for which you want the data (or NULL)
  80. * @return A (16-bit) value for MSI MDR register.
  81. */
  82. extern uint16_t pcie_msi_mdr(unsigned int irq,
  83. msi_vector_t *vector);
  84. /**
  85. * @brief Configure the given PCI endpoint to generate MSIs.
  86. *
  87. * @param bdf the target PCI endpoint
  88. * @param vectors an array of allocated vector(s)
  89. * @param n_vector the size of the vector array
  90. * @param irq The IRQ we wish to trigger via MSI.
  91. * @return true if the endpoint supports MSI, false otherwise.
  92. */
  93. extern bool pcie_msi_enable(pcie_bdf_t bdf,
  94. msi_vector_t *vectors,
  95. uint8_t n_vector,
  96. unsigned int irq);
  97. /*
  98. * The first word of the MSI capability is shared with the
  99. * capability ID and list link. The high 16 bits are the MCR.
  100. */
  101. #define PCIE_MSI_MCR 0U
  102. #define PCIE_MSI_MCR_EN 0x00010000U /* enable MSI */
  103. #define PCIE_MSI_MCR_MMC 0x000E0000U /* Multi Messages Capable mask */
  104. #define PCIE_MSI_MCR_MMC_SHIFT 17
  105. #define PCIE_MSI_MCR_MME 0x00700000U /* mask of # of enabled IRQs */
  106. #define PCIE_MSI_MCR_MME_SHIFT 20
  107. #define PCIE_MSI_MCR_64 0x00800000U /* 64-bit MSI */
  108. /*
  109. * The MAP follows the MCR. If PCIE_MSI_MCR_64, then the MAP
  110. * is two words long. The MDR follows immediately after the MAP.
  111. */
  112. #define PCIE_MSI_MAP0 1U
  113. #define PCIE_MSI_MAP1_64 2U
  114. #define PCIE_MSI_MDR_32 2U
  115. #define PCIE_MSI_MDR_64 3U
  116. /*
  117. * As for MSI, he first word of the MSI-X capability is shared
  118. * with the capability ID and list link. The high 16 bits are the MCR.
  119. */
  120. #define PCIE_MSIX_MCR 0U
  121. #define PCIE_MSIX_MCR_EN 0x80000000U /* Enable MSI-X */
  122. #define PCIE_MSIX_MCR_FMASK 0x40000000U /* Function Mask */
  123. #define PCIE_MSIX_MCR_TSIZE 0x07FF0000U /* Table size mask */
  124. #define PCIE_MSIX_MCR_TSIZE_SHIFT 16
  125. #define PCIE_MSIR_TABLE_ENTRY_SIZE 16
  126. #define PCIE_MSIX_TR 1U
  127. #define PCIE_MSIX_TR_BIR 0x00000007U /* Table BIR mask */
  128. #define PCIE_MSIX_TR_OFFSET 0xFFFFFFF8U /* Offset mask */
  129. #define PCIE_MSIX_PBA 2U
  130. #define PCIE_MSIX_PBA_BIR 0x00000007U /* PBA BIR mask */
  131. #define PCIE_MSIX_PBA_OFFSET 0xFFFFFFF8U /* Offset mask */
  132. #define PCIE_VTBL_MA 0U /* Msg Address offset */
  133. #define PCIE_VTBL_MUA 4U /* Msg Upper Address offset */
  134. #define PCIE_VTBL_MD 8U /* Msg Data offset */
  135. #define PCIE_VTBL_VCTRL 12U /* Vector control offset */
  136. #ifdef __cplusplus
  137. }
  138. #endif
  139. #endif /* ZEPHYR_INCLUDE_DRIVERS_PCIE_MSI_H_ */