2
0

fpgasvc.c 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219
  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(uint8_t cmd, uint32_t addr, void *data, size_t len)
  119. {
  120. spi_transaction_ext_t trans;
  121. esp_err_t err;
  122. if ((len|addr) & 3) {
  123. printf("[FPGA] ERROR: I/O must be aligned dwords\n");
  124. return ESP_FAIL;
  125. }
  126. memset(&trans, 0, sizeof trans);
  127. trans.base.flags =
  128. SPI_TRANS_MODE_DIO |
  129. SPI_TRANS_VARIABLE_DUMMY |
  130. SPI_TRANS_MULTILINE_CMD |
  131. SPI_TRANS_MULTILINE_ADDR;
  132. trans.command_bits = 8;
  133. trans.address_bits = 32;
  134. trans.base.cmd = cmd;
  135. trans.base.addr = addr;
  136. if (cmd & FPGA_CMD_RD) {
  137. trans.dummy_bits = 16; /* 16 cycles = 32 bits */
  138. trans.base.rx_buffer = data;
  139. trans.base.rxlength = len << 3;
  140. } else {
  141. trans.base.tx_buffer = data;
  142. trans.base.length = len << 3;
  143. }
  144. xSemaphoreTake(spi_mutex, portMAX_DELAY);
  145. err = spi_device_transmit(spi_handle, (spi_transaction_t *)&trans);
  146. xSemaphoreGive(spi_mutex);
  147. return err;
  148. }
  149. static void fpga_service_task(void *dummy)
  150. {
  151. (void)dummy;
  152. struct dram_io_head head;
  153. /* If the FPGA is already up, need to issue our own active handshake */
  154. fpga_io(FPGA_CMD_WR|FPGA_CMD_IRQ(RV_IRQ_HELLO), 0, NULL, 0);
  155. while (1) {
  156. while (!digitalRead(PIN_FPGA_INT)) {
  157. printf("[FPGA] FPGA signals ready\n");
  158. fpga_io(FPGA_CMD_RD|FPGA_CMD_ACK(ESP_IRQ_READY), FPGA_HDR_ADDR,
  159. &head, sizeof head);
  160. if (head.magic == DRAM_IO_MAGIC && head.hlen >= sizeof head) {
  161. printf("[FPGA] Ready, board = %u.%u\n",
  162. (uint8_t)(head.board >> 24), (uint8_t)(head.board >> 16));
  163. }
  164. char signature_string[head.signature_len + 3];
  165. fpga_io(FPGA_CMD_RD, (size_t)head.signature,
  166. signature_string, (head.signature_len + 3) & ~3);
  167. signature_string[head.signature_len] = '\0';
  168. printf("[FPGA] \"%s\"\n", signature_string);
  169. }
  170. /* Wait until an interrupt is received */
  171. xTaskNotifyWaitIndexed(NOTIFY_INDEX, 0, 1, NULL, portMAX_DELAY);
  172. }
  173. }