fpgasvc.c 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264
  1. #include "common.h"
  2. #include "config.h"
  3. #include "fpga.h"
  4. #include "esplink.h"
  5. #include <driver/gpio.h>
  6. #include <driver/spi_common.h>
  7. #include <driver/spi_master.h>
  8. #define PIN_FPGA_INT 9
  9. #define PIN_FPGA_CS 10
  10. #define PIN_FPGA_IO0 11
  11. #define PIN_FPGA_CLK 12
  12. #define PIN_FPGA_IO1 13
  13. #define FPGA_SPI_HOST FSPI /* SPI2 */
  14. #define FPGA_PRIORITY 3
  15. static spi_bus_config_t spi_bus_config = {
  16. .data0_io_num = PIN_FPGA_IO0,
  17. .data1_io_num = PIN_FPGA_IO1,
  18. .sclk_io_num = PIN_FPGA_CLK,
  19. .data2_io_num = -1,
  20. .data3_io_num = -1,
  21. .data4_io_num = -1,
  22. .data5_io_num = -1,
  23. .data6_io_num = -1,
  24. .data7_io_num = -1,
  25. .max_transfer_sz = 4096,
  26. .flags = SPICOMMON_BUSFLAG_MASTER | SPICOMMON_BUSFLAG_DUAL
  27. };
  28. static const spi_device_interface_config_t spi_device_interface_config = {
  29. .command_bits = 8,
  30. .address_bits = 32,
  31. .dummy_bits = 0,
  32. .mode = 0,
  33. .cs_ena_pretrans = 0,
  34. .cs_ena_posttrans = 0,
  35. .clock_speed_hz = SPI_MASTER_FREQ_40M,
  36. .spics_io_num = PIN_FPGA_CS,
  37. .flags = SPI_DEVICE_HALFDUPLEX,
  38. .queue_size = 4 /* Maybe? */
  39. };
  40. static spi_device_handle_t spi_handle;
  41. static TaskHandle_t fpga_task;
  42. static SemaphoreHandle_t spi_mutex;
  43. #define NOTIFY_INDEX 0
  44. static void ARDUINO_ISR_ATTR fpga_interrupt(void)
  45. {
  46. BaseType_t do_wakeup = pdFALSE;
  47. if (!fpga_task)
  48. return;
  49. xTaskNotifyIndexedFromISR(fpga_task, NOTIFY_INDEX, 1, eSetBits, &do_wakeup);
  50. if (do_wakeup)
  51. portYIELD_FROM_ISR(do_wakeup);
  52. }
  53. static void fpga_service_task(void *);
  54. int fpga_service_start(void)
  55. {
  56. esp_err_t err;
  57. pinMode(PIN_FPGA_INT, INPUT_PULLUP);
  58. if (!spi_mutex) {
  59. spi_mutex = xSemaphoreCreateMutex();
  60. if (!spi_mutex)
  61. goto failed;
  62. }
  63. xSemaphoreTake(spi_mutex, portMAX_DELAY);
  64. err = spi_bus_initialize(FPGA_SPI_HOST, &spi_bus_config, SPI_DMA_CH_AUTO);
  65. if (err)
  66. goto failed;
  67. err = spi_bus_add_device(FPGA_SPI_HOST, &spi_device_interface_config,
  68. &spi_handle);
  69. if (err)
  70. goto failed;
  71. /* Only device on this bus, so acquire it permanently */
  72. spi_device_acquire_bus(spi_handle, portMAX_DELAY);
  73. xSemaphoreGive(spi_mutex);
  74. if (!fpga_task) {
  75. if (xTaskCreate(fpga_service_task, "fpga_svc", 4096, NULL,
  76. FPGA_PRIORITY, &fpga_task) != pdPASS)
  77. goto failed;
  78. attachInterrupt(PIN_FPGA_INT, fpga_interrupt, FALLING);
  79. esp_register_shutdown_handler(fpga_service_stop);
  80. }
  81. /* All good! */
  82. printf("[FPGA] FPGA services started\n");
  83. return 0;
  84. failed:
  85. printf("[FPGA] Failed to initialize FPGA SPI bus\n");
  86. if (spi_mutex)
  87. xSemaphoreGive(spi_mutex);
  88. fpga_service_stop();
  89. return -1;
  90. }
  91. void fpga_service_stop(void)
  92. {
  93. if (spi_mutex) {
  94. xSemaphoreTake(spi_mutex, portMAX_DELAY);
  95. if (fpga_task) {
  96. esp_unregister_shutdown_handler(fpga_service_stop);
  97. detachInterrupt(PIN_FPGA_INT);
  98. vTaskDelete(fpga_task);
  99. fpga_task = NULL;
  100. }
  101. if (spi_handle) {
  102. spi_device_release_bus(spi_handle);
  103. spi_bus_remove_device(spi_handle);
  104. spi_bus_free(FPGA_SPI_HOST);
  105. spi_handle = NULL;
  106. }
  107. xSemaphoreGive(spi_mutex);
  108. }
  109. printf("[FPGA] FPGA services stopped\n");
  110. }
  111. #define FPGA_CMD_IRQ(x) ((x) << 0)
  112. #define FPGA_CMD_ACK(x) ((x) << 2)
  113. #define FPGA_CMD_ACK2 (2 << 2)
  114. #define FPGA_CMD_ACK3 (3 << 2)
  115. #define FPGA_CMD_WR (0 << 4)
  116. #define FPGA_CMD_RD (1 << 4)
  117. #define FPGA_HDR_ADDR 0x40000000
  118. static esp_err_t fpga_io_write(uint8_t cmd, uint32_t addr,
  119. const void *data, size_t len)
  120. {
  121. spi_transaction_ext_t trans;
  122. esp_err_t err;
  123. if (!len)
  124. return ESP_OK;
  125. memset(&trans, 0, sizeof trans);
  126. trans.base.flags =
  127. SPI_TRANS_MODE_DIO |
  128. SPI_TRANS_MULTILINE_CMD |
  129. SPI_TRANS_MULTILINE_ADDR;
  130. trans.base.cmd = cmd | ~FPGA_CMD_RD;
  131. trans.base.addr = addr;
  132. trans.base.tx_buffer = data;
  133. trans.base.length = len << 3;
  134. xSemaphoreTake(spi_mutex, portMAX_DELAY);
  135. err = spi_device_transmit(spi_handle, (spi_transaction_t *)&trans);
  136. xSemaphoreGive(spi_mutex);
  137. return err;
  138. }
  139. static esp_err_t fpga_io_read(uint8_t cmd, uint32_t addr,
  140. void *data, size_t len)
  141. {
  142. spi_transaction_ext_t trans;
  143. esp_err_t err;
  144. if (!len)
  145. return ESP_OK;
  146. memset(&trans, 0, sizeof trans);
  147. trans.base.flags =
  148. SPI_TRANS_MODE_DIO |
  149. SPI_TRANS_VARIABLE_DUMMY |
  150. SPI_TRANS_MULTILINE_CMD |
  151. SPI_TRANS_MULTILINE_ADDR;
  152. trans.base.cmd = cmd | FPGA_CMD_RD;
  153. trans.base.addr = addr;
  154. trans.base.rx_buffer = data;
  155. /* Emulate partial word read by adding dummy bits for offset */
  156. trans.dummy_bits = 16 + ((addr & 3) << 2);
  157. trans.base.rxlength = len << 3;
  158. xSemaphoreTake(spi_mutex, portMAX_DELAY);
  159. err = spi_device_transmit(spi_handle, (spi_transaction_t *)&trans);
  160. xSemaphoreGive(spi_mutex);
  161. return err;
  162. }
  163. static uint32_t fpga_io_status(void)
  164. {
  165. spi_transaction_ext_t trans;
  166. esp_err_t err;
  167. memset(&trans, 0, sizeof trans);
  168. trans.base.flags =
  169. SPI_TRANS_MODE_DIO |
  170. SPI_TRANS_MULTILINE_CMD |
  171. SPI_TRANS_MULTILINE_ADDR |
  172. SPI_TRANS_USE_RXDATA;
  173. trans.base.cmd = FPGA_CMD_RD;
  174. trans.base.rxlength = 32;
  175. xSemaphoreTake(spi_mutex, portMAX_DELAY);
  176. err = spi_device_transmit(spi_handle, (spi_transaction_t *)&trans);
  177. xSemaphoreGive(spi_mutex);
  178. return err ? 0 : ntohl(*(const uint32_t *)&trans.base.rx_data);
  179. }
  180. static void fpga_service_task(void *dummy)
  181. {
  182. (void)dummy;
  183. struct dram_io_head head;
  184. /* If the FPGA is already up, need to issue our own active handshake */
  185. fpga_io_write(FPGA_CMD_IRQ(RV_IRQ_HELLO), 0, NULL, 0);
  186. while (1) {
  187. while (!digitalRead(PIN_FPGA_INT)) {
  188. printf("[FPGA] FPGA signals ready\n");
  189. uint32_t status = fpga_io_status();
  190. printf("[FPGA] Link status bits = 0x%08x\n", status);
  191. fpga_io_read(FPGA_CMD_ACK(ESP_IRQ_READY), FPGA_HDR_ADDR,
  192. &head, sizeof head);
  193. if (head.magic == DRAM_IO_MAGIC && head.hlen >= sizeof head) {
  194. printf("[FPGA] Ready, board = %u.%u fixes %02x fpga %u\n",
  195. head.board.major, head.board.minor,
  196. head.board.fixes, head.board.fpga);
  197. }
  198. char signature_string[head.signature_len+1];
  199. fpga_io_read(0, (size_t)head.signature,
  200. signature_string, head.signature_len);
  201. signature_string[head.signature_len] = '\0';
  202. printf("[FPGA] \"%s\"\n", signature_string);
  203. }
  204. /* Wait until an interrupt is received */
  205. xTaskNotifyWaitIndexed(NOTIFY_INDEX, 0, 1, NULL, portMAX_DELAY);
  206. }
  207. }