audioout_hal.c 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446
  1. /*
  2. * Copyright (c) 2020 Actions Semiconductor Co., Ltd
  3. *
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. /**
  7. * @file
  8. * @brief Audio Out HAL
  9. */
  10. #ifndef SYS_LOG_DOMAIN
  11. #define SYS_LOG_DOMAIN "hal_aout"
  12. #endif
  13. #include <audio_hal.h>
  14. hal_audio_out_context_t hal_audio_out_context;
  15. static inline hal_audio_out_context_t* _hal_audio_out_get_context(void)
  16. {
  17. return &hal_audio_out_context;
  18. }
  19. int hal_audio_out_init(void)
  20. {
  21. hal_audio_out_context_t* audio_out = _hal_audio_out_get_context();
  22. audio_out->aout_dev = (struct device *)device_get_binding(CONFIG_AUDIO_OUT_ACTS_DEV_NAME);
  23. if (!audio_out->aout_dev) {
  24. SYS_LOG_ERR("device not found\n");
  25. return -ENODEV;
  26. }
  27. SYS_LOG_INF("success \n");
  28. return 0;
  29. }
  30. void* hal_aout_channel_open(audio_out_init_param_t *init_param)
  31. {
  32. aout_param_t aout_param = {0};
  33. dac_setting_t dac_setting = {0};
  34. #ifdef CONFIG_AUDIO_OUT_I2STX_SUPPORT
  35. i2stx_setting_t i2stx_setting = {0};
  36. #endif
  37. #ifdef CONFIG_AUDIO_OUT_SPDIFTX_SUPPORT
  38. spdiftx_setting_t spdiftx_setting = {0};
  39. #endif
  40. audio_reload_t reload_setting = {0};
  41. hal_audio_out_context_t* audio_out = _hal_audio_out_get_context();
  42. assert(audio_out->aout_dev);
  43. assert(init_param);
  44. aout_param.sample_rate = init_param->sample_rate;
  45. aout_param.channel_type = init_param->channel_type;
  46. aout_param.outfifo_type = init_param->channel_id;
  47. SYS_LOG_INF("sample rate %d, channel type %d, out fifo %d",
  48. aout_param.sample_rate, aout_param.channel_type, aout_param.outfifo_type);
  49. aout_param.callback = init_param->callback;
  50. aout_param.cb_data = init_param->callback_data;
  51. if(init_param->data_width == 16) {
  52. aout_param.channel_width = CHANNEL_WIDTH_16BITS;
  53. } else {
  54. aout_param.channel_width = CHANNEL_WIDTH_24BITS;
  55. }
  56. if (init_param->dma_reload) {
  57. reload_setting.reload_addr = init_param->reload_addr;
  58. reload_setting.reload_len = init_param->reload_len;
  59. aout_param.reload_setting = &reload_setting;
  60. }
  61. if (!init_param->channel_type) {
  62. SYS_LOG_ERR("invalid channel type %d", init_param->channel_type);
  63. return NULL;
  64. }
  65. if (init_param->channel_type & AUDIO_CHANNEL_DAC) {
  66. dac_setting.volume.left_volume = init_param->left_volume;
  67. dac_setting.volume.right_volume = init_param->right_volume;
  68. aout_param.dac_setting = &dac_setting;
  69. dac_setting.channel_mode = init_param->channel_mode;
  70. }
  71. #ifdef CONFIG_AUDIO_OUT_I2STX_SUPPORT
  72. if (init_param->channel_type & AUDIO_CHANNEL_I2STX) {
  73. i2stx_setting.mode = I2S_MASTER_MODE;
  74. if (I2S_SLAVE_MODE == i2stx_setting.mode) {
  75. i2stx_setting.srd_callback = NULL;
  76. }
  77. aout_param.i2stx_setting = &i2stx_setting;
  78. if (AOUT_FIFO_DAC0 == aout_param.outfifo_type) {
  79. aout_param.dac_setting = &dac_setting;
  80. }
  81. }
  82. #endif
  83. #ifdef CONFIG_AUDIO_OUT_SPDIFTX_SUPPORT
  84. if (init_param->channel_type & AUDIO_CHANNEL_SPDIFTX) {
  85. if (AOUT_FIFO_DAC0 == aout_param.outfifo_type) {
  86. aout_param.dac_setting = &dac_setting;
  87. }
  88. aout_param.spdiftx_setting= &spdiftx_setting;
  89. }
  90. #endif
  91. return audio_out_open(audio_out->aout_dev, (void *)&aout_param);
  92. }
  93. int hal_aout_channel_start(void* aout_channel_handle)
  94. {
  95. int result;
  96. hal_audio_out_context_t* audio_out = _hal_audio_out_get_context();
  97. assert(audio_out->aout_dev);
  98. result = audio_out_start(audio_out->aout_dev, aout_channel_handle);
  99. return result;
  100. }
  101. int hal_aout_channel_write_data(void* aout_channel_handle, u8_t *data, u32_t data_size)
  102. {
  103. int result;
  104. hal_audio_out_context_t* audio_out = _hal_audio_out_get_context();
  105. assert(audio_out->aout_dev);
  106. result = audio_out_write(audio_out->aout_dev, aout_channel_handle, data, data_size);
  107. return result;
  108. }
  109. int hal_aout_channel_stop(void* aout_channel_handle)
  110. {
  111. int result;
  112. hal_audio_out_context_t* audio_out = _hal_audio_out_get_context();
  113. assert(audio_out->aout_dev);
  114. result = audio_out_stop(audio_out->aout_dev, aout_channel_handle);
  115. return result;
  116. }
  117. int hal_aout_channel_close(void* aout_channel_handle)
  118. {
  119. int result;
  120. hal_audio_out_context_t* audio_out = _hal_audio_out_get_context();
  121. assert(audio_out->aout_dev);
  122. result = audio_out_close(audio_out->aout_dev, aout_channel_handle);
  123. return result;
  124. }
  125. int hal_aout_channel_set_pa_vol_level(void* aout_channel_handle, int vol_level)
  126. {
  127. volume_setting_t volume;
  128. hal_audio_out_context_t* audio_out = _hal_audio_out_get_context();
  129. assert(audio_out->aout_dev);
  130. volume.left_volume = vol_level;
  131. volume.right_volume = vol_level;
  132. return audio_out_control(audio_out->aout_dev, aout_channel_handle, AOUT_CMD_SET_VOLUME, (void *)&volume);
  133. }
  134. int hal_aout_channel_set_aps(void *aout_channel_handle, unsigned int aps_level, unsigned int aps_mode)
  135. {
  136. hal_audio_out_context_t* audio_out = _hal_audio_out_get_context();
  137. assert(audio_out->aout_dev);
  138. assert(aout_channel_handle);
  139. return audio_out_control(audio_out->aout_dev, aout_channel_handle, AOUT_CMD_SET_APS, &aps_level);
  140. }
  141. int hal_aout_channel_mute_ctl(void *aout_channel_handle, u8_t mode)
  142. {
  143. hal_audio_out_context_t* audio_out = _hal_audio_out_get_context();
  144. assert(audio_out->aout_dev);
  145. assert(aout_channel_handle);
  146. return audio_out_control(audio_out->aout_dev, aout_channel_handle, AOUT_CMD_OUT_MUTE, &mode);
  147. }
  148. u32_t hal_aout_channel_get_sample_cnt(void *aout_channel_handle)
  149. {
  150. hal_audio_out_context_t* audio_out = _hal_audio_out_get_context();
  151. u32_t cnt = 0;
  152. assert(audio_out->aout_dev);
  153. assert(aout_channel_handle);
  154. if (audio_out_control(audio_out->aout_dev, aout_channel_handle, AOUT_CMD_GET_SAMPLE_CNT, (void *)&cnt)) {
  155. SYS_LOG_ERR("Get FIFO counter error");
  156. }
  157. return cnt;
  158. }
  159. int hal_aout_channel_enable_sample_cnt(void *aout_channel_handle, bool enable)
  160. {
  161. int ret = 0;
  162. hal_audio_out_context_t* audio_out = _hal_audio_out_get_context();
  163. assert(audio_out->aout_dev);
  164. assert(aout_channel_handle);
  165. if (enable) {
  166. ret = audio_out_control(audio_out->aout_dev, aout_channel_handle, AOUT_CMD_ENABLE_SAMPLE_CNT, NULL);
  167. } else {
  168. ret = audio_out_control(audio_out->aout_dev, aout_channel_handle, AOUT_CMD_DISABLE_SAMPLE_CNT, NULL);
  169. }
  170. return ret;
  171. }
  172. int hal_aout_channel_reset_sample_cnt(void *aout_channel_handle)
  173. {
  174. hal_audio_out_context_t* audio_out = _hal_audio_out_get_context();
  175. assert(audio_out->aout_dev);
  176. assert(aout_channel_handle);
  177. return audio_out_control(audio_out->aout_dev, aout_channel_handle, AOUT_CMD_RESET_SAMPLE_CNT, NULL);
  178. }
  179. int hal_aout_open_pa(void)
  180. {
  181. hal_audio_out_context_t* audio_out = _hal_audio_out_get_context();
  182. assert(audio_out->aout_dev);
  183. return audio_out_control(audio_out->aout_dev, NULL, AOUT_CMD_OPEN_PA, NULL);
  184. }
  185. int hal_aout_close_pa(void)
  186. {
  187. hal_audio_out_context_t* audio_out = _hal_audio_out_get_context();
  188. assert(audio_out->aout_dev);
  189. return audio_out_control(audio_out->aout_dev, NULL, AOUT_CMD_CLOSE_PA, NULL);
  190. }
  191. int hal_aout_channel_check_fifo_underflow(void *aout_channel_handle)
  192. {
  193. int samples = 0;
  194. hal_audio_out_context_t* audio_out = _hal_audio_out_get_context();
  195. assert(audio_out->aout_dev);
  196. return audio_out_control(audio_out->aout_dev, aout_channel_handle, AOUT_CMD_GET_FIFO_LEN, &samples);
  197. }
  198. int hal_aout_pa_class_select(u8_t pa_mode)
  199. {
  200. hal_audio_out_context_t* audio_out = _hal_audio_out_get_context();
  201. assert(audio_out->aout_dev);
  202. return audio_out_control(audio_out->aout_dev, NULL, AOUT_CMD_PA_CLASS_SEL, &pa_mode);
  203. }
  204. int hal_aout_set_pcm_threshold(void *aout_channel_handle, int he_thres, int hf_thres)
  205. {
  206. hal_audio_out_context_t* audio_out = _hal_audio_out_get_context();
  207. dac_threshold_setting_t dac_threshold = {
  208. .he_thres = he_thres,
  209. .hf_thres = hf_thres,
  210. };
  211. assert(audio_out->aout_dev);
  212. assert(aout_channel_handle);
  213. return audio_out_control(audio_out->aout_dev, aout_channel_handle, AOUT_CMD_SET_DAC_THRESHOLD, &dac_threshold);
  214. }
  215. int hal_aout_set_fifo_src(void *aout_channel_handle, u8_t channel_id, bool from_dsp, void *dsp_audio_set_param)
  216. {
  217. hal_audio_out_context_t* audio_out = _hal_audio_out_get_context();
  218. dac_fifosrc_setting_t dac_fifosrc = {
  219. .fifo_idx = channel_id,
  220. .fifo_from_dsp = from_dsp,
  221. .dsp_audio_set_param = dsp_audio_set_param,
  222. };
  223. assert(audio_out->aout_dev);
  224. assert(aout_channel_handle);
  225. return audio_out_control(audio_out->aout_dev, aout_channel_handle, AOUT_CMD_SET_FIFO_SRC, &dac_fifosrc);
  226. }
  227. int hal_aout_lr_channel_enable(void *aout_channel_handle, bool l_enable, bool r_enable)
  228. {
  229. int ret;
  230. uint8_t lr_sel;
  231. hal_audio_out_context_t* audio_out = _hal_audio_out_get_context();
  232. assert(audio_out->aout_dev);
  233. assert(aout_channel_handle);
  234. lr_sel = (l_enable ? LEFT_CHANNEL_SEL : 0) | (r_enable ? RIGHT_CHANNEL_SEL : 0);
  235. ret = audio_out_control(audio_out->aout_dev, aout_channel_handle, AOUT_CMD_SELECT_DAC_ENABLE_CHANNEL, &lr_sel);
  236. return ret;
  237. }
  238. int hal_aout_channel_get_buffer_size(void *aout_channel_handle)
  239. {
  240. int samples = 0;
  241. hal_audio_out_context_t* audio_out = _hal_audio_out_get_context();
  242. assert(audio_out->aout_dev);
  243. audio_out_control(audio_out->aout_dev, aout_channel_handle, AOUT_CMD_GET_FIFO_LEN, &samples);
  244. return samples;
  245. }
  246. int hal_aout_channel_get_buffer_space(void *aout_channel_handle)
  247. {
  248. int samples = 0;
  249. hal_audio_out_context_t* audio_out = _hal_audio_out_get_context();
  250. assert(audio_out->aout_dev);
  251. audio_out_control(audio_out->aout_dev, aout_channel_handle, AOUT_CMD_GET_FIFO_AVAILABLE_LEN, &samples);
  252. return samples;
  253. }
  254. uint32_t hal_aout_channel_get_sdm_cnt(void *aout_channel_handle)
  255. {
  256. uint32_t samples = 0;
  257. hal_audio_out_context_t* audio_out = _hal_audio_out_get_context();
  258. assert(audio_out->aout_dev);
  259. audio_out_control(audio_out->aout_dev, aout_channel_handle, AOUT_CMD_GET_DAC_SDM_SAMPLE_CNT, &samples);
  260. return samples;
  261. }
  262. uint32_t hal_aout_channel_get_saved_sdm_cnt(void *aout_channel_handle)
  263. {
  264. uint32_t samples = 0;
  265. hal_audio_out_context_t* audio_out = _hal_audio_out_get_context();
  266. assert(audio_out->aout_dev);
  267. audio_out_control(audio_out->aout_dev, aout_channel_handle, AOUT_CMD_GET_DAC_SDM_STABLE_SAMPLE_CNT, &samples);
  268. return samples;
  269. }
  270. int hal_aout_channel_enable_sdm_cnt(void *aout_channel_handle, bool enable)
  271. {
  272. int ret = 0;
  273. hal_audio_out_context_t* audio_out = _hal_audio_out_get_context();
  274. assert(audio_out->aout_dev);
  275. assert(aout_channel_handle);
  276. if (enable) {
  277. ret = audio_out_control(audio_out->aout_dev, aout_channel_handle, AOUT_CMD_ENABLE_DAC_SDM_SAMPLE_CNT, NULL);
  278. } else {
  279. ret = audio_out_control(audio_out->aout_dev, aout_channel_handle, AOUT_CMD_DISABLE_DAC_SDM_SAMPLE_CNT, NULL);
  280. }
  281. return ret;
  282. }
  283. int hal_aout_channel_reset_sdm_cnt(void *aout_channel_handle)
  284. {
  285. hal_audio_out_context_t* audio_out = _hal_audio_out_get_context();
  286. assert(audio_out->aout_dev);
  287. assert(aout_channel_handle);
  288. return audio_out_control(audio_out->aout_dev, aout_channel_handle, AOUT_CMD_RESET_DAC_SDM_SAMPLE_CNT, NULL);
  289. }
  290. int hal_aout_trigger_src_control(void *aout_channel_handle, dac_ext_trigger_ctl_t *trigger_ctl)
  291. {
  292. hal_audio_out_context_t* audio_out = _hal_audio_out_get_context();
  293. assert(audio_out->aout_dev);
  294. return audio_out_control(audio_out->aout_dev, aout_channel_handle, AOUT_CMD_DAC_TRIGGER_CONTROL, trigger_ctl);
  295. }
  296. int hal_aout_channel_set_dac_trigger_src(void *aout_channel_handle, audio_trigger_src src)
  297. {
  298. hal_audio_out_context_t* audio_out = _hal_audio_out_get_context();
  299. assert(audio_out->aout_dev);
  300. return audio_out_control(audio_out->aout_dev, aout_channel_handle, AOUT_CMD_SET_DAC_TRIGGER_SRC, &src);
  301. }
  302. int hal_aout_channel_set_dac_enable(void *aout_channel_handle)
  303. {
  304. hal_audio_out_context_t* audio_out = _hal_audio_out_get_context();
  305. assert(audio_out->aout_dev);
  306. return audio_out_control(audio_out->aout_dev, aout_channel_handle, AOUT_CMD_DAC_FORCE_START, 0);
  307. }