qspi_core.c 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270
  1. /*
  2. * Copyright (c) 2006-2023, RT-Thread Development Team
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Change Logs:
  7. * Date Author Notes
  8. * 2018-11-16 zylx first version.
  9. */
  10. #include <drivers/spi.h>
  11. rt_err_t rt_qspi_configure(struct rt_qspi_device *device, struct rt_qspi_configuration *cfg)
  12. {
  13. RT_ASSERT(device != RT_NULL);
  14. RT_ASSERT(cfg != RT_NULL);
  15. /* If the configurations are the same, we don't need to set again. */
  16. if (device->config.medium_size == cfg->medium_size &&
  17. device->config.ddr_mode == cfg->ddr_mode &&
  18. device->config.qspi_dl_width == cfg->qspi_dl_width &&
  19. device->config.parent.data_width == cfg->parent.data_width &&
  20. device->config.parent.mode == (cfg->parent.mode & RT_SPI_MODE_MASK) &&
  21. device->config.parent.max_hz == cfg->parent.max_hz)
  22. {
  23. return RT_EOK;
  24. }
  25. /* copy configuration items */
  26. device->config.parent.mode = cfg->parent.mode;
  27. device->config.parent.max_hz = cfg->parent.max_hz;
  28. device->config.parent.data_width = cfg->parent.data_width;
  29. device->config.parent.reserved = cfg->parent.reserved;
  30. device->config.medium_size = cfg->medium_size;
  31. device->config.ddr_mode = cfg->ddr_mode;
  32. device->config.qspi_dl_width = cfg->qspi_dl_width;
  33. return rt_spi_bus_configure(&device->parent);
  34. }
  35. rt_err_t rt_qspi_bus_register(struct rt_spi_bus *bus, const char *name, const struct rt_spi_ops *ops)
  36. {
  37. rt_err_t result = RT_EOK;
  38. result = rt_spi_bus_register(bus, name, ops);
  39. if(result == RT_EOK)
  40. {
  41. /* set SPI bus to qspi modes */
  42. bus->mode = RT_SPI_BUS_MODE_QSPI;
  43. }
  44. return result;
  45. }
  46. rt_size_t rt_qspi_transfer_message(struct rt_qspi_device *device, struct rt_qspi_message *message)
  47. {
  48. rt_err_t result;
  49. RT_ASSERT(device != RT_NULL);
  50. RT_ASSERT(message != RT_NULL);
  51. result = rt_mutex_take(&(device->parent.bus->lock), RT_WAITING_FOREVER);
  52. if (result != RT_EOK)
  53. {
  54. rt_set_errno(-RT_EBUSY);
  55. return 0;
  56. }
  57. /* reset errno */
  58. rt_set_errno(RT_EOK);
  59. /* configure SPI bus */
  60. if (device->parent.bus->owner != &device->parent)
  61. {
  62. /* not the same owner as current, re-configure SPI bus */
  63. result = device->parent.bus->ops->configure(&device->parent, &device->parent.config);
  64. if (result == RT_EOK)
  65. {
  66. /* set SPI bus owner */
  67. device->parent.bus->owner = &device->parent;
  68. }
  69. else
  70. {
  71. /* configure SPI bus failed */
  72. rt_set_errno(-RT_EIO);
  73. goto __exit;
  74. }
  75. }
  76. /* transmit each SPI message */
  77. result = device->parent.bus->ops->xfer(&device->parent, &message->parent);
  78. if (result == 0)
  79. {
  80. rt_set_errno(-RT_EIO);
  81. }
  82. __exit:
  83. /* release bus lock */
  84. rt_mutex_release(&(device->parent.bus->lock));
  85. return result;
  86. }
  87. rt_err_t rt_qspi_send_then_recv(struct rt_qspi_device *device, const void *send_buf, rt_size_t send_length, void *recv_buf, rt_size_t recv_length)
  88. {
  89. RT_ASSERT(send_buf);
  90. RT_ASSERT(recv_buf);
  91. RT_ASSERT(send_length != 0);
  92. struct rt_qspi_message message;
  93. unsigned char *ptr = (unsigned char *)send_buf;
  94. rt_size_t count = 0;
  95. rt_err_t result = 0;
  96. message.instruction.content = ptr[0];
  97. message.instruction.qspi_lines = 1;
  98. count++;
  99. /* get address */
  100. if (send_length > 1)
  101. {
  102. if (device->config.medium_size > 0x1000000 && send_length >= 5)
  103. {
  104. /* medium size greater than 16Mb, address size is 4 Byte */
  105. message.address.content = (ptr[1] << 24) | (ptr[2] << 16) | (ptr[3] << 8) | (ptr[4]);
  106. message.address.size = 32;
  107. count += 4;
  108. }
  109. else if (send_length >= 4)
  110. {
  111. /* address size is 3 Byte */
  112. message.address.content = (ptr[1] << 16) | (ptr[2] << 8) | (ptr[3]);
  113. message.address.size = 24;
  114. count += 3;
  115. }
  116. else
  117. {
  118. return -RT_ERROR;
  119. }
  120. message.address.qspi_lines = 1;
  121. }
  122. else
  123. {
  124. /* no address stage */
  125. message.address.content = 0 ;
  126. message.address.qspi_lines = 0;
  127. message.address.size = 0;
  128. }
  129. message.alternate_bytes.content = 0;
  130. message.alternate_bytes.size = 0;
  131. message.alternate_bytes.qspi_lines = 0;
  132. /* set dummy cycles */
  133. if (count != send_length)
  134. {
  135. message.dummy_cycles = (send_length - count) * 8;
  136. }
  137. else
  138. {
  139. message.dummy_cycles = 0;
  140. }
  141. /* set recv buf and recv size */
  142. message.parent.recv_buf = recv_buf;
  143. message.parent.send_buf = RT_NULL;
  144. message.parent.length = recv_length;
  145. message.parent.cs_take = 1;
  146. message.parent.cs_release = 1;
  147. message.qspi_data_lines = 1;
  148. result = rt_qspi_transfer_message(device, &message);
  149. if (result == 0)
  150. {
  151. result = -RT_EIO;
  152. }
  153. else
  154. {
  155. result = recv_length;
  156. }
  157. return result;
  158. }
  159. rt_err_t rt_qspi_send(struct rt_qspi_device *device, const void *send_buf, rt_size_t length)
  160. {
  161. RT_ASSERT(send_buf);
  162. RT_ASSERT(length != 0);
  163. struct rt_qspi_message message;
  164. unsigned char *ptr = (unsigned char *)send_buf;
  165. rt_size_t count = 0;
  166. rt_err_t result = 0;
  167. message.instruction.content = ptr[0];
  168. message.instruction.qspi_lines = 1;
  169. count++;
  170. /* get address */
  171. if (length > 1)
  172. {
  173. if (device->config.medium_size > 0x1000000 && length >= 5)
  174. {
  175. /* medium size greater than 16Mb, address size is 4 Byte */
  176. message.address.content = (ptr[1] << 24) | (ptr[2] << 16) | (ptr[3] << 8) | (ptr[4]);
  177. message.address.size = 32;
  178. message.address.qspi_lines = 1;
  179. count += 4;
  180. }
  181. else if (length >= 4)
  182. {
  183. /* address size is 3 Byte */
  184. message.address.content = (ptr[1] << 16) | (ptr[2] << 8) | (ptr[3]);
  185. message.address.size = 24;
  186. message.address.qspi_lines = 1;
  187. count += 3;
  188. }
  189. else
  190. {
  191. return -RT_ERROR;
  192. }
  193. }
  194. else
  195. {
  196. /* no address stage */
  197. message.address.content = 0 ;
  198. message.address.qspi_lines = 0;
  199. message.address.size = 0;
  200. }
  201. message.alternate_bytes.content = 0;
  202. message.alternate_bytes.size = 0;
  203. message.alternate_bytes.qspi_lines = 0;
  204. message.dummy_cycles = 0;
  205. /* determine if there is data to send */
  206. if (length - count > 0)
  207. {
  208. message.qspi_data_lines = 1;
  209. }
  210. else
  211. {
  212. message.qspi_data_lines = 0;
  213. }
  214. /* set send buf and send size */
  215. message.parent.send_buf = ptr + count;
  216. message.parent.recv_buf = RT_NULL;
  217. message.parent.length = length - count;
  218. message.parent.cs_take = 1;
  219. message.parent.cs_release = 1;
  220. result = rt_qspi_transfer_message(device, &message);
  221. if (result == 0)
  222. {
  223. result = -RT_EIO;
  224. }
  225. else
  226. {
  227. result = length;
  228. }
  229. return result;
  230. }