spimt_leopard.c 25 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788
  1. /*
  2. * Copyright (c) 2018 Actions Semiconductor Co., Ltd
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief SPIMT driver for Actions SoC
  9. */
  10. //#define DT_DRV_COMPAT actions_acts_spimt
  11. #define LOG_LEVEL CONFIG_SPI_LOG_LEVEL
  12. #include <logging/log.h>
  13. LOG_MODULE_REGISTER(spimt_acts);
  14. #include "spi_context.h"
  15. #include <errno.h>
  16. #include <device.h>
  17. #include <drivers/ipmsg.h>
  18. #include <drivers/spimt.h>
  19. #include <rbuf/rbuf_msg_sc.h>
  20. #include <soc.h>
  21. #include <board_cfg.h>
  22. #include <linker/linker-defs.h>
  23. #define SPIMT_CLOCK (4000000)
  24. //#define SPI_BUF_LEN (4096)
  25. #define SPI_MAX_BC (32*1024)
  26. #define SPI_DELAY (8)
  27. /**
  28. * @brief SPI Module (SPIMT)
  29. */
  30. typedef struct {
  31. volatile uint32_t CTL; /*!< (@ 0x00000000) TASK Control Register */
  32. volatile uint32_t DMA_CTL; /*!< (@ 0x00000004) TASK DMA Control Register */
  33. volatile uint32_t DMA_ADD; /*!< (@ 0x00000008) TASK DMA Address Register */
  34. volatile uint32_t DMA_CNT; /*!< (@ 0x0000000C) TASK DMA Counter Register */
  35. volatile uint32_t RESERVED[12];
  36. } SPIMT_AUTO_TASK_Type; /*!< Size = 64 (0x40) */
  37. typedef struct { /*!< (@ 0x40080000) SPIMT Structure */
  38. volatile uint32_t CTL; /*!< (@ 0x00000000) Control Register */
  39. volatile uint32_t NML_STA; /*!< (@ 0x00000004) Status Register */
  40. volatile uint32_t NML_TXDAT; /*!< (@ 0x00000008) TX FIFO DATA Register */
  41. volatile uint32_t NML_RXDAT; /*!< (@ 0x0000000C) RX FIFO DATA Register */
  42. volatile uint32_t NML_BC; /*!< (@ 0x00000010) Bytes Count Register */
  43. volatile uint32_t AUTO_TASK_STAT; /*!< (@ 0x00000014) Auto task status Register */
  44. volatile uint32_t AUTO_TASK_IE; /*!< (@ 0x00000018) Auto task IRQ enable Register */
  45. volatile uint32_t AUTO_TASK_IP; /*!< (@ 0x0000001C) Auto task IRQ pending Register */
  46. volatile uint32_t TASK_CS_DLY; /*!< (@ 0x00000020) Task CS Delay Register */
  47. volatile uint32_t TASK_CS_SEL; /*!< (@ 0x00000024) Task CS Select Register */
  48. volatile uint32_t RESERVED[54];
  49. volatile SPIMT_AUTO_TASK_Type AUTO_TASK[8]; /*!< (@ 0x00000100) TASK[0..7] Group */
  50. volatile uint32_t RESERVED1[3904];
  51. } SPIMT_Type; /*!< Size = 768 (0x300) */
  52. typedef SPIMT_Type SPIMT_ARRAYType[1]; /*!< max. 2 instances available */
  53. #define SPIMT ((SPIMT_ARRAYType*) SPIMT0_REG_BASE)
  54. /* ========================================================== CTL ========================================================== */
  55. #define SPIMT_CTL_CS_SEL_Msk (0x40000UL) /*!< CS_SEL (Bitfield-Mask: 0x01) */
  56. #define SPIMT_CTL_MODSEL_Msk (0x20000UL) /*!< MODSEL (Bitfield-Mask: 0x01) */
  57. #define SPIMT_CTL_MODE_SEL_Msk (0x10000UL) /*!< MODE_SEL (Bitfield-Mask: 0x01) */
  58. #define SPIMT_CTL_RXWR_SEL_Msk (0x4000UL) /*!< RXWR_SEL (Bitfield-Mask: 0x01) */
  59. #define SPIMT_CTL_DELAY_Msk (0x3c00UL) /*!< DELAY (Bitfield-Mask: 0x0f) */
  60. #define SPIMT_CTL_3WIRE_Msk (0x80UL) /*!< 3WIRE (Bitfield-Mask: 0x01) */
  61. #define SPIMT_CTL_REQ_Msk (0x40UL) /*!< REQ (Bitfield-Mask: 0x01) */
  62. #define SPIMT_CTL_TXFIFO_EN_Msk (0x20UL) /*!< TXFIFO_EN (Bitfield-Mask: 0x01) */
  63. #define SPIMT_CTL_RXFIFO_EN_Msk (0x10UL) /*!< RXFIFO_EN (Bitfield-Mask: 0x01) */
  64. #define SPIMT_CTL_SS_Msk (0x8UL) /*!< SS (Bitfield-Mask: 0x01) */
  65. #define SPIMT_CTL_LOOP_Msk (0x4UL) /*!< LOOP (Bitfield-Mask: 0x01) */
  66. #define SPIMT_CTL_WR_Msk (0x3UL) /*!< WR (Bitfield-Mask: 0x03) */
  67. #define SPIMT_CTL_CS_SEL_Pos (18UL) /*!< CS_SEL (Bit 18) */
  68. #define SPIMT_CTL_DELAY_Pos (10UL) /*!< DELAY (Bit 10) */
  69. #define SPIMT_CTL_WR_Pos (0UL) /*!< WR (Bit 0) */
  70. /* ======================================================== NML_STA ======================================================== */
  71. #define SPIMT_NML_STA_BUSY_Msk (0x10UL) /*!< BUSY (Bitfield-Mask: 0x01) */
  72. #define SPIMT_NML_STA_TXFU_Msk (0x8UL) /*!< TXFU (Bitfield-Mask: 0x01) */
  73. #define SPIMT_NML_STA_TXEM_Msk (0x4UL) /*!< TXEM (Bitfield-Mask: 0x01) */
  74. #define SPIMT_NML_STA_RXFU_Msk (0x2UL) /*!< RXFU (Bitfield-Mask: 0x01) */
  75. #define SPIMT_NML_STA_RXEM_Msk (0x1UL) /*!< RXEM (Bitfield-Mask: 0x01) */
  76. /* ===================================================== AUTO_TASK_IE ====================================================== */
  77. #define SPIMT_AUTO_TASK_IE_TSK0HFIE_Pos (8UL) /*!< TSK0HFIE (Bit 8) */
  78. #define SPIMT_AUTO_TASK_IE_TSK0TCIE_Pos (0UL) /*!< TSK0TCIE (Bit 0) */
  79. /* ===================================================== AUTO_TASK_IP ====================================================== */
  80. #define SPIMT_AUTO_TASK_IP_TSK0HFIP_Pos (8UL) /*!< TSK0HFIP (Bit 8) */
  81. #define SPIMT_AUTO_TASK_IP_TSK0TCIP_Pos (0UL) /*!< TSK0TCIP (Bit 0) */
  82. /* ====================================================== TASK_CS_SEL ====================================================== */
  83. #define SPIMT_TASK_CS_SEL_TSK0CS_SEL_Pos (0UL) /*!< TSK0CS_SEL (Bit 0) */
  84. #define SPIMT_TASK_CS_SEL_TSK0CS_SEL_Msk (0x3UL) /*!< TSK0CS_SEL (Bitfield-Mask: 0x03) */
  85. /* ========================================================== CTL ========================================================== */
  86. #define SPIMT_AUTO_TASK_CTL_SOFT_EN_Msk (0x80000000UL) /*!< SOFT_EN (Bitfield-Mask: 0x01) */
  87. #define SPIMT_AUTO_TASK_CTL_SOFT_ST_Msk (0x40000000UL) /*!< SOFT_ST (Bitfield-Mask: 0x01) */
  88. /* ======================================================== DMA_CTL ======================================================== */
  89. #define SPIMT_AUTO_TASK_DMA_CTL_DMARELD_Pos (1UL) /*!< DMARELD (Bit 1) */
  90. #define SPIMT_AUTO_TASK_DMA_CTL_DMASTART_Msk (0x1UL) /*!< DMASTART (Bitfield-Mask: 0x01) */
  91. struct acts_spimt_config {
  92. uint32_t ctl_reg;
  93. uint8_t bus_id;
  94. uint8_t clock_id;
  95. uint8_t reset_id;
  96. void (*irq_config_func)(void);
  97. };
  98. struct acts_spimt_data {
  99. struct spi_context ctx;
  100. uint8_t soft_started;
  101. spi_task_callback_t task_cb[SPI_TASK_NUM];
  102. void *task_cb_ctx[SPI_TASK_NUM];
  103. spi_task_t *task_attr[SPI_TASK_NUM];
  104. uint8_t *task_buf[SPI_TASK_NUM];
  105. uint32_t task_len[SPI_TASK_NUM];
  106. };
  107. #define DEV_CFG(dev) \
  108. ((const struct acts_spimt_config *const)(dev)->config)
  109. #define DEV_DATA(dev) \
  110. ((struct acts_spimt_data *const)(dev)->data)
  111. #if IS_ENABLED(CONFIG_SPIMT_0) || IS_ENABLED(CONFIG_SPIMT_1)
  112. // switch to auto mode
  113. static void spimt_auto_mode_set(int spi_dev)
  114. {
  115. /* clear task IRQ pending */
  116. if((SPIMT[spi_dev]->CTL & SPIMT_CTL_MODSEL_Msk) == 0)
  117. {
  118. SPIMT[spi_dev]->AUTO_TASK_IP = SPIMT[spi_dev]->AUTO_TASK_IP;
  119. /* set spi auto mode */
  120. SPIMT[spi_dev]->CTL |= SPIMT_CTL_MODSEL_Msk;
  121. }
  122. }
  123. // switch to normal mode
  124. static void spimt_auto_mode_cancel(int spi_dev)
  125. {
  126. /* cancel spi auto mode */
  127. SPIMT[spi_dev]->CTL &= ~SPIMT_CTL_MODSEL_Msk;
  128. }
  129. // CS pin set low
  130. static void spimt_cs_set_low(int spi_dev)
  131. {
  132. /* cancel spi auto mode */
  133. spimt_auto_mode_cancel(spi_dev);
  134. /* set low */
  135. SPIMT[spi_dev]->CTL &= ~SPIMT_CTL_SS_Msk;
  136. }
  137. // CS pin set high
  138. static void spimt_cs_set_high(int spi_dev)
  139. {
  140. /* set high */
  141. SPIMT[spi_dev]->CTL |= SPIMT_CTL_SS_Msk;
  142. /* enable spi auto mode */
  143. spimt_auto_mode_set(spi_dev);
  144. }
  145. // CS pin set high/low
  146. static void spimt_cs_set(int spi_dev, int enable)
  147. {
  148. if (enable) {
  149. spimt_cs_set_low(spi_dev);
  150. } else {
  151. spimt_cs_set_high(spi_dev);
  152. }
  153. }
  154. // select CS pin num
  155. static void spimt_cs_select(int spi_dev, int cs_num)
  156. {
  157. SPIMT[spi_dev]->CTL &= ~SPIMT_CTL_CS_SEL_Msk;
  158. SPIMT[spi_dev]->CTL |= (cs_num << SPIMT_CTL_CS_SEL_Pos);
  159. }
  160. // normal read
  161. static void spimt_read(int spi_dev, unsigned char *buf, unsigned int len)
  162. {
  163. int left = len;
  164. int rx_len = 0;
  165. int i;
  166. /* clear status */
  167. SPIMT[spi_dev]->NML_STA = SPIMT[spi_dev]->NML_STA;
  168. while (left) {
  169. /* set byte count */
  170. rx_len = left > SPI_MAX_BC ? SPI_MAX_BC : left;
  171. left -= rx_len;
  172. SPIMT[spi_dev]->NML_BC = rx_len;
  173. /* read-only mode */
  174. SPIMT[spi_dev]->CTL |= (1 << SPIMT_CTL_WR_Pos);
  175. /* wait for data */
  176. for (i = 0; i < rx_len; i++) {
  177. while(SPIMT[spi_dev]->NML_STA & SPIMT_NML_STA_RXEM_Msk);
  178. buf[i] = SPIMT[spi_dev]->NML_RXDAT;
  179. }
  180. /* disable mode */
  181. SPIMT[spi_dev]->CTL &= ~SPIMT_CTL_WR_Msk;
  182. }
  183. }
  184. // normal write
  185. static void spimt_write(int spi_dev, unsigned char *buf, unsigned int len)
  186. {
  187. int i;
  188. /* clear status */
  189. SPIMT[spi_dev]->NML_STA = SPIMT[spi_dev]->NML_STA;
  190. /* enable write-only mode */
  191. SPIMT[spi_dev]->CTL |= (2 << SPIMT_CTL_WR_Pos);
  192. /* write data */
  193. for (i = 0; i < len; i ++) {
  194. while (SPIMT[spi_dev]->NML_STA & SPIMT_NML_STA_TXFU_Msk);
  195. SPIMT[spi_dev]->NML_TXDAT = buf[i];
  196. }
  197. /* wait for tx empty*/
  198. while(!(SPIMT[spi_dev]->NML_STA & SPIMT_NML_STA_TXEM_Msk));
  199. /* wait for bus idle*/
  200. while(SPIMT[spi_dev]->NML_STA & SPIMT_NML_STA_BUSY_Msk);
  201. /* disable write-only mode */
  202. SPIMT[spi_dev]->CTL &= ~SPIMT_CTL_WR_Msk;
  203. }
  204. /**
  205. * @breif config spi task
  206. *
  207. * @Note
  208. * task 0,1,2,3 route to CS0
  209. * task 4,5,6,7 route to CS1
  210. */
  211. static void spimt_auto_task_config(int spi_dev, int task_id, const spi_task_t *task_attr, uint32_t addr)
  212. {
  213. uint8_t *pdata = (uint8_t *)&task_attr->ctl;
  214. volatile uint32_t ctl = *(volatile unsigned int*)pdata;
  215. /* config task dma first */
  216. SPIMT[spi_dev]->AUTO_TASK[task_id].DMA_ADD = addr;//task_attr->dma.addr;
  217. SPIMT[spi_dev]->AUTO_TASK[task_id].DMA_CNT = task_attr->dma.len;
  218. if (task_attr->dma.len > 0) {
  219. SPIMT[spi_dev]->AUTO_TASK[task_id].DMA_CTL = SPIMT_AUTO_TASK_DMA_CTL_DMASTART_Msk
  220. | (task_attr->dma.reload << SPIMT_AUTO_TASK_DMA_CTL_DMARELD_Pos);
  221. }
  222. if (task_attr->irq_type & SPI_TASK_IRQ_CMPLT) {
  223. /* enable task DMA transmission complete IRQ */
  224. SPIMT[spi_dev]->AUTO_TASK_IE |= 1 << (SPIMT_AUTO_TASK_IE_TSK0TCIE_Pos + task_id);
  225. } else {
  226. /* disable task DMA transmission complete IRQ */
  227. SPIMT[spi_dev]->AUTO_TASK_IE &= ~(1 << (SPIMT_AUTO_TASK_IE_TSK0TCIE_Pos + task_id));
  228. }
  229. if (task_attr->irq_type & SPI_TASK_IRQ_HALF_CMPLT) {
  230. /* enable task DMA Half transmission complete IRQ */
  231. SPIMT[spi_dev]->AUTO_TASK_IE |= 1 << (SPIMT_AUTO_TASK_IE_TSK0HFIE_Pos + task_id);
  232. } else {
  233. /* disable task DMA Half transmission complete IRQ */
  234. SPIMT[spi_dev]->AUTO_TASK_IE &= ~(1 << (SPIMT_AUTO_TASK_IE_TSK0HFIE_Pos + task_id));
  235. }
  236. /* config task ctl*/
  237. SPIMT[spi_dev]->AUTO_TASK[task_id].CTL = (ctl & ~(SPIMT_AUTO_TASK_CTL_SOFT_EN_Msk | SPIMT_AUTO_TASK_CTL_SOFT_ST_Msk));
  238. /* config task cs*/
  239. SPIMT[spi_dev]->TASK_CS_SEL &= ~(SPIMT_TASK_CS_SEL_TSK0CS_SEL_Msk << (task_id * 2));
  240. SPIMT[spi_dev]->TASK_CS_SEL |= ((task_attr->task_cs) << (task_id * 2));
  241. }
  242. // force trigger spi task by software if don't use externtal trigger sources
  243. static void spimt_auto_task_soft_start(int spi_dev, int task_id)
  244. {
  245. /* trigger task by software */
  246. // SPIMT[spi_dev]->AUTO_TASK[task_id].CTL &= ~SPIMT_AUTO_TASK_CTL_SOFT_ST_Msk;
  247. SPIMT[spi_dev]->AUTO_TASK[task_id].CTL |= (SPIMT_AUTO_TASK_CTL_SOFT_EN_Msk | SPIMT_AUTO_TASK_CTL_SOFT_ST_Msk);
  248. }
  249. // force trigger spi task by software if don't use externtal trigger sources
  250. static void spimt_auto_task_soft_stop(int spi_dev, int task_id)
  251. {
  252. /* trigger task by software */
  253. SPIMT[spi_dev]->AUTO_TASK[task_id].CTL &= ~(SPIMT_AUTO_TASK_CTL_SOFT_EN_Msk | SPIMT_AUTO_TASK_CTL_SOFT_ST_Msk);
  254. }
  255. //get the irq pending
  256. __sleepfunc static int spimt_auto_task_irq_get_pending(int spi_dev)
  257. {
  258. return SPIMT[spi_dev]->AUTO_TASK_IP & SPIMT[spi_dev]->AUTO_TASK_IE;
  259. }
  260. __sleepfunc static int spimt_auto_task_irq_mask(int task_id, int task_irq_type)
  261. {
  262. int irq_pending_mask = 0;
  263. switch (task_irq_type) {
  264. case SPI_TASK_IRQ_CMPLT:
  265. irq_pending_mask = 1 << (SPIMT_AUTO_TASK_IP_TSK0TCIP_Pos + task_id);
  266. break;
  267. case SPI_TASK_IRQ_HALF_CMPLT:
  268. irq_pending_mask = 1 << (SPIMT_AUTO_TASK_IP_TSK0HFIP_Pos + task_id);
  269. break;
  270. default:
  271. break;
  272. }
  273. return irq_pending_mask;
  274. }
  275. //check if the irq is pending
  276. //static int spimt_auto_task_irq_is_pending(int spi_dev, int task_id, int task_irq_type)
  277. //{
  278. // return SPIMT[spi_dev]->AUTO_TASK_IP & spimt_auto_task_irq_mask(task_id, task_irq_type);
  279. //}
  280. //clear task irq pending
  281. __sleepfunc static void spimt_auto_task_irq_clr_pending(int spi_dev, int task_id, int task_irq_type)
  282. {
  283. SPIMT[spi_dev]->AUTO_TASK_IP = spimt_auto_task_irq_mask(task_id, task_irq_type);
  284. }
  285. bool spimt_acts_transfer_ongoing(struct acts_spimt_data *data)
  286. {
  287. return spi_context_tx_on(&data->ctx) || spi_context_rx_on(&data->ctx);
  288. }
  289. static int spimt_acts_transfer_data(const struct acts_spimt_config *cfg, struct acts_spimt_data *data)
  290. {
  291. struct spi_context *ctx = &data->ctx;
  292. // if ((ctx->tx_len > SPI_BUF_LEN) || (ctx->rx_len > SPI_BUF_LEN)) {
  293. // LOG_ERR("buffer overflow");
  294. // return -1;
  295. // }
  296. LOG_DBG("tx_len %d, rx_len %d", ctx->tx_len, ctx->rx_len);
  297. if (ctx->tx_len) {
  298. spimt_write(cfg->bus_id, (uint8_t*)ctx->tx_buf, ctx->tx_len);
  299. spi_context_update_tx(ctx, 1, ctx->tx_len);
  300. }
  301. if (ctx->rx_len) {
  302. spimt_read(cfg->bus_id, (uint8_t*)ctx->rx_buf, ctx->rx_len);
  303. spi_context_update_rx(ctx, 1, ctx->rx_len);
  304. }
  305. return 0;
  306. }
  307. static int spimt_acts_configure(const struct acts_spimt_config *cfg,
  308. struct acts_spimt_data *spi,
  309. const struct spi_config *config)
  310. {
  311. if (spi_context_configured(&spi->ctx, config)) {
  312. /* Nothing to do */
  313. return 0;
  314. }
  315. // code implement by user
  316. return 0;
  317. }
  318. static int transceive(const struct device *dev,
  319. const struct spi_config *config,
  320. const struct spi_buf_set *tx_bufs,
  321. const struct spi_buf_set *rx_bufs,
  322. bool asynchronous,
  323. struct k_poll_signal *signal)
  324. {
  325. const struct acts_spimt_config *cfg =DEV_CFG(dev);
  326. struct acts_spimt_data *data = DEV_DATA(dev);
  327. int ret;
  328. spi_context_lock(&data->ctx, asynchronous, signal);
  329. /* Configure */
  330. ret = spimt_acts_configure(cfg, data, config);
  331. if (ret) {
  332. goto out;
  333. }
  334. /* Set buffers info */
  335. spi_context_buffers_setup(&data->ctx, tx_bufs, rx_bufs, 1);
  336. /* Set cs active */
  337. spimt_cs_set(cfg->bus_id, 1);
  338. /* transfer data */
  339. do {
  340. ret = spimt_acts_transfer_data(cfg, data);
  341. } while (!ret && spimt_acts_transfer_ongoing(data));
  342. /* Set cs deactive */
  343. spimt_cs_set(cfg->bus_id, 0);
  344. out:
  345. spi_context_release(&data->ctx, ret);
  346. return ret;
  347. }
  348. static int spimt_acts_transceive(const struct device *dev,
  349. const struct spi_config *config,
  350. const struct spi_buf_set *tx_bufs,
  351. const struct spi_buf_set *rx_bufs)
  352. {
  353. return transceive(dev, config, tx_bufs, rx_bufs, false, NULL);
  354. }
  355. static int spimt_acts_release(const struct device *dev,
  356. const struct spi_config *config)
  357. {
  358. struct acts_spimt_data *data = dev->data;
  359. spi_context_unlock_unconditionally(&data->ctx);
  360. return 0;
  361. }
  362. static void spimt_acts_register_callback(const struct device *dev, int task_id,
  363. spi_task_callback_t cb, void *context)
  364. {
  365. struct acts_spimt_data *data = DEV_DATA(dev);
  366. data->task_cb[task_id] = cb;
  367. data->task_cb_ctx[task_id] = context;
  368. }
  369. static uint8_t* spimt_task_buf_start(struct acts_spimt_data *data,
  370. int task_id, uint32_t addr, uint16_t len, uint8_t rd)
  371. {
  372. rbuf_t *rbuf;
  373. uint8_t *sbuf;
  374. uint16_t slen;
  375. /* check task len */
  376. if (len == 0) {
  377. LOG_ERR("task[%d] length %d error!", task_id, len);
  378. return NULL;
  379. }
  380. /* get task buf */
  381. sbuf = data->task_buf[task_id];
  382. /* check task buf len */
  383. if ((sbuf != NULL) && (len > data->task_len[task_id])) {
  384. /* free task buf */
  385. rbuf = RBUF_FR_BUF(sbuf);
  386. RB_MSG_FREE(rbuf);
  387. sbuf = NULL;
  388. data->task_buf[task_id] = NULL;
  389. data->task_len[task_id] = 0;
  390. }
  391. /* alloc task buf */
  392. if (sbuf == NULL) {
  393. slen = (len < 16) ? 16 : len;
  394. rbuf = RB_MSG_ALLOC(slen);
  395. if (rbuf != NULL) {
  396. sbuf = (uint8_t*)RBUF_TO_BUF(rbuf);
  397. data->task_buf[task_id] = sbuf;
  398. data->task_len[task_id] = slen;
  399. } else {
  400. LOG_ERR("task[%d] malloc(%d) error!", task_id, len);
  401. }
  402. }
  403. /* copy buf before writing */
  404. if (!rd && (addr != 0) && sbuf) {
  405. memcpy(sbuf, (void*)addr, len);
  406. }
  407. return sbuf;
  408. }
  409. static uint8_t* spimt_task_buf_stop(struct acts_spimt_data *data,
  410. int task_id, uint32_t addr, uint32_t len, uint8_t rd)
  411. {
  412. uint8_t *sbuf;
  413. sbuf = data->task_buf[task_id];
  414. /* copy buf after reading */
  415. if (rd && (addr != 0) && sbuf) {
  416. memcpy((void*)addr, sbuf, len);
  417. }
  418. return sbuf;
  419. }
  420. static int spimt_acts_task_start(const struct device *dev, int task_id,
  421. const spi_task_t *attr)
  422. {
  423. const struct acts_spimt_config *cfg =DEV_CFG(dev);
  424. struct acts_spimt_data *data = DEV_DATA(dev);
  425. uint8_t *buf;
  426. /* start dma buffer */
  427. buf = spimt_task_buf_start(data, task_id, attr->dma.addr, attr->dma.len, attr->ctl.rwsel);
  428. if (buf == NULL) {
  429. return -1;
  430. }
  431. /* save attr */
  432. data->task_attr[task_id] = (spi_task_t*)attr;
  433. /* select spi auto mode */
  434. spimt_auto_mode_set(cfg->bus_id);
  435. /* config spi task */
  436. spimt_auto_task_config(cfg->bus_id, task_id, attr, (uint32_t)buf);
  437. /* config ppi */
  438. if (!attr->ctl.soft) {
  439. /* disable ppi trigger */
  440. ppi_trig_src_en(attr->trig.trig, 0);
  441. /* clear ppi pending */
  442. ppi_trig_src_clr_pending(attr->trig.trig);
  443. /* config ppi trigger */
  444. ppi_task_trig_config(attr->trig.chan, attr->trig.task, attr->trig.trig);
  445. /* enable ppi trigger */
  446. ppi_trig_src_en(attr->trig.trig, attr->trig.en);
  447. } else {
  448. /* soft trigger */
  449. data->soft_started = 1;
  450. spimt_auto_task_soft_start(cfg->bus_id, task_id);
  451. }
  452. return 0;
  453. }
  454. static int spimt_acts_task_stop(const struct device *dev, int task_id)
  455. {
  456. const struct acts_spimt_config *cfg =DEV_CFG(dev);
  457. struct acts_spimt_data *data = DEV_DATA(dev);
  458. const spi_task_t *attr = data->task_attr[task_id];
  459. /* disable ppi trigger */
  460. if ((attr != NULL) && (!attr->ctl.soft)) {
  461. ppi_trig_src_en(attr->trig.trig, 0);
  462. } else {
  463. spimt_auto_task_soft_stop(cfg->bus_id, task_id);
  464. data->soft_started = 0;
  465. }
  466. /* stop dma buffer */
  467. spimt_task_buf_stop(data, task_id, attr->dma.addr, attr->dma.len, attr->ctl.rwsel);
  468. /* clear attr */
  469. data->task_attr[task_id] = NULL;
  470. return 0;
  471. }
  472. static int spimt_acts_cs_select(const struct device *dev, int cs_num)
  473. {
  474. const struct acts_spimt_config *cfg = DEV_CFG(dev);
  475. spimt_cs_select(cfg->bus_id, cs_num);
  476. return 0;
  477. }
  478. static int spimt_acts_ctl_reset(const struct device *dev)
  479. {
  480. const struct acts_spimt_config *cfg = DEV_CFG(dev);
  481. // reset spimt
  482. acts_reset_peripheral(cfg->reset_id);
  483. /* set spi mode3 (default mode) */
  484. SPIMT[cfg->bus_id]->CTL = SPIMT_CTL_SS_Msk | (SPI_DELAY << SPIMT_CTL_DELAY_Pos)
  485. | SPIMT_CTL_RXFIFO_EN_Msk | SPIMT_CTL_TXFIFO_EN_Msk
  486. | (1 << (16UL));//spimt mode 0 == 1 << 16 ; spimt mode3 == 0 << 16
  487. return 0;
  488. }
  489. static const unsigned short spi_irq_list[2] = {
  490. SPI_TASK_IRQ_CMPLT,
  491. SPI_TASK_IRQ_HALF_CMPLT,
  492. };
  493. static void spimt_acts_isr(const struct device *dev)
  494. {
  495. const struct acts_spimt_config *cfg = DEV_CFG(dev);
  496. struct acts_spimt_data *data = DEV_DATA(dev);
  497. int task_id, irq_type, len;
  498. int pending = spimt_auto_task_irq_get_pending(cfg->bus_id);
  499. int pos = find_msb_set(pending) - 1;
  500. spi_task_callback_t cb;
  501. const spi_task_t *attr;
  502. uint8_t *buf;
  503. void *ctx;
  504. while (pos >= 0) {
  505. task_id = (pos % 8);
  506. irq_type = spi_irq_list[pos / 8];
  507. attr = data->task_attr[task_id];
  508. /* clear task pending */
  509. spimt_auto_task_irq_clr_pending(cfg->bus_id, task_id, irq_type);
  510. /* clear ppi pending */
  511. ppi_trig_src_clr_pending(SPIMT0_TASK0_CIP + attr->trig.task);
  512. if (!attr->ctl.soft) {
  513. ppi_trig_src_clr_pending(attr->trig.trig);
  514. if (attr->trig.trig <= TIMER4) {
  515. //timer_clear_pd(attr->trig.trig);
  516. }
  517. }
  518. /* call handler */
  519. cb = data->task_cb[task_id];
  520. if (cb != NULL) {
  521. /* get buffer */
  522. ctx = data->task_cb_ctx[task_id];
  523. buf = data->task_buf[task_id];
  524. len = attr->dma.len / 2;
  525. switch(irq_type) {
  526. case SPI_TASK_IRQ_CMPLT:
  527. buf += len;
  528. break;
  529. case SPI_TASK_IRQ_HALF_CMPLT:
  530. break;
  531. }
  532. cb(buf, len, ctx);
  533. }
  534. /* find msb */
  535. pending = spimt_auto_task_irq_get_pending(cfg->bus_id);
  536. pos = find_msb_set(pending) - 1;
  537. }
  538. }
  539. __sleepfunc uint8_t* spi_task_get_data(int bus_id, int task_id, int trig, int *plen)
  540. {
  541. int len = 0;
  542. uint8_t *buf = NULL;
  543. int pending = spimt_auto_task_irq_get_pending(bus_id);
  544. /* clear task pending */
  545. if (pending & spimt_auto_task_irq_mask(task_id, SPI_TASK_IRQ_HALF_CMPLT)) {
  546. spimt_auto_task_irq_clr_pending(bus_id, task_id, SPI_TASK_IRQ_HALF_CMPLT);
  547. len = SPIMT[bus_id]->AUTO_TASK[task_id].DMA_CNT / 2;
  548. buf = (uint8_t *)SPIMT[bus_id]->AUTO_TASK[task_id].DMA_ADD;
  549. } else if (pending & spimt_auto_task_irq_mask(task_id, SPI_TASK_IRQ_CMPLT)) {
  550. spimt_auto_task_irq_clr_pending(bus_id, task_id, SPI_TASK_IRQ_CMPLT);
  551. len = SPIMT[bus_id]->AUTO_TASK[task_id].DMA_CNT / 2;
  552. buf = (uint8_t *)SPIMT[bus_id]->AUTO_TASK[task_id].DMA_ADD + len;
  553. }
  554. /* clear ppi pending */
  555. if (buf) {
  556. ppi_trig_src_clr_pending(SPIMT0_TASK0_CIP+task_id);
  557. if (trig >= 0) {
  558. ppi_trig_src_clr_pending(trig);
  559. }
  560. }
  561. if (plen) {
  562. *plen = len;
  563. }
  564. return buf;
  565. }
  566. #ifdef CONFIG_PM_DEVICE
  567. int spimt_pm_control(const struct device *dev, enum pm_device_action action)
  568. {
  569. struct acts_spimt_data *data = DEV_DATA(dev);
  570. switch (action) {
  571. case PM_DEVICE_ACTION_RESUME:
  572. break;
  573. case PM_DEVICE_ACTION_SUSPEND:
  574. if(data->soft_started) {
  575. LOG_ERR("spimt busy, cannot suspend");
  576. return -1;
  577. }
  578. break;
  579. default:
  580. return 0;
  581. }
  582. return 0;
  583. }
  584. #else
  585. #define spimt_pm_control NULL
  586. #endif
  587. int spimt_acts_init(const struct device *dev)
  588. {
  589. const struct acts_spimt_config *cfg = DEV_CFG(dev);
  590. struct acts_spimt_data *data = DEV_DATA(dev);
  591. // enable clock
  592. acts_clock_peripheral_enable(cfg->clock_id);
  593. // reset spimt
  594. spimt_acts_ctl_reset(dev);
  595. // set clock
  596. clk_set_rate(cfg->clock_id, SPIMT_CLOCK);
  597. spi_context_unlock_unconditionally(&data->ctx);
  598. /* irq init */
  599. cfg->irq_config_func();
  600. data->soft_started = 0;
  601. return 0;
  602. }
  603. static const struct spimt_driver_api spimt_acts_driver_api = {
  604. .spi_api = {
  605. .transceive = spimt_acts_transceive,
  606. .release = spimt_acts_release,
  607. },
  608. .register_callback = spimt_acts_register_callback,
  609. .task_start = spimt_acts_task_start,
  610. .task_stop = spimt_acts_task_stop,
  611. .cs_select = spimt_acts_cs_select,
  612. .ctl_reset = spimt_acts_ctl_reset,
  613. };
  614. #endif /*#if IS_ENABLED(CONFIG_SPIMT_0)*/
  615. #define SPIMT_ACTS_DEFINE_CONFIG(n) \
  616. static const struct device DEVICE_NAME_GET(spimt##n##_acts); \
  617. \
  618. static void spimt##n##_acts_irq_config(void) \
  619. { \
  620. IRQ_CONNECT(IRQ_ID_SPI##n##MT, CONFIG_SPIMT_##n##_IRQ_PRI, \
  621. spimt_acts_isr, \
  622. DEVICE_GET(spimt##n##_acts), 0); \
  623. irq_enable(IRQ_ID_SPI##n##MT); \
  624. } \
  625. static const struct acts_spimt_config spimt##n##_acts_config = { \
  626. .ctl_reg = SPIMT##n##_REG_BASE,\
  627. .bus_id = n,\
  628. .clock_id = CLOCK_ID_SPIMT##n,\
  629. .reset_id = RESET_ID_SPIMT##n,\
  630. .irq_config_func = spimt##n##_acts_irq_config, \
  631. }
  632. #define SPIMT_ACTS_DEFINE_DATA(n) \
  633. static __act_s2_sleep_data struct acts_spimt_data spimt##n##_acts_dev_data = { \
  634. SPI_CONTEXT_INIT_LOCK(spimt##n##_acts_dev_data, ctx), \
  635. SPI_CONTEXT_INIT_SYNC(spimt##n##_acts_dev_data, ctx), \
  636. }
  637. #define SPIMT_ACTS_DEVICE_INIT(n) \
  638. SPIMT_ACTS_DEFINE_CONFIG(n); \
  639. SPIMT_ACTS_DEFINE_DATA(n); \
  640. DEVICE_DEFINE(spimt##n##_acts, \
  641. CONFIG_SPIMT_##n##_NAME, \
  642. &spimt_acts_init, spimt_pm_control, &spimt##n##_acts_dev_data, \
  643. &spimt##n##_acts_config, POST_KERNEL, \
  644. CONFIG_SPI_INIT_PRIORITY, &spimt_acts_driver_api);
  645. #if IS_ENABLED(CONFIG_SPIMT_0)
  646. SPIMT_ACTS_DEVICE_INIT(0)
  647. #endif
  648. #if IS_ENABLED(CONFIG_SPIMT_1)
  649. SPIMT_ACTS_DEVICE_INIT(1)
  650. #endif