cmd_ioloop.c 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295
  1. /*
  2. * (C) Copyright 2014
  3. * Dirk Eibach, Guntermann & Drunck GmbH, dirk.eibach@gdsys.cc
  4. *
  5. * SPDX-License-Identifier: GPL-2.0+
  6. */
  7. #include <common.h>
  8. #include <command.h>
  9. #include <gdsys_fpga.h>
  10. enum {
  11. STATE_TX_PACKET_BUILDING = 1<<0,
  12. STATE_TX_TRANSMITTING = 1<<1,
  13. STATE_TX_BUFFER_FULL = 1<<2,
  14. STATE_TX_ERR = 1<<3,
  15. STATE_RECEIVE_TIMEOUT = 1<<4,
  16. STATE_PROC_RX_STORE_TIMEOUT = 1<<5,
  17. STATE_PROC_RX_RECEIVE_TIMEOUT = 1<<6,
  18. STATE_RX_DIST_ERR = 1<<7,
  19. STATE_RX_LENGTH_ERR = 1<<8,
  20. STATE_RX_FRAME_CTR_ERR = 1<<9,
  21. STATE_RX_FCS_ERR = 1<<10,
  22. STATE_RX_PACKET_DROPPED = 1<<11,
  23. STATE_RX_DATA_LAST = 1<<12,
  24. STATE_RX_DATA_FIRST = 1<<13,
  25. STATE_RX_DATA_AVAILABLE = 1<<15,
  26. };
  27. enum {
  28. CTRL_PROC_RECEIVE_ENABLE = 1<<12,
  29. CTRL_FLUSH_TRANSMIT_BUFFER = 1<<15,
  30. };
  31. enum {
  32. IRQ_CPU_TRANSMITBUFFER_FREE_STATUS = 1<<5,
  33. IRQ_CPU_PACKET_TRANSMITTED_EVENT = 1<<6,
  34. IRQ_NEW_CPU_PACKET_RECEIVED_EVENT = 1<<7,
  35. IRQ_CPU_RECEIVE_DATA_AVAILABLE_STATUS = 1<<8,
  36. };
  37. struct io_generic_packet {
  38. u16 target_address;
  39. u16 source_address;
  40. u8 packet_type;
  41. u8 bc;
  42. u16 packet_length;
  43. } __attribute__((__packed__));
  44. unsigned long long rx_ctr;
  45. unsigned long long tx_ctr;
  46. unsigned long long err_ctr;
  47. static void io_check_status(unsigned int fpga, u16 status, bool silent)
  48. {
  49. u16 mask = STATE_RX_DIST_ERR | STATE_RX_LENGTH_ERR |
  50. STATE_RX_FRAME_CTR_ERR | STATE_RX_FCS_ERR |
  51. STATE_RX_PACKET_DROPPED | STATE_TX_ERR;
  52. if (!(status & mask)) {
  53. FPGA_SET_REG(fpga, ep.rx_tx_status, status);
  54. return;
  55. }
  56. err_ctr++;
  57. FPGA_SET_REG(fpga, ep.rx_tx_status, status);
  58. if (silent)
  59. return;
  60. if (status & STATE_RX_PACKET_DROPPED)
  61. printf("RX_PACKET_DROPPED, status %04x\n", status);
  62. if (status & STATE_RX_DIST_ERR)
  63. printf("RX_DIST_ERR\n");
  64. if (status & STATE_RX_LENGTH_ERR)
  65. printf("RX_LENGTH_ERR\n");
  66. if (status & STATE_RX_FRAME_CTR_ERR)
  67. printf("RX_FRAME_CTR_ERR\n");
  68. if (status & STATE_RX_FCS_ERR)
  69. printf("RX_FCS_ERR\n");
  70. if (status & STATE_TX_ERR)
  71. printf("TX_ERR\n");
  72. }
  73. static void io_send(unsigned int fpga, unsigned int size)
  74. {
  75. unsigned int k;
  76. struct io_generic_packet packet = {
  77. .source_address = 1,
  78. .packet_type = 1,
  79. .packet_length = size,
  80. };
  81. u16 *p = (u16 *)&packet;
  82. for (k = 0; k < sizeof(packet) / 2; ++k)
  83. FPGA_SET_REG(fpga, ep.transmit_data, *p++);
  84. for (k = 0; k < (size + 1) / 2; ++k)
  85. FPGA_SET_REG(fpga, ep.transmit_data, k);
  86. FPGA_SET_REG(fpga, ep.rx_tx_control,
  87. CTRL_PROC_RECEIVE_ENABLE | CTRL_FLUSH_TRANSMIT_BUFFER);
  88. tx_ctr++;
  89. }
  90. static void io_receive(unsigned int fpga)
  91. {
  92. unsigned int k = 0;
  93. u16 rx_tx_status;
  94. FPGA_GET_REG(fpga, ep.rx_tx_status, &rx_tx_status);
  95. while (rx_tx_status & STATE_RX_DATA_AVAILABLE) {
  96. u16 rx;
  97. if (rx_tx_status & STATE_RX_DATA_LAST)
  98. rx_ctr++;
  99. FPGA_GET_REG(fpga, ep.receive_data, &rx);
  100. FPGA_GET_REG(fpga, ep.rx_tx_status, &rx_tx_status);
  101. ++k;
  102. }
  103. }
  104. static void io_reflect(unsigned int fpga)
  105. {
  106. u16 buffer[128];
  107. unsigned int k = 0;
  108. unsigned int n;
  109. u16 rx_tx_status;
  110. FPGA_GET_REG(fpga, ep.rx_tx_status, &rx_tx_status);
  111. while (rx_tx_status & STATE_RX_DATA_AVAILABLE) {
  112. FPGA_GET_REG(fpga, ep.receive_data, &buffer[k++]);
  113. if (rx_tx_status & STATE_RX_DATA_LAST)
  114. break;
  115. FPGA_GET_REG(fpga, ep.rx_tx_status, &rx_tx_status);
  116. }
  117. if (!k)
  118. return;
  119. for (n = 0; n < k; ++n)
  120. FPGA_SET_REG(fpga, ep.transmit_data, buffer[n]);
  121. FPGA_SET_REG(fpga, ep.rx_tx_control,
  122. CTRL_PROC_RECEIVE_ENABLE | CTRL_FLUSH_TRANSMIT_BUFFER);
  123. tx_ctr++;
  124. }
  125. /*
  126. * FPGA io-endpoint reflector
  127. *
  128. * Syntax:
  129. * ioreflect {fpga} {reportrate}
  130. */
  131. int do_ioreflect(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
  132. {
  133. unsigned int fpga;
  134. unsigned int rate = 0;
  135. unsigned long long last_seen = 0;
  136. if (argc < 2)
  137. return CMD_RET_USAGE;
  138. fpga = simple_strtoul(argv[1], NULL, 10);
  139. /*
  140. * If another parameter, it is the report rate in packets.
  141. */
  142. if (argc > 2)
  143. rate = simple_strtoul(argv[2], NULL, 10);
  144. /* enable receive path */
  145. FPGA_SET_REG(fpga, ep.rx_tx_control, CTRL_PROC_RECEIVE_ENABLE);
  146. /* set device address to dummy 1*/
  147. FPGA_SET_REG(fpga, ep.device_address, 1);
  148. rx_ctr = 0; tx_ctr = 0; err_ctr = 0;
  149. while (1) {
  150. u16 top_int;
  151. u16 rx_tx_status;
  152. FPGA_GET_REG(fpga, top_interrupt, &top_int);
  153. FPGA_GET_REG(fpga, ep.rx_tx_status, &rx_tx_status);
  154. io_check_status(fpga, rx_tx_status, true);
  155. if ((top_int & IRQ_CPU_RECEIVE_DATA_AVAILABLE_STATUS) &&
  156. (top_int & IRQ_CPU_TRANSMITBUFFER_FREE_STATUS))
  157. io_reflect(fpga);
  158. if (rate) {
  159. if (!(tx_ctr % rate) && (tx_ctr != last_seen))
  160. printf("refl %llu, err %llu\n", tx_ctr,
  161. err_ctr);
  162. last_seen = tx_ctr;
  163. }
  164. if (ctrlc())
  165. break;
  166. }
  167. return 0;
  168. }
  169. /*
  170. * FPGA io-endpoint looptest
  171. *
  172. * Syntax:
  173. * ioloop {fpga} {size} {rate}
  174. */
  175. #define DISP_LINE_LEN 16
  176. int do_ioloop(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[])
  177. {
  178. unsigned int fpga;
  179. unsigned int size;
  180. unsigned int rate = 0;
  181. if (argc < 3)
  182. return CMD_RET_USAGE;
  183. /*
  184. * FPGA is specified since argc > 2
  185. */
  186. fpga = simple_strtoul(argv[1], NULL, 10);
  187. /*
  188. * packet size is specified since argc > 2
  189. */
  190. size = simple_strtoul(argv[2], NULL, 10);
  191. /*
  192. * If another parameter, it is the test rate in packets per second.
  193. */
  194. if (argc > 3)
  195. rate = simple_strtoul(argv[3], NULL, 10);
  196. /* enable receive path */
  197. FPGA_SET_REG(fpga, ep.rx_tx_control, CTRL_PROC_RECEIVE_ENABLE);
  198. /* set device address to dummy 1*/
  199. FPGA_SET_REG(fpga, ep.device_address, 1);
  200. rx_ctr = 0; tx_ctr = 0; err_ctr = 0;
  201. while (1) {
  202. u16 top_int;
  203. u16 rx_tx_status;
  204. FPGA_GET_REG(fpga, top_interrupt, &top_int);
  205. FPGA_GET_REG(fpga, ep.rx_tx_status, &rx_tx_status);
  206. io_check_status(fpga, rx_tx_status, false);
  207. if (top_int & IRQ_CPU_TRANSMITBUFFER_FREE_STATUS)
  208. io_send(fpga, size);
  209. if (top_int & IRQ_CPU_RECEIVE_DATA_AVAILABLE_STATUS)
  210. io_receive(fpga);
  211. if (rate) {
  212. if (ctrlc())
  213. break;
  214. udelay(1000000 / rate);
  215. if (!(tx_ctr % rate))
  216. printf("d %lld, tx %llu, rx %llu, err %llu\n",
  217. tx_ctr - rx_ctr, tx_ctr, rx_ctr,
  218. err_ctr);
  219. }
  220. }
  221. return 0;
  222. }
  223. U_BOOT_CMD(
  224. ioloop, 4, 0, do_ioloop,
  225. "fpga io-endpoint looptest",
  226. "fpga packetsize [packets/sec]"
  227. );
  228. U_BOOT_CMD(
  229. ioreflect, 3, 0, do_ioreflect,
  230. "fpga io-endpoint reflector",
  231. "fpga reportrate"
  232. );