phy_audio_common.h 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430
  1. /*
  2. * Copyright (c) 2020 Actions Semiconductor Co., Ltd
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief AUDIO physical common API
  9. */
  10. #ifndef __PHY_AUDIO_COMMON_H__
  11. #define __PHY_AUDIO_COMMON_H__
  12. #include <device.h>
  13. #ifdef CONFIG_SOC_SERIES_LARK
  14. #include "audio_acts_utils.h"
  15. #endif
  16. #ifdef CONFIG_SOC_SERIES_LEOPARD
  17. #include "phy_leopard/audio_acts_utils.h"
  18. #endif
  19. #ifndef BIT
  20. #define BIT(n) (1UL << (n))
  21. #endif
  22. /* @brief Definition for the physical audio control internal commands */
  23. /************************ Common IO commands **************************/
  24. #define PHY_CMD_BASE (0xFF)
  25. #define PHY_CMD_OFFSET (64)
  26. #define PHY_CMD_DUMP_REGS (PHY_CMD_BASE + 1)
  27. /*! Dump the audio in/out controller regiser for debug
  28. * int phy_audio_control(dev, #PHY_CMD_DUMP_REGS, NULL)
  29. */
  30. #define PHY_CMD_FIFO_GET (PHY_CMD_BASE + 2)
  31. /*! Get a FIFO for using from the speicified audio channel
  32. * int phy_audio_control(dev, #PHY_CMD_FIFO_GET, void *param)
  33. */
  34. #define PHY_CMD_FIFO_PUT (PHY_CMD_BASE + 3)
  35. /*! Put the FIFO to the speicified audio channel
  36. * int phy_audio_control(dev, #PHY_CMD_FIFO_PUT, NULL)
  37. */
  38. #define PHY_CMD_FIFO_DRQ_LEVEL_GET (PHY_CMD_BASE + 16)
  39. /*! Enable the specified index of the DAC FIFO sample counter function.
  40. * int phy_audio_control(dev, #PHY_CMD_FIFO_DRQ_LEVEL_GET, uint32_t *io)
  41. * For #io can refer to #PHY_FIFO_CMD(i, x)
  42. */
  43. #define PHY_CMD_FIFO_DRQ_LEVEL_SET (PHY_CMD_BASE + 17)
  44. /*! Enable the specified index of the DAC FIFO sample counter function.
  45. * int phy_audio_control(dev, #PHY_CMD_FIFO_DRQ_LEVEL_SET, uint32_t *io)
  46. * #io can refer to #PHY_FIFO_CMD(i, x)
  47. */
  48. #define PHY_CMD_GET_AIN_DMA_INFO (PHY_CMD_BASE + 18)
  49. /*! Get the audio in DMA information such as dma slot id from phydical audio device.
  50. * int phy_audio_control(dev, #PHY_CMD_GET_AIN_DMA_INFO, struct audio_in_dma_info *info)
  51. */
  52. #define PHY_CMD_GET_AOUT_DMA_INFO (PHY_CMD_BASE + 19)
  53. /*! Get the audio out DMA information such as dma slot id from phydical audio device.
  54. * int phy_audio_control(dev, #PHY_CMD_GET_AOUT_DMA_INFO, struct audio_out_dma_info *info)
  55. */
  56. #define PHY_CMD_CHANNEL_START (PHY_CMD_BASE + 20)
  57. /*! Start the phydical audio device.
  58. * int phy_audio_control(dev, #PHY_CMD_CHANNEL_START, void *param)
  59. */
  60. /************************ DAC IO commands **************************/
  61. #define PHY_CMD_DAC_BASE (PHY_CMD_BASE + PHY_CMD_OFFSET)
  62. #define PHY_CMD_DAC_WAIT_EMPTY (PHY_CMD_DAC_BASE + 1)
  63. /*! Check and wait the empty pending of DAC FIFO by DAC FIFO index
  64. * int phy_audio_control(dev, #PHY_CMD_DAC_CHECK_EMPTY, uint8_t *fifo_idx)
  65. */
  66. #define PHY_CMD_DAC_FIFO_GET_SAMPLE_CNT (PHY_CMD_DAC_BASE + 2)
  67. /*! Get the DAC FIFO sample counter by DAC FIFO index.
  68. * int phy_audio_control(dev, #PHY_CMD_DAC_FIFO0_GET_SAMPLE_CNT, uint32_t *idx)
  69. */
  70. #define PHY_CMD_DAC_FIFO_RESET_SAMPLE_CNT (PHY_CMD_DAC_BASE + 3)
  71. /*! Reset the specified index of the DAC FIFO sample counter function.
  72. * int phy_audio_control(dev, #PHY_CMD_DAC_FIFO_RESET_SAMPLE_CNT, uint8_t *idx)
  73. */
  74. #define PHY_CMD_DAC_FIFO_DISABLE_SAMPLE_CNT (PHY_CMD_DAC_BASE + 4)
  75. /*! Disable the specified index of the DAC FIFO sample counter function.
  76. * int phy_audio_control(dev, #PHY_CMD_DAC_FIFO_RESET_SAMPLE_CNT, uint8_t *idx)
  77. */
  78. #define PHY_CMD_DAC_FIFO_ENABLE_SAMPLE_CNT (PHY_CMD_DAC_BASE + 5)
  79. /*! Enable the specified index of the DAC FIFO sample counter function.
  80. * int phy_audio_control(dev, #PHY_CMD_DAC_FIFO_RESET_SAMPLE_CNT, uint8_t *idx)
  81. */
  82. #define PHY_CMD_DAC_FIFO_VOLUME_GET (PHY_CMD_DAC_BASE + 6)
  83. /*! Get the DAC FIFO volume value.
  84. * int phy_audio_control(dev, #PHY_CMD_DAC_FIFO_VOLUME_GET, uint32_t *io)
  85. * #io can refer to #PHY_FIFO_CMD(i, x)
  86. */
  87. #define PHY_CMD_DAC_FIFO_VOLUME_SET (PHY_CMD_DAC_BASE + 7)
  88. /*! Enable the specified index of the DAC FIFO sample counter function.
  89. * int phy_audio_control(dev, #PHY_CMD_DAC_FIFO_VOLUME_SET, uint32_t *io)
  90. * #io can refer to #PHY_FIFO_CMD(i, x)
  91. */
  92. #define PHY_CMD_CLAIM_WITH_128FS (PHY_CMD_DAC_BASE + 8)
  93. /*! Claim that current audio channels exist 128fs channel typically spdiftx which used in multi-linkage mode
  94. * int phy_audio_control(dev, #PHY_CMD_CLAIM_WITH_128FS, NULL)
  95. */
  96. #define PHY_CMD_CLAIM_WITHOUT_128FS (PHY_CMD_DAC_BASE + 9)
  97. /*! Claim that current audio channels do not exist 128fs channel typically spdiftx which used in multi-linkage mode
  98. * int phy_audio_control(dev, #PHY_CMD_CLAIM_WITHOUT_128FS, NULL)
  99. */
  100. #define PHY_CMD_ANC_MIX2_DAC_ENABLE (PHY_CMD_DAC_BASE + 10)
  101. /*!Enable ANC data stream MIX to DAC left/right channels.
  102. * int phy_audio_control(dev, #PHY_CMD_ANC_MIX2_DAC_ENABLE, uint8_t *lr_sel)
  103. * If lr_sel == (LEFT_CHANNEL_SEL | RIGHT_CHANNEL_SEL) stands for both ANC left and right data will mix to DAC.
  104. */
  105. #define PHY_CMD_ANC_MIX2_DAC_LR_INVERSE (PHY_CMD_DAC_BASE + 11)
  106. /*! Enable the ANC data stream will polarity inverse when mixed to DAC .
  107. * int phy_audio_control(dev, #PHY_CMD_ANC_MIX2_DAC_LR_INVERSE, uint8_t *en)
  108. * If (en == 1) ANC data will inverse others will not inverse.
  109. */
  110. /************************ I2STX IO commands **************************/
  111. #define PHY_CMD_I2STX_BASE (PHY_CMD_DAC_BASE + PHY_CMD_OFFSET)
  112. #define PHY_CMD_I2STX_IS_OPENED (PHY_CMD_I2STX_BASE + 1)
  113. /*! Get the open status of i2stx channel
  114. * int phy_audio_control(dev, #PHY_CMD_I2STX_IS_OPENED, uint8_t *status)
  115. */
  116. #define PHY_CMD_I2STX_DISABLE_DEVICE (PHY_CMD_I2STX_BASE + 2)
  117. /*! Disable I2STX hardware resouces include BCLK/LRCLK output signals.
  118. * int phy_audio_control(dev, #PHY_CMD_I2STX_DISABLE_DEVICE, NULL)
  119. */
  120. #define PHY_CMD_I2STX_CLK_SET (PHY_CMD_I2STX_BASE + 3)
  121. /*! Set the I2STX clock by other physical audio device such as I2SRX.
  122. * int phy_audio_control(dev, #PHY_CMD_I2STX_CLK_SET, uint8_t *sr)
  123. */
  124. #define PHY_CMD_I2STX_IS_MCLK_128FS (PHY_CMD_I2STX_BASE + 4)
  125. /*! Check if the MCLK of I2STX is a 128FS.
  126. * int phy_audio_control(dev, #PHY_CMD_I2STX_IS_MCLK_128FS, uint8_t *is_en)
  127. */
  128. #define PHY_CMD_I2S_LOOPBACK (PHY_CMD_I2STX_BASE + 5)
  129. /*! I2STX send clk and data to I2SRX.
  130. * int phy_audio_control(dev, #PHY_CMD_I2S_LOOPBACK, uint8_t *is_en)
  131. */
  132. /************************ I2SRX IO commands **************************/
  133. #define PHY_CMD_I2SRX_BASE (PHY_CMD_I2STX_BASE + PHY_CMD_OFFSET)
  134. #define PHY_CMD_I2SRX_IS_OPENED (PHY_CMD_I2SRX_BASE + 1)
  135. /*! Get the open status of i2srx channel
  136. * int phy_audio_control(dev, #PHY_CMD_I2SRX_IS_OPENED, uint8_t *status)
  137. */
  138. /************************ ADC IO commands **************************/
  139. #define PHY_CMD_ADC_BASE (PHY_CMD_I2SRX_BASE + PHY_CMD_OFFSET)
  140. #define PHY_CMD_ADC_DIGITAL_ENABLE (PHY_CMD_ADC_BASE + 1)
  141. /*! Enable the ADC channels 0/1/2 at the same time. For now only support 2 input device to run.
  142. * int phy_audio_control(dev, #PHY_CMD_ADC_DIGITAL_ENABLE, struct aduio_in_adc_en *ctl)
  143. */
  144. #define PHY_CMD_GET_AUDIOPLL_IDX (PHY_CMD_ADC_BASE + 2)
  145. /*! Get the AUDIOPLL index that used by physical audio device.
  146. * int phy_audio_control(dev, #PHY_CMD_GET_AUDIOPLL_IDX, uint8_t *idx)
  147. */
  148. #define PHY_CMD_ADC_GAIN_CONFIG (PHY_CMD_ADC_BASE + 3)
  149. /*! Config the ADC GAIN to physical ADC device.
  150. * int phy_audio_control(dev, #PHY_CMD_ADC_GAIN_CONFIG, adc_setting_t *setting)
  151. */
  152. #define PHY_CMD_IS_ADC_BUSY (PHY_CMD_DAC_BASE + 4)
  153. /*! Query ADC is busy or not.
  154. * int phy_audio_control(dev, #PHY_CMD_IS_ADC_BUSY, uint8_t *is_busy)
  155. */
  156. /**
  157. * enum audio_fifouse_sel_e
  158. * @brief FIFO use object select
  159. */
  160. typedef enum {
  161. FIFO_SEL_CPU = 0, /* FIFO's user is CPU */
  162. FIFO_SEL_DMA, /* FIFO's user is DMA */
  163. FIFO_SEL_ASRC, /* FIFO's user is ASRC */
  164. FIFO_SEL_DSP, /* FIFO's user is DSP */
  165. FIFO_SEL_DSP_DMA /* FIFO's user is DSP DMA */
  166. } audio_fifouse_sel_e;
  167. /*
  168. * enum audio_dma_width_e
  169. * @brief DMA transfer width configuration
  170. */
  171. typedef enum {
  172. DMA_WIDTH_32BITS = 0,
  173. DMA_WIDTH_16BITS
  174. } audio_dma_width_e;
  175. /*
  176. * enum audio_i2s_srd_period_e
  177. * @brief I2S sample rate detect period selection
  178. */
  179. typedef enum {
  180. I2S_SRD_2LRCLK = 0,
  181. I2S_SRD_4LRCLK
  182. } audio_i2s_srd_period_e;
  183. /*
  184. * enum audio_anc_work_mode
  185. * @brief ANC work mode selection
  186. */
  187. typedef enum {
  188. FF_IIR_US = 0, /* FF filter => IIR => US => DAC */
  189. FF_DSFIFO_USFIFO, /* FF filter => DSFIFO => DSP =>USFIFO => DAC */
  190. } audio_anc_work_mode;
  191. /**
  192. * struct audio_dma_dt
  193. * @brief audio dma resource from device tree.
  194. */
  195. struct audio_dma_dt {
  196. const char *dma_dev_name; /* DMA device name */
  197. uint32_t dma_chan; /* DMA channel */
  198. uint8_t dma_id; /* DMA slot id */
  199. };
  200. /**
  201. * struct audio_in_dma_info
  202. * @brief audio in dma information structure
  203. */
  204. struct audio_in_dma_info {
  205. uint16_t input_dev; /* input audio device */
  206. struct audio_dma_dt dma_info; /* dma info */
  207. };
  208. /**
  209. * struct aduio_in_adc_en
  210. * @brief audio in adc enable control
  211. */
  212. struct aduio_in_adc_en {
  213. uint16_t *input_dev_array;
  214. uint8_t input_dev_num;
  215. };
  216. /**
  217. * struct audio_out_dma_info
  218. * @brief audio out dma information structure
  219. */
  220. struct audio_out_dma_info {
  221. uint8_t fifo_type; /* audio out fifo type */
  222. struct audio_dma_dt dma_info; /* dma info */
  223. };
  224. /**
  225. * struct audio_debug_trace
  226. * @brief audio debug trace object.
  227. */
  228. struct audio_debug_trace_t {
  229. #define AUDIO_DEBUG_TRACE_START_FLAG BIT(0)
  230. uint32_t sec_timestamp;
  231. uint32_t trace_exec_timestamp;
  232. uint32_t trace_start_timestamp;
  233. uint32_t max_exec_time;
  234. uint32_t total_exec_time;
  235. uint32_t counter_per_sec;
  236. uint32_t counter_per_sec_bak;
  237. uint32_t total_counter;
  238. uint8_t flags;
  239. };
  240. extern struct audio_debug_trace_t audio_debug_trace;
  241. static inline void audio_debug_trace_info(void)
  242. {
  243. printk("audio trace total %dus {max_exec_time:%dus,total_exec_time:%dus}\n",
  244. k_cyc_to_us_floor32(k_cycle_get_32() - audio_debug_trace.trace_start_timestamp),
  245. k_cyc_to_us_floor32(audio_debug_trace.max_exec_time),
  246. k_cyc_to_us_floor32(audio_debug_trace.total_exec_time));
  247. printk("audio trace coutner_per_sec_bak:%d total_counter:%d\n",
  248. audio_debug_trace.counter_per_sec_bak, audio_debug_trace.total_counter);
  249. }
  250. static inline void audio_debug_trace_start(void)
  251. {
  252. audio_debug_trace.trace_exec_timestamp = k_cycle_get_32();
  253. audio_debug_trace.counter_per_sec++;
  254. audio_debug_trace.total_counter++;
  255. if (!(audio_debug_trace.flags & AUDIO_DEBUG_TRACE_START_FLAG)) {
  256. audio_debug_trace.trace_start_timestamp = audio_debug_trace.trace_exec_timestamp;
  257. audio_debug_trace.flags |= AUDIO_DEBUG_TRACE_START_FLAG;
  258. }
  259. }
  260. static inline void audio_debug_trace_end(void)
  261. {
  262. uint32_t delta, cur_time = k_cycle_get_32();
  263. delta = cur_time - audio_debug_trace.trace_exec_timestamp;
  264. if (delta > audio_debug_trace.max_exec_time)
  265. audio_debug_trace.max_exec_time = delta;
  266. audio_debug_trace.total_exec_time += delta;
  267. delta = k_cyc_to_us_floor32(cur_time - audio_debug_trace.sec_timestamp);
  268. if (delta > 1000000UL) {
  269. //audio_debug_trace_info();
  270. audio_debug_trace.sec_timestamp = cur_time;
  271. audio_debug_trace.counter_per_sec_bak = audio_debug_trace.counter_per_sec;
  272. audio_debug_trace.counter_per_sec = 0;
  273. }
  274. }
  275. static inline void audio_debug_trace_clear(void)
  276. {
  277. memset(&audio_debug_trace, 0, sizeof(struct audio_debug_trace_t));
  278. }
  279. /**
  280. * @brief The macro to combine with audio dma fifo structure and driver config dma information.
  281. */
  282. #define AUDIO_DMA_FIFO_DEF(m, n) \
  283. .dma_fifo##n = { \
  284. .dma_dev_name = CONFIG_DMA_0_NAME, \
  285. .dma_chan = CONFIG_AUDIO_##m##_0_FIFO##n##_DMA_CHAN, \
  286. .dma_id = CONFIG_AUDIO_##m##_0_FIFO##n##_DMA_ID \
  287. }
  288. /* @brief The macro to extend the device features in configuration */
  289. #define PHY_DEV_FEATURE_DEF(x) .features.v.x
  290. #define PHY_DEV_FEATURE(x) (cfg->features.v.x)
  291. /* @brief the physical IO commands which append with FIFO index */
  292. #define PHY_FIFO_INDEX_OFFSET (16)
  293. #define PHY_FIFO_CMD(i, x) (((i) << PHY_FIFO_INDEX_OFFSET) | (x))
  294. #define PHY_GET_FIFO_CMD_INDEX(x) ((x) >> PHY_FIFO_INDEX_OFFSET)
  295. #define PHY_GET_FIFO_CMD_VAL(x) ((x) & 0xFFFF)
  296. /* @brief the macro to show the infomation from device tree */
  297. #define PHY_DEV_SHOW_DT_INFO (0)
  298. #ifdef CONFIG_CFG_DRV
  299. /* @brief the macro to get the external configuration generated by PC tool */
  300. #define PHY_AUDIO_CFG(x, item_key, item) \
  301. { uint32_t val = 0; \
  302. if (!cfg_get_by_key(item_key, &val, sizeof((x).item))) \
  303. return __LINE__; \
  304. LOG_INF("%s:%d", #item, val); \
  305. (x).item = val; \
  306. }
  307. #define PHY_AUDIO_PIN_NUM_CFG(x) ((x) & 0xFF)
  308. #define PHY_AUDIO_PIN_MFP_CFG(x) ((x) >> 8)
  309. #endif
  310. /*
  311. * @struct phy_audio_driver_api
  312. * @brief Audio physical layer common API that standards the common behaviors of audio-out and audio-in channels.
  313. */
  314. struct phy_audio_driver_api {
  315. int (*audio_enable)(struct device *dev, void *param);
  316. int (*audio_disable)(struct device *dev, void *param);
  317. int (*audio_ioctl)(struct device *dev, uint32_t cmd, void *param);
  318. };
  319. /*
  320. * @brief Enable one audio in/out channel by the specified paremeters.
  321. * @param dev: The physical audio device handler.
  322. * @param param: The parameters to enable physical audio in/out channel.
  323. * @return 0 on success, negative errno code on fail.
  324. */
  325. static inline int phy_audio_enable(struct device *dev, void *param)
  326. {
  327. if (!dev)
  328. return -EINVAL;
  329. const struct phy_audio_driver_api *api = dev->api;
  330. return api->audio_enable(dev, param);
  331. }
  332. /*
  333. * @brief Disable the audio in/out channel by the specified device handler.
  334. * @param dev: The physical audio device handler to close.
  335. * @param param: The parameters to disable physical audio in/out channel.
  336. * @return 0 on success, negative errno code on fail.
  337. */
  338. static inline int phy_audio_disable(struct device *dev, void *param)
  339. {
  340. if (!dev)
  341. return -EINVAL;
  342. const struct phy_audio_driver_api *api = dev->api;
  343. return api->audio_disable(dev, param);
  344. }
  345. /*
  346. * @brief The io-commands that coresponding with the appropriative channel to control the audio channel dynamically.
  347. * @param dev: The physical audio device handler.
  348. * @param param: The io-commands to control physical audio channel.
  349. * @return 0 on success, negative errno code on fail.
  350. */
  351. static inline int phy_audio_control(struct device *dev, uint32_t cmd, void *param)
  352. {
  353. if (!dev)
  354. return -EINVAL;
  355. const struct phy_audio_driver_api *api = dev->api;
  356. return api->audio_ioctl(dev, cmd, param);
  357. }
  358. #endif /* __PHY_AUDIO_COMMON_H__ */