phy_audio_i2stx.c 56 KB

12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538
  1. /*
  2. * Copyright (c) 2020 Actions Semiconductor Co., Ltd
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief Audio I2STX physical implementation
  9. */
  10. /*
  11. * Features
  12. * - Support master and slave mode
  13. * - Support I2S 2ch format
  14. * - I2STX FIFO(32 x 24bits level) and DAC FIFO(8 x 24 bits level)
  15. * - Support 3 format: left-justified format, right-justified format, I2S format
  16. * - Support 16/20/24 effective data width; BCLK support 32/64fs, and MCLK=4BCLK
  17. * - Support sample rate auto detect in slave mode
  18. * - Sample rate support 8k/12k/11.025k/16k/22.05k/24k/32k/44.1k/48k/88.2k/96k/192k
  19. */
  20. #include <kernel.h>
  21. #include <device.h>
  22. #include <string.h>
  23. #include <errno.h>
  24. #include <soc.h>
  25. #include <board_cfg.h>
  26. #include "../phy_audio_common.h"
  27. #include <drivers/audio/audio_out.h>
  28. #include <board.h>
  29. #ifdef CONFIG_CFG_DRV
  30. #include <config.h>
  31. #include <drivers/cfg_drv/driver_config.h>
  32. #endif
  33. #include <logging/log.h>
  34. LOG_MODULE_REGISTER(i2stx0, LOG_LEVEL_DBG);
  35. // #define DEBUG_I2STX_TO_RX
  36. /***************************************************************************************************
  37. * I2STX_CTL
  38. */
  39. #define I2ST0_CTL_TX_ER_PD BIT(31) /* PD */
  40. #define I2ST0_CTL_TX_ER_IE BIT(30) /* IE */
  41. #define I2ST0_CTL_PCM_SYNC BIT(26) /* PCM frame select*/
  42. #define I2ST0_CTL_PCM_CHAN_SHIFT (24)
  43. #define I2ST0_CTL_PCM_CHAN_MASK (0x3 << I2ST0_CTL_PCM_CHAN_SHIFT)
  44. #define I2ST0_CTL_PCM_CHAN(x) ((x) << I2ST0_CTL_PCM_CHAN_SHIFT) /* pcm slot select*/
  45. #define I2ST0_CTL_TXD_DELAY_SHIFT (21) /* tx data output delay 0: 2 mclk after bclk rising edge; 1: 3 mclk; 2: 4 mclk; 3" 5 mclk */
  46. #define I2ST0_CTL_TXD_DELAY_MASK (0x3 << I2ST0_CTL_TXD_DELAY_SHIFT)
  47. #define I2ST0_CTL_TXD_DELAY(x) ((x) << I2ST0_CTL_TXD_DELAY_SHIFT)
  48. #define I2ST0_CTL_TDMTX_CHAN BIT(20) /* 4 channel or 8 channel */
  49. #define I2ST0_CTL_TDMTX_MODE BIT(19) /* (TDM-A) I2S format or (TDM-B) left-justified format */
  50. #define I2ST0_CTL_TDMTX_SYNC_SHIFT (17) /* the type of LRCLK setting at the begining of the data frame */
  51. #define I2ST0_CTL_TDMTX_SYNC_MASK (0x3 << I2ST0_CTL_TDMTX_SYNC_SHIFT)
  52. #define I2ST0_CTL_TDMTX_SYNC(x) ((x) << I2ST0_CTL_TDMTX_SYNC_SHIFT)
  53. // #define I2ST0_CTL_MULT_DEVICE BIT(16) /* multi device selection */
  54. #define I2ST0_CTL_LPEN0 BIT(8) /* I2STX and I2SRX data loopback enable */
  55. #define I2ST0_CTL_PCMTX0_EN BIT(6) /* PCM TX ENABEL */
  56. #define I2ST0_CTL_TXWIDTH_SHIFT (4) /* effective data width */
  57. #define I2ST0_CTL_TXWIDTH_MASK (0x3 << I2ST0_CTL_TXWIDTH_SHIFT)
  58. #define I2ST0_CTL_TXWIDTH(x) ((x) << I2ST0_CTL_TXWIDTH_SHIFT)
  59. #define I2ST0_CTL_TXBCLKSET BIT(3) /* bits of BCLK with LRCLK */
  60. #define I2ST0_CTL_TXMODELSEL_SHIFT (1) /* I2S transfer format select */
  61. #define I2ST0_CTL_TXMODELSEL_MASK (0x3 << I2ST0_CTL_TXMODELSEL_SHIFT)
  62. #define I2ST0_CTL_TXMODELSEL(x) ((x) << I2ST0_CTL_TXMODELSEL_SHIFT)
  63. #define I2ST0_CTL_TXEN BIT(0) /* I2S TX Enable */
  64. /***************************************************************************************************
  65. * I2STX_FIFOCTL
  66. */
  67. // #define I2ST0_FIFOCTL_FIFO0_VOL_SHIFT (8) /* FIFO volume control */
  68. // #define I2ST0_FIFOCTL_FIFO0_VOL_MASK (0xF << I2ST0_FIFOCTL_FIFO0_VOL_SHIFT)
  69. // #define I2ST0_FIFOCTL_FIFO0_VOL(x) ((x) << I2ST0_FIFOCTL_FIFO0_VOL_SHIFT)
  70. #define I2ST0_FIFOCTL_TXFIFO_DMAWIDTH BIT(7) /* dma transfer width */
  71. #define I2ST0_FIFOCTL_FIFO_IN_SEL_SHIFT (4) /* I2STX FIFO input select */
  72. #define I2ST0_FIFOCTL_FIFO_IN_SEL_MASK (0x4 << I2ST0_FIFOCTL_FIFO_IN_SEL_SHIFT)
  73. #define I2ST0_FIFOCTL_FIFO_IN_SEL(x) ((x) << I2ST0_FIFOCTL_FIFO_IN_SEL_SHIFT)
  74. // #define I2ST0_FIFOCTL_FIFO_SEL BIT(3) /* I2S/SPDIF module FIFO selection */
  75. #define I2ST0_FIFOCTL_DSP_DMA_DEN BIT(3)
  76. #define I2ST0_FIFOCTL_FIFO_IEN BIT(2) /* I2STX FIFO half empty irq enable */
  77. #define I2ST0_FIFOCTL_FIFO_DEN BIT(1) /* I2STX FIFO half empty drq enable */
  78. #define I2ST0_FIFOCTL_FIFO_RST BIT(0) /* I2STX FIFO reset */
  79. /***************************************************************************************************
  80. * I2STX_FIFOSTAT
  81. */
  82. #define I2ST0_FIFOSTA_FIFO_ER BIT(8) /* FIFO error */
  83. #define I2ST0_FIFOSTA_IP BIT(7) /* half empty irq pending bit */
  84. #define I2ST0_FIFOSTA_TFFU BIT(6) /* fifo full flag */
  85. #define I2ST0_FIFOSTA_STA_SHIFT (0) /* fifo status */
  86. #define I2ST0_FIFOSTA_STA_MASK (0x3F << I2ST0_FIFOSTA_STA_SHIFT)
  87. /***************************************************************************************************
  88. * I2STX_DAT
  89. */
  90. #define I2ST0_DAT_DAT_SHIFT (8) /* I2STX FIFO is 24bits x 32 levels */
  91. #define I2ST0_DAT_DAT_MASK (0xFFFFFF << I2ST0_DAT_DAT_SHIFT)
  92. /***************************************************************************************************
  93. * I2STX_SRDCTL - I2STX Slave mode sample rate detect register
  94. */
  95. #define I2ST0_SRDCTL_MUTE_EN BIT(12) /* If detect sample rate or channel width changing, mute the TX output as 0 */
  96. #define I2ST0_SRDCTL_TO_IE BIT(11)
  97. // #define I2ST0_SRDCTL_SRD_IE BIT(8) /* sample rate detect result change interrupt enable */
  98. #define I2ST0_SRDCTL_SRC_IE BIT(10)
  99. #define I2ST0_SRDCTL_CHW_IE BIT(8)
  100. #define I2ST0_SRDCTL_CNT_TIM_SHIFT (4) /* slave mode sample rate detect counter period select */
  101. #define I2ST0_SRDCTL_CNT_TIM_MASK (0x3 << I2ST0_SRDCTL_CNT_TIM_SHIFT)
  102. #define I2ST0_SRDCTL_CNT_TIM(x) ((x) << I2ST0_SRDCTL_CNT_TIM_SHIFT)
  103. #define I2ST0_SRDCTL_SRD_TH_SHIFT (1) /* sample rate detecting sensitivity setting */
  104. #define I2ST0_SRDCTL_SRD_TH_MASK (0x7 << I2ST0_SRDCTL_SRD_TH_SHIFT)
  105. #define I2ST0_SRDCTL_SRD_TH(x) ((x) << I2ST0_SRDCTL_SRD_TH_SHIFT)
  106. #define I2ST0_SRDCTL_SRD_EN BIT(0) /* slave mode sample rate detect enable */
  107. /***************************************************************************************************
  108. * I2STX_SRDSTA
  109. */
  110. #define I2ST0_SRDSTA_CNT_SHIFT (12) /* CNT of LRCLK which sampling by SRC_CLK */
  111. #define I2ST0_SRDSTA_CNT_MASK (0x1FFF << I2ST0_SRDSTA_CNT_SHIFT)
  112. #define I2ST0_SRDSTA_CNT(x) ((x) << I2ST0_SRDSTA_CNT_SHIFT)
  113. #define I2ST0_SRDSTA_TO_PD BIT(11) /* SRD timput irq pending */
  114. #define I2ST0_SRDSTA_SRC_PD BIT(10) /* sample rate changing detection interrupt pending */
  115. #define I2ST0_SRDSTA_CHW_PD BIT(8) /* channel width change irq pending */
  116. #define I2ST0_SRDSTA_WL_SHIFT (0) /* channel word lenght */
  117. #define I2ST0_SRDSTA_WL_MASK (0x7 << I2ST0_SRDSTA_WL_SHIFT)
  118. #define I2ST0_SRDSTA_WL(x) ((x) << I2ST0_SRDSTA_WL_SHIFT)
  119. /***************************************************************************************************
  120. * I2STX_FIFO_CNT
  121. */
  122. #define I2ST0_FIFO_CNT_IP BIT(18) /* I2STX FIFO counter overflow irq pending */
  123. #define I2ST0_FIFO_CNT_IE BIT(17) /* I2STX FIFO counter overflow irq enable */
  124. #define I2ST0_FIFO_CNT_EN BIT(16) /* I2STX FIFO counter enable */
  125. #define I2ST0_FIFO_CNT_CNT_SHIFT (0) /* I2STX FIFO counter */
  126. #define I2ST0_FIFO_CNT_CNT_MASK (0xFFFF << I2ST0_FIFO_CNT_CNT_SHIFT)
  127. /***************************************************************************************************
  128. * i2STX FEATURES CONGIURATION
  129. */
  130. /* The sensitivity of the SRD */
  131. #define I2STX_SRD_TH_DEFAULT (7)
  132. #define I2STX_BCLK_DIV_DEFAULT (0) /* MCLK = 4BCLK */
  133. #define I2STX_FIFO_LEVEL (32)
  134. #define I2STX_SRD_CONFIG_TIMEOUT_US (500000)
  135. #define I2STX_FIFO_MAX_VOL_LEVEL (0xF)
  136. #define I2STX_FIFO_VOL_LEVEL_DEFAULT (0x3) /* 0db */
  137. /*
  138. * enum a_i2s_mclk_clksrc_e
  139. * @brief The MCLK clock source of i2stx selection
  140. */
  141. typedef enum {
  142. CLK_SRCTX_I2STX = 0, /* I2STX clock source from I2STX MCLK */
  143. CLK_SRCTX_DAC_256FS, /* I2STX clock source from DAC 256FS */
  144. CLK_SRCTX_DAC_128FS, /* I2STX clock source from DAC 256FS */
  145. CLK_SRCTX_I2STX_EXT, /* I2STX clock source from I2STX extern MCLK */
  146. } a_i2stx_mclk_clksrc_e;
  147. /*
  148. * @struct acts_audio_i2stx0
  149. * @brief I2STX controller hardware register
  150. */
  151. struct acts_audio_i2stx {
  152. volatile uint32_t tx_ctl; /* I2STX control */
  153. volatile uint32_t fifoctl; /* I2STX FIFO control */
  154. volatile uint32_t fifostat; /* I2STX FIFO state */
  155. volatile uint32_t dat; /* I2STX FIFO data */
  156. volatile uint32_t srdctl; /* I2S slave mode TX sample rate detect control */
  157. volatile uint32_t srdstat; /* I2S slave mode TX sample rate detect state */
  158. volatile uint32_t fifocnt; /* I2STX out FIFO counter */
  159. };
  160. #ifdef CONFIG_CFG_DRV
  161. /**
  162. * struct phy_i2stx_external_config
  163. * @brief The I2STX external configuration which generated by configuration tool.
  164. */
  165. struct phy_i2stx_external_config {
  166. CFG_Type_I2S_Select_GPIO I2STX_Select_GPIO; /* I2STX PINs setting */
  167. };
  168. #endif
  169. /*
  170. * struct phy_i2stx_drv_data
  171. * @brief The software related data that used by physical i2stx driver.
  172. */
  173. struct phy_i2stx_drv_data {
  174. uint8_t mclksrc; /* MCLK clock source selection refer to #a_i2stx_mclk_clksrc_e */
  175. uint32_t fifo_cnt; /* I2STX FIFO hardware counter max value is 0xFFFF */
  176. int (*srd_callback)(void *cb_data, uint32_t cmd, void *param);
  177. void *cb_data;
  178. uint8_t srd_wl; /* The width length detected by SRD */
  179. uint8_t fifo_use; /* Record the used FIFO */
  180. #ifdef CONFIG_CFG_DRV
  181. struct phy_i2stx_external_config external_config; /* I2STX external configuration */
  182. #endif
  183. uint8_t link_with_dac : 1; /* The flag of linkage with dac */
  184. uint8_t channel_opened : 1; /* flag of channel opened */
  185. };
  186. /**
  187. * union phy_i2stx_features
  188. * @brief The infomation from DTS to control the I2STX features to enable or nor.
  189. */
  190. typedef union {
  191. uint32_t raw;
  192. struct {
  193. uint32_t srd_en : 1; /* SRD function enable or not */
  194. uint32_t mode : 1; /* master mode or slaver mode; 0: master 1: slaver */
  195. uint32_t channel_num : 4; /* channel number and only support 2/4/8 channels */
  196. uint32_t slave_internal_clk : 1; /* slave mode MCLK to use internal clock */
  197. uint32_t lrclk_proc : 1; /* LRCLK process 0: 50% DUTY; 1: 1 BCLK */
  198. uint32_t mclk_reverse : 1; /* mclk reverse */
  199. uint32_t always_open : 1; /* BCLK/LRCLK always existed */
  200. uint32_t format : 2; /* I2S transfer format */
  201. uint32_t tdm_format : 1; /* TDM format */
  202. uint32_t tdm_frame : 2; /* TDM frame start position */
  203. uint32_t bclk_width : 1; /* bclk width */
  204. uint32_t txd_delay : 2; /* I2STX data delay */
  205. uint32_t pcm_en : 1; /* pcm select */
  206. uint32_t pcm_frame : 1; /* pcm frame */
  207. uint32_t pcm_slot : 2; /* pcm slot */
  208. } v;
  209. } phy_i2stx_features;
  210. /**
  211. * struct phy_i2stx_config_data
  212. * @brief The hardware related data that used by physical i2stx driver.
  213. */
  214. struct phy_i2stx_config_data {
  215. uint32_t reg_base; /* I2STX controller register base address */
  216. uint8_t clk_id; /* I2STX devclk id */
  217. uint8_t srd_clk_id; /* I2S SRD clock id */
  218. uint8_t hclk_clk_id; /* I2S HCLK clock id */
  219. uint8_t rst_id; /* I2STX reset id */
  220. struct audio_dma_dt dma_fifo0; /* DMA resource for SPDIFRX */
  221. void (*irq_config)(void); /* IRQ configuration function */
  222. phy_i2stx_features features; /* I2STX features */
  223. };
  224. /* @brief Get the I2STX controller base address */
  225. static inline struct acts_audio_i2stx *get_i2stx_reg_base(struct device *dev)
  226. {
  227. const struct phy_i2stx_config_data *cfg = dev->config;
  228. return (struct acts_audio_i2stx *)cfg->reg_base;
  229. }
  230. /* @brief Dump the I2STX relative registers */
  231. static void i2stx_dump_register(struct device *dev)
  232. {
  233. struct acts_audio_i2stx *i2stx_reg = get_i2stx_reg_base(dev);
  234. LOG_INF("** i2stx contoller regster **");
  235. LOG_INF(" COREPLL_CTL: %08x", sys_read32(COREPLL_CTL));
  236. LOG_INF(" CMU_SYSCLK: %08x", sys_read32(CMU_SYSCLK));
  237. LOG_INF(" CMUA_AVDDLDO_CTL: %08x", sys_read32(AVDDLDO_CTL));
  238. LOG_INF(" CMUA_A_audio_debug: %08x", sys_read32(CMUA_REG_BASE+0x48));
  239. LOG_INF(" CMUA_AUDIOPLL0_CTL: %08x", sys_read32(AUDIO_PLL0_CTL));
  240. LOG_INF(" CMUD_DEVCLKEN1: %08x", sys_read32(CMU_DEVCLKEN1));
  241. LOG_INF(" CMUD_DEVCLKEN0:DMA %08x", sys_read32(CMU_DEVCLKEN0));
  242. LOG_INF(" RMU_MRCR1: %08x", sys_read32(RMU_MRCR1));
  243. LOG_INF(" RMU_MRCR0: %08x", sys_read32(RMU_MRCR0));
  244. LOG_INF(" CMU_DACCLK: %08x", sys_read32(CMU_DACCLK));
  245. LOG_INF(" CMU_I2STXCLK: %08x", sys_read32(CMU_I2STXCLK));
  246. LOG_INF(" BASE: %08x", (uint32_t)i2stx_reg);
  247. LOG_INF(" I2STX_CTL: %08x", i2stx_reg->tx_ctl);
  248. LOG_INF(" I2STX_FIFOCTL: %08x", i2stx_reg->fifoctl);
  249. LOG_INF(" I2STX_FIFOSTA: %08x", i2stx_reg->fifostat);
  250. LOG_INF(" I2STX_DAT: %08x", i2stx_reg->dat);
  251. LOG_INF(" I2STX_SRDCTL: %08x", i2stx_reg->srdctl);
  252. LOG_INF(" I2STX_SRDSTA: %08x", i2stx_reg->srdstat);
  253. LOG_INF(" I2STX_FIFOCNT: %08x", i2stx_reg->fifocnt);
  254. LOG_INF(" DMA7_CTL: %08x", sys_read32(0x4001c800));
  255. LOG_INF(" DMA7_START: %08x", sys_read32(0x4001c804));
  256. LOG_INF(" DMA7_sadd0: %08x", sys_read32(0x4001c808));
  257. LOG_INF(" DMA7_sadd1: %08x", sys_read32(0x4001c80c));
  258. LOG_INF(" DMA7_dadd0: %08x", sys_read32(0x4001c810));
  259. LOG_INF(" DMA7_dadd1: %08x", sys_read32(0x4001c814));
  260. LOG_INF(" DMA7_bc: %08x", sys_read32(0x4001c818));
  261. LOG_INF(" DMA7_rc: %08x", sys_read32(0x4001c81c));
  262. }
  263. /* @brief I2STX sample rate config */
  264. static int i2stx_sample_rate_set(struct device *dev, uint16_t sr_khz,
  265. a_i2stx_mclk_clksrc_e mclk_src, bool direct_set)
  266. {
  267. const struct phy_i2stx_config_data *cfg = dev->config;
  268. int ret;
  269. uint8_t clk_div, series, pll_index;
  270. uint8_t div, fir_div, fir2x_div, cic_div;
  271. uint32_t reg = 0, reg1;
  272. uint32_t lrclk_div;
  273. a_mclk_type_e mclk = 0;
  274. switch (I2STX_BCLK_DIV_DEFAULT)
  275. {
  276. case 0:
  277. if (!PHY_DEV_FEATURE(bclk_width))
  278. mclk = MCLK_256FS;
  279. else
  280. mclk = MCLK_128FS;
  281. break;
  282. case 1:
  283. if (!PHY_DEV_FEATURE(bclk_width))
  284. mclk = MCLK_384FS;
  285. else
  286. mclk = MCLK_192FS;
  287. break;
  288. case 2:
  289. if (!PHY_DEV_FEATURE(bclk_width))
  290. mclk = MCLK_512FS;
  291. else
  292. mclk = MCLK_256FS;
  293. break;
  294. case 3:
  295. if (!PHY_DEV_FEATURE(bclk_width))
  296. mclk = MCLK_768FS;
  297. else
  298. mclk = MCLK_384FS;
  299. break;
  300. default:
  301. if (!PHY_DEV_FEATURE(bclk_width))
  302. mclk = MCLK_256FS;
  303. else
  304. mclk = MCLK_128FS;
  305. break;
  306. }
  307. if (direct_set) {
  308. reg = sys_read32(CMU_I2STXCLK) & ~CMU_I2STXCLK_I2SG0MCLKSRC_MASK;
  309. goto out;
  310. }
  311. if (CLK_SRCTX_I2STX_EXT != mclk_src) {
  312. /* make external MCLK clock phase reverse */
  313. if (PHY_DEV_FEATURE(mclk_reverse))
  314. reg |= CMU_I2STXCLK_I2SG0MCLKEXTREV;
  315. /* I2S master mode to enable MCLK output to PAD */
  316. if (!PHY_DEV_FEATURE(mode)) {
  317. reg |= CMU_I2SG0LRCLOKEN | CMU_I2SG0BCLKOEN | CMU_I2SG0MCLKOEN;
  318. } else {
  319. if (PHY_DEV_FEATURE(slave_internal_clk))
  320. sr_khz *= 2; /* slave mode use internal clk condition: MCLK > 6BCLK */
  321. /* BCLK/LRCLK source from external PAD */
  322. reg |= CMU_I2STXCLK_I2SG0BLRCLKSRC;
  323. }
  324. reg |= CMU_I2STXCLK_I2SG0LRCLKPROC(PHY_DEV_FEATURE(lrclk_proc));
  325. /* BCLK = FS x channel number x channel bit width */
  326. if (!PHY_DEV_FEATURE(bclk_width))
  327. lrclk_div = PHY_DEV_FEATURE(channel_num) * 32;
  328. else
  329. lrclk_div = PHY_DEV_FEATURE(channel_num) * 16;
  330. /* LRCLK divisor 0:32; 1:64; 2: 128; 3:256 */
  331. if (32 == lrclk_div) {
  332. lrclk_div = 0;
  333. } else if (64 == lrclk_div) {
  334. lrclk_div = 1;
  335. } else if (128 == lrclk_div) {
  336. lrclk_div = 2;
  337. } else if (256 == lrclk_div) {
  338. lrclk_div = 3;
  339. } else {
  340. LOG_ERR("invalid lrclk divisor:%d", lrclk_div);
  341. return -EINVAL;
  342. }
  343. reg |= CMU_I2STXCLK_I2SG0LRCLKDIV(lrclk_div);
  344. /* MCLK = 4 x BCLK */
  345. reg |= CMU_I2STXCLK_I2SG0BCLKDIV(I2STX_BCLK_DIV_DEFAULT);
  346. sr_khz = PHY_DEV_FEATURE(channel_num) / 2 * sr_khz;
  347. }
  348. if ((CLK_SRCTX_DAC_256FS == mclk_src)
  349. || (CLK_SRCTX_DAC_128FS == mclk_src)) {
  350. /* Get audio PLL setting */
  351. ret = audio_get_pll_setting_dac(sr_khz, &div, &fir_div, &fir2x_div, &cic_div, &series);
  352. if (ret) {
  353. LOG_DBG("get pll setting error:%d", ret);
  354. return ret;
  355. }
  356. ret = audio_pll_check_config(series, &pll_index);
  357. if (ret) {
  358. LOG_DBG("check pll config error:%d", ret);
  359. return ret;
  360. }
  361. reg1 = sys_read32(CMU_DACCLK) & ~0x1FF;
  362. /*enable dac audio*/
  363. reg1 |= (CMU_DACCLK_DACSDMCLOCK | CMU_DACCLK_DACCICFIRCLOCK | CMU_DACCLK_DACFIFO0CLKEN);
  364. /* Select audio_clk_div*/
  365. reg1 |= (pll_index & 0x1) << CMU_DACCLK_DACCLKSRC;
  366. reg1 |= (CMU_DACCLK_DACCLKDIV(div)| (fir_div << CMU_DACCLK_DACFIRCLKDIV)
  367. | CMU_DACCLK_DACFIR2XCLKDIV(fir2x_div) | (cic_div << CMU_DACCLK_DACCICCLCKDIV));
  368. reg1 |= CMU_DACCLK_DACHUMDIV;
  369. sys_write32(reg1, CMU_DACCLK);
  370. } else if (CLK_SRCTX_I2STX == mclk_src) {
  371. /* Get audio PLL setting */
  372. ret = audio_get_pll_setting_i2s(sr_khz, mclk, &clk_div, &series);
  373. if (ret) {
  374. LOG_DBG("get pll setting error:%d", ret);
  375. return ret;
  376. }
  377. ret = audio_pll_check_config(series, &pll_index);
  378. if (ret) {
  379. LOG_DBG("check pll config error:%d", ret);
  380. return ret;
  381. }
  382. printk("clk_div:%d,pll_index:%d\n",clk_div,pll_index);
  383. // reg |= (pll_index & 0x1) << CMU_I2STXCLK_I2SG0CLKSRC;
  384. reg |= clk_div << CMU_I2STXCLK_I2SG0CLKDIV_SHIFT;
  385. } else if (CLK_SRCTX_I2STX_EXT == mclk_src) {
  386. /* BCLK/LRCLK source from external PAD */
  387. reg |= CMU_I2STXCLK_I2SG0BLRCLKSRC;
  388. LOG_INF("I2STX clock source from external");
  389. } else {
  390. LOG_ERR("Invalid i2stx clk source %d", mclk_src);
  391. return -EINVAL;
  392. }
  393. out:
  394. /* Select the i2stx mclk source */
  395. reg |= CMU_I2STXCLK_I2SG0MCLKSRC(mclk_src & 0x3);
  396. sys_write32(reg, CMU_I2STXCLK);
  397. LOG_DBG("i2stx_sample_rate_set:mclk:%d,direct_set:%d",(uint8_t)mclk,direct_set);
  398. return 0;
  399. }
  400. /* @brief Get the sample rate from the I2STX config */
  401. static int i2stx_sample_rate_get(struct device *dev)
  402. {
  403. const struct phy_i2stx_config_data *cfg = dev->config;
  404. uint8_t clk_div, pll_index, mclk_src;
  405. uint8_t div, fir_div, fir2x_div, cic_div;
  406. uint32_t reg;
  407. int ret = -1;
  408. a_mclk_type_e mclk;
  409. if (!PHY_DEV_FEATURE(bclk_width))
  410. mclk = MCLK_256FS;
  411. else
  412. mclk = MCLK_128FS;
  413. reg = sys_read32(CMU_I2STXCLK);
  414. mclk_src = (reg & CMU_I2STXCLK_I2SG0MCLKSRC_MASK) >> CMU_I2STXCLK_I2SG0MCLKSRC_SHIFT;
  415. if ((CLK_SRCTX_DAC_256FS == mclk_src)
  416. || (CLK_SRCTX_DAC_128FS == mclk_src)) {
  417. reg = sys_read32(CMU_DACCLK);
  418. pll_index = 0;//leopard只能选择PLL0
  419. div = (reg & CMU_DACCLK_DACCLKDIV_MASK);
  420. fir_div = (reg & (1<<CMU_DACCLK_DACFIRCLKDIV)>>CMU_DACCLK_DACFIRCLKDIV);
  421. fir2x_div = (reg & CMU_DACCLK_DACFIR2XCLKDIV_MASK)>>CMU_DACCLK_DACFIR2XCLKDIV_SHIFT;
  422. cic_div = (reg & (1<<CMU_DACCLK_DACCICCLCKDIV)>>CMU_DACCLK_DACCICCLCKDIV);
  423. ret = audio_get_pll_sample_rate_dac(div, fir_div, fir2x_div, cic_div, 0);
  424. } else if (CLK_SRCTX_I2STX == mclk_src) {
  425. reg = sys_read32(CMU_I2STXCLK);
  426. clk_div = reg & 0xF;
  427. pll_index = 0;//leopard只能选择PLL0
  428. ret = audio_get_pll_sample_rate_i2s(mclk, clk_div, pll_index);
  429. } else if (CLK_SRCTX_I2STX_EXT == mclk_src) {
  430. LOG_INF("I2STX is using the external clock");
  431. ret = -ENOENT;
  432. }
  433. return ret;
  434. }
  435. /* @brief Get the AUDIO_PLL APS used by I2STX */
  436. static int i2stx_get_pll_aps(struct device *dev)
  437. {
  438. uint32_t reg;
  439. uint8_t pll_index, mclk_src;
  440. int ret = -1;
  441. reg = sys_read32(CMU_I2STXCLK);
  442. mclk_src = (reg & CMU_I2STXCLK_I2SG0MCLKSRC_MASK) >> CMU_I2STXCLK_I2SG0MCLKSRC_SHIFT;
  443. /* If I2STX CLK from DAC_CLK, the APS regulation will be called at DAC APIs */
  444. if (CLK_SRCTX_I2STX == mclk_src) {
  445. pll_index = 0;//leopard只能选择PLL0;
  446. ret = audio_pll_get_aps((a_pll_type_e)pll_index);
  447. } else if (CLK_SRCTX_I2STX_EXT == mclk_src) {
  448. LOG_INF("I2STX is using the external clock source");
  449. ret = -ENOENT;
  450. } else {
  451. ret = -EPERM;
  452. }
  453. return ret;
  454. }
  455. /* @brief Set the AUDIO_PLL APS used by I2STX */
  456. static int i2stx_set_pll_aps(struct device *dev, audio_aps_level_e level)
  457. {
  458. uint32_t reg;
  459. uint8_t pll_index, mclk_src;
  460. int ret = -1;
  461. reg = sys_read32(CMU_I2STXCLK);
  462. mclk_src = (reg & CMU_I2STXCLK_I2SG0MCLKSRC_MASK) >> CMU_I2STXCLK_I2SG0MCLKSRC_SHIFT;
  463. /* If I2STX CLK from DAC_CLK, the APS regulation will be called at DAC APIs */
  464. if (CLK_SRCTX_I2STX == mclk_src) {
  465. pll_index = 0;//leopard只能选择PLL0
  466. ret = audio_pll_set_aps((a_pll_type_e)pll_index, level);
  467. } else if (CLK_SRCTX_I2STX_EXT == mclk_src) {
  468. LOG_INF("I2STX is using the external clock");
  469. ret = -ENOENT;
  470. } else {
  471. ret = -EPERM;
  472. }
  473. return ret;
  474. }
  475. /* @brief Disable the I2STX FIFO */
  476. static void i2stx_fifo_disable(struct device *dev)
  477. {
  478. struct acts_audio_i2stx *i2stx_reg = get_i2stx_reg_base(dev);
  479. i2stx_reg->fifoctl = 0;
  480. }
  481. /* @brief Reset the I2STX FIFO */
  482. static void i2stx_fifo_reset(struct device *dev)
  483. {
  484. struct acts_audio_i2stx *i2stx_reg = get_i2stx_reg_base(dev);
  485. i2stx_reg->fifoctl &= ~I2ST0_FIFOCTL_FIFO_RST;
  486. i2stx_reg->fifoctl |= I2ST0_FIFOCTL_FIFO_RST;
  487. }
  488. /* @brief Check the I2STX FIFO is working or not */
  489. static bool is_i2stx_fifo_working(struct device *dev)
  490. {
  491. struct acts_audio_i2stx *i2stx_reg = get_i2stx_reg_base(dev);
  492. return !!(i2stx_reg->fifoctl & I2ST0_FIFOCTL_FIFO_RST);
  493. }
  494. /* @brief Check the I2STX FIFO is error */
  495. static bool check_i2stx_fifo_error(struct device *dev)
  496. {
  497. struct acts_audio_i2stx *i2stx_reg = get_i2stx_reg_base(dev);
  498. if (i2stx_reg->fifostat & I2ST0_FIFOSTA_FIFO_ER) {
  499. i2stx_reg->fifostat |= I2ST0_FIFOSTA_FIFO_ER;
  500. return true;
  501. }
  502. return false;
  503. }
  504. /* @brief Get the I2STX FIFO status which indicates how many samples not filled */
  505. static uint32_t i2stx_fifo_status(struct device *dev)
  506. {
  507. struct acts_audio_i2stx *i2stx_reg = get_i2stx_reg_base(dev);
  508. return (i2stx_reg->fifostat & I2ST0_FIFOSTA_STA_MASK) >> I2ST0_FIFOSTA_STA_SHIFT;
  509. }
  510. /* @brief Enable the I2STX FIFO */
  511. static void i2stx_fifo_enable(struct device *dev, audio_fifouse_sel_e sel, audio_dma_width_e width, bool this_use)
  512. {
  513. struct acts_audio_i2stx *i2stx_reg = get_i2stx_reg_base(dev);
  514. uint32_t reg = 0;
  515. LOG_DBG("I2STX sel:%d width:%d this_use:%d", sel, width, this_use);
  516. /* If the I2STX FIFO is used by SPDIFTX, the FIFO_SEL shall set to DAC FIFO source, otherwise will get no dma IRQ */
  517. // if (!this_use)
  518. // reg = I2ST0_FIFOCTL_FIFO_SEL; /* 0: I2STX FIFO; 1: DAC FIFO */
  519. /* set I2STX FIFO default volume*/
  520. // reg &= ~I2ST0_FIFOCTL_FIFO0_VOL_MASK;
  521. // reg |= I2ST0_FIFOCTL_FIFO0_VOL(I2STX_FIFO_VOL_LEVEL_DEFAULT);
  522. if (DMA_WIDTH_16BITS == width)
  523. reg |= I2ST0_FIFOCTL_TXFIFO_DMAWIDTH;
  524. if (FIFO_SEL_CPU == sel) {
  525. reg |= (I2ST0_FIFOCTL_FIFO_IEN | I2ST0_FIFOCTL_FIFO_RST);
  526. } else if (FIFO_SEL_DMA == sel) {
  527. reg |= (I2ST0_FIFOCTL_FIFO_DEN | I2ST0_FIFOCTL_FIFO_RST
  528. | I2ST0_FIFOCTL_FIFO_IN_SEL(1));
  529. } else if (FIFO_SEL_DSP == sel) {
  530. reg |= I2ST0_FIFOCTL_FIFO_IN_SEL(3) | I2ST0_FIFOCTL_FIFO_RST;
  531. } else {
  532. LOG_ERR("invalid fifo sel %d", sel);
  533. }
  534. i2stx_reg->fifoctl = reg;
  535. }
  536. /* @brief Enable the I2STX FIFO counter function */
  537. static void i2stx_fifocount_enable(struct device *dev)
  538. {
  539. struct acts_audio_i2stx *i2stx_reg = get_i2stx_reg_base(dev);
  540. uint32_t reg = 0;
  541. reg = I2ST0_FIFO_CNT_EN;
  542. reg |= I2ST0_FIFO_CNT_IE;
  543. reg |= I2ST0_FIFO_CNT_IP;
  544. i2stx_reg->fifocnt = reg;
  545. LOG_INF("I2STX sample counter enable");
  546. }
  547. /* @brief Disable the I2STX FIFO counter function */
  548. static void i2stx_fifocount_disable(struct device *dev)
  549. {
  550. struct acts_audio_i2stx *i2stx_reg = get_i2stx_reg_base(dev);
  551. i2stx_reg->fifocnt &= ~(I2ST0_FIFO_CNT_EN | I2ST0_FIFO_CNT_IE);
  552. }
  553. /* @brief Reset the I2STX FIFO counter function and by default to enable IRQ */
  554. static void i2stx_fifocount_reset(struct device *dev)
  555. {
  556. struct acts_audio_i2stx *i2stx_reg = get_i2stx_reg_base(dev);
  557. i2stx_reg->fifocnt &= ~(I2ST0_FIFO_CNT_EN | I2ST0_FIFO_CNT_IE);
  558. i2stx_reg->fifocnt |= I2ST0_FIFO_CNT_EN;
  559. i2stx_reg->fifocnt |= I2ST0_FIFO_CNT_IE;
  560. }
  561. /* @brief Enable the I2STX FIFO counter function */
  562. static uint32_t i2stx_read_fifocount(struct device *dev)
  563. {
  564. struct acts_audio_i2stx *i2stx_reg = get_i2stx_reg_base(dev);
  565. return i2stx_reg->fifocnt & I2ST0_FIFO_CNT_CNT_MASK;
  566. }
  567. /* @brief set the I2STX FIFO volume */
  568. // static int i2stx_fifo_volume_set(struct device *dev, uint8_t vol)
  569. // {
  570. // struct acts_audio_i2stx *i2stx_reg = get_i2stx_reg_base(dev);
  571. // uint32_t reg = i2stx_reg->fifoctl;
  572. // /**
  573. // * FIFO VOLUME LEVEL <=> db
  574. // * - 0: 3db
  575. // * - 1: 2db
  576. // * - 2: 1db
  577. // * ... ...
  578. // * - 0xe: -11db
  579. // * - 0xf: -12db
  580. // */
  581. // if (vol > I2STX_FIFO_MAX_VOL_LEVEL)
  582. // vol = I2STX_FIFO_MAX_VOL_LEVEL;
  583. // // reg &= ~I2ST0_FIFOCTL_FIFO0_VOL_MASK;
  584. // // reg |= I2ST0_FIFOCTL_FIFO0_VOL(vol);
  585. // i2stx_reg->fifoctl = reg;
  586. // return 0;
  587. // }
  588. /* @brief get the I2STX FIFO volume */
  589. // static int i2stx_fifo_volume_get(struct device *dev)
  590. // {
  591. // struct acts_audio_i2stx *i2stx_reg = get_i2stx_reg_base(dev);
  592. // uint32_t reg = i2stx_reg->fifoctl;
  593. // int val;
  594. // val = (reg & I2ST0_FIFOCTL_FIFO0_VOL_MASK) >> I2ST0_FIFOCTL_FIFO0_VOL_SHIFT;
  595. // return val;
  596. // }
  597. /* @brief I2STX digital function control enable */
  598. static void i2stx_digital_enable(struct device *dev, audio_ch_width_e width, bool en)
  599. {
  600. const struct phy_i2stx_config_data *cfg = dev->config;
  601. struct acts_audio_i2stx *i2stx_reg = get_i2stx_reg_base(dev);
  602. uint32_t reg = i2stx_reg->tx_ctl & ~(0x3E);
  603. uint8_t tx_width, fmt = PHY_DEV_FEATURE(format);
  604. uint8_t pcm_frame = PHY_DEV_FEATURE(pcm_frame);
  605. uint8_t pcm_slot = PHY_DEV_FEATURE(pcm_slot);
  606. /* By default TXD = 2MCLK */
  607. reg |= I2ST0_CTL_TXD_DELAY(PHY_DEV_FEATURE(txd_delay));
  608. if (CHANNEL_WIDTH_16BITS == width)
  609. tx_width = 0;
  610. else if (CHANNEL_WIDTH_20BITS == width)
  611. tx_width = 1;
  612. else
  613. tx_width = 2;
  614. if (PHY_DEV_FEATURE(channel_num) != 2) {
  615. fmt = 3; /* TDM mode */
  616. reg &= ~ (0xF << I2ST0_CTL_TDMTX_SYNC_SHIFT);
  617. if (PHY_DEV_FEATURE(channel_num) == 8)
  618. reg |= I2ST0_CTL_TDMTX_CHAN; /* 0: 4 channels; 1: 8 channels */
  619. if (PHY_DEV_FEATURE(tdm_format))
  620. reg |= I2ST0_CTL_TDMTX_MODE; /* 0: I2S format; 1: Left-justified format */
  621. reg |= I2ST0_CTL_TDMTX_SYNC(PHY_DEV_FEATURE(tdm_frame));
  622. }
  623. reg |= (I2ST0_CTL_TXMODELSEL(fmt) | I2ST0_CTL_TXWIDTH(tx_width));
  624. if (PHY_DEV_FEATURE(bclk_width))
  625. reg |= I2ST0_CTL_TXBCLKSET;
  626. i2stx_reg->tx_ctl = reg;
  627. /* I2S PCM TX enable */
  628. if(en)
  629. {
  630. // i2stx_reg->tx_ctl |= I2ST0_CTL_TXEN | I2ST0_CTL_LPEN0;//loopback
  631. if(PHY_DEV_FEATURE(pcm_en))
  632. {
  633. i2stx_reg->tx_ctl |= I2ST0_CTL_PCM_CHAN(pcm_slot);
  634. i2stx_reg->tx_ctl |= I2ST0_CTL_PCMTX0_EN;
  635. //pcm frame set 1 is long frame
  636. if(pcm_frame)
  637. {
  638. i2stx_reg->tx_ctl |= I2ST0_CTL_PCM_SYNC;
  639. sys_write32(sys_read32(CMU_I2STXCLK)|
  640. CMU_I2STXCLK_I2SG0LRCLKPROC(2),
  641. CMU_I2STXCLK);
  642. }
  643. else
  644. sys_write32(sys_read32(CMU_I2STXCLK)|
  645. CMU_I2STXCLK_I2SG0LRCLKPROC(3),
  646. CMU_I2STXCLK);
  647. }
  648. else
  649. i2stx_reg->tx_ctl |= I2ST0_CTL_TXEN;
  650. }
  651. printk("%s.%d,reg:%08x,tx_ctl:%08x,en:%d\n",__func__,__LINE__,reg,i2stx_reg->tx_ctl,en);
  652. }
  653. /* @brief I2STX digital function control disable */
  654. static void i2stx_digital_disable(struct device *dev)
  655. {
  656. struct acts_audio_i2stx *i2stx_reg = get_i2stx_reg_base(dev);
  657. i2stx_reg->tx_ctl = 0;
  658. }
  659. /* @brief I2STX sample rate detect function configuration */
  660. static int i2stx_srd_cfg(struct device *dev, uint8_t srd_th, audio_i2s_srd_period_e period, bool irq_en, bool mute)
  661. {
  662. const struct phy_i2stx_config_data *cfg = dev->config;
  663. struct acts_audio_i2stx *i2stx_reg = get_i2stx_reg_base(dev);
  664. uint32_t reg;
  665. uint8_t _wl, wl;
  666. uint32_t start_time, curr_time;
  667. reg = (I2ST0_SRDCTL_SRD_TH(srd_th & 0x7) | I2ST0_SRDCTL_CNT_TIM(period));
  668. if (irq_en)
  669. reg |= I2ST0_SRDCTL_SRC_IE | I2ST0_SRDCTL_CHW_IE | I2ST0_SRDCTL_TO_IE;
  670. if (mute)
  671. reg |= I2ST0_SRDCTL_MUTE_EN;
  672. i2stx_reg->srdctl = reg;
  673. /* enable slave mode sample rate detect */
  674. i2stx_reg->srdctl |= I2ST0_SRDCTL_SRD_EN;
  675. if (!PHY_DEV_FEATURE(bclk_width))
  676. wl = SRDSTA_WL_64RATE;
  677. else
  678. wl = SRDSTA_WL_32RATE;
  679. start_time = k_cycle_get_32();
  680. _wl = i2stx_reg->srdstat & I2ST0_SRDSTA_WL_MASK;
  681. while (_wl != wl) {
  682. curr_time = k_cycle_get_32();
  683. if (k_cyc_to_us_floor32(curr_time - start_time) > I2STX_SRD_CONFIG_TIMEOUT_US) {
  684. LOG_ERR("Wait SRD WL status timeout");
  685. return -ETIMEDOUT;
  686. }
  687. _wl = i2stx_reg->srdstat & I2ST0_SRDSTA_WL_MASK;
  688. k_sleep(K_MSEC(2));
  689. }
  690. return 0;
  691. }
  692. /* @brief get the I2STX sample rate detect counter */
  693. static uint32_t read_i2stx_srd_count(struct device *dev)
  694. {
  695. struct acts_audio_i2stx *i2stx_reg = get_i2stx_reg_base(dev);
  696. /* CNT of LRCLK which sampling by SRC_CLK */
  697. return ((i2stx_reg->srdstat & I2ST0_SRDSTA_CNT_MASK) >> I2ST0_SRDSTA_CNT_SHIFT);
  698. }
  699. /* @brief I2STX in slave mode handle sample rate detect change */
  700. void i2stx_srd_fs_change(struct device *dev)
  701. {
  702. const struct phy_i2stx_config_data *cfg = dev->config;
  703. struct phy_i2stx_drv_data *data = dev->data;
  704. uint32_t cnt, fs;
  705. audio_sr_sel_e sr;
  706. cnt = read_i2stx_srd_count(dev);
  707. /* CNT = SRD_CLK / LRCLK and SRD_CLK uses HOSC which is a 32MHz clock source*/
  708. fs = 32000000 / cnt;
  709. /* Allow a 1% deviation */
  710. if ((fs > 7920) && (fs < 8080)) { /* 8kfs */
  711. sr = SAMPLE_RATE_8KHZ;
  712. } else if ((fs > 10915) && (fs < 11135)) { /* 11.025kfs */
  713. sr = SAMPLE_RATE_11KHZ;
  714. } else if ((fs > 11880) && (fs < 12120)) { /* 12kfs */
  715. sr = SAMPLE_RATE_12KHZ;
  716. } else if ((fs > 15840) && (fs < 16160)) { /* 16kfs */
  717. sr = SAMPLE_RATE_16KHZ;
  718. } else if ((fs > 21830) && (fs < 22270)) { /* 22.05kfs */
  719. sr = SAMPLE_RATE_22KHZ;
  720. } else if ((fs > 23760) && (fs < 24240)) { /* 24kfs */
  721. sr = SAMPLE_RATE_24KHZ;
  722. } else if ((fs > 31680) && (fs < 32320)) { /* 32kfs */
  723. sr = SAMPLE_RATE_32KHZ;
  724. } else if ((fs > 43659) && (fs < 44541)) { /* 44.1kfs */
  725. sr = SAMPLE_RATE_44KHZ;
  726. } else if ((fs > 47520) && (fs < 48480)) { /* 48kfs */
  727. sr = SAMPLE_RATE_48KHZ;
  728. } else if ((fs > 63360) && (fs < 64640)) { /* 64kfs */
  729. sr = SAMPLE_RATE_64KHZ;
  730. } else if ((fs > 87318) && (fs < 89082)) { /* 88.2kfs */
  731. sr = SAMPLE_RATE_88KHZ;
  732. } else if ((fs > 95040) && (fs < 96960)) { /* 96kfs */
  733. sr = SAMPLE_RATE_96KHZ;
  734. } else if ((fs > 174636) && (fs < 178164)) { /* 176.4kfs */
  735. sr = SAMPLE_RATE_176KHZ;
  736. } else if((fs > 190080) && (fs < 193920)) { /* 192kfs */
  737. sr = SAMPLE_RATE_192KHZ;
  738. } else {
  739. LOG_ERR("Invalid sample rate %d", fs);
  740. return ;
  741. }
  742. LOG_INF("Detect new sample rate %d -> %d", fs, sr);
  743. /* FIXME: If not do the fifo reset, the left and right channel will exchange probably. */
  744. if (data->fifo_use == AOUT_FIFO_I2STX0)
  745. i2stx_fifo_reset(dev);
  746. if (PHY_DEV_FEATURE(slave_internal_clk))
  747. i2stx_sample_rate_set(dev, sr, data->mclksrc, false);
  748. if (data->srd_callback)
  749. data->srd_callback(data->cb_data, I2STX_SRD_FS_CHANGE, (void *)&sr);
  750. }
  751. /* @brief I2STX SRD width length change operation */
  752. void i2stx_srd_wl_change(struct device *dev)
  753. {
  754. struct phy_i2stx_drv_data *data = dev->data;
  755. struct acts_audio_i2stx *i2stx_reg = get_i2stx_reg_base(dev);
  756. uint8_t width;
  757. width = i2stx_reg->srdstat & I2ST0_SRDSTA_WL_MASK;
  758. LOG_INF("Detect new width length: %d", width);
  759. if (((data->srd_wl == SRDSTA_WL_64RATE) && (width == SRDSTA_WL_32RATE))
  760. || ((data->srd_wl == SRDSTA_WL_32RATE) && (width == SRDSTA_WL_64RATE))) {
  761. data->srd_wl = width;
  762. if (data->srd_callback)
  763. data->srd_callback(data->cb_data, I2STX_SRD_WL_CHANGE, (void *)&data->srd_wl);
  764. }
  765. }
  766. static int phy_i2stx_prepare_enable(struct device *dev, aout_param_t *out_param)
  767. {
  768. const struct phy_i2stx_config_data *cfg = dev->config;
  769. if ((!out_param) || (!out_param->i2stx_setting)
  770. || (!out_param->sample_rate)) {
  771. LOG_ERR("Invalid parameters");
  772. return -EINVAL;
  773. }
  774. if (!(out_param->channel_type & AUDIO_CHANNEL_I2STX)) {
  775. LOG_ERR("Invalid channel type %d", out_param->channel_type);
  776. return -EINVAL;
  777. }
  778. if ((PHY_DEV_FEATURE(channel_num) != 2)
  779. && (PHY_DEV_FEATURE(channel_num) != 4)
  780. && (PHY_DEV_FEATURE(channel_num) != 8)) {
  781. LOG_ERR("Invalid channel num %d", PHY_DEV_FEATURE(channel_num));
  782. return -EINVAL;
  783. }
  784. #ifndef CONFIG_CFG_DRV
  785. {
  786. struct board_pinmux_info pinmux_info;
  787. board_get_i2stx0_pinmux_info(&pinmux_info);
  788. /* Config the i2stx pin state */
  789. acts_pinmux_setup_pins(pinmux_info.pins_config, pinmux_info.pins_num);
  790. }
  791. #endif
  792. acts_clock_peripheral_enable(cfg->hclk_clk_id);
  793. acts_clock_peripheral_enable(cfg->clk_id);
  794. acts_clock_peripheral_enable(CLOCK_ID_DAC);//dacclk en.
  795. if (PHY_DEV_FEATURE(srd_en))
  796. acts_clock_peripheral_enable(cfg->srd_clk_id);
  797. return 0;
  798. }
  799. //8_mclk,9_blck,4_lr,5_out;ping config for debug_config
  800. //rx:16_mclk,17_blck,21_lr,13_in
  801. // #ifdef DEBUG_I2STX_TO_RX
  802. // static int set_gpio_mfc(int type, int gpio_m_num,int gpio__b_num,int gpio_lr_num,int gpio_io_num)
  803. // {
  804. // if(type==0)
  805. // {
  806. // //i2s tx gpio_mfc_config
  807. // sys_write32(0x0000100c,GPIO_REG_BASE + gpio_m_num * 0x4);
  808. // sys_write32(0x0000100c,GPIO_REG_BASE + gpio__b_num * 0x4);
  809. // sys_write32(0x0000100c,GPIO_REG_BASE + gpio_lr_num * 0x4);
  810. // sys_write32(0x0000100c,GPIO_REG_BASE + gpio_io_num * 0x4);
  811. // printk("%08x\n",(sys_read32(GPIO_REG_BASE + gpio_m_num * 0x4)));
  812. // printk("%08x\n",(sys_read32(GPIO_REG_BASE + gpio__b_num * 0x4)));
  813. // printk("%08x\n",(sys_read32(GPIO_REG_BASE + gpio_lr_num * 0x4)));
  814. // printk("%08x\n",(sys_read32(GPIO_REG_BASE + gpio_io_num * 0x4)));
  815. // }
  816. // else
  817. // {
  818. // //i2s rx gpio_mfc_config
  819. // sys_write32(0x0000100d,GPIO_REG_BASE + gpio_m_num * 0x4);
  820. // sys_write32(0x0000100d,GPIO_REG_BASE + gpio__b_num * 0x4);
  821. // sys_write32(0x0000100d,GPIO_REG_BASE + gpio_lr_num * 0x4);
  822. // sys_write32(0x0000100d,GPIO_REG_BASE + gpio_io_num * 0x4);
  823. // printk("%08x\n",*((volatile unsigned int*)(GPIO_REG_BASE + gpio_m_num * 0x4)));
  824. // printk("%08x\n",*((volatile unsigned int*)(GPIO_REG_BASE + gpio__b_num * 0x4)));
  825. // printk("%08x\n",*((volatile unsigned int*)(GPIO_REG_BASE + gpio_lr_num * 0x4)));
  826. // printk("%08x\n",*((volatile unsigned int*)(GPIO_REG_BASE + gpio_io_num * 0x4)));
  827. // }
  828. // return 0;
  829. // }
  830. // #endif
  831. /* @brief physical I2STX device enable */
  832. static int phy_i2stx_enable(struct device *dev, void *param)
  833. {
  834. const struct phy_i2stx_config_data *cfg = dev->config;
  835. aout_param_t *out_param = (aout_param_t *)param;
  836. i2stx_setting_t *i2stx_setting = out_param->i2stx_setting;
  837. struct acts_audio_i2stx *i2stx_reg = get_i2stx_reg_base(dev);
  838. struct phy_i2stx_drv_data *data = dev->data;
  839. int ret;
  840. bool link_dac = false, link_spdif = false;
  841. printk("phy_i2stx_enable=out_param->channel_type:%d,out_param->outfifo_type:%d\n",
  842. out_param->channel_type,out_param->outfifo_type);
  843. soc_powergate_set(POWERGATE_DSP_AU_PG_DEV, true);
  844. ret = phy_i2stx_prepare_enable(dev, out_param);
  845. if (ret) {
  846. LOG_ERR("Failed to prepare enable i2stx err=%d", ret);
  847. soc_powergate_set(POWERGATE_DSP_AU_PG_DEV, false);
  848. return ret;
  849. }
  850. if (out_param->channel_type & AUDIO_CHANNEL_DAC) {
  851. link_dac = true;
  852. data->link_with_dac = 1;
  853. data->mclksrc = CLK_SRCTX_DAC_256FS;
  854. } else if (out_param->channel_type & AUDIO_CHANNEL_SPDIFTX) {
  855. link_spdif = true;
  856. if ((AOUT_FIFO_DAC0 == out_param->outfifo_type)
  857. || (AOUT_FIFO_DAC1 == out_param->outfifo_type)) {
  858. data->mclksrc = CLK_SRCTX_DAC_256FS;
  859. /* I2STX use DAC FIFO shall set FIFO_SET to 1 */
  860. // i2stx_reg->fifoctl |= I2ST0_FIFOCTL_FIFO_SEL;
  861. } else {
  862. data->mclksrc = CLK_SRCTX_I2STX;
  863. }
  864. } else {
  865. if ((AOUT_FIFO_DAC0 == out_param->outfifo_type)
  866. || (AOUT_FIFO_DAC1 == out_param->outfifo_type)) {
  867. data->mclksrc = CLK_SRCTX_DAC_256FS;
  868. data->fifo_use = out_param->outfifo_type;
  869. /* I2STX use DAC FIFO shall set FIFO_SET to 1 */
  870. // i2stx_reg->fifoctl |= I2ST0_FIFOCTL_FIFO_SEL;
  871. } else {
  872. data->mclksrc = CLK_SRCTX_I2STX;
  873. }
  874. /* slave mode */
  875. if (PHY_DEV_FEATURE(mode)) {
  876. /* MCLK use external clock */
  877. if (!PHY_DEV_FEATURE(slave_internal_clk))
  878. data->mclksrc = CLK_SRCTX_I2STX_EXT;
  879. }
  880. }
  881. printk("phy_i2stx_enable=data->mclksrc:%d,link_dac:%d\n",data->mclksrc,link_dac);
  882. //sysconfig
  883. // sys_write32(sys_read32(AUDIOLDO_CTL) | 0x1, AUDIOLDO_CTL);//使能LD0
  884. // sys_write32(sys_read32(COREPLL_CTL) | 0x1<<7 | 0x0c, COREPLL_CTL);////10*8M=80M.
  885. // sys_write32(sys_read32(CMU_SYSCLK) |(0<<8)|(9<<4)|0x2, CMU_SYSCLK);//// cpu clk = corepll/2/10=4M
  886. /* I2STX sample rate set */
  887. if (i2stx_sample_rate_set(dev, out_param->sample_rate,
  888. data->mclksrc, link_dac ? true : false)) {
  889. LOG_ERR("Failed to config sample rate %d",
  890. out_param->sample_rate);
  891. soc_powergate_set(POWERGATE_DSP_AU_PG_DEV, false);
  892. return -ESRCH;
  893. }
  894. /*enable txfifo clock*/
  895. sys_write32(sys_read32(CMU_I2STXCLK) | CMU_I2STX0_FIFOCLKEN, CMU_I2STXCLK);
  896. LOG_INF(" phy_i2stx_enable: CMU_I2STXCLK:%08x", sys_read32(CMU_I2STXCLK));
  897. /* Linkage with DAC will force the I2STX use the DAC FIFO */
  898. if ((!link_dac) && (AOUT_FIFO_I2STX0 == out_param->outfifo_type)) {
  899. if (is_i2stx_fifo_working(dev)) {
  900. LOG_ERR("I2STX FIFO now is using ...");
  901. soc_powergate_set(POWERGATE_DSP_AU_PG_DEV, false);
  902. return -EACCES;
  903. }
  904. i2stx_fifo_enable(dev, FIFO_SEL_DMA, (out_param->channel_width == CHANNEL_WIDTH_16BITS)
  905. ? DMA_WIDTH_16BITS : DMA_WIDTH_32BITS, true);
  906. data->fifo_use = AOUT_FIFO_I2STX0;
  907. }
  908. /* I2STX linkage with SPDIFTX */
  909. // if (link_spdif)
  910. // i2stx_reg->tx_ctl |= I2ST0_CTL_MULT_DEVICE;
  911. if (!link_dac) {
  912. i2stx_digital_enable(dev, out_param->channel_width, true);
  913. } else {
  914. /* linkage with DAC shall set FIFO_SET to 1 */
  915. // i2stx_reg->fifoctl |= I2ST0_FIFOCTL_FIFO_SEL;
  916. i2stx_digital_enable(dev, out_param->channel_width, false);
  917. }
  918. if (PHY_DEV_FEATURE(srd_en)) {
  919. if ((!link_dac) && (PHY_DEV_FEATURE(mode))) {
  920. LOG_INF("I2STX SRD enable");
  921. i2stx_srd_cfg(dev, I2STX_SRD_TH_DEFAULT, I2S_SRD_2LRCLK, true, false);
  922. data->srd_callback = i2stx_setting->srd_callback;
  923. data->cb_data = i2stx_setting->cb_data;
  924. if (!PHY_DEV_FEATURE(bclk_width))
  925. data->srd_wl = SRDSTA_WL_64RATE;
  926. else
  927. data->srd_wl = SRDSTA_WL_32RATE;
  928. }
  929. }
  930. if (AOUT_FIFO_I2STX0 == data->fifo_use) {
  931. /* Clear FIFO ERROR */
  932. if (i2stx_reg->fifostat & I2ST0_FIFOSTA_FIFO_ER)
  933. i2stx_reg->fifostat |= I2ST0_FIFOSTA_FIFO_ER;
  934. }
  935. /* set flag of channel opened */
  936. data->channel_opened = 1;
  937. // //pin config for -debug
  938. // #ifdef DEBUG_I2STX_TO_RX
  939. // set_gpio_mfc(0,8,9,4,5);//tx
  940. // set_gpio_mfc(1,16,17,21,15);//rx
  941. // #endif
  942. return 0;
  943. }
  944. /* @brief physical I2STX device disable */
  945. static int phy_i2stx_disable(struct device *dev, void *param)
  946. {
  947. const struct phy_i2stx_config_data *cfg = dev->config;
  948. struct phy_i2stx_drv_data *data = dev->data;
  949. if (data->fifo_use == AOUT_FIFO_I2STX0)
  950. i2stx_fifo_disable(dev);
  951. /* Don't disable I2STX until the user call iotcl #PHY_CMD_I2STX_DISABLE_DEVICE */
  952. if (!PHY_DEV_FEATURE(always_open) && !PHY_DEV_FEATURE(mode)) {
  953. i2stx_digital_disable(dev);
  954. acts_clock_peripheral_disable(cfg->clk_id);
  955. }
  956. if (PHY_DEV_FEATURE(srd_en))
  957. acts_clock_peripheral_disable(cfg->srd_clk_id);
  958. data->srd_callback = NULL;
  959. data->cb_data = NULL;
  960. if (!PHY_DEV_FEATURE(bclk_width))
  961. data->srd_wl = SRDSTA_WL_64RATE;
  962. else
  963. data->srd_wl = SRDSTA_WL_32RATE;
  964. data->fifo_use = AUDIO_FIFO_INVALID_TYPE;
  965. data->fifo_cnt = 0;
  966. data->link_with_dac = 0;
  967. data->channel_opened = 0;
  968. soc_powergate_set(POWERGATE_DSP_AU_PG_DEV, false);
  969. return 0;
  970. }
  971. static int phy_i2stx_ioctl(struct device *dev, uint32_t cmd, void *param)
  972. {
  973. const struct phy_i2stx_config_data *cfg = dev->config;
  974. struct phy_i2stx_drv_data *data = dev->data;
  975. // struct acts_audio_i2stx *i2stx_reg = get_i2stx_reg_base(dev);
  976. int ret = 0;
  977. switch (cmd) {
  978. case PHY_CMD_DUMP_REGS:
  979. {
  980. i2stx_dump_register(dev);
  981. break;
  982. }
  983. case AOUT_CMD_GET_SAMPLERATE:
  984. {
  985. ret = i2stx_sample_rate_get(dev);
  986. if (ret < 0) {
  987. LOG_ERR("Failed to get I2STX sample rate err=%d", ret);
  988. return ret;
  989. }
  990. *(audio_sr_sel_e *)param = (audio_sr_sel_e)ret;
  991. ret = 0;
  992. break;
  993. }
  994. case AOUT_CMD_SET_SAMPLERATE:
  995. {
  996. audio_sr_sel_e val = *(audio_sr_sel_e *)param;
  997. ret = i2stx_sample_rate_set(dev, val, data->mclksrc,
  998. data->link_with_dac ? true : false);
  999. if (ret) {
  1000. LOG_ERR("Failed to set I2STX sample rate err=%d", ret);
  1001. return ret;
  1002. }
  1003. break;
  1004. }
  1005. case AOUT_CMD_GET_SAMPLE_CNT:
  1006. {
  1007. uint32_t val;
  1008. val = i2stx_read_fifocount(dev);
  1009. *(uint32_t *)param = val + data->fifo_cnt;
  1010. LOG_DBG("I2STX FIFO counter: %d", *(uint32_t *)param);
  1011. break;
  1012. }
  1013. case AOUT_CMD_RESET_SAMPLE_CNT:
  1014. {
  1015. data->fifo_cnt = 0;
  1016. i2stx_fifocount_reset(dev);
  1017. break;
  1018. }
  1019. case AOUT_CMD_ENABLE_SAMPLE_CNT:
  1020. {
  1021. i2stx_fifocount_enable(dev);
  1022. break;
  1023. }
  1024. case AOUT_CMD_DISABLE_SAMPLE_CNT:
  1025. {
  1026. i2stx_fifocount_disable(dev);
  1027. break;
  1028. }
  1029. case AOUT_CMD_GET_CHANNEL_STATUS:
  1030. {
  1031. uint8_t status = 0;
  1032. if (check_i2stx_fifo_error(dev)) {
  1033. status = AUDIO_CHANNEL_STATUS_ERROR;
  1034. LOG_DBG("I2STX FIFO ERROR");
  1035. }
  1036. if (i2stx_fifo_status(dev) < (I2STX_FIFO_LEVEL - 1))
  1037. status |= AUDIO_CHANNEL_STATUS_BUSY;
  1038. *(uint8_t *)param = status;
  1039. break;
  1040. }
  1041. case AOUT_CMD_GET_FIFO_LEN:
  1042. {
  1043. *(uint32_t *)param = I2STX_FIFO_LEVEL;
  1044. break;
  1045. }
  1046. case AOUT_CMD_GET_FIFO_AVAILABLE_LEN:
  1047. {
  1048. *(uint32_t *)param = i2stx_fifo_status(dev);
  1049. break;
  1050. }
  1051. case AOUT_CMD_GET_APS:
  1052. {
  1053. ret = i2stx_get_pll_aps(dev);
  1054. if (ret < 0) {
  1055. LOG_ERR("Failed to get audio pll APS err=%d", ret);
  1056. return ret;
  1057. }
  1058. *(audio_aps_level_e *)param = (audio_aps_level_e)ret;
  1059. ret = 0;
  1060. break;
  1061. }
  1062. case AOUT_CMD_SET_APS:
  1063. {
  1064. audio_aps_level_e level = *(audio_aps_level_e *)param;
  1065. ret = i2stx_set_pll_aps(dev, level);
  1066. if (ret) {
  1067. LOG_ERR("Failed to set audio pll APS err=%d", ret);
  1068. return ret;
  1069. }
  1070. break;
  1071. }
  1072. case PHY_CMD_FIFO_GET:
  1073. {
  1074. aout_param_t *out_param = (aout_param_t *)param;
  1075. if (!out_param)
  1076. return -EINVAL;
  1077. /* enable i2stx clock */
  1078. acts_clock_peripheral_enable(cfg->clk_id);
  1079. acts_clock_peripheral_enable(cfg->hclk_clk_id);
  1080. if (is_i2stx_fifo_working(dev)) {
  1081. LOG_ERR("I2STX FIFO is using");
  1082. return -EBUSY;
  1083. }
  1084. /* SPDIFTX use I2STX FIFO shall set FIFO_SET to 1 */
  1085. // i2stx_reg->fifoctl |= I2ST0_FIFOCTL_FIFO_SEL;
  1086. i2stx_fifo_enable(dev, FIFO_SEL_DMA,
  1087. (out_param->channel_width == CHANNEL_WIDTH_16BITS)
  1088. ? DMA_WIDTH_16BITS : DMA_WIDTH_32BITS, false);
  1089. break;
  1090. }
  1091. case PHY_CMD_FIFO_PUT:
  1092. {
  1093. if (is_i2stx_fifo_working(dev))
  1094. i2stx_fifo_disable(dev);
  1095. data->fifo_cnt = 0;
  1096. break;
  1097. }
  1098. case PHY_CMD_I2STX_IS_OPENED:
  1099. {
  1100. *(uint8_t *)param = data->channel_opened;
  1101. break;
  1102. }
  1103. case PHY_CMD_I2STX_DISABLE_DEVICE:
  1104. {
  1105. i2stx_digital_disable(dev);
  1106. acts_clock_peripheral_disable(cfg->clk_id);
  1107. acts_clock_peripheral_disable(cfg->hclk_clk_id);
  1108. acts_clock_peripheral_disable(cfg->srd_clk_id);
  1109. break;
  1110. }
  1111. case PHY_CMD_I2STX_CLK_SET:
  1112. {
  1113. uint8_t sr = *(uint8_t *)param;
  1114. struct board_pinmux_info pinmux_info;
  1115. acts_clock_peripheral_enable(cfg->clk_id);
  1116. board_get_i2stx0_pinmux_info(&pinmux_info);
  1117. acts_pinmux_setup_pins(pinmux_info.pins_config, pinmux_info.pins_num);
  1118. ret = i2stx_sample_rate_set(dev, sr, CLK_SRCTX_I2STX, false);
  1119. break;
  1120. }
  1121. case PHY_CMD_GET_AOUT_DMA_INFO:
  1122. {
  1123. struct audio_out_dma_info *info = (struct audio_out_dma_info *)param;
  1124. info->dma_info.dma_chan = cfg->dma_fifo0.dma_chan;
  1125. info->dma_info.dma_dev_name = cfg->dma_fifo0.dma_dev_name;
  1126. info->dma_info.dma_id = cfg->dma_fifo0.dma_id;
  1127. break;
  1128. }
  1129. case PHY_CMD_I2STX_IS_MCLK_128FS:
  1130. {
  1131. if (PHY_DEV_FEATURE(bclk_width))
  1132. *(uint8_t *)param = 1;
  1133. else
  1134. *(uint8_t *)param = 0;
  1135. break;
  1136. }
  1137. case PHY_CMD_I2S_LOOPBACK:
  1138. {
  1139. uint8_t loopback_en = *(uint8_t *)param;
  1140. struct acts_audio_i2stx *i2stx_reg = get_i2stx_reg_base(dev);
  1141. if(loopback_en == 1)
  1142. i2stx_reg->tx_ctl |= I2ST0_CTL_LPEN0;
  1143. else
  1144. i2stx_reg->tx_ctl &= ~I2ST0_CTL_LPEN0;
  1145. LOG_INF("PHY_CMD_I2S_LOOPBACK===enabel:%d\n", loopback_en);
  1146. break;
  1147. }
  1148. // case PHY_CMD_DAC_FIFO_VOLUME_SET:
  1149. // {
  1150. // uint8_t volume = *(uint8_t *)param;
  1151. // ret = i2stx_fifo_volume_set(dev, volume);
  1152. // break;
  1153. // }
  1154. // case PHY_CMD_DAC_FIFO_VOLUME_GET:
  1155. // {
  1156. // ret = i2stx_fifo_volume_get(dev);
  1157. // if (ret < 0) {
  1158. // LOG_ERR("Get I2STX FIFO volume error=%d", ret);
  1159. // return ret;
  1160. // }
  1161. // *(uint8_t *)param = ret;
  1162. // ret = 0;
  1163. // break;
  1164. // }
  1165. default:
  1166. LOG_ERR("Unsupport command %d", cmd);
  1167. return -ENOTSUP;
  1168. }
  1169. return ret;
  1170. }
  1171. const struct phy_audio_driver_api phy_i2stx_drv_api = {
  1172. .audio_enable = phy_i2stx_enable,
  1173. .audio_disable = phy_i2stx_disable,
  1174. .audio_ioctl = phy_i2stx_ioctl,
  1175. };
  1176. /* dump i2stx device tree infomation */
  1177. static void __i2stx_dt_dump_info(const struct phy_i2stx_config_data *cfg)
  1178. {
  1179. #if (PHY_DEV_SHOW_DT_INFO == 1)
  1180. LOG_INF("** I2STX BASIC INFO **");
  1181. LOG_INF(" BASE: %08x", cfg->reg_base);
  1182. LOG_INF(" CLK-ID: %08x", cfg->clk_id);
  1183. LOG_INF("SRDCLK-ID: %08x", cfg->srd_clk_id);
  1184. LOG_INF(" RST-ID: %08x", cfg->rst_id);
  1185. LOG_INF("DMA0-NAME: %s", cfg->dma_fifo0.dma_dev_name);
  1186. LOG_INF(" DMA0-ID: %08x", cfg->dma_fifo0.dma_id);
  1187. LOG_INF(" DMA0-CH: %08x", cfg->dma_fifo0.dma_chan);
  1188. LOG_INF("** I2STX FEATURES **");
  1189. LOG_INF(" SRD-EN: %d", PHY_DEV_FEATURE(srd_en));
  1190. LOG_INF(" MODE: %d", PHY_DEV_FEATURE(mode));
  1191. LOG_INF(" BCLK-WIDTH: %d", PHY_DEV_FEATURE(bclk_width));
  1192. LOG_INF(" CH-NUM: %d", PHY_DEV_FEATURE(channel_num));
  1193. LOG_INF("INTERNAL-CLK: %d", PHY_DEV_FEATURE(slave_internal_clk));
  1194. LOG_INF(" LRCLK-PROC: %d", PHY_DEV_FEATURE(lrclk_proc));
  1195. LOG_INF("MCLK-REVERSE: %d", PHY_DEV_FEATURE(mclk_reverse));
  1196. LOG_INF(" ALWAYS-OPEN: %d", PHY_DEV_FEATURE(always_open));
  1197. LOG_INF(" TDM-FORMAT: %d", PHY_DEV_FEATURE(tdm_format));
  1198. LOG_INF(" TDM-FRAME: %d", PHY_DEV_FEATURE(tdm_frame));
  1199. LOG_INF(" TDM-FRAME: %d", PHY_DEV_FEATURE(txd_delay));
  1200. #endif
  1201. }
  1202. static void phy_i2stx_isr(const void *arg)
  1203. {
  1204. struct device *dev = (struct device *)arg;
  1205. struct acts_audio_i2stx *i2stx_reg = get_i2stx_reg_base(dev);
  1206. struct phy_i2stx_drv_data *data = dev->data;
  1207. LOG_DBG("fifocnt 0x%x srdstat 0x%x", i2stx_reg->fifocnt, i2stx_reg->srdstat);
  1208. if (i2stx_reg->fifocnt & I2ST0_FIFO_CNT_IP) {
  1209. data->fifo_cnt += (AOUT_FIFO_CNT_MAX + 1);
  1210. /* Here we need to wait 100us for the synchronization of audio clock fields */
  1211. k_busy_wait(100);
  1212. i2stx_reg->fifocnt |= I2ST0_FIFO_CNT_IP;
  1213. }
  1214. /* Sample rate detection timeout irq pending */
  1215. if (i2stx_reg->srdstat & I2ST0_SRDSTA_TO_PD) {
  1216. i2stx_reg->srdstat |= I2ST0_SRDSTA_TO_PD;
  1217. if (data->srd_callback)
  1218. data->srd_callback(data->cb_data, I2STX_SRD_TIMEOUT, NULL);
  1219. }
  1220. /* Sample rate changed detection irq pending */
  1221. if (i2stx_reg->srdstat & I2ST0_SRDSTA_SRC_PD) {
  1222. i2stx_reg->srdstat |= I2ST0_SRDSTA_SRC_PD;
  1223. i2stx_srd_fs_change(dev);
  1224. }
  1225. /* Channel width change irq pending */
  1226. if (i2stx_reg->srdstat & I2ST0_SRDSTA_CHW_PD) {
  1227. i2stx_reg->srdstat |= I2ST0_SRDSTA_CHW_PD;
  1228. i2stx_srd_wl_change(dev);
  1229. }
  1230. }
  1231. #ifdef CONFIG_CFG_DRV
  1232. /* @brief initialize I2STX external configuration */
  1233. static int phy_i2stx_config_init(const struct device *dev)
  1234. {
  1235. struct phy_i2stx_drv_data *data = dev->data;
  1236. struct acts_pin_config i2stx_pins_state[4] = {0};
  1237. int ret;
  1238. ret = cfg_get_by_key(ITEM_AUDIO_I2STX_SELECT_GPIO,
  1239. &data->external_config.I2STX_Select_GPIO, sizeof(data->external_config.I2STX_Select_GPIO));
  1240. if (ret) {
  1241. LOG_INF("** I2STX PINMUX **");
  1242. LOG_INF("I2S_MCLK:%d", data->external_config.I2STX_Select_GPIO.I2S_MCLK);
  1243. LOG_INF("I2S_LRCLK:%d", data->external_config.I2STX_Select_GPIO.I2S_LRCLK);
  1244. LOG_INF("I2S_BCLK:%d", data->external_config.I2STX_Select_GPIO.I2S_BCLK);
  1245. LOG_INF("I2S_DOUT:%d", data->external_config.I2STX_Select_GPIO.I2S_DOUT);
  1246. }
  1247. if ((data->external_config.I2STX_Select_GPIO.I2S_MCLK != GPIO_NONE)
  1248. && (data->external_config.I2STX_Select_GPIO.I2S_LRCLK != GPIO_NONE)
  1249. && (data->external_config.I2STX_Select_GPIO.I2S_BCLK != GPIO_NONE)
  1250. && (data->external_config.I2STX_Select_GPIO.I2S_DOUT != GPIO_NONE)) {
  1251. i2stx_pins_state[0].pin_num = data->external_config.I2STX_Select_GPIO.I2S_MCLK;
  1252. i2stx_pins_state[0].mode = 12;
  1253. i2stx_pins_state[1].pin_num = data->external_config.I2STX_Select_GPIO.I2S_LRCLK;
  1254. i2stx_pins_state[1].mode = 12;
  1255. i2stx_pins_state[2].pin_num = data->external_config.I2STX_Select_GPIO.I2S_BCLK;
  1256. i2stx_pins_state[2].mode = 12;
  1257. i2stx_pins_state[3].pin_num = data->external_config.I2STX_Select_GPIO.I2S_DOUT;
  1258. i2stx_pins_state[3].mode = 12;
  1259. acts_pinmux_setup_pins(i2stx_pins_state, ARRAY_SIZE(i2stx_pins_state));
  1260. }
  1261. return 0;
  1262. }
  1263. #endif
  1264. static int phy_i2stx_init(const struct device *dev)
  1265. {
  1266. const struct phy_i2stx_config_data *cfg = dev->config;
  1267. struct phy_i2stx_drv_data *data = dev->data;
  1268. /* clear driver data */
  1269. memset(data, 0, sizeof(struct phy_i2stx_drv_data));
  1270. __i2stx_dt_dump_info(cfg);
  1271. #ifdef CONFIG_CFG_DRV
  1272. phy_i2stx_config_init(dev);
  1273. #endif
  1274. /* reset I2STX controller */
  1275. acts_reset_peripheral(cfg->rst_id);
  1276. acts_reset_peripheral(RESET_ID_DAC);//RESET DAC
  1277. if (cfg->irq_config)
  1278. cfg->irq_config();
  1279. data->fifo_use = AUDIO_FIFO_INVALID_TYPE;
  1280. printk("I2STX init successfully\n");
  1281. return 0;
  1282. }
  1283. static void phy_i2stx_irq_config(void);
  1284. /* physical i2stx driver data */
  1285. static struct phy_i2stx_drv_data phy_i2stx_drv_data0;
  1286. /* physical i2stx config data CLOCK_ID_DAC RESET_ID_DAC RESET_ID_ADC*/
  1287. static const struct phy_i2stx_config_data phy_i2stx_config_data0 = {
  1288. .reg_base = AUDIO_I2STX0_REG_BASE,
  1289. .clk_id = CLOCK_ID_I2STX,
  1290. .srd_clk_id = CLOCK_ID_I2SSRDCLK,
  1291. .hclk_clk_id = CLOCK_ID_I2SHCLKEN,
  1292. .rst_id = RESET_ID_I2STX,
  1293. AUDIO_DMA_FIFO_DEF(I2STX, 0),
  1294. .irq_config = phy_i2stx_irq_config,
  1295. PHY_DEV_FEATURE_DEF(format) = CONFIG_AUDIO_I2STX_0_FORMAT,
  1296. PHY_DEV_FEATURE_DEF(srd_en) = CONFIG_AUDIO_I2STX_0_SRD_EN,
  1297. PHY_DEV_FEATURE_DEF(mode) = CONFIG_AUDIO_I2STX_0_MODE,
  1298. PHY_DEV_FEATURE_DEF(bclk_width) = CONFIG_AUDIO_I2STX_0_BCLK_WIDTH,
  1299. PHY_DEV_FEATURE_DEF(channel_num) = CONFIG_AUDIO_I2STX_0_CHANNEL_NUM,
  1300. PHY_DEV_FEATURE_DEF(slave_internal_clk) = CONFIG_AUDIO_I2STX_0_SLAVE_INTERNAL_CLK,
  1301. PHY_DEV_FEATURE_DEF(lrclk_proc) = CONFIG_AUDIO_I2STX_0_LRCLK_PROC,
  1302. PHY_DEV_FEATURE_DEF(mclk_reverse) = CONFIG_AUDIO_I2STX_0_MCLK_REVERSE,
  1303. PHY_DEV_FEATURE_DEF(always_open) = CONFIG_AUDIO_I2STX_0_ALWAYS_OPEN,
  1304. PHY_DEV_FEATURE_DEF(tdm_format) = CONFIG_AUDIO_I2STX_0_TDM_FORMAT,
  1305. PHY_DEV_FEATURE_DEF(tdm_frame) = CONFIG_AUDIO_I2STX_0_TDM_FRAME,
  1306. PHY_DEV_FEATURE_DEF(txd_delay) = CONFIG_AUDIO_I2STX_0_TX_DELAY,
  1307. PHY_DEV_FEATURE_DEF(pcm_en) = CONFIG_AUDIO_PCMTX_0_EN,
  1308. PHY_DEV_FEATURE_DEF(pcm_frame) = CONFIG_AUDIO_PCMTX_0_FORMART,
  1309. PHY_DEV_FEATURE_DEF(pcm_slot) = CONFIG_AUDIO_PCMTX_0_SLOT,
  1310. };
  1311. #if IS_ENABLED(CONFIG_AUDIO_I2STX_0)
  1312. DEVICE_DEFINE(i2stx0, CONFIG_AUDIO_I2STX_0_NAME, phy_i2stx_init, NULL,
  1313. &phy_i2stx_drv_data0, &phy_i2stx_config_data0,
  1314. POST_KERNEL, CONFIG_KERNEL_INIT_PRIORITY_OBJECTS, &phy_i2stx_drv_api);
  1315. #endif
  1316. /*
  1317. * @brief Enable I2STX IRQ
  1318. * @note I2STX IRQ source as shown below:
  1319. * - I2STX FIFO Half Empty IRQ
  1320. * - I2STX FIFO CNT IRQ
  1321. * - I2STX SRDTO IRQ
  1322. * - I2STX SRDSR IRQ
  1323. * - I2STX SRDCHW IRQ
  1324. */
  1325. static void phy_i2stx_irq_config(void)
  1326. {
  1327. IRQ_CONNECT(IRQ_ID_I2S0, CONFIG_AUDIO_I2STX_0_IRQ_PRI,
  1328. phy_i2stx_isr,
  1329. DEVICE_GET(i2stx0), 0);
  1330. irq_enable(IRQ_ID_I2S0);
  1331. }