drv_soft_spi.c 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303
  1. /*
  2. * Copyright (c) 2006-2021, RT-Thread Development Team
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Change Logs:
  7. * Date Author Notes
  8. * 2021-10-11 kyle first implementation.
  9. */
  10. #include "drv_soft_spi.h"
  11. #include <board.h>
  12. #include <rtthread.h>
  13. #if defined(RT_USING_SPI) && defined(RT_USING_SPI_BITOPS) && defined(RT_USING_PIN)
  14. #include <rtdevice.h>
  15. #include "spi-bit-ops.h"
  16. #define DBG_TAG "drv.SPI"
  17. #ifdef RT_SPI_DEBUG
  18. #define DBG_LVL DBG_LOG
  19. #else
  20. #define DBG_LVL DBG_WARNING
  21. #endif
  22. #include <rtdbg.h>
  23. #ifndef ITEM_NUM
  24. #define ITEM_NUM(items) sizeof(items) / sizeof(items[0])
  25. #endif
  26. struct gd32_spi_bit_data
  27. {
  28. struct
  29. {
  30. rcu_periph_enum clk;
  31. rt_uint32_t port;
  32. rt_uint32_t pin;
  33. } sclk, mosi, miso;
  34. };
  35. rt_inline FlagStatus GPIO_OUTPUT_BIT_GET(uint32_t gpio_periph, uint32_t pin)
  36. {
  37. if((uint32_t)RESET !=(GPIO_OCTL(gpio_periph)&(pin))){
  38. return SET;
  39. }else{
  40. return RESET;
  41. }
  42. }
  43. rt_inline void GPIO_BIT_RESET(uint32_t gpio_periph, uint32_t pin)
  44. {
  45. GPIO_BC(gpio_periph) = (uint32_t)pin;
  46. }
  47. rt_inline void GPIO_BIT_SET(uint32_t gpio_periph, uint32_t pin)
  48. {
  49. GPIO_BOP(gpio_periph) = (uint32_t)pin;
  50. }
  51. rt_inline FlagStatus GPIO_INPUT_BIT_GET(uint32_t gpio_periph,uint32_t pin)
  52. {
  53. if((uint32_t)RESET != (GPIO_ISTAT(gpio_periph)&(pin))){
  54. return SET;
  55. }else{
  56. return RESET;
  57. }
  58. }
  59. rt_inline void GPIO_INIT(uint32_t gpio_periph, uint32_t mode, uint32_t speed, uint32_t pin)
  60. {
  61. uint16_t i;
  62. uint32_t temp_mode = 0U;
  63. uint32_t reg = 0U;
  64. /* GPIO mode configuration */
  65. temp_mode = (uint32_t)(mode & ((uint32_t)0x0FU));
  66. /* GPIO speed configuration */
  67. if(((uint32_t)0x00U) != ((uint32_t)mode & ((uint32_t)0x10U))){
  68. /* output mode max speed:10MHz,2MHz,50MHz */
  69. temp_mode |= (uint32_t)speed;
  70. }
  71. /* configure the eight low port pins with GPIO_CTL0 */
  72. for(i = 0U;i < 8U;i++){
  73. if((1U << i) & pin){
  74. reg = GPIO_CTL0(gpio_periph);
  75. /* clear the specified pin mode bits */
  76. reg &= ~GPIO_MODE_MASK(i);
  77. /* set the specified pin mode bits */
  78. reg |= GPIO_MODE_SET(i, temp_mode);
  79. /* set IPD or IPU */
  80. if(GPIO_MODE_IPD == mode){
  81. /* reset the corresponding OCTL bit */
  82. GPIO_BC(gpio_periph) = (uint32_t)((1U << i) & pin);
  83. }else{
  84. /* set the corresponding OCTL bit */
  85. if(GPIO_MODE_IPU == mode){
  86. GPIO_BOP(gpio_periph) = (uint32_t)((1U << i) & pin);
  87. }
  88. }
  89. /* set GPIO_CTL0 register */
  90. GPIO_CTL0(gpio_periph) = reg;
  91. }
  92. }
  93. /* configure the eight high port pins with GPIO_CTL1 */
  94. for(i = 8U;i < 16U;i++){
  95. if((1U << i) & pin){
  96. reg = GPIO_CTL1(gpio_periph);
  97. /* clear the specified pin mode bits */
  98. reg &= ~GPIO_MODE_MASK(i - 8U);
  99. /* set the specified pin mode bits */
  100. reg |= GPIO_MODE_SET(i - 8U, temp_mode);
  101. /* set IPD or IPU */
  102. if(GPIO_MODE_IPD == mode){
  103. /* reset the corresponding OCTL bit */
  104. GPIO_BC(gpio_periph) = (uint32_t)((1U << i) & pin);
  105. }else{
  106. /* set the corresponding OCTL bit */
  107. if(GPIO_MODE_IPU == mode){
  108. GPIO_BOP(gpio_periph) = (uint32_t)((1U << i) & pin);
  109. }
  110. }
  111. /* set GPIO_CTL1 register */
  112. GPIO_CTL1(gpio_periph) = reg;
  113. }
  114. }
  115. }
  116. #define GPIO_SET_OUTPUT(port, pin) GPIO_INIT(port, GPIO_MODE_OUT_PP, GPIO_OSPEED_50MHZ, pin)
  117. #define GPIO_SET_INPUT(port, pin) GPIO_INIT(port, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_50MHZ, pin)
  118. static void gpio_tog_sclk(void *data)
  119. {
  120. struct gd32_spi_bit_data *bd = data;
  121. if (GPIO_OUTPUT_BIT_GET(bd->sclk.port, bd->sclk.pin) == SET)
  122. {
  123. GPIO_BIT_RESET(bd->sclk.port, bd->sclk.pin);
  124. }
  125. else
  126. {
  127. GPIO_BIT_SET(bd->sclk.port, bd->sclk.pin);
  128. }
  129. }
  130. static void gpio_set_sclk(void *data, rt_int32_t state)
  131. {
  132. struct gd32_spi_bit_data *bd = data;
  133. if (state)
  134. {
  135. GPIO_BIT_SET(bd->sclk.port, bd->sclk.pin);
  136. }
  137. else
  138. {
  139. GPIO_BIT_RESET(bd->sclk.port, bd->sclk.pin);
  140. }
  141. }
  142. static void gpio_set_mosi(void *data, rt_int32_t state)
  143. {
  144. struct gd32_spi_bit_data *bd = data;
  145. if (state)
  146. {
  147. GPIO_BIT_SET(bd->mosi.port, bd->mosi.pin);
  148. }
  149. else
  150. {
  151. GPIO_BIT_RESET(bd->mosi.port, bd->mosi.pin);
  152. }
  153. }
  154. static void gpio_set_miso(void *data, rt_int32_t state)
  155. {
  156. struct gd32_spi_bit_data *bd = data;
  157. if (state)
  158. {
  159. GPIO_BIT_SET(bd->miso.port, bd->miso.pin);
  160. }
  161. else
  162. {
  163. GPIO_BIT_RESET(bd->miso.port, bd->miso.pin);
  164. }
  165. }
  166. static rt_int32_t gpio_get_sclk(void *data)
  167. {
  168. struct gd32_spi_bit_data *bd = data;
  169. return GPIO_INPUT_BIT_GET(bd->sclk.port, bd->sclk.pin);
  170. }
  171. static rt_int32_t gpio_get_mosi(void *data)
  172. {
  173. struct gd32_spi_bit_data *bd = data;
  174. return GPIO_INPUT_BIT_GET(bd->mosi.port, bd->mosi.pin);
  175. }
  176. static rt_int32_t gpio_get_miso(void *data)
  177. {
  178. struct gd32_spi_bit_data *bd = data;
  179. return GPIO_INPUT_BIT_GET(bd->miso.port, bd->miso.pin);
  180. }
  181. static void gpio_dir_mosi(void *data, rt_int32_t state)
  182. {
  183. struct gd32_spi_bit_data *bd = data;
  184. if (state)
  185. {
  186. GPIO_SET_INPUT(bd->mosi.port, bd->mosi.pin);
  187. }
  188. else
  189. {
  190. GPIO_SET_OUTPUT(bd->mosi.port, bd->mosi.pin);
  191. }
  192. }
  193. static void gpio_dir_miso(void *data, rt_int32_t state)
  194. {
  195. struct gd32_spi_bit_data *bd = data;
  196. if (state)
  197. {
  198. GPIO_SET_INPUT(bd->miso.port, bd->miso.pin);
  199. }
  200. else
  201. {
  202. GPIO_SET_OUTPUT(bd->miso.port, bd->miso.pin);
  203. }
  204. }
  205. static void gpio_udelay(rt_uint32_t us)
  206. {
  207. int i = ((rcu_clock_freq_get(CK_SYS) / 4000000) * us);
  208. while (i)
  209. {
  210. i--;
  211. }
  212. }
  213. static void soft_spi_gpio_init(const struct gd32_spi_bit_data *bd)
  214. {
  215. rcu_periph_clock_enable(bd->sclk.clk);
  216. rcu_periph_clock_enable(bd->mosi.clk);
  217. rcu_periph_clock_enable(bd->miso.clk);
  218. gpio_init(bd->sclk.port, GPIO_MODE_OUT_PP, GPIO_OSPEED_50MHZ, bd->sclk.pin);
  219. gpio_init(bd->mosi.port, GPIO_MODE_OUT_PP, GPIO_OSPEED_50MHZ, bd->mosi.pin);
  220. gpio_init(bd->miso.port, GPIO_MODE_IN_FLOATING, GPIO_OSPEED_50MHZ, bd->miso.pin);
  221. GPIO_BIT_SET(bd->sclk.port, bd->sclk.pin);
  222. GPIO_BIT_SET(bd->mosi.port, bd->mosi.pin);
  223. GPIO_BIT_SET(bd->miso.port, bd->miso.pin);
  224. }
  225. int rt_soft_spi_init(void)
  226. {
  227. int result = 0;
  228. {
  229. static const struct gd32_spi_bit_data spi1_bdata =
  230. {
  231. .sclk = { RCU_GPIOB, GPIOB, GPIO_PIN_13},
  232. .mosi = { RCU_GPIOB, GPIOB, GPIO_PIN_15},
  233. .miso = { RCU_GPIOB, GPIOB, GPIO_PIN_14},
  234. };
  235. static struct rt_spi_bit_ops spi1_bops =
  236. {
  237. .data = (void *)&spi1_bdata,
  238. .tog_sclk = gpio_tog_sclk,
  239. .set_sclk = gpio_set_sclk,
  240. .set_mosi = gpio_set_mosi,
  241. .set_miso = gpio_set_miso,
  242. .get_sclk = gpio_get_sclk,
  243. .get_mosi = gpio_get_mosi,
  244. .get_miso = gpio_get_miso,
  245. .dir_mosi = gpio_dir_mosi,
  246. .dir_miso = gpio_dir_miso,
  247. .udelay = gpio_udelay,
  248. };
  249. struct rt_spi_bit_obj spi1_obj;
  250. soft_spi_gpio_init(&spi1_bdata);
  251. rt_spi_bit_add_bus(&spi1_obj, "spi3", &spi1_bops);
  252. }
  253. return result;
  254. }
  255. INIT_BOARD_EXPORT(rt_soft_spi_init);
  256. #endif