qcaspi-pseudo-code.txt 8.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300
  1. <section id="pseudo-code">
  2. <title>
  3. Serial Driver Pseudo Code
  4. </title>
  5. <para>
  6. This chapter presents pseudo code for the QCA7000 SPI driver. Qualcomm Atheros provides a precoded driver for Linux but customers may want to write their own driver for a non-Linux environment. In either case, the following steps are required for efficient SPI communications.
  7. </para>
  8. <screen>
  9. function receive frame
  10. allocate an input buffer;
  11. if no memory available then
  12. return failure;
  13. end if;
  14. read the read buffer space available register;
  15. if read buffer is empty then
  16. return an error;
  17. end if;
  18. write frame length into buffer size register;
  19. if mode is legacy then
  20. send SPI read command;
  21. end if;
  22. while read buffer is not empty do
  23. read read buffer;
  24. extract Ethernet frame from your frame buffer;
  25. update any read statistics;
  26. end while;
  27. return success;
  28. end function;
  29. </screen>
  30. <screen>
  31. function transmit frame do
  32. read the write buffer space available register;
  33. while transmit queue is not empty and write buffer space is available do
  34. encapsulate transmit frame;
  35. write transmit frame to write buffer;
  36. update transmit statistics;
  37. if error then
  38. return failure;
  39. end if;
  40. update transmit statistics;
  41. remove frame from transmit queue;
  42. end while;
  43. return success;
  44. end function;
  45. </screen>
  46. <para>
  47. The interrupt service routine suspends, waiting for an interrupt from the QCA7000. Use of an interrupt means that the host need not continuously poll the SPI status and space available registers to detect if an operation has completed successfully before starting the next operation. Once an interrupt is detected by the host, this routine disables further interrupts and determines shy the interrupt occured by inspecting the <varname>INTR_CAUSE</varname> and <varname>SPI_STATUS</varname> registers. It then handles the interrupt by calling appropriate functions. When done, it clears the interrupt cause and SPI status registers and suspends.
  48. </para>
  49. <screen>
  50. function interrupt service routine do
  51. while terminate is false do
  52. set thread state to interruptable;
  53. if no interrupts and synchronization state is synchronized and transmit queue is empty then
  54. allow other tasks to run;
  55. end if;
  56. set thread state to busy;
  57. check synchronization state;
  58. if syncrhonization state is not synchronized then
  59. flush transmit queue;
  60. suspend for a while;
  61. end if;
  62. if interrupt occurred then
  63. disable interrupts;
  64. read interrupt cause register;
  65. if interrupt cause is CPU on then
  66. update synchronization state;
  67. if synchronization state is synchronized then
  68. continue;
  69. end if;
  70. end if;
  71. if interrupt cause is packet available then
  72. if synchronization state is synchronized then
  73. call receive frame function;
  74. end if;
  75. end if;
  76. if interrupt cause is read buffer error then
  77. set synchronization state to unknown;
  78. continue;
  79. end if;
  80. if interrupt cause is write buffer error then
  81. set synchronization state to unknown;
  82. continue;
  83. end if;
  84. clear interrupt cause register;
  85. clear SPI status register;
  86. end if;
  87. if transmit queue is not empty then
  88. call transmit frame function;
  89. end if;
  90. end while;
  91. set thread state to dormant;
  92. return;
  93. end function;
  94. </screen>
  95. <screen>
  96. function synchronize slave do
  97. allocate a static reset counter;
  98. if synchronization state is CPU on then
  99. read QCA7000 signature register;
  100. read QCA7000 signature register;
  101. if signature is invalid then
  102. set synchronization state to unknown;
  103. else
  104. read write buffer space available register;
  105. if write buffer is empty then
  106. set qca.SynchState to QCASPI_SYNC_READY;
  107. set synchronization state to ready;
  108. return;
  109. else
  110. set synchronization state to unknown;
  111. end if;
  112. end if;
  113. end if;
  114. if synchronization state is ready then
  115. if mode is legacy then
  116. return;
  117. end if;
  118. read QCA7000 signature register;
  119. if signature is invalid then
  120. set synchronization state to unknown;
  121. return;
  122. end if;
  123. end if;
  124. if synchronization state is unknown then
  125. if mode is legacy then
  126. use GPIO to reset QCA7000;
  127. else
  128. read QCA7000 signature register;
  129. if signature is invalid then
  130. return;
  131. end if;
  132. set soc_core_reset bit in QCA SPI configuration register;
  133. end if;
  134. set synchronization state to reset;
  135. clear reset counter;
  136. return;
  137. end if;
  138. if synchronization state is reset then
  139. increment reset counter;
  140. if reset counter exceeds reset limit then
  141. set synchronization state to unknown;
  142. end if;
  143. end if;
  144. return;
  145. end function;
  146. </screen>
  147. <screen>
  148. function interrupt handler do
  149. increment interrupt count;
  150. if thread is available and thread is dormant then
  151. wake up thread to service interrupt;
  152. end if;
  153. return success;
  154. end function;
  155. </screen>
  156. </section>
  157. <section id="qcaspi1">
  158. <title>
  159. SPI Serial Driver
  160. </title>
  161. <section id="qcaspi_spi_thread">
  162. <title>
  163. qcaspi_spi_thread
  164. </title>
  165. <funcsynopsis>
  166. <funcprototype>
  167. <funcdef>static int <function>qcaspi_spi_thread</function></funcdef>
  168. <paramdef>char void * <parameter>data</parameter></paramdef>
  169. </funcprototype>
  170. </funcsynopsis>
  171. <para>
  172. Manages synchronization with the exteranl QCA7000.
  173. Handles interrupts fomr the external QCA7000.
  174. Transmits frames for the transmit queue to the QCA7000.
  175. </para>
  176. </section>
  177. <section id="qcaspi_qca7k_sync">
  178. <title>
  179. qcaspi_qca7k_sync
  180. </title>
  181. <funcsynopsis>
  182. <funcprototype>
  183. <funcdef>void <function>qca_qca7k_sync</function></funcdef>
  184. <paramdef>char struct qcaspi * <parameter>qca</parameter></paramdef>
  185. <paramdef>int <parameter>event</parameter></paramdef>
  186. </funcprototype>
  187. </funcsynopsis>
  188. <para>
  189. Keeps track of the current synchonization state.
  190. </para>
  191. </section>
  192. </section>
  193. <section id="qcaspi2">
  194. <title>
  195. Register Functions
  196. </title>
  197. <section id="qcaspi_read_register">
  198. <title>
  199. qcaspi_read_register
  200. </title>
  201. <funcsynopsis>
  202. <funcprototype>
  203. <funcdef>uint16_t <function>qcaspi_read_register</function></funcdef>
  204. <paramdef>struct qcaspi * <parameter>qca</parameter></paramdef>
  205. <paramdef>uint16_t <parameter>reg</parameter></paramdef>
  206. </funcprototype>
  207. </funcsynopsis>
  208. <para>
  209. Reads a QCA7000 register and returns register content.
  210. </para>
  211. </section>
  212. <section id="qcaspi_write_register">
  213. <title>
  214. qcaspi_write_register
  215. </title>
  216. <funcsynopsis>
  217. <funcprototype>
  218. <funcdef>void <function>qcaspi_write_register</function></funcdef>
  219. <paramdef>struct qcaspi * <parameter>qca</parameter></paramdef>
  220. <paramdef>uint16_t <parameter>reg</parameter></paramdef>
  221. <paramdef>uint16_t <parameter>value</parameter></paramdef>
  222. </funcprototype>
  223. </funcsynopsis>
  224. <para>
  225. Write a value into a QCA7000 register.
  226. </para>
  227. </section>
  228. <section id="qcaspi_tx_cmd">
  229. <title>
  230. qcaspi_tx_cmd
  231. </title>
  232. <funcsynopsis>
  233. <funcprototype>
  234. <funcdef>int <function>qcaspi_tx_cmd</function></funcdef>
  235. <paramdef>struct qcaspi * <parameter>qca</parameter></paramdef>
  236. <paramdef>uint16_t <parameter>cmd</parameter></paramdef>
  237. </funcprototype>
  238. </funcsynopsis>
  239. <para>
  240. Transmit a 16-bit command to the QCA7000. This is currently used when performing a legacy DMA read or write.
  241. </para>
  242. </section>
  243. </section>
  244. <section id="qcaspi3">
  245. <title>
  246. Interrupt Functions
  247. </title>
  248. <section id="disable_spi_interrupts">
  249. <title>
  250. disable_spi_interrupts
  251. </title>
  252. <funcsynopsis>
  253. <funcprototype>
  254. <funcdef>uint32_t <function>disable_spi_interrupts</function></funcdef>
  255. <paramdef>struct qcaspi * <parameter>qca</parameter></paramdef>
  256. </funcprototype>
  257. </funcsynopsis>
  258. <para>
  259. Disables interrupts by writing <constant>0</constant> to the QCA7000 <varname>INTR_ENABLE</varname> register.
  260. </para>
  261. </section>
  262. <section id="enable_spi_interrupts">
  263. <title>
  264. enable_spi_interrupts
  265. </title>
  266. <funcsynopsis>
  267. <funcprototype>
  268. <funcdef>uint32_t <function>enable_spi_interrupts</function></funcdef>
  269. <paramdef>struct qcaspi * <parameter>qca</parameter></paramdef>
  270. <paramdef>uint32_t <parameter>intr_enable</parameter></paramdef>
  271. </funcprototype>
  272. </funcsynopsis>
  273. <para>
  274. Enables interrupts specified by writing to the QCA7000 <varname>INTR_ENABLE</varname> register and returns the previous register value.
  275. </para>
  276. </section>
  277. <section id="qcaspi_intr_handler">
  278. <title>
  279. qcaspi_intr_handler
  280. </title>
  281. <funcsynopsis>
  282. <funcprototype>
  283. <funcdef>static irqreturn_t <function>qcaspi_intr_handler</function></funcdef>
  284. <paramdef>int <parameter>irq</parameter></paramdef>
  285. <paramdef>void * <parameter>data</parameter></paramdef>
  286. </funcprototype>
  287. </funcsynopsis>
  288. <para>
  289. Called to service interrupts on rising edge of the QCA7000 interrupt line.
  290. </para>
  291. </section>
  292. </section>