i2cmt_leopard.c 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799
  1. /*
  2. * Copyright (c) 2017 Actions Semiconductor Co., Ltd
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief I2CMT master driver for Actions SoC
  9. */
  10. //#define DT_DRV_COMPAT actions_acts_i2cmt
  11. #include <errno.h>
  12. #include <sys/__assert.h>
  13. #include <stdbool.h>
  14. #include <kernel.h>
  15. #include <device.h>
  16. #include <init.h>
  17. #include <drivers/ipmsg.h>
  18. #include <drivers/i2cmt.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 LOG_LEVEL CONFIG_I2C_LOG_LEVEL
  24. #include <logging/log.h>
  25. LOG_MODULE_REGISTER(i2cmt_acts);
  26. #define I2CMT_CLOCK (4000000)
  27. #define CMU_DIV_STEP_NUM (4)
  28. #define MIN_I2CMT_DIV (5)
  29. #define MAX_I2CMT_DIV (31)
  30. /**
  31. * @brief I2C Module (I2CMT)
  32. */
  33. typedef struct {
  34. volatile uint32_t CTL; /*!< (@ 0x00000000) TASK Control Register */
  35. volatile uint32_t DMA_CTL; /*!< (@ 0x00000004) TASK DMA Control Register */
  36. volatile uint32_t DMA_ADD; /*!< (@ 0x00000008) TASK DMA Address Register */
  37. volatile uint32_t DMA_CNT; /*!< (@ 0x0000000C) TASK DMA CNT Register */
  38. volatile uint32_t DMA_RC; /*!< (@ 0x00000010) TASK DMA remain counter Register */
  39. volatile uint32_t SLAVE_ADDR;
  40. volatile uint32_t RESERVED[10];
  41. } I2CMT_AUTO_TASK_Type; /*!< Size = 64 (0x40) */
  42. typedef struct { /*!< (@ 0x40088000) I2CMT Structure */
  43. volatile uint32_t CTL; /*!< (@ 0x00000000) I2C Control Register */
  44. volatile uint32_t NML_STA; /*!< (@ 0x00000004) I2C Status Register */
  45. volatile uint32_t NML_DAT; /*!< (@ 0x00000008) I2C TX/RX DATA Register */
  46. volatile uint32_t AUTO_TASK_STAT; /*!< (@ 0x0000000C) Auto task status Register */
  47. volatile uint32_t AUTO_TASK_IE; /*!< (@ 0x00000010) Auto task IRQ enable Register */
  48. volatile uint32_t AUTO_TASK_IP; /*!< (@ 0x00000014) Auto task IRQ pending Register */
  49. volatile uint32_t RESERVED1[2];
  50. volatile uint32_t AUTO_TASK_CFG; /*!< (@ 0x00000020) Auto task CFG Register */
  51. volatile uint32_t RESERVED2[55];
  52. volatile I2CMT_AUTO_TASK_Type AUTO_TASK[4]; /*!< (@ 0x00000100) TASK[0..3] Group */
  53. volatile uint32_t RESERVED3[3968];
  54. } I2CMT_Type; /*!< Size = 512 (0x200) */
  55. typedef I2CMT_Type I2CMT_ARRAYType[1]; /*!< max. 2 instances available */
  56. #define I2CMT ((I2CMT_ARRAYType*) I2CMT0_REG_BASE)
  57. /* ========================================================== CTL ========================================================== */
  58. #define I2CMT_CTL_CLKDIV_Msk (0xf80UL) /*!< CLKDIV (Bitfield-Mask: 0x1f) */
  59. #define I2CMT_CTL_MODSEL_Msk (0x40UL) /*!< MODSEL (Bitfield-Mask: 0x01) */
  60. #define I2CMT_CTL_EN_Msk (0x20UL) /*!< EN (Bitfield-Mask: 0x01) */
  61. #define I2CMT_CTL_IRQE_Msk (0x10UL) /*!< IRQE (Bitfield-Mask: 0x01) */
  62. #define I2CMT_CTL_GBCC_Msk (0xcUL) /*!< GBCC (Bitfield-Mask: 0x03) */
  63. #define I2CMT_CTL_RB_Msk (0x2UL) /*!< RB (Bitfield-Mask: 0x01) */
  64. #define I2CMT_CTL_GACK_Msk (0x1UL) /*!< GACK (Bitfield-Mask: 0x01) */
  65. #define I2CMT_CTL_CLKDIV_Pos (7UL) /*!< CLKDIV (Bit 7) */
  66. #define I2CMT_CTL_MODSEL_Pos (6UL) /*!< MODSEL (Bit 6) */
  67. #define I2CMT_CTL_EN_Pos (5UL) /*!< EN (Bit 5) */
  68. #define I2CMT_CTL_IRQE_Pos (4UL) /*!< IRQE (Bit 4) */
  69. #define I2CMT_CTL_GBCC_Pos (2UL) /*!< GBCC (Bit 2) */
  70. #define I2CMT_CTL_RB_Pos (1UL) /*!< RB (Bit 1) */
  71. #define I2CMT_CTL_GACK_Pos (0UL) /*!< GACK (Bit 0) */
  72. /* ======================================================== NML_STA ======================================================== */
  73. #define I2CMT_NML_STA_TCB_Msk (0x100UL) /*!< TCB (Bitfield-Mask: 0x01) */
  74. #define I2CMT_NML_STA_STPD_Msk (0x80UL) /*!< STPD (Bitfield-Mask: 0x01) */
  75. #define I2CMT_NML_STA_STAD_Msk (0x40UL) /*!< STAD (Bitfield-Mask: 0x01) */
  76. #define I2CMT_NML_STA_RWST_Msk (0x20UL) /*!< RWST (Bitfield-Mask: 0x01) */
  77. #define I2CMT_NML_STA_LBST_Msk (0x10UL) /*!< LBST (Bitfield-Mask: 0x01) */
  78. #define I2CMT_NML_STA_IRQP_Msk (0x8UL) /*!< IRQP (Bitfield-Mask: 0x01) */
  79. #define I2CMT_NML_STA_BBB_Msk (0x4UL) /*!< BBB (Bitfield-Mask: 0x01) */
  80. #define I2CMT_NML_STA_BEB_Msk (0x2UL) /*!< BEB (Bitfield-Mask: 0x01) */
  81. #define I2CMT_NML_STA_RACK_Msk (0x1UL) /*!< RACK (Bitfield-Mask: 0x01) */
  82. /* ===================================================== AUTO_TASK_IE ====================================================== */
  83. #define I2CMT_AUTO_TASK_IE_TSK0BERIE_Pos (12UL) /*!< TSK0BERIE (Bit 12) */
  84. #define I2CMT_AUTO_TASK_IE_TSK0NAKIE_Pos (8UL) /*!< TSK0NAKIE (Bit 8) */
  85. #define I2CMT_AUTO_TASK_IE_TSK0HFIE_Pos (4UL) /*!< TSK0HFIE (Bit 4) */
  86. #define I2CMT_AUTO_TASK_IE_TSK0TCIE_Pos (0UL) /*!< TSK0TCIE (Bit 0) */
  87. /* ===================================================== AUTO_TASK_IP ====================================================== */
  88. #define I2CMT_AUTO_TASK_IP_TSK0BERIP_Pos (12UL) /*!< TSK0BERIP (Bit 12) */
  89. #define I2CMT_AUTO_TASK_IP_TSK0NAKIP_Pos (8UL) /*!< TSK0NAKIP (Bit 8) */
  90. #define I2CMT_AUTO_TASK_IP_TSK0HFIP_Pos (4UL) /*!< TSK0HFIP (Bit 4) */
  91. #define I2CMT_AUTO_TASK_IP_TSK0TCIP_Pos (0UL) /*!< TSK0TCIP (Bit 0) */
  92. /* ========================================================== CTL ========================================================== */
  93. #define I2CMT_AUTO_TASK_CTL_SOFT_ST_Msk (0x80000000UL) /*!< SOFT_ST (Bitfield-Mask: 0x01) */
  94. /* ======================================================== DMA_CTL ======================================================== */
  95. #define I2CMT_AUTO_TASK_DMA_CTL_DMARELD_Pos (1UL) /*!< DMARELD (Bit 1) */
  96. #define I2CMT_AUTO_TASK_DMA_CTL_DMASTART_Msk (0x1UL) /*!< DMASTART (Bitfield-Mask: 0x01) */
  97. // I2C BUS STAT
  98. #define MASK (I2CMT_CTL_GBCC_Msk | I2CMT_CTL_RB_Msk | I2CMT_CTL_GACK_Msk)
  99. #define START (1 << I2CMT_CTL_GBCC_Pos)
  100. #define STOP (2 << I2CMT_CTL_GBCC_Pos)
  101. #define RESTA (3 << I2CMT_CTL_GBCC_Pos)
  102. #define REBUS I2CMT_CTL_RB_Msk
  103. #define ACK 0
  104. #define NACK I2CMT_CTL_GACK_Msk
  105. enum i2cmt_state {
  106. STATE_OK = 0,
  107. STATE_ERR,
  108. };
  109. struct acts_i2cmt_config {
  110. uint32_t ctl_reg;
  111. uint32_t clk_freq;
  112. uint16_t max_task_len;
  113. uint8_t bus_id;
  114. uint8_t clock_id;
  115. uint8_t reset_id;
  116. void (*irq_config_func)(void);
  117. };
  118. /* Device run time data */
  119. struct acts_i2cmt_data {
  120. struct k_mutex mutex;
  121. struct k_sem complete_sem;
  122. enum i2cmt_state state;
  123. uint8_t suspended;
  124. uint8_t soft_started;
  125. /* task data */
  126. i2c_task_callback_t task_cb[I2C_TASK_NUM];
  127. void *task_cb_ctx[I2C_TASK_NUM];
  128. i2c_task_t *task_attr[I2C_TASK_NUM];
  129. uint8_t *task_buf[I2C_TASK_NUM];
  130. uint32_t task_len[I2C_TASK_NUM];
  131. };
  132. #define DEV_NAME(dev) ((dev)->name)
  133. #define DEV_CFG(dev) \
  134. ((const struct acts_i2cmt_config *const)(dev)->config)
  135. #define DEV_DATA(dev) \
  136. ((struct acts_i2cmt_data *const)(dev)->data)
  137. static void i2cmt_set_rate(int i2c_dev, int rate)
  138. {
  139. unsigned int val, cmu_div, i2cmt_div, div, real_rate, cmu_div_reg;
  140. uint8_t cmu_clksrc_div[CMU_DIV_STEP_NUM] = {1, 2, 4, 8};
  141. div = (I2CMT_CLOCK + rate * 2 - 1) / (rate * 2);
  142. /* foreach cmu_clksrc_div and match proper i2cmt_div */
  143. for (int i = 0; i < CMU_DIV_STEP_NUM; i++) {
  144. cmu_div_reg = i;
  145. cmu_div = cmu_clksrc_div[i];
  146. i2cmt_div = (div + cmu_div - 1) / cmu_div;
  147. if ((i2cmt_div >= MIN_I2CMT_DIV) && (i2cmt_div <= MAX_I2CMT_DIV)) {
  148. break;
  149. } else if (i2cmt_div < MIN_I2CMT_DIV) {
  150. i2cmt_div = MIN_I2CMT_DIV;
  151. break;
  152. } else {
  153. i2cmt_div = MAX_I2CMT_DIV;
  154. }
  155. }
  156. real_rate = I2CMT_CLOCK / cmu_div / (2 * i2cmt_div);
  157. LOG_INF("I2CMT%d expected rate %dHz, real rate %dHz, cmudiv=%d, i2cmtdiv=%d",
  158. i2c_dev, rate, real_rate, cmu_div, i2cmt_div);
  159. /* fill in RC4M clock src and div */
  160. val = (0 << 8) | (cmu_div_reg << 0);
  161. sys_write32(val, CMU_I2CMT0CLK + i2c_dev * 4);
  162. I2CMT[i2c_dev]->CTL &= ~I2CMT_CTL_CLKDIV_Msk;
  163. I2CMT[i2c_dev]->CTL |= i2cmt_div << I2CMT_CTL_CLKDIV_Pos;
  164. }
  165. // switch to auto mode
  166. static void i2cmt_auto_mode_set(int i2c_dev)
  167. {
  168. /* clear task IRQ pending */
  169. I2CMT[i2c_dev]->AUTO_TASK_IP = I2CMT[i2c_dev]->AUTO_TASK_IP;
  170. /* set auto mode */
  171. I2CMT[i2c_dev]->CTL |= I2CMT_CTL_MODSEL_Msk;
  172. }
  173. // configure i2c task
  174. static void i2cmt_auto_task_config(int i2c_dev, int task_id, i2c_task_t *task_attr, uint32_t addr)
  175. {
  176. uint8_t *pdata = (uint8_t *)&task_attr->ctl;
  177. volatile uint32_t ctl = *(volatile unsigned int*)pdata;
  178. pdata = (uint8_t *)&task_attr->slavedev;
  179. volatile uint32_t sdev = *(volatile unsigned int*)pdata;
  180. /* config task dma first */
  181. if (task_attr->ctl.swen && (!task_attr->slavedev.rwsel)) {
  182. I2CMT[i2c_dev]->AUTO_TASK[task_id].DMA_CTL &= ~(I2CMT_AUTO_TASK_DMA_CTL_DMASTART_Msk);
  183. } else {
  184. I2CMT[i2c_dev]->AUTO_TASK[task_id].DMA_ADD = addr;//task_attr->dma.addr;
  185. I2CMT[i2c_dev]->AUTO_TASK[task_id].DMA_CNT = task_attr->dma.len;
  186. if (task_attr->dma.len > 0) {
  187. I2CMT[i2c_dev]->AUTO_TASK[task_id].DMA_CTL = I2CMT_AUTO_TASK_DMA_CTL_DMASTART_Msk
  188. | (task_attr->dma.reload << I2CMT_AUTO_TASK_DMA_CTL_DMARELD_Pos);
  189. }
  190. }
  191. if(task_attr->irq_type & I2C_TASK_IRQ_CMPLT) {
  192. /* enable task DMA transmission complete IRQ */
  193. I2CMT[i2c_dev]->AUTO_TASK_IE |= 1 << (I2CMT_AUTO_TASK_IE_TSK0TCIE_Pos + task_id);
  194. } else {
  195. /* disable task DMA transmission complete IRQ */
  196. I2CMT[i2c_dev]->AUTO_TASK_IE &= ~(1 << (I2CMT_AUTO_TASK_IE_TSK0TCIE_Pos + task_id));
  197. }
  198. if(task_attr->irq_type & I2C_TASK_IRQ_HALF_CMPLT) {
  199. /* enable task DMA Half transmission complete IRQ */
  200. I2CMT[i2c_dev]->AUTO_TASK_IE |= 1 << (I2CMT_AUTO_TASK_IE_TSK0HFIE_Pos + task_id);
  201. } else {
  202. /* disable task DMA Half transmission complete IRQ */
  203. I2CMT[i2c_dev]->AUTO_TASK_IE &= ~(1 << (I2CMT_AUTO_TASK_IE_TSK0HFIE_Pos + task_id));
  204. }
  205. if(task_attr->irq_type & I2C_TASK_IRQ_NACK) {
  206. /* enable task DMA receive NACK IRQ */
  207. I2CMT[i2c_dev]->AUTO_TASK_IE |= 1 << (I2CMT_AUTO_TASK_IE_TSK0NAKIE_Pos + task_id);
  208. } else {
  209. /* disable task DMA receive NACK IRQ */
  210. I2CMT[i2c_dev]->AUTO_TASK_IE &= ~(1 << (I2CMT_AUTO_TASK_IE_TSK0NAKIE_Pos + task_id));
  211. }
  212. if(task_attr->irq_type & I2C_TASK_IRQ_BUS_ERROR) {
  213. /* enable task DMA bus error IRQ */
  214. I2CMT[i2c_dev]->AUTO_TASK_IE |= 1 << (I2CMT_AUTO_TASK_IE_TSK0BERIE_Pos + task_id);
  215. } else {
  216. /* disable task DMA bus error IRQ */
  217. I2CMT[i2c_dev]->AUTO_TASK_IE &= ~(1 << (I2CMT_AUTO_TASK_IE_TSK0BERIE_Pos + task_id));
  218. }
  219. /* config task ctl*/
  220. I2CMT[i2c_dev]->AUTO_TASK[task_id].CTL = (ctl & ~I2CMT_AUTO_TASK_CTL_SOFT_ST_Msk);
  221. I2CMT[i2c_dev]->AUTO_TASK[task_id].SLAVE_ADDR = sdev;
  222. }
  223. // configure i2c stop
  224. static void i2cmt_auto_task_stop_en(int i2c_dev, int task_id, int en)
  225. {
  226. if (en) {
  227. I2CMT[i2c_dev]->AUTO_TASK_CFG &= ~(1 << task_id);
  228. } else {
  229. I2CMT[i2c_dev]->AUTO_TASK_CFG |= (1 << task_id);
  230. }
  231. }
  232. // force trigger i2c task by software if don't use externtal trigger sources
  233. static void i2cmt_auto_task_soft_start(int i2c_dev, int task_id)
  234. {
  235. /* trigger task by software */
  236. I2CMT[i2c_dev]->AUTO_TASK[task_id].CTL &= ~I2CMT_AUTO_TASK_CTL_SOFT_ST_Msk;
  237. I2CMT[i2c_dev]->AUTO_TASK[task_id].CTL |= I2CMT_AUTO_TASK_CTL_SOFT_ST_Msk;
  238. }
  239. // force trigger i2c task by software if don't use externtal trigger sources
  240. static void i2cmt_auto_task_soft_stop(int i2c_dev, int task_id)
  241. {
  242. /* stop dma */
  243. I2CMT[i2c_dev]->AUTO_TASK[task_id].DMA_CTL = 0;
  244. /* trigger task by software */
  245. I2CMT[i2c_dev]->AUTO_TASK[task_id].CTL &= ~I2CMT_AUTO_TASK_CTL_SOFT_ST_Msk;
  246. }
  247. //get the irq pending
  248. static __sleepfunc int i2cmt_auto_task_irq_get_pending(int i2c_dev)
  249. {
  250. return I2CMT[i2c_dev]->AUTO_TASK_IP & I2CMT[i2c_dev]->AUTO_TASK_IE;
  251. }
  252. static __sleepfunc int i2cmt_auto_task_irq_mask(int task_id, int task_irq_type)
  253. {
  254. int irq_pending_mask = 0;
  255. switch (task_irq_type) {
  256. case I2C_TASK_IRQ_CMPLT:
  257. irq_pending_mask = 1 << (I2CMT_AUTO_TASK_IP_TSK0TCIP_Pos + task_id);
  258. break;
  259. case I2C_TASK_IRQ_HALF_CMPLT:
  260. irq_pending_mask = 1 << (I2CMT_AUTO_TASK_IP_TSK0HFIP_Pos + task_id);
  261. break;
  262. case I2C_TASK_IRQ_NACK:
  263. irq_pending_mask = 1 << (I2CMT_AUTO_TASK_IP_TSK0NAKIP_Pos + task_id);
  264. break;
  265. case I2C_TASK_IRQ_BUS_ERROR:
  266. irq_pending_mask = 1 << (I2CMT_AUTO_TASK_IP_TSK0BERIP_Pos + task_id);
  267. break;
  268. default:
  269. break;
  270. }
  271. return irq_pending_mask;
  272. }
  273. //check if the irq is pending
  274. //static __sleepfunc int i2cmt_auto_task_irq_is_pending(int i2c_dev, int task_id, int task_irq_type)
  275. //{
  276. // return I2CMT[i2c_dev]->AUTO_TASK_IP & i2cmt_auto_task_irq_mask(task_id, task_irq_type);
  277. //}
  278. //clear task irq pending
  279. static __sleepfunc void i2cmt_auto_task_irq_clr_pending(int i2c_dev, int task_id, int task_irq_type)
  280. {
  281. I2CMT[i2c_dev]->AUTO_TASK_IP = i2cmt_auto_task_irq_mask(task_id, task_irq_type);
  282. }
  283. static void i2cmt_acts_register_callback(struct device *dev, int task_id,
  284. i2c_task_callback_t cb, void *context)
  285. {
  286. struct acts_i2cmt_data *data = DEV_DATA(dev);
  287. data->task_cb[task_id] = cb;
  288. data->task_cb_ctx[task_id] = context;
  289. }
  290. static uint8_t* i2cmt_task_buf_start(const struct device *dev,
  291. int task_id, uint32_t addr, uint16_t len, uint8_t rd)
  292. {
  293. const struct acts_i2cmt_config *cfg =DEV_CFG(dev);
  294. struct acts_i2cmt_data *data = DEV_DATA(dev);
  295. rbuf_t *rbuf;
  296. uint8_t *sbuf;
  297. uint16_t slen;
  298. /* check task len */
  299. if ((len == 0) || (len > cfg->max_task_len)) {
  300. LOG_ERR("task[%d] length %d error!", task_id, len);
  301. return NULL;
  302. }
  303. /* get task buf */
  304. sbuf = data->task_buf[task_id];
  305. /* check task buf len */
  306. if ((sbuf != NULL) && (len > data->task_len[task_id])) {
  307. /* free task buf */
  308. rbuf = RBUF_FR_BUF(sbuf);
  309. RB_MSG_FREE(rbuf);
  310. sbuf = NULL;
  311. data->task_buf[task_id] = NULL;
  312. data->task_len[task_id] = 0;
  313. }
  314. /* alloc task buf */
  315. if (sbuf == NULL) {
  316. if (task_id == I2C_TASK_NUM - 1) {
  317. slen = cfg->max_task_len;
  318. } else {
  319. slen = (len < 16) ? 16 : len;
  320. }
  321. rbuf = RB_MSG_ALLOC(slen);
  322. if (rbuf != NULL) {
  323. sbuf = (uint8_t*)RBUF_TO_BUF(rbuf);
  324. data->task_buf[task_id] = sbuf;
  325. data->task_len[task_id] = slen;
  326. } else {
  327. LOG_ERR("task[%d] malloc(%d) error!", task_id, len);
  328. }
  329. }
  330. /* copy buf before writing */
  331. if (!rd && (addr != 0) && sbuf) {
  332. memcpy(sbuf, (void*)addr, len);
  333. }
  334. return sbuf;
  335. }
  336. static uint8_t* i2cmt_task_buf_stop(struct acts_i2cmt_data *data,
  337. int task_id, uint32_t addr, uint32_t len, uint8_t rd)
  338. {
  339. uint8_t *sbuf;
  340. sbuf = data->task_buf[task_id];
  341. /* copy buf after reading */
  342. if (rd && (addr != 0) && sbuf) {
  343. memcpy((void*)addr, sbuf, len);
  344. }
  345. return sbuf;
  346. }
  347. static int i2cmt_acts_task_start(struct device *dev, int task_id,
  348. const i2c_task_t *attr)
  349. {
  350. const struct acts_i2cmt_config *cfg =DEV_CFG(dev);
  351. struct acts_i2cmt_data *data = DEV_DATA(dev);
  352. uint8_t *buf;
  353. /* start dma buffer */
  354. buf = i2cmt_task_buf_start(dev, task_id, attr->dma.addr, attr->dma.len, attr->slavedev.rwsel);
  355. if (buf == NULL) {
  356. return -1;
  357. }
  358. /* save attr */
  359. data->task_attr[task_id] = (i2c_task_t*)attr;
  360. /* select i2c auto mode */
  361. i2cmt_auto_mode_set(cfg->bus_id);
  362. /* config i2c task */
  363. i2cmt_auto_task_config(cfg->bus_id, task_id, (i2c_task_t*)attr, (uint32_t)buf);
  364. /* config ppi */
  365. if (!attr->ctl.soft) {
  366. /* disable ppi trigger */
  367. ppi_trig_src_en(attr->trig.trig, 0);
  368. /* clear ppi pending */
  369. ppi_trig_src_clr_pending(attr->trig.trig);
  370. /* config ppi trigger */
  371. ppi_task_trig_config(attr->trig.chan, attr->trig.task, attr->trig.trig);
  372. /* enable ppi trigger */
  373. ppi_trig_src_en(attr->trig.trig, attr->trig.en);
  374. } else {
  375. /* soft trigger */
  376. data->soft_started = 1;
  377. i2cmt_auto_task_soft_start(cfg->bus_id, task_id);
  378. }
  379. return 0;
  380. }
  381. static int i2cmt_acts_task_stop(struct device *dev, int task_id)
  382. {
  383. const struct acts_i2cmt_config *cfg =DEV_CFG(dev);
  384. struct acts_i2cmt_data *data = DEV_DATA(dev);
  385. const i2c_task_t *attr = data->task_attr[task_id];
  386. /* check attr */
  387. if (attr == NULL) {
  388. return 0;
  389. }
  390. /* disable ppi trigger */
  391. if ((attr != NULL) && (!attr->ctl.soft)) {
  392. ppi_trig_src_en(attr->trig.trig, 0);
  393. /* disable dma transmission */
  394. I2CMT[cfg->bus_id]->AUTO_TASK[task_id].DMA_CTL = 0;
  395. } else {
  396. i2cmt_auto_task_soft_stop(cfg->bus_id, task_id);
  397. data->soft_started = 0;
  398. }
  399. /* stop dma buffer */
  400. i2cmt_task_buf_stop(data, task_id, attr->dma.addr, attr->dma.len, attr->slavedev.rwsel);
  401. /* clear attr */
  402. data->task_attr[task_id] = NULL;
  403. return 0;
  404. }
  405. static int i2cmt_acts_ctl_reset(const struct device *dev)
  406. {
  407. const struct acts_i2cmt_config *cfg = DEV_CFG(dev);
  408. // reset i2cmt
  409. acts_reset_peripheral(cfg->reset_id);
  410. /* enable i2c */
  411. I2CMT[cfg->bus_id]->CTL |= I2CMT_CTL_EN_Msk;
  412. return 0;
  413. }
  414. #if IS_ENABLED(CONFIG_I2CMT_0) || IS_ENABLED(CONFIG_I2CMT_1)
  415. static const unsigned short i2c_irq_list[4] = {
  416. I2C_TASK_IRQ_CMPLT,
  417. I2C_TASK_IRQ_HALF_CMPLT,
  418. I2C_TASK_IRQ_NACK,
  419. I2C_TASK_IRQ_BUS_ERROR,
  420. };
  421. static void i2cmt_acts_isr(struct device *dev)
  422. {
  423. const struct acts_i2cmt_config *cfg = DEV_CFG(dev);;
  424. struct acts_i2cmt_data *data = DEV_DATA(dev);
  425. int task_id, irq_type, len;
  426. int pending = i2cmt_auto_task_irq_get_pending(cfg->bus_id);
  427. int pos = find_msb_set(pending) - 1;
  428. i2c_task_callback_t cb;
  429. const i2c_task_t *attr;
  430. uint8_t *buf;
  431. void *ctx;
  432. while (pos >= 0) {
  433. task_id = (pos % 4);
  434. irq_type = i2c_irq_list[pos / 4];
  435. attr = data->task_attr[task_id];
  436. /* clear task pending */
  437. i2cmt_auto_task_irq_clr_pending(cfg->bus_id, task_id, irq_type);
  438. /* clear ppi pending */
  439. ppi_trig_src_clr_pending(SPIMT0_TASK0_CIP + attr->trig.task);
  440. if (!attr->ctl.soft) {
  441. ppi_trig_src_clr_pending(attr->trig.trig);
  442. if (attr->trig.trig <= TIMER4) {
  443. //timer_clear_pd(attr->trig.trig);
  444. }
  445. }
  446. /* call handler */
  447. cb = data->task_cb[task_id];
  448. if (cb != NULL) {
  449. /* get buffer */
  450. ctx = data->task_cb_ctx[task_id];
  451. buf = data->task_buf[task_id];
  452. len = attr->dma.len / 2;
  453. switch(irq_type) {
  454. case I2C_TASK_IRQ_CMPLT:
  455. buf += len;
  456. break;
  457. case I2C_TASK_IRQ_HALF_CMPLT:
  458. break;
  459. case I2C_TASK_IRQ_NACK:
  460. case I2C_TASK_IRQ_BUS_ERROR:
  461. buf = NULL;
  462. break;
  463. }
  464. cb(buf, len, ctx);
  465. }
  466. /* find msb */
  467. pending = i2cmt_auto_task_irq_get_pending(cfg->bus_id);
  468. pos = find_msb_set(pending) - 1;
  469. }
  470. }
  471. #endif
  472. __sleepfunc uint8_t* i2c_task_get_data(int bus_id, int task_id, int trig, int *plen)
  473. {
  474. int len = 0;
  475. uint8_t *buf = NULL;
  476. int pending = i2cmt_auto_task_irq_get_pending(bus_id);
  477. /* clear task pending */
  478. if (pending & i2cmt_auto_task_irq_mask(task_id, I2C_TASK_IRQ_HALF_CMPLT)) {
  479. i2cmt_auto_task_irq_clr_pending(bus_id, task_id, I2C_TASK_IRQ_HALF_CMPLT);
  480. len = I2CMT[bus_id]->AUTO_TASK[task_id].DMA_CNT / 2;
  481. buf = (uint8_t *)I2CMT[bus_id]->AUTO_TASK[task_id].DMA_ADD;
  482. } else if (pending & i2cmt_auto_task_irq_mask(task_id, I2C_TASK_IRQ_CMPLT)) {
  483. i2cmt_auto_task_irq_clr_pending(bus_id, task_id, I2C_TASK_IRQ_CMPLT);
  484. len = I2CMT[bus_id]->AUTO_TASK[task_id].DMA_CNT / 2;
  485. buf = (uint8_t *)I2CMT[bus_id]->AUTO_TASK[task_id].DMA_ADD + len;
  486. }
  487. /* clear ppi pending */
  488. if (buf) {
  489. ppi_trig_src_clr_pending(I2CMT0_TASK0_CIP+task_id);
  490. if (trig >= 0) {
  491. ppi_trig_src_clr_pending(trig);
  492. }
  493. }
  494. if (plen) {
  495. *plen = len;
  496. }
  497. return buf;
  498. }
  499. static uint32_t i2cmt_get_timeout(const struct device *dev, int len)
  500. {
  501. const struct acts_i2cmt_config *cfg = DEV_CFG(dev);
  502. /* > 50ms */
  503. return ((len << 15) / cfg->clk_freq + 50);
  504. }
  505. static void i2cmt_xfer_callback(unsigned char *buf, int len, void *ctx)
  506. {
  507. struct acts_i2cmt_data *data = DEV_DATA((const struct device *)ctx);
  508. // save status
  509. data->state = (buf != NULL) ? STATE_OK : STATE_ERR;
  510. // complete xfer
  511. k_sem_give(&data->complete_sem);
  512. }
  513. static int i2cmt_xfer_task(struct device *dev, uint32_t task_id, i2c_task_t *task)
  514. {
  515. int ret;
  516. uint32_t pre_time, cur_time;
  517. uint32_t timeout_ms = i2cmt_get_timeout(dev, task->dma.len + 3);
  518. const struct acts_i2cmt_config *cfg = DEV_CFG(dev);
  519. struct acts_i2cmt_data *data = DEV_DATA(dev);
  520. /* reset sem */
  521. if (!data->suspended) {
  522. k_sem_reset(&data->complete_sem);
  523. }
  524. /* start task */
  525. i2cmt_acts_register_callback(dev, task_id, i2cmt_xfer_callback, (void*)dev);
  526. ret = i2cmt_acts_task_start(dev, task_id, task);
  527. if(ret) {
  528. return -1;
  529. }
  530. /* wait task */
  531. if (data->suspended) {
  532. pre_time = (uint32_t)soc_sys_uptime_get();
  533. while(!i2c_task_get_data(cfg->bus_id, task_id, -1, NULL)) {
  534. cur_time = (uint32_t)soc_sys_uptime_get();
  535. // timeout
  536. if ((cur_time - pre_time) > timeout_ms) {
  537. data->state = STATE_ERR;
  538. break;
  539. }
  540. }
  541. } else {
  542. ret = k_sem_take(&data->complete_sem, K_MSEC(timeout_ms));
  543. if (ret) {
  544. data->state = STATE_ERR;
  545. LOG_ERR("i2cmt[%d] timeout(%d ms)", cfg->bus_id, timeout_ms);
  546. }
  547. }
  548. /* stop task */
  549. i2cmt_acts_task_stop(dev, task_id);
  550. i2cmt_acts_register_callback(dev, task_id, NULL, NULL);
  551. return data->state;
  552. }
  553. static int i2cmt_acts_configure(const struct device *dev, uint32_t config)
  554. {
  555. LOG_ERR("Change I2CMT clock is not supported");
  556. return 0;
  557. }
  558. static int i2cmt_acts_transfer(const struct device *dev, struct i2c_msg *msgs,
  559. uint8_t num_msgs, uint16_t addr)
  560. {
  561. const struct acts_i2cmt_config *cfg = DEV_CFG(dev);
  562. struct acts_i2cmt_data *data = DEV_DATA(dev);
  563. int ret = 0, i;
  564. i2c_task_t i2c_task = {0};
  565. if (!data->suspended) {
  566. k_mutex_lock(&data->mutex, K_FOREVER);
  567. }
  568. for (i = 0; i < num_msgs; i ++) {
  569. /* config task */
  570. i2c_task.irq_type = I2C_TASK_IRQ_CMPLT | I2C_TASK_IRQ_NACK;
  571. i2c_task.slavedev.rwsel = ((msgs[i].flags & I2C_MSG_RW_MASK) == I2C_MSG_READ);
  572. i2c_task.slavedev.sdevaddr = addr;
  573. i2c_task.ctl.rdlen_wsdat = msgs[i].len;
  574. i2c_task.ctl.tdataddr = 1,
  575. i2c_task.ctl.soft = 1;
  576. i2c_task.dma.reload = 0;
  577. i2c_task.dma.addr = (uint32_t)msgs[i].buf;
  578. i2c_task.dma.len = msgs[i].len;
  579. /* config stop */
  580. i2cmt_auto_task_stop_en(cfg->bus_id, I2C_TASK_NUM - 1, (msgs[i].flags & I2C_MSG_STOP));
  581. /* xfer task */
  582. ret = i2cmt_xfer_task((struct device *)dev, I2C_TASK_NUM - 1, &i2c_task);
  583. if (ret != STATE_OK) {
  584. break;
  585. }
  586. }
  587. if (!data->suspended) {
  588. k_mutex_unlock(&data->mutex);
  589. }
  590. return (ret != STATE_OK);
  591. }
  592. #ifdef CONFIG_PM_DEVICE
  593. int i2cmt_pm_control(const struct device *dev, enum pm_device_action action)
  594. {
  595. struct acts_i2cmt_data *data = DEV_DATA(dev);
  596. switch (action) {
  597. case PM_DEVICE_ACTION_RESUME:
  598. data->suspended = 0;
  599. break;
  600. case PM_DEVICE_ACTION_SUSPEND:
  601. if((data->mutex.lock_count != 0U) || data->soft_started) {
  602. LOG_ERR("i2cmt busy, cannot suspend");
  603. return -1;
  604. }
  605. data->suspended = 1;
  606. break;
  607. default:
  608. return 0;
  609. }
  610. return 0;
  611. }
  612. #else
  613. #define i2cmt_pm_control NULL
  614. #endif
  615. int i2cmt_acts_init(const struct device *dev)
  616. {
  617. const struct acts_i2cmt_config *cfg = DEV_CFG(dev);;
  618. struct acts_i2cmt_data *data = DEV_DATA(dev);
  619. // enable clock
  620. acts_clock_peripheral_enable(cfg->clock_id);
  621. // reset i2cmt
  622. i2cmt_acts_ctl_reset(dev);
  623. // set clock
  624. i2cmt_set_rate(cfg->bus_id, cfg->clk_freq);
  625. /* irq init */
  626. cfg->irq_config_func();
  627. /* data init */
  628. k_mutex_init(&data->mutex);
  629. k_sem_init(&data->complete_sem, 0, UINT_MAX);
  630. data->state = STATE_OK;
  631. data->suspended = 0;
  632. data->soft_started = 0;
  633. return 0;
  634. }
  635. const struct i2cmt_driver_api i2cmt_acts_driver_api = {
  636. .i2c_api = {
  637. .configure = i2cmt_acts_configure,
  638. .transfer = i2cmt_acts_transfer,
  639. },
  640. .register_callback = i2cmt_acts_register_callback,
  641. .task_start = i2cmt_acts_task_start,
  642. .task_stop = i2cmt_acts_task_stop,
  643. .ctl_reset = i2cmt_acts_ctl_reset,
  644. };
  645. #define I2CMT_ACTS_DEFINE_CONFIG(n) \
  646. static const struct device DEVICE_NAME_GET(i2cmt##n##_acts); \
  647. \
  648. static void i2cmt##n##_acts_irq_config(void) \
  649. { \
  650. IRQ_CONNECT(IRQ_ID_IIC##n##MT, CONFIG_I2CMT_##n##_IRQ_PRI, \
  651. i2cmt_acts_isr, \
  652. DEVICE_GET(i2cmt##n##_acts), 0); \
  653. irq_enable(IRQ_ID_IIC##n##MT); \
  654. } \
  655. static const struct acts_i2cmt_config i2cmt##n##_acts_config = { \
  656. .ctl_reg = I2CMT##n##_REG_BASE,\
  657. .clk_freq = CONFIG_I2CMT_##n##_CLK_FREQ, \
  658. .bus_id = n,\
  659. .clock_id = CLOCK_ID_I2CMT##n,\
  660. .reset_id = RESET_ID_I2CMT##n,\
  661. .irq_config_func = i2cmt##n##_acts_irq_config, \
  662. .max_task_len = CONFIG_I2CMT_##n##_MAX_TASK_LEN, \
  663. }
  664. #define I2CMT_ACTS_DEFINE_DATA(n) \
  665. static __act_s2_sleep_data struct acts_i2cmt_data i2cmt##n##_acts_dev_data;
  666. #define I2CMT_ACTS_DEVICE_INIT(n) \
  667. I2CMT_ACTS_DEFINE_CONFIG(n); \
  668. I2CMT_ACTS_DEFINE_DATA(n); \
  669. DEVICE_DEFINE(i2cmt##n##_acts, \
  670. CONFIG_I2CMT_##n##_NAME, \
  671. i2cmt_acts_init, i2cmt_pm_control, &i2cmt##n##_acts_dev_data, \
  672. &i2cmt##n##_acts_config, POST_KERNEL, \
  673. CONFIG_I2C_INIT_PRIORITY, &i2cmt_acts_driver_api);
  674. #if IS_ENABLED(CONFIG_I2CMT_0)
  675. I2CMT_ACTS_DEVICE_INIT(0)
  676. #endif
  677. #if IS_ENABLED(CONFIG_I2CMT_1)
  678. I2CMT_ACTS_DEVICE_INIT(1)
  679. #endif