msi.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320
  1. #include "msi.h"
  2. #include "pci.h"
  3. #include <common/errno.h>
  4. #include <mm/mmio.h>
  5. /**
  6. * @brief 生成msi消息
  7. *
  8. * @param msi_desc msi描述符
  9. * @return struct msi_msg_t* msi消息指针(在描述符内)
  10. */
  11. extern struct msi_msg_t *msi_arch_get_msg(struct msi_desc_t *msi_desc);
  12. /**
  13. * @brief 读取msix的capability list
  14. *
  15. * @param msi_desc msi描述符
  16. * @param cap_off capability list的offset
  17. * @return struct pci_msix_cap_t 对应的capability list
  18. */
  19. static __always_inline struct pci_msix_cap_t __msi_read_msix_cap_list(struct msi_desc_t *msi_desc, uint32_t cap_off)
  20. {
  21. struct pci_msix_cap_t cap_list = {0};
  22. uint32_t dw0;
  23. dw0 = pci_read_config(msi_desc->pci_dev->bus, msi_desc->pci_dev->device, msi_desc->pci_dev->func, cap_off);
  24. cap_list.cap_id = dw0 & 0xff;
  25. cap_list.next_off = (dw0 >> 8) & 0xff;
  26. cap_list.msg_ctrl = (dw0 >> 16) & 0xffff;
  27. cap_list.dword1 = pci_read_config(msi_desc->pci_dev->bus, msi_desc->pci_dev->device, msi_desc->pci_dev->func, cap_off + 0x4);
  28. cap_list.dword2 = pci_read_config(msi_desc->pci_dev->bus, msi_desc->pci_dev->device, msi_desc->pci_dev->func, cap_off + 0x8);
  29. return cap_list;
  30. }
  31. static __always_inline struct pci_msi_cap_t __msi_read_cap_list(struct msi_desc_t *msi_desc, uint32_t cap_off)
  32. {
  33. struct pci_msi_cap_t cap_list;
  34. uint32_t dw0;
  35. dw0 = pci_read_config(msi_desc->pci_dev->bus, msi_desc->pci_dev->device, msi_desc->pci_dev->func, cap_off);
  36. cap_list.cap_id = dw0 & 0xff;
  37. cap_list.next_off = (dw0 >> 8) & 0xff;
  38. cap_list.msg_ctrl = (dw0 >> 16) & 0xffff;
  39. cap_list.msg_addr_lo = pci_read_config(msi_desc->pci_dev->bus, msi_desc->pci_dev->device, msi_desc->pci_dev->func, cap_off + 0x4);
  40. uint16_t msg_data_off = 0xc;
  41. if (cap_list.msg_ctrl & (1 << 7)) // 64位
  42. {
  43. cap_list.msg_addr_hi = pci_read_config(msi_desc->pci_dev->bus, msi_desc->pci_dev->device, msi_desc->pci_dev->func, cap_off + 0x8);
  44. }
  45. else
  46. {
  47. cap_list.msg_addr_hi = 0;
  48. msg_data_off = 0x8;
  49. }
  50. cap_list.msg_data = pci_read_config(msi_desc->pci_dev->bus, msi_desc->pci_dev->device, msi_desc->pci_dev->func, cap_off + msg_data_off) & 0xffff;
  51. cap_list.mask = pci_read_config(msi_desc->pci_dev->bus, msi_desc->pci_dev->device, msi_desc->pci_dev->func, cap_off + 0x10);
  52. cap_list.pending = pci_read_config(msi_desc->pci_dev->bus, msi_desc->pci_dev->device, msi_desc->pci_dev->func, cap_off + 0x14);
  53. return cap_list;
  54. }
  55. /**
  56. * @brief 映射设备的msix表
  57. *
  58. * @param pci_dev pci设备信息结构体
  59. * @param msix_cap msix capability list的结构体
  60. * @return int 错误码
  61. */
  62. static __always_inline int __msix_map_table(struct pci_device_structure_header_t *pci_dev, struct pci_msix_cap_t *msix_cap)
  63. {
  64. // 计算bar寄存器的offset
  65. uint32_t bar_off = 0x10 + 4 * (msix_cap->dword1 & 0x7);
  66. // msix table相对于bar寄存器中存储的地址的offset
  67. pci_dev->msix_offset = msix_cap->dword1 & (~0x7);
  68. pci_dev->msix_table_size = (msix_cap->msg_ctrl & 0x7ff) + 1;
  69. pci_dev->msix_mmio_size = pci_dev->msix_table_size * 16 + pci_dev->msix_offset;
  70. // 申请mmio空间
  71. mmio_create(pci_dev->msix_mmio_size, VM_IO | VM_DONTCOPY, &pci_dev->msix_mmio_vaddr, &pci_dev->msix_mmio_size);
  72. pci_dev->msix_mmio_vaddr &= (~0xf);
  73. uint32_t bar = pci_read_config(pci_dev->bus, pci_dev->device, pci_dev->func, bar_off);
  74. kdebug("pci_dev->msix_mmio_vaddr=%#018lx, bar=%#010lx, table offset=%#010lx, table_size=%#010lx, mmio_size=%d", pci_dev->msix_mmio_vaddr, bar, pci_dev->msix_offset, pci_dev->msix_table_size, pci_dev->msix_mmio_size);
  75. // 将msix table映射到页表
  76. mm_map(&initial_mm, pci_dev->msix_mmio_vaddr, pci_dev->msix_mmio_size, bar);
  77. return 0;
  78. }
  79. /**
  80. * @brief 将msi_desc中的数据填写到msix表的指定表项处
  81. *
  82. * @param pci_dev pci设备结构体
  83. * @param msi_desc msi描述符
  84. */
  85. static __always_inline void __msix_set_entry(struct msi_desc_t *msi_desc)
  86. {
  87. uint64_t *ptr = (uint64_t *)(msi_desc->pci_dev->msix_mmio_vaddr + msi_desc->pci_dev->msix_offset + msi_desc->msi_index * 16);
  88. *ptr = (msi_desc->msg.address_hi << 32) | (msi_desc->msg.address_lo);
  89. ++ptr;
  90. *ptr = (msi_desc->msg.vector_control << 32) | (msi_desc->msg.data);
  91. }
  92. /**
  93. * @brief 清空设备的msix table的指定表项
  94. *
  95. * @param pci_dev pci设备
  96. * @param msi_index 表项号
  97. */
  98. static __always_inline void __msix_clear_entry(struct pci_device_structure_header_t *pci_dev, uint16_t msi_index)
  99. {
  100. uint64_t *ptr = (uint64_t *)(pci_dev->msix_mmio_vaddr + pci_dev->msix_offset + msi_index * 16);
  101. *ptr = 0;
  102. ++ptr;
  103. *ptr = 0;
  104. }
  105. /**
  106. * @brief 启用 Message Signaled Interrupts
  107. *
  108. * @param header 设备header
  109. * @param vector 中断向量号
  110. * @param processor 要投递到的处理器
  111. * @param edge_trigger 是否边缘触发
  112. * @param assert 是否高电平触发
  113. *
  114. * @return 返回码
  115. */
  116. int pci_enable_msi(struct msi_desc_t *msi_desc)
  117. {
  118. struct pci_device_structure_header_t *ptr = msi_desc->pci_dev;
  119. uint32_t cap_ptr;
  120. uint32_t tmp;
  121. uint16_t message_control;
  122. uint64_t message_addr;
  123. // 先尝试获取msi-x,若不存在,则获取msi capability
  124. if (msi_desc->pci.msi_attribute.is_msix)
  125. {
  126. cap_ptr = pci_enumerate_capability_list(ptr, 0x11);
  127. if (((int32_t)cap_ptr) < 0)
  128. {
  129. cap_ptr = pci_enumerate_capability_list(ptr, 0x05);
  130. if (((int32_t)cap_ptr) < 0)
  131. return -ENOSYS;
  132. msi_desc->pci.msi_attribute.is_msix = 0;
  133. }
  134. }
  135. else
  136. {
  137. cap_ptr = pci_enumerate_capability_list(ptr, 0x05);
  138. if (((int32_t)cap_ptr) < 0)
  139. return -ENOSYS;
  140. msi_desc->pci.msi_attribute.is_msix = 0;
  141. }
  142. // 获取msi消息
  143. msi_arch_get_msg(msi_desc);
  144. if (msi_desc->pci.msi_attribute.is_msix) // MSI-X
  145. {
  146. // 读取msix的信息
  147. struct pci_msix_cap_t cap = __msi_read_msix_cap_list(msi_desc, cap_ptr);
  148. // 映射msix table
  149. __msix_map_table(msi_desc->pci_dev, &cap);
  150. // 设置msix的中断
  151. __msix_set_entry(msi_desc);
  152. // 使能msi
  153. tmp = pci_read_config(ptr->bus, ptr->device, ptr->func, cap_ptr); // 读取cap+0x0处的值
  154. tmp |= (1 << 16);
  155. pci_write_config(ptr->bus, ptr->device, ptr->func, cap_ptr, tmp);
  156. }
  157. else
  158. {
  159. tmp = pci_read_config(ptr->bus, ptr->device, ptr->func, cap_ptr); // 读取cap+0x0处的值
  160. message_control = (tmp >> 16) & 0xffff;
  161. // 写入message address
  162. message_addr = ((((uint64_t)msi_desc->msg.address_hi) << 32) | msi_desc->msg.address_lo); // 获取message address
  163. pci_write_config(ptr->bus, ptr->device, ptr->func, cap_ptr + 0x4, (uint32_t)(message_addr & 0xffffffff));
  164. if (message_control & (1 << 7)) // 64位
  165. pci_write_config(ptr->bus, ptr->device, ptr->func, cap_ptr + 0x8, (uint32_t)((message_addr >> 32) & 0xffffffff));
  166. // 写入message data
  167. tmp = msi_desc->msg.data;
  168. if (message_control & (1 << 7)) // 64位
  169. pci_write_config(ptr->bus, ptr->device, ptr->func, cap_ptr + 0xc, tmp);
  170. else
  171. pci_write_config(ptr->bus, ptr->device, ptr->func, cap_ptr + 0x8, tmp);
  172. // 使能msi
  173. tmp = pci_read_config(ptr->bus, ptr->device, ptr->func, cap_ptr); // 读取cap+0x0处的值
  174. tmp |= (1 << 16);
  175. pci_write_config(ptr->bus, ptr->device, ptr->func, cap_ptr, tmp);
  176. }
  177. return 0;
  178. }
  179. /**
  180. * @brief 在已配置好msi寄存器的设备上,使能msi
  181. *
  182. * @param header 设备头部
  183. * @return int 返回码
  184. */
  185. int pci_start_msi(void *header)
  186. {
  187. struct pci_device_structure_header_t *ptr = (struct pci_device_structure_header_t *)header;
  188. uint32_t cap_ptr;
  189. uint32_t tmp;
  190. switch (ptr->HeaderType)
  191. {
  192. case 0x00: // general device
  193. if (!(ptr->Status & 0x10))
  194. return -ENOSYS;
  195. cap_ptr = ((struct pci_device_structure_general_device_t *)ptr)->Capabilities_Pointer;
  196. tmp = pci_read_config(ptr->bus, ptr->device, ptr->func, cap_ptr); // 读取cap+0x0处的值
  197. if (tmp & 0xff != 0x5)
  198. return -ENOSYS;
  199. // 使能msi
  200. tmp = pci_read_config(ptr->bus, ptr->device, ptr->func, cap_ptr); // 读取cap+0x0处的值
  201. tmp |= (1 << 16);
  202. pci_write_config(ptr->bus, ptr->device, ptr->func, cap_ptr, tmp);
  203. break;
  204. case 0x01: // pci to pci bridge
  205. if (!(ptr->Status & 0x10))
  206. return -ENOSYS;
  207. cap_ptr = ((struct pci_device_structure_pci_to_pci_bridge_t *)ptr)->Capability_Pointer;
  208. tmp = pci_read_config(ptr->bus, ptr->device, ptr->func, cap_ptr); // 读取cap+0x0处的值
  209. if (tmp & 0xff != 0x5)
  210. return -ENOSYS;
  211. // 使能msi
  212. tmp = pci_read_config(ptr->bus, ptr->device, ptr->func, cap_ptr); // 读取cap+0x0处的值
  213. tmp |= (1 << 16);
  214. pci_write_config(ptr->bus, ptr->device, ptr->func, cap_ptr, tmp);
  215. break;
  216. case 0x02: // pci to card bus bridge
  217. return -ENOSYS;
  218. break;
  219. default: // 不应该到达这里
  220. return -EINVAL;
  221. break;
  222. }
  223. return 0;
  224. }
  225. /**
  226. * @brief 禁用指定设备的msi
  227. *
  228. * @param header pci header
  229. * @return int
  230. */
  231. int pci_disable_msi(void *header)
  232. {
  233. struct pci_device_structure_header_t *ptr = (struct pci_device_structure_header_t *)header;
  234. uint32_t cap_ptr;
  235. uint32_t tmp;
  236. switch (ptr->HeaderType)
  237. {
  238. case 0x00: // general device
  239. if (!(ptr->Status & 0x10))
  240. return -ENOSYS;
  241. cap_ptr = ((struct pci_device_structure_general_device_t *)ptr)->Capabilities_Pointer;
  242. tmp = pci_read_config(ptr->bus, ptr->device, ptr->func, cap_ptr); // 读取cap+0x0处的值
  243. if (tmp & 0xff != 0x5)
  244. return -ENOSYS;
  245. // 禁用msi
  246. tmp = pci_read_config(ptr->bus, ptr->device, ptr->func, cap_ptr); // 读取cap+0x0处的值
  247. tmp &= (~(1 << 16));
  248. pci_write_config(ptr->bus, ptr->device, ptr->func, cap_ptr, tmp);
  249. break;
  250. case 0x01: // pci to pci bridge
  251. if (!(ptr->Status & 0x10))
  252. return -ENOSYS;
  253. cap_ptr = ((struct pci_device_structure_pci_to_pci_bridge_t *)ptr)->Capability_Pointer;
  254. tmp = pci_read_config(ptr->bus, ptr->device, ptr->func, cap_ptr); // 读取cap+0x0处的值
  255. if (tmp & 0xff != 0x5)
  256. return -ENOSYS;
  257. // 禁用msi
  258. tmp = pci_read_config(ptr->bus, ptr->device, ptr->func, cap_ptr); // 读取cap+0x0处的值
  259. tmp &= (~(1 << 16));
  260. pci_write_config(ptr->bus, ptr->device, ptr->func, cap_ptr, tmp);
  261. break;
  262. case 0x02: // pci to card bus bridge
  263. return -ENOSYS;
  264. break;
  265. default: // 不应该到达这里
  266. return -EINVAL;
  267. break;
  268. }
  269. return 0;
  270. }