audio_in.h 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498
  1. /*
  2. * Copyright (c) 2021 Actions Semiconductor Co., Ltd
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file audio_in.h
  8. * @brief Audio input channels common interface.
  9. *
  10. * <b> Audio input channels </b> are used to record a uncompressed PCM data
  11. * through analog(e.g line-in) or digital physical interface(e.g. SPDIF).
  12. *
  13. * This public audio input interface aims to define a unified structures and functions
  14. * for vary audio input channels.
  15. */
  16. #ifndef __AUDIO_IN_H__
  17. #define __AUDIO_IN_H__
  18. #include <zephyr/types.h>
  19. #include <stddef.h>
  20. #include <device.h>
  21. #include <board.h>
  22. #include <drivers/audio/audio_common.h>
  23. #include <drivers/cfg_drv/dev_config.h>
  24. #ifdef __cplusplus
  25. extern "C" {
  26. #endif
  27. /**
  28. * @defgroup audio_in_apis Audio In Device APIs
  29. * @ingroup driver_apis
  30. * @{
  31. */
  32. /**
  33. * @brief The <b> Audio input channels </b> contains the following channels.
  34. * - ADC (analog convert to digital)
  35. * - I2SRX
  36. * - SPDIFRX
  37. *
  38. * All audio input channels follow the same interfaces to record a PCM stream.
  39. * The different operations between the input channels are the parameter
  40. * structures that used to open or control an dedicated audio input channel.
  41. * For example open an ADC channel need to inplement the structure #adc_setting_t,
  42. * whileas the structure to open an I2SRX channel is #i2srx_setting_t.
  43. * Moreover, Some special control commands are dedicated for some input channel,
  44. * such as #AIN_CMD_SPDIF_IS_PCM_STREAM only work for SPDIFRX channel.
  45. *
  46. * Example:
  47. *
  48. * @code
  49. *
  50. * #include <drivers/audio/audio_in.h>
  51. *
  52. * // Note that all audio input channels only support DMA reload mode.
  53. *
  54. * #define SHELL_AUDIO_BUFFER_SIZE (1024)
  55. * static uint8_t audio_pcm_buffer[SHELL_AUDIO_BUFFER_SIZE];
  56. *
  57. * // Audio input DMA callback function.
  58. * static int audio_rec_read_data_cb(void *callback_data, uint32_t reason) {
  59. * uint32_t len = sizeof(audio_pcm_buffer) / 2;
  60. * uint8_t *buf = NULL;
  61. *
  62. * if (AIN_DMA_IRQ_HF == reason) {
  63. * buf = audio_pcm_buffer;
  64. * } else {
  65. * buf = audio_pcm_buffer + len;
  66. * }
  67. *
  68. * // The recorded data is in the memory of buf, and its length is len.
  69. * }
  70. *
  71. * //1. Get a audio input device.
  72. * struct device *ain_dev = device_get_binding(CONFIG_AUDIO_IN_ACTS_DEV_NAME);
  73. * if (!ain_dev) {
  74. * printk("failed to get audio input device\n");
  75. * return -ENODEV;
  76. * }
  77. *
  78. * // 2. Open a ADC channel
  79. * ain_param_t ain_param = {0};
  80. * adc_setting_t adc_setting = {0};
  81. * void *ain_handle;
  82. *
  83. * ain_param.sample_rate = SAMPLE_RATE_16KHZ;
  84. * ain_param.callback = audio_rec_read_data_cb;
  85. * ain_param.cb_data = NULL;
  86. * ain_param.reload_setting.reload_addr = audio_pcm_buffer;
  87. * ain_param.reload_setting.reload_len = sizeof(audio_pcm_buffer);
  88. * ain_param.channel_type = AUDIO_CHANNEL_ADC;
  89. * ain_param.channel_width = CHANNEL_WIDTH_16BITS;
  90. *
  91. * adc_setting.device = AUDIO_ANALOG_MIC0;
  92. * uint8_t i;
  93. * for (i = 0; i < ADC_CH_NUM_MAX; i++) {
  94. * adc_setting.gain.ch_gain[i] = 365;
  95. * }
  96. * ain_param.adc_setting = &adc_setting;
  97. *
  98. * ain_handle = audio_in_open(ain_dev, &ain_param);
  99. * if (!ain_handle) {
  100. * printk("failed to open the audio in channel");
  101. * return -EIO;
  102. * }
  103. *
  104. * // 3. Use audio_out_control to send extra comands to the opened ADC channel if needed.
  105. *
  106. * // 4. Start to play the ADC channel
  107. * ret = audio_record_start(ain_dev, ain_handle);
  108. *
  109. * @endcode
  110. *
  111. * @note For more detailed audio input channel example, please make a reference to the file:audio_driver_shell.c.
  112. *
  113. */
  114. /**
  115. * @name Definition for the audio in control commands
  116. * @{
  117. */
  118. #define AIN_CMD_SET_ADC_GAIN (1)
  119. /*!< Set the ADC left channel gain value which in dB format.
  120. * int audio_in_control(dev, handle, #AIN_CMD_SET_ADC_GAIN, adc_gain *gain)
  121. * Returns 0 if successful and negative errno code if error.
  122. *
  123. * @note You can get the dB range from #AIN_CMD_GET_ADC_LEFT_GAIN_RANGE and AIN_CMD_GET_ADC_RIGHT_GAIN_RANGE.
  124. */
  125. #define AIN_CMD_GET_ADC_LEFT_GAIN_RANGE (2)
  126. /*!< Get the ADC left channel gain range in dB format.
  127. * int audio_in_control(dev, handle, #AIN_CMD_GET_ADC_LEFT_GAIN_RANGE, adc_gain_range *range)
  128. * Returns 0 if successful and negative errno code if error.
  129. */
  130. #define AIN_CMD_GET_ADC_RIGHT_GAIN_RANGE (3)
  131. /*!< Get the ADC right channel gain range in dB format.
  132. * int audio_in_control(dev, handle, #AIN_CMD_GET_ADC_RIGHT_GAIN_RANGE, adc_gain_range *range)
  133. * Returns 0 if successful and negative errno code if error.
  134. */
  135. #define AIN_CMD_SPDIF_GET_CHANNEL_STATUS (4)
  136. /* Get the SPDIFRX channel status.
  137. * int audio_in_control(dev, handle, #AIN_CMD_SPDIF_GET_CHANNEL_STATUS, audio_spdif_ch_status_t *sts)
  138. * Returns 0 if successful and negative errno code if error.
  139. */
  140. #define AIN_CMD_SPDIF_IS_PCM_STREAM (5)
  141. /* Check if the stream that received from spdifrx is the pcm format
  142. * int audio_in_control(dev, handle, #AIN_CMD_SPDIF_IS_PCM_STREAM, bool *is_pcm)
  143. * Returns 0 if successful and negative errno code if error.
  144. */
  145. #define AIN_CMD_SPDIF_CHECK_DECODE_ERR (6)
  146. /* Check if there is spdif decode error happened
  147. * int audio_in_control(dev, handle, #AIN_CMD_SPDIF_CHECK_DECODE_ERR, bool *is_err)
  148. * Returns 0 if successful and negative errno code if error.
  149. */
  150. #define AIN_CMD_I2SRX_QUERY_SAMPLE_RATE (7)
  151. /* Query the i2c master device sample rate and i2srx works in slave mode.
  152. * int audio_in_control(dev, handle, #AIN_CMD_I2SRX_QUERY_SAMPLE_RATE, audio_sr_sel_e *is_err)
  153. * Returns 0 if successful and negative errno code if error.
  154. */
  155. #define AIN_CMD_GET_SAMPLERATE (8)
  156. /*!< Get the channel audio sample rate by specified the audio channel handler.
  157. * int audio_in_control(dev, handle, #AIN_CMD_GET_SAMPLERATE, audio_sr_sel_e *sr)
  158. * Returns 0 if successful and negative errno code if error.
  159. */
  160. #define AIN_CMD_SET_SAMPLERATE (9)
  161. /*!< Set the channel audio sample rate by the giving audio channel handler.
  162. * int audio_in_control(dev, handle, #AIN_CMD_SET_SAMPLERATE, audio_sr_sel_e *sr)
  163. * Returns 0 if successful and negative errno code if error.
  164. */
  165. #define AIN_CMD_GET_APS (10)
  166. /*!< Get the AUDIO_PLL APS
  167. * int audio_in_control(dev, handle, #AIN_CMD_GET_APS, audio_aps_level_e *aps)
  168. * Returns 0 if successful and negative errno code if error.
  169. */
  170. #define AIN_CMD_SET_APS (11)
  171. /*!< Set the AUDIO_PLL APS for the sample rate tuning
  172. * int audio_in_control(dev, handle, #AIN_CMD_SET_APS, audio_aps_level_e *aps)
  173. * Returns 0 if successful and negative errno code if error.
  174. */
  175. #define AIN_CMD_BIND_CHANNEL (12)
  176. /*!< Bind the different audio in channels that can start simutaneously.
  177. * int audio_in_control(dev, handle, #AIN_CMD_BIND_CHANNEL, void *hdl)
  178. * Returns 0 if successful and negative errno code if error.
  179. */
  180. #define AIN_CMD_GET_ADC_FIFO_DRQ_LEVEL (13)
  181. /*!< Get the ADC FIFO DRQ level.
  182. * int audio_in_control(dev, NULL, #AIN_CMD_GET_ADC_FIFO_DRQ_LEVEL, uint8_t *level)
  183. * Returns 0 if successful and negative errno code if error.
  184. */
  185. #define AIN_CMD_SET_ADC_FIFO_DRQ_LEVEL (14)
  186. /*!< Set the ADC FIFO DRQ level.
  187. * int audio_in_control(dev, NULL, #AIN_CMD_SET_ADC_FIFO_DRQ_LEVEL, uint8_t *level)
  188. * Returns 0 if successful and negative errno code if error.
  189. *
  190. * @note The parameter level is range from 0 to 15;
  191. */
  192. #define AIN_CMD_DEBUG_PERFORMANCE_CTL (15)
  193. /*!< Control to enable or disable to dump the perfornamce infomation for debug.
  194. * int audio_in_control(dev, NULL, #AOUT_CMD_DEBUG_PERFORMANCE_CTL, uint8_t *en)
  195. * Returns 0 if successful and negative errno code if error.
  196. *
  197. * @note en: 0 to disable; 1 to enable
  198. */
  199. #define AIN_CMD_DEBUG_PERFORMANCE_CTL_ALL (16)
  200. /*!< Control all sessions to enable or disable to dump the perfornamce infomation for debug.
  201. * int audio_in_control(dev, NULL, #AIN_CMD_DEBUG_PERFORMANCE_CTL_ALL, uint8_t *en)
  202. * Returns 0 if successful and negative errno code if error.
  203. *
  204. * @note en: 0 to disable; 1 to enable
  205. */
  206. #define AIN_CMD_DEBUG_DUMP_LENGTH (17)
  207. /*!< Set the length of play buffer to print out per-second.
  208. * int audio_in_control(dev, NULL, #AIN_CMD_DEBUG_DUMP_LENGTH, uint8_t *len)
  209. * Returns 0 if successful and negative errno code if error.
  210. */
  211. #define AIN_CMD_DEBUG_DUMP_LENGTH_ALL (18)
  212. /*!< Set the length of all sessions play buffer to print out per-second.
  213. * int audio_in_control(dev, NULL, #AIN_CMD_DEBUG_DUMP_LENGTH_ALL, uint8_t *len)
  214. * Returns 0 if successful and negative errno code if error.
  215. */
  216. #define AIN_CMD_SET_ADC_TRIGGER_SRC (19)
  217. /*!< Set the source of trigger ADC to start by external IRQ signal.
  218. * int audio_in_control(dev, handle, #AIN_CMD_SET_ADC_TRIGGER_SRC, uint8_t *trigger_src)
  219. * The parameter trigger_src can refer to #audio_trigger_src.
  220. * Returns 0 if successful and negative errno code if error.
  221. *
  222. * @note The parameter trigger_src can refer to #audio_trigger_src.
  223. */
  224. #define AIN_CMD_ANC_CONTROL (20)
  225. /*!< Control to enable or disable ANC function.
  226. * int audio_in_control(dev, NULL, #AIN_CMD_ANC_CONTROL, adc_anc_ctl_t *anc_ctl)
  227. * Returns 0 if successful and negative errno code if error.
  228. */
  229. #define AIN_CMD_SET_SEPARATED_MODE (21)
  230. /*!< Set DMA as separated mode for mute left/right channel individually or store the PCM data by separated format.
  231. * int audio_in_control(dev, handle, #AIN_CMD_SET_SEPARATED_MODE, audio_interleave_mode_e *p_mode)
  232. * Returns 0 if successful and negative errno code if error.
  233. */
  234. #define AIN_CMD_AEC_CONTROL (22)
  235. /*!< Control to enable or disable AEC function that send DAC data to ADC.
  236. * int audio_in_control(dev, handle, #AIN_CMD_AEC_CONTROL, uint8_t *en)
  237. * Returns 0 if successful and negative errno code if error.
  238. */
  239. /** @} */
  240. /*!
  241. * enum audio_interleave_mode_e
  242. * @brief Audio DMA interleave mode setting
  243. */
  244. typedef enum {
  245. LEFT_MONO_RIGHT_MUTE_MODE = 0,
  246. LEFT_MUTE_RIGHT_MONO_MODE,
  247. LEFT_RIGHT_SEPERATE
  248. } audio_interleave_mode_e;
  249. /*!
  250. * struct adc_gain_range
  251. * @brief The ADC min and max gain range
  252. */
  253. typedef struct {
  254. int16_t min;
  255. /*!< min gain */
  256. int16_t max;
  257. /*!< max gain */
  258. } adc_gain_range;
  259. /*!
  260. * struct adc_gain
  261. * @brief The ADC gain setting
  262. */
  263. typedef struct {
  264. #define ADC_GAIN_INVALID (0xFFFF)
  265. int16_t ch_gain[ADC_CH_NUM_MAX];
  266. /*!< The gain value shall be set by 10 multiple of actual value e.g. ch_gain[0]=100 means channel 0 gain is 10 db;
  267. * If gain value equal to #ADC_GAIN_INVALID that the setting will be ignored.
  268. */
  269. } adc_gain;
  270. /*!
  271. * struct adc_setting_t
  272. * @brief The ADC setting parameters
  273. */
  274. typedef struct {
  275. uint16_t device;
  276. /*!< ADC input device chooses */
  277. adc_gain gain;
  278. /*!< ADC gain setting */
  279. } adc_setting_t;
  280. /*!
  281. * struct i2srx_setting_t
  282. * @brief The I2SRX setting parameters
  283. */
  284. typedef struct {
  285. #define I2SRX_SRD_FS_CHANGE (1 << 0)
  286. /*!< I2SRX SRD(sample rate detect) captures the event that the sample rate has changed
  287. * int callback(cb_data, #I2STX_SRD_FS_CHANGE, audio_sr_sel_e *sr)
  288. */
  289. #define I2SRX_SRD_WL_CHANGE (1 << 1)
  290. /*!< I2SRX SRD(sample rate detect) captures the event that the effective width length has changed
  291. * int callback(cb_data, #I2STX_SRD_WL_CHANGE, audio_i2s_srd_wl_e *wl)
  292. */
  293. #define I2SRX_SRD_TIMEOUT (1 << 2)
  294. /*!< I2SRX SRD(sample rate detect) captures the timeout (disconnection) event
  295. * int callback(cb_data, #I2STX_SRD_TIMEOUT, NULL)
  296. */
  297. int (*srd_callback)(void *cb_data, uint32_t cmd, void *param);
  298. /*!< The callback function from I2SRX SRD module which worked in the slave mode */
  299. void *cb_data;
  300. /*!< Callback user data */
  301. audio_i2s_mode_e mode;
  302. } i2srx_setting_t;
  303. /*!
  304. * struct spdifrx_setting_t
  305. * @brief The SPDIFRX setting parameters
  306. */
  307. typedef struct {
  308. #define SPDIFRX_SRD_FS_CHANGE (1 << 0)
  309. /*!< SPDIFRX SRD(sample rate detect) captures the event that the sample rate has changed.
  310. * int callback(cb_data, #SPDIFRX_SRD_FS_CHANGE, audio_sr_sel_e *sr)
  311. */
  312. #define SPDIFRX_SRD_TIMEOUT (1 << 1)
  313. /*!< SPDIFRX SRD(sample rate detect) timeout (disconnect) event.
  314. * int callback(cb_data, #SPDIFRX_SRD_TIMEOUT, NULL)
  315. */
  316. int (*srd_callback)(void *cb_data, uint32_t cmd, void *param);
  317. /*!< sample rate detect callback */
  318. void *cb_data;
  319. /*!< callback user data */
  320. } spdifrx_setting_t;
  321. /*!
  322. * struct ain_param_t
  323. * @brief The audio in configuration parameters
  324. */
  325. typedef struct {
  326. #define AIN_DMA_IRQ_HF (1 << 0)
  327. /*!< DMA irq half full flag */
  328. #define AIN_DMA_IRQ_TC (1 << 1)
  329. /*!< DMA irq transfer completly flag */
  330. uint8_t sample_rate;
  331. /*!< The sample rate setting refer to enum audio_sr_sel_e */
  332. uint16_t channel_type;
  333. /*!< Indicates the channel type selection and can refer to #AUDIO_CHANNEL_ADC, #AUDIO_CHANNEL_I2SRX, #AUDIO_CHANNEL_SPDIFRX*/
  334. audio_ch_width_e channel_width;
  335. /*!< The channel effective data width */
  336. adc_setting_t *adc_setting;
  337. /*!< The ADC function setting if has */
  338. i2srx_setting_t *i2srx_setting;
  339. /*!< The I2SRX function setting if has */
  340. spdifrx_setting_t *spdifrx_setting;
  341. /*!< The SPDIFRX function setting if has */
  342. int (*callback)(void *cb_data, uint32_t reason);
  343. /*!< The callback function which called when #AIN_DMA_IRQ_HF or #AIN_DMA_IRQ_TC events happened */
  344. void *cb_data;
  345. /*!< callback user data */
  346. audio_reload_t reload_setting;
  347. /*!< The reload mode setting which is mandatory*/
  348. } ain_param_t;
  349. /*!
  350. * struct ain_driver_api
  351. * @brief Public API for audio in driver
  352. */
  353. struct ain_driver_api {
  354. void* (*ain_open)(struct device *dev, ain_param_t *param);
  355. int (*ain_close)(struct device *dev, void *handle);
  356. int (*ain_start)(struct device *dev, void *handle);
  357. int (*ain_stop)(struct device *dev, void *handle);
  358. int (*ain_control)(struct device *dev, void *handle, int cmd, void *param);
  359. };
  360. /*!
  361. * @brief Open the audio input channel by specified parameters
  362. *
  363. * @param dev Pointer to the device structure for the audio input channel instance.
  364. *
  365. * @param setting Pointer to the audio input channel parameter.
  366. *
  367. * @return The audio input channel instance handle.
  368. */
  369. static inline void* audio_in_open(struct device *dev, ain_param_t *setting)
  370. {
  371. const struct ain_driver_api *api = dev->api;
  372. return api->ain_open(dev, setting);
  373. }
  374. /*!
  375. * @brief Close the audio input channel by the specified handle.
  376. *
  377. * @param dev Pointer to the device structure for the audio input channel instance.
  378. *
  379. * @param handle The audio input channel instance handle.
  380. *
  381. * @return 0 on success, negative errno code on fail.
  382. *
  383. * @note the handle shall be the same as the retval of #audio_in_open.
  384. */
  385. static inline int audio_in_close(struct device *dev, void *handle)
  386. {
  387. const struct ain_driver_api *api = dev->api;
  388. return api->ain_close(dev, handle);
  389. }
  390. /*!
  391. * @brief Control the audio input channel by the specified handle.
  392. *
  393. * @param dev Pointer to the device structure for the audio input channel instance.
  394. *
  395. * @param handle The audio input channel instance handle.
  396. *
  397. * @param cmd The control command that sent to the audio input channel.
  398. *
  399. * @param param The audio out in/out parameters which corresponding with the commands
  400. *
  401. * @return 0 on success, negative errno code on fail.
  402. *
  403. * @note the handle shall be the same as the retval of #audio_in_open.
  404. */
  405. static inline int audio_in_control(struct device *dev, void *handle, int cmd, void *param)
  406. {
  407. const struct ain_driver_api *api = dev->api;
  408. return api->ain_control(dev, handle, cmd, param);
  409. }
  410. /*!
  411. * @brief Start the audio input channel by the specified handle.
  412. *
  413. * @param dev Pointer to the device structure for the audio input channel instance.
  414. *
  415. * @param handle The audio input channel instance handle.
  416. *
  417. * @return 0 on success, negative errno code on fail.
  418. *
  419. * @note the handle shall be the same as the retval of #audio_in_open.
  420. */
  421. static inline int audio_in_start(struct device *dev, void *handle)
  422. {
  423. const struct ain_driver_api *api = dev->api;
  424. return api->ain_start(dev, handle);
  425. }
  426. /*!
  427. * @brief Stop the audio input channel by the specified handle.
  428. *
  429. * @param dev Pointer to the device structure for the audio input channel instance.
  430. *
  431. * @param handle The audio input channel instance handle.
  432. *
  433. * @return 0 on success, negative errno code on fail.
  434. *
  435. * @note the handle shall be the same as the retval of #audio_in_open.
  436. */
  437. static inline int audio_in_stop(struct device *dev, void *handle)
  438. {
  439. const struct ain_driver_api *api = dev->api;
  440. return api->ain_stop(dev, handle);
  441. }
  442. #ifdef __cplusplus
  443. }
  444. #endif
  445. /**
  446. * @} end defgroup audio_in_apis
  447. */
  448. #endif /* __AUDIO_IN_H__ */