dev_qspi_core.c 7.5 KB

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