drv_codec.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403
  1. /*
  2. * Copyright (c) 2006-2018, RT-Thread Development Team
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. *
  6. * Change Logs:
  7. * Date Author Notes
  8. */
  9. #include <rthw.h>
  10. #include <rtthread.h>
  11. #include <rtdevice.h>
  12. #include "board.h"
  13. #include "drv_codec.h"
  14. #include "fsl_wm8960.h"
  15. #include <fsl_sai.h>
  16. #include <fsl_sai_edma.h>
  17. #include <fsl_lpi2c.h>
  18. #include <fsl_dmamux.h>
  19. #define DEMO_CODEC_WM8960
  20. #define DEMO_SAI SAI1
  21. #define DEMO_SAI_IRQ SAI1_IRQn
  22. #define SAI_TxIRQHandler SAI1_IRQHandler
  23. /* Select Audio/Video PLL (786.48 MHz) as sai1 clock source */
  24. #define DEMO_SAI1_CLOCK_SOURCE_SELECT (2U)
  25. /* Clock pre divider for sai1 clock source */
  26. #define DEMO_SAI1_CLOCK_SOURCE_PRE_DIVIDER (1U)
  27. /* Clock divider for sai1 clock source */
  28. #define DEMO_SAI1_CLOCK_SOURCE_DIVIDER (63U)
  29. /* Get frequency of sai1 clock */
  30. #define DEMO_SAI_CLK_FREQ (CLOCK_GetFreq(kCLOCK_AudioPllClk) / (DEMO_SAI1_CLOCK_SOURCE_DIVIDER + 1U) / (DEMO_SAI1_CLOCK_SOURCE_PRE_DIVIDER + 1U))
  31. /* I2C instance and clock */
  32. #define DEMO_I2C LPI2C1
  33. /* Select USB1 PLL (480 MHz) as master lpi2c clock source */
  34. #define DEMO_LPI2C_CLOCK_SOURCE_SELECT (0U)
  35. /* Clock divider for master lpi2c clock source */
  36. #define DEMO_LPI2C_CLOCK_SOURCE_DIVIDER (5U)
  37. /* Get frequency of lpi2c clock */
  38. #define DEMO_I2C_CLK_FREQ ((CLOCK_GetFreq(kCLOCK_Usb1PllClk) / 8) / (DEMO_LPI2C_CLOCK_SOURCE_DIVIDER + 1U))
  39. /* DMA */
  40. #define DMAMUX0 DMAMUX
  41. #define EXAMPLE_DMA DMA0
  42. #define EXAMPLE_CHANNEL (0U)
  43. #define EXAMPLE_SAI_TX_SOURCE kDmaRequestMuxSai1Tx
  44. struct imxcodec
  45. {
  46. I2S_Type *sai;
  47. sai_edma_handle_t txHandle;
  48. wm8960_handle_t codecHandle;
  49. edma_handle_t dmaHandle;
  50. lpi2c_master_handle_t i2cHandle;
  51. sai_transfer_format_t format;
  52. };
  53. static void _InitPins(void)
  54. {
  55. CLOCK_EnableClock(kCLOCK_Iomuxc);
  56. IOMUXC_SetPinMux(IOMUXC_GPIO_AD_B1_00_LPI2C1_SCL, 1);
  57. IOMUXC_SetPinMux(IOMUXC_GPIO_AD_B1_01_LPI2C1_SDA, 1);
  58. IOMUXC_SetPinConfig( IOMUXC_GPIO_AD_B1_00_LPI2C1_SCL, 0xD8B0u);
  59. IOMUXC_SetPinConfig( IOMUXC_GPIO_AD_B1_01_LPI2C1_SDA, 0xD8B0u);
  60. IOMUXC_SetPinMux(IOMUXC_GPIO_AD_B1_09_SAI1_MCLK, 1U);
  61. IOMUXC_SetPinMux(IOMUXC_GPIO_AD_B1_13_SAI1_TX_DATA00, 1U);
  62. IOMUXC_SetPinMux(IOMUXC_GPIO_AD_B1_14_SAI1_TX_BCLK, 1U);
  63. IOMUXC_SetPinMux(IOMUXC_GPIO_AD_B1_15_SAI1_TX_SYNC, 1U);
  64. IOMUXC_SetPinConfig(IOMUXC_GPIO_AD_B1_09_SAI1_MCLK, 0x10B0u);
  65. IOMUXC_SetPinConfig(IOMUXC_GPIO_AD_B1_13_SAI1_TX_DATA00, 0x10B0u);
  66. IOMUXC_SetPinConfig(IOMUXC_GPIO_AD_B1_14_SAI1_TX_BCLK, 0x10B0u);
  67. IOMUXC_SetPinConfig(IOMUXC_GPIO_AD_B1_15_SAI1_TX_SYNC, 0x10B0u);
  68. }
  69. static void BOARD_EnableSaiMclkOutput(bool enable)
  70. {
  71. if (enable)
  72. {
  73. IOMUXC_GPR->GPR1 |= IOMUXC_GPR_GPR1_SAI1_MCLK_DIR_MASK;
  74. }
  75. else
  76. {
  77. IOMUXC_GPR->GPR1 &= (~IOMUXC_GPR_GPR1_SAI1_MCLK_DIR_MASK);
  78. }
  79. }
  80. static void saidma_callback(I2S_Type *base, sai_edma_handle_t *handle, status_t status, void *userData)
  81. {
  82. int ind = 0;
  83. rt_uint8_t *saddr;
  84. ind = handle->queueDriver;
  85. saddr = (rt_uint8_t*)handle->saiQueue[ind].data;
  86. rt_audio_tx_complete(userData, saddr);
  87. }
  88. /*********************************************************************************************************
  89. ** Audio device
  90. *********************************************************************************************************/
  91. static rt_err_t icodec_getcaps(struct rt_audio_device *audio,struct rt_audio_caps *caps)
  92. {
  93. rt_err_t result = RT_EOK;
  94. struct imxcodec *icodec = (struct imxcodec *)audio->parent.user_data;
  95. switch (caps->main_type)
  96. {
  97. case AUDIO_TYPE_QUERY: /* qurey the types of hw_codec device */
  98. {
  99. switch (caps->sub_type)
  100. {
  101. case AUDIO_TYPE_QUERY:
  102. caps->udata.mask = AUDIO_TYPE_OUTPUT | AUDIO_TYPE_MIXER;
  103. break;
  104. default:
  105. result = -RT_ERROR;
  106. break;
  107. }
  108. break;
  109. }
  110. case AUDIO_TYPE_OUTPUT: /* Provide capabilities of OUTPUT unit */
  111. switch (caps->sub_type)
  112. {
  113. case AUDIO_DSP_PARAM:
  114. if (audio->replay == NULL)
  115. {
  116. result = -RT_ERROR;
  117. break;
  118. }
  119. caps->udata.config.channels = 1;
  120. caps->udata.config.samplefmt = 1;
  121. caps->udata.config.samplerate = 1;
  122. caps->udata.config.samplefmts = 1;
  123. break;
  124. default:
  125. result = -RT_ERROR;
  126. break;
  127. }
  128. break;
  129. case AUDIO_TYPE_MIXER: /* report the Mixer Units */
  130. switch (caps->sub_type)
  131. {
  132. case AUDIO_MIXER_QUERY:
  133. caps->udata.mask = AUDIO_MIXER_VOLUME | AUDIO_MIXER_DIGITAL | AUDIO_MIXER_LINE;
  134. break;
  135. case AUDIO_MIXER_VOLUME:
  136. caps->udata.value = WM8960_GetVolume(&icodec->codecHandle, kWM8960_ModuleDAC);
  137. break;
  138. case AUDIO_MIXER_DIGITAL:
  139. break;
  140. case AUDIO_MIXER_LINE:
  141. break;
  142. default:
  143. result = -RT_ERROR;
  144. break;
  145. }
  146. break;
  147. default:
  148. result = -RT_ERROR;
  149. break;
  150. }
  151. return result;
  152. }
  153. static rt_err_t icodec_configure(struct rt_audio_device *audio,struct rt_audio_caps *caps)
  154. {
  155. rt_err_t result = RT_EOK;
  156. struct imxcodec *icodec = (struct imxcodec *)audio->parent.user_data;
  157. switch (caps->main_type)
  158. {
  159. case AUDIO_TYPE_MIXER:
  160. {
  161. switch (caps->sub_type)
  162. {
  163. case AUDIO_MIXER_VOLUME:
  164. {
  165. WM8960_SetVolume(&icodec->codecHandle, kWM8960_ModuleDAC,
  166. caps->udata.value);
  167. }
  168. break;
  169. default:
  170. {
  171. result = -RT_ERROR;
  172. }
  173. break;
  174. }
  175. }
  176. break;
  177. case AUDIO_TYPE_OUTPUT:
  178. {
  179. switch (caps->sub_type)
  180. {
  181. case AUDIO_DSP_PARAM:
  182. {
  183. } break;
  184. case AUDIO_DSP_SAMPLERATE:
  185. {
  186. int rate = caps->udata.value;
  187. icodec->format.sampleRate_Hz = rate;
  188. SAI_TxSetFormat(icodec->sai, &icodec->format, icodec->format.masterClockHz, icodec->format.masterClockHz);
  189. }
  190. break;
  191. default:
  192. {
  193. result = -RT_ERROR;
  194. }
  195. break;
  196. }
  197. }
  198. break;
  199. default:
  200. result = -RT_ERROR;
  201. break;
  202. }
  203. return result;
  204. }
  205. static rt_err_t icodec_init(struct rt_audio_device *audio)
  206. {
  207. sai_config_t config;
  208. uint32_t mclkSourceClockHz = 0U;
  209. edma_config_t dmaConfig = {0};
  210. lpi2c_master_config_t i2cConfig = {0};
  211. uint32_t i2cSourceClock;
  212. clock_audio_pll_config_t audioPllConfig = {32, 1, 77, 100};
  213. struct imxcodec *icodec = audio->parent.user_data;
  214. sai_transfer_format_t *format;
  215. icodec->sai = DEMO_SAI;
  216. format = &icodec->format;
  217. _InitPins();
  218. CLOCK_InitAudioPll(&audioPllConfig);
  219. /*Clock setting for LPI2C*/
  220. CLOCK_SetMux(kCLOCK_Lpi2cMux, DEMO_LPI2C_CLOCK_SOURCE_SELECT);
  221. CLOCK_SetDiv(kCLOCK_Lpi2cDiv, DEMO_LPI2C_CLOCK_SOURCE_DIVIDER);
  222. /*Clock setting for SAI1*/
  223. CLOCK_SetMux(kCLOCK_Sai1Mux, DEMO_SAI1_CLOCK_SOURCE_SELECT);
  224. CLOCK_SetDiv(kCLOCK_Sai1PreDiv, DEMO_SAI1_CLOCK_SOURCE_PRE_DIVIDER);
  225. CLOCK_SetDiv(kCLOCK_Sai1Div, DEMO_SAI1_CLOCK_SOURCE_DIVIDER);
  226. /*Enable MCLK clock*/
  227. BOARD_EnableSaiMclkOutput(true);
  228. /* Create EDMA handle */
  229. EDMA_GetDefaultConfig(&dmaConfig);
  230. EDMA_Init(EXAMPLE_DMA, &dmaConfig);
  231. EDMA_CreateHandle(&icodec->dmaHandle, EXAMPLE_DMA, EXAMPLE_CHANNEL);
  232. DMAMUX_Init(DMAMUX0);
  233. DMAMUX_SetSource(DMAMUX0, EXAMPLE_CHANNEL, EXAMPLE_SAI_TX_SOURCE);
  234. DMAMUX_EnableChannel(DMAMUX0, EXAMPLE_CHANNEL);
  235. /* Init SAI module */
  236. SAI_TxGetDefaultConfig(&config);
  237. config.protocol = kSAI_BusLeftJustified;
  238. SAI_TxInit(DEMO_SAI, &config);
  239. /* Configure the audio format */
  240. format->bitWidth = kSAI_WordWidth16bits;
  241. format->channel = 0U;
  242. format->sampleRate_Hz = kSAI_SampleRate48KHz;
  243. format->masterClockHz = DEMO_SAI_CLK_FREQ;
  244. format->protocol = config.protocol;
  245. format->stereo = kSAI_Stereo;
  246. format->isFrameSyncCompact = 0;
  247. format->watermark = FSL_FEATURE_SAI_FIFO_COUNT / 2U;
  248. /* Configure Sgtl5000 I2C */
  249. icodec->codecHandle.base = DEMO_I2C;
  250. icodec->codecHandle.i2cHandle = &icodec->i2cHandle;
  251. i2cSourceClock = DEMO_I2C_CLK_FREQ;
  252. LPI2C_MasterGetDefaultConfig(&i2cConfig);
  253. LPI2C_MasterInit(DEMO_I2C, &i2cConfig, i2cSourceClock);
  254. LPI2C_MasterTransferCreateHandle(DEMO_I2C, &icodec->i2cHandle, NULL, NULL);
  255. WM8960_Init(&icodec->codecHandle, NULL);
  256. WM8960_ConfigDataFormat(&icodec->codecHandle, format->masterClockHz, format->sampleRate_Hz, format->bitWidth);
  257. SAI_TransferTxCreateHandleEDMA(icodec->sai, &icodec->txHandle, saidma_callback, audio, &icodec->dmaHandle);
  258. mclkSourceClockHz = DEMO_SAI_CLK_FREQ;
  259. SAI_TransferTxSetFormatEDMA(icodec->sai, &icodec->txHandle, format, mclkSourceClockHz, format->masterClockHz);
  260. return RT_EOK;
  261. }
  262. static rt_err_t icodec_shutdown(struct rt_audio_device *audio)
  263. {
  264. return RT_EOK;
  265. }
  266. rt_err_t icodec_start(struct rt_audio_device *audio,int stream)
  267. {
  268. return RT_EOK;
  269. }
  270. rt_err_t icodec_stop(struct rt_audio_device *audio,int stream)
  271. {
  272. return RT_EOK;
  273. }
  274. static rt_err_t icodec_suspend(struct rt_audio_device *audio,int stream)
  275. {
  276. return RT_EOK;
  277. }
  278. static rt_err_t icodec_resume(struct rt_audio_device *audio,int stream)
  279. {
  280. return RT_EOK;
  281. }
  282. static rt_err_t icodec_control (struct rt_audio_device *audio, int cmd, void *args)
  283. {
  284. rt_err_t result = RT_EOK;
  285. switch (cmd)
  286. {
  287. case AUDIO_CTL_HWRESET:
  288. break;
  289. default:
  290. result = -RT_ERROR;
  291. break;
  292. }
  293. return result;
  294. }
  295. static rt_size_t icodec_transmit(struct rt_audio_device *audio, const void *writeBuf, void *readBuf, rt_size_t size)
  296. {
  297. struct imxcodec *icodec = (struct imxcodec *)audio->parent.user_data;
  298. if(writeBuf != RT_NULL)
  299. {
  300. sai_transfer_t xfer;
  301. xfer.data = (uint8_t *)writeBuf;
  302. xfer.dataSize = size;
  303. if (size%32 == 0)
  304. icodec->txHandle.count = 16;
  305. else
  306. icodec->txHandle.count = 1;
  307. rt_hw_cpu_dcache_ops(RT_HW_CACHE_FLUSH, (void*)writeBuf, size);
  308. if (SAI_TransferSendEDMA(icodec->sai, &icodec->txHandle, &xfer) != kStatus_Success)
  309. return 0;
  310. return size;
  311. }
  312. return 0;
  313. }
  314. static struct imxcodec _g_imxcodec;
  315. static struct rt_audio_device _g_audio_device;
  316. const struct rt_audio_ops _g_audio_ops =
  317. {
  318. .getcaps = icodec_getcaps,
  319. .configure = icodec_configure,
  320. .init = icodec_init,
  321. .shutdown = icodec_shutdown,
  322. .start = icodec_start,
  323. .stop = icodec_stop,
  324. .suspend = icodec_suspend,
  325. .resume = icodec_resume,
  326. .control = icodec_control,
  327. .transmit = icodec_transmit,
  328. };
  329. int rt_hw_codec_init(void)
  330. {
  331. int result;
  332. struct rt_audio_device *audio = &_g_audio_device;
  333. audio->ops = (struct rt_audio_ops*)&_g_audio_ops;
  334. _g_imxcodec.sai = DEMO_SAI;
  335. result = rt_audio_register(audio,"sound0", RT_DEVICE_FLAG_WRONLY, &_g_imxcodec);
  336. return result;
  337. }
  338. INIT_DEVICE_EXPORT(rt_hw_codec_init);