luat_multimedia_audio.c 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448
  1. #include "luat_base.h"
  2. #include "luat_gpio.h"
  3. #ifdef LUAT_USE_DRV_GPIO
  4. #include "luat/drv_gpio.h"
  5. #endif
  6. #include "luat_i2s.h"
  7. #include "luat_audio.h"
  8. #include "luat_multimedia.h"
  9. #include "luat_mcu.h"
  10. #define LUAT_LOG_TAG "audio"
  11. #include "luat_log.h"
  12. LUAT_WEAK luat_audio_conf_t *luat_audio_get_config(uint8_t multimedia_id){
  13. return NULL;
  14. }
  15. LUAT_WEAK int luat_audio_play_multi_files(uint8_t multimedia_id, uData_t *info, uint32_t files_num, uint8_t error_stop){
  16. return -1;
  17. }
  18. LUAT_WEAK int luat_audio_play_file(uint8_t multimedia_id, const char *path){
  19. return -1;
  20. }
  21. LUAT_WEAK uint8_t luat_audio_is_finish(uint8_t multimedia_id){
  22. luat_audio_conf_t* audio_conf = luat_audio_get_config(multimedia_id);
  23. if (audio_conf){
  24. if (audio_conf->bus_type == LUAT_AUDIO_BUS_I2S){
  25. luat_i2s_conf_t * i2s_conf = luat_i2s_get_config(audio_conf->codec_conf.i2s_id);
  26. i2s_conf->state=LUAT_I2S_STATE_STOP?1:0;
  27. }
  28. }
  29. return -1;
  30. }
  31. LUAT_WEAK int luat_audio_play_stop(uint8_t multimedia_id){
  32. luat_audio_conf_t* audio_conf = luat_audio_get_config(multimedia_id);
  33. if (audio_conf){
  34. if (audio_conf->bus_type == LUAT_AUDIO_BUS_I2S){
  35. return luat_i2s_close(audio_conf->codec_conf.i2s_id);
  36. }
  37. }
  38. return -1;
  39. }
  40. LUAT_WEAK int luat_audio_play_get_last_error(uint8_t multimedia_id){
  41. return 0;
  42. }
  43. LUAT_WEAK int luat_audio_start_raw(uint8_t multimedia_id, uint8_t audio_format, uint8_t num_channels, uint32_t sample_rate, uint8_t bits_per_sample, uint8_t is_signed){
  44. luat_audio_conf_t* audio_conf = luat_audio_get_config(multimedia_id);
  45. if (audio_conf){
  46. if (audio_conf->bus_type == LUAT_AUDIO_BUS_I2S){
  47. luat_i2s_conf_t * i2s_conf = luat_i2s_get_config(audio_conf->codec_conf.i2s_id);
  48. i2s_conf->data_bits = bits_per_sample;
  49. i2s_conf->sample_rate = sample_rate,
  50. luat_i2s_modify(audio_conf->codec_conf.i2s_id,i2s_conf->channel_format,i2s_conf->data_bits, i2s_conf->sample_rate);
  51. audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_SET_RATE,sample_rate);
  52. luat_audio_pa(multimedia_id,1,0);
  53. }
  54. }
  55. return 0;
  56. }
  57. LUAT_WEAK int luat_audio_write_raw(uint8_t multimedia_id, uint8_t *data, uint32_t len){
  58. luat_audio_conf_t* audio_conf = luat_audio_get_config(multimedia_id);
  59. if (audio_conf){
  60. if (audio_conf->bus_type == LUAT_AUDIO_BUS_I2S){
  61. int send_bytes = 0;
  62. while (send_bytes < len) {
  63. int length = luat_i2s_send(audio_conf->codec_conf.i2s_id,data + send_bytes, len - send_bytes);
  64. if (length > 0) {
  65. send_bytes += length;
  66. }
  67. luat_rtos_task_sleep(1);
  68. }
  69. }
  70. }
  71. return 0;
  72. }
  73. LUAT_WEAK int luat_audio_stop_raw(uint8_t multimedia_id){
  74. luat_audio_conf_t* audio_conf = luat_audio_get_config(multimedia_id);
  75. if (audio_conf){
  76. if (audio_conf->bus_type == LUAT_AUDIO_BUS_I2S){
  77. return luat_i2s_close(audio_conf->codec_conf.i2s_id);
  78. }
  79. }
  80. return -1;
  81. }
  82. LUAT_WEAK int luat_audio_pause_raw(uint8_t multimedia_id, uint8_t is_pause){
  83. luat_audio_conf_t* audio_conf = luat_audio_get_config(multimedia_id);
  84. if (audio_conf){
  85. if (audio_conf->bus_type == LUAT_AUDIO_BUS_I2S){
  86. if (is_pause){
  87. luat_audio_pa(multimedia_id,0,0);
  88. luat_i2s_pause(audio_conf->codec_conf.i2s_id);
  89. }else{
  90. luat_audio_pa(multimedia_id,1,0);
  91. luat_i2s_resume(audio_conf->codec_conf.i2s_id);
  92. }
  93. return 0;
  94. }
  95. }
  96. return -1;
  97. }
  98. //以上函数通用方法待实现
  99. LUAT_WEAK void luat_audio_config_pa(uint8_t multimedia_id, uint32_t pin, int level, uint32_t dummy_time_len, uint32_t pa_delay_time){
  100. luat_audio_conf_t* audio_conf = luat_audio_get_config(multimedia_id);
  101. if (audio_conf){
  102. if (pin != LUAT_GPIO_NONE && pin<LUAT_GPIO_PIN_MAX && pin>0){
  103. audio_conf->pa_pin = pin;
  104. audio_conf->pa_on_level = level;
  105. luat_gpio_mode(pin, Luat_GPIO_OUTPUT, Luat_GPIO_DEFAULT, !level);
  106. #ifdef LUAT_USE_DRV_GPIO
  107. luat_drv_gpio_set(pin, !level);
  108. #else
  109. luat_gpio_set(pin, !level);
  110. #endif
  111. audio_conf->pa_is_control_enable = 1;
  112. luat_rtos_timer_create(&audio_conf->pa_delay_timer);
  113. }else{
  114. audio_conf->pa_pin = LUAT_GPIO_NONE;
  115. }
  116. audio_conf->after_sleep_ready_time = dummy_time_len;
  117. audio_conf->pa_delay_time = pa_delay_time;
  118. }
  119. }
  120. LUAT_WEAK void luat_audio_config_dac(uint8_t multimedia_id, int pin, int level, uint32_t dac_off_delay_time){
  121. luat_audio_conf_t* audio_conf = luat_audio_get_config(multimedia_id);
  122. if (audio_conf){
  123. if (pin != LUAT_GPIO_NONE){
  124. luat_gpio_mode(pin, Luat_GPIO_OUTPUT, Luat_GPIO_DEFAULT, level);
  125. audio_conf->power_pin = pin;
  126. audio_conf->power_on_level = level;
  127. audio_conf->power_off_delay_time = dac_off_delay_time;
  128. }else{
  129. audio_conf->power_pin = LUAT_GPIO_NONE;
  130. }
  131. }
  132. }
  133. static LUAT_RT_RET_TYPE pa_delay_timer_cb(LUAT_RT_CB_PARAM){
  134. uint8_t multimedia_id = (uint8_t)(uint32_t)param;
  135. luat_audio_pa(multimedia_id,1, 0);
  136. }
  137. LUAT_WEAK void luat_audio_pa(uint8_t multimedia_id,uint8_t on, uint32_t delay){
  138. luat_audio_conf_t* audio_conf = luat_audio_get_config(multimedia_id);
  139. if (audio_conf){
  140. if (!audio_conf->pa_is_control_enable) return;
  141. if (audio_conf->pa_delay_timer!=NULL&&delay>0){
  142. luat_rtos_timer_start(audio_conf->pa_delay_timer,delay,0,pa_delay_timer_cb,(void*)(uint32_t)multimedia_id);
  143. }
  144. else{
  145. #ifdef LUAT_USE_DRV_GPIO
  146. luat_drv_gpio_set(audio_conf->pa_pin, on?audio_conf->pa_on_level:!audio_conf->pa_on_level);
  147. #else
  148. luat_gpio_set(audio_conf->pa_pin, on?audio_conf->pa_on_level:!audio_conf->pa_on_level);
  149. #endif
  150. //LLOGD("PA %d,%d,%d", audio_conf->pa_pin, audio_conf->pa_on_level, on);
  151. if (on) audio_conf->pa_on_enable = 1;
  152. }
  153. }
  154. }
  155. LUAT_WEAK void luat_audio_power(uint8_t multimedia_id,uint8_t on){
  156. luat_audio_conf_t* audio_conf = luat_audio_get_config(multimedia_id);
  157. if (audio_conf){
  158. if (audio_conf->power_pin == LUAT_GPIO_NONE) return;
  159. #ifdef LUAT_USE_DRV_GPIO
  160. luat_drv_gpio_set(audio_conf->power_pin, on?audio_conf->power_on_level:!audio_conf->power_on_level);
  161. #else
  162. luat_gpio_set(audio_conf->power_pin, on?audio_conf->power_on_level:!audio_conf->power_on_level);
  163. #endif
  164. }
  165. }
  166. LUAT_WEAK uint16_t luat_audio_vol(uint8_t multimedia_id, uint16_t vol){
  167. luat_audio_conf_t* audio_conf = luat_audio_get_config(multimedia_id);
  168. if (audio_conf == NULL || vol < 0 || vol > 1000) return -1;
  169. if (audio_conf->codec_conf.codec_opts->no_control) {
  170. audio_conf->soft_vol = vol;
  171. return vol;
  172. }
  173. audio_conf->soft_vol = vol<=100?100:vol;
  174. if (audio_conf->bus_type == LUAT_AUDIO_BUS_I2S){
  175. uint8_t sleep_mode = audio_conf->sleep_mode;
  176. audio_conf->last_vol = vol;
  177. if (sleep_mode && audio_conf->codec_conf.codec_opts->no_control!=1) luat_audio_pm_request(multimedia_id,LUAT_AUDIO_PM_RESUME);
  178. if (vol <= 100){
  179. audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_SET_VOICE_VOL,vol);
  180. vol = audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_GET_VOICE_VOL,0);
  181. }else{
  182. audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_SET_VOICE_VOL,100);
  183. }
  184. if (sleep_mode && audio_conf->codec_conf.codec_opts->no_control!=1) luat_audio_pm_request(multimedia_id,sleep_mode);
  185. return vol;
  186. }
  187. return -1;
  188. }
  189. LUAT_WEAK uint8_t luat_audio_mic_vol(uint8_t multimedia_id, uint16_t vol){
  190. if(vol < 0 || vol > 100) return -1;
  191. luat_audio_conf_t* audio_conf = luat_audio_get_config(multimedia_id);
  192. if (audio_conf){
  193. if (audio_conf->bus_type == LUAT_AUDIO_BUS_I2S){
  194. if (audio_conf->codec_conf.codec_opts->no_control) return -1;
  195. uint8_t sleep_mode = audio_conf->sleep_mode;
  196. if (sleep_mode) luat_audio_pm_request(multimedia_id,LUAT_AUDIO_PM_RESUME);
  197. audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_SET_MIC_VOL,vol);
  198. vol = audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_GET_MIC_VOL,0);
  199. if (sleep_mode) luat_audio_pm_request(multimedia_id,sleep_mode);
  200. audio_conf->last_mic_vol = vol;
  201. return vol;
  202. }
  203. }
  204. return -1;
  205. }
  206. //通用方式待实现
  207. LUAT_WEAK int luat_audio_play_blank(uint8_t multimedia_id, uint8_t on_off){
  208. return -1;
  209. }
  210. LUAT_WEAK uint8_t luat_audio_mute(uint8_t multimedia_id, uint8_t on){
  211. luat_audio_conf_t* audio_conf = luat_audio_get_config(multimedia_id);
  212. if (audio_conf){
  213. if (audio_conf->bus_type == LUAT_AUDIO_BUS_I2S)
  214. {
  215. if (audio_conf->codec_conf.codec_opts->no_control)
  216. {
  217. luat_audio_play_blank(multimedia_id, 1);
  218. }
  219. else
  220. {
  221. audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_SET_MUTE,on);
  222. return audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_GET_MUTE,0);
  223. }
  224. }
  225. }
  226. return -1;
  227. }
  228. LUAT_WEAK int luat_audio_setup_codec(uint8_t multimedia_id, const luat_audio_codec_conf_t *codec_conf){
  229. luat_audio_conf_t* audio_conf = luat_audio_get_config(multimedia_id);
  230. if (audio_conf){
  231. if (audio_conf->bus_type == LUAT_AUDIO_BUS_I2S){
  232. audio_conf->codec_conf= *codec_conf;
  233. return 0;
  234. }
  235. }
  236. return -1;
  237. }
  238. LUAT_WEAK int luat_audio_init(uint8_t multimedia_id, uint16_t init_vol, uint16_t init_mic_vol){
  239. luat_audio_conf_t* audio_conf = luat_audio_get_config(multimedia_id);
  240. if (audio_conf == NULL) return -1;
  241. audio_conf->last_wakeup_time_ms = luat_mcu_tick64_ms();
  242. audio_conf->last_vol = init_vol;
  243. audio_conf->last_mic_vol = init_mic_vol;
  244. if (audio_conf->bus_type == LUAT_AUDIO_BUS_I2S){
  245. if (audio_conf->codec_conf.codec_opts->no_control)
  246. {
  247. audio_conf->sleep_mode = LUAT_AUDIO_PM_SHUTDOWN;
  248. return 0;
  249. }
  250. luat_audio_power_keep_ctrl_by_bsp(1);
  251. int result = audio_conf->codec_conf.codec_opts->init(&audio_conf->codec_conf, LUAT_CODEC_MODE_SLAVE);
  252. if (result) return result;
  253. LLOGD("codec init %s ",audio_conf->codec_conf.codec_opts->name);
  254. result = audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_SET_RATE, LUAT_I2S_HZ_16k);
  255. if (result) return result;
  256. result = audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_SET_BITS, LUAT_I2S_BITS_16);
  257. if (result) return result;
  258. result = audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_SET_FORMAT,LUAT_CODEC_FORMAT_I2S);
  259. if (result) return result;
  260. result = audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_SET_VOICE_VOL, init_vol);
  261. if (result) return result;
  262. result = audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_SET_MIC_VOL, init_mic_vol);
  263. if (result) return result;
  264. if (!audio_conf->pa_is_control_enable) //PA无法控制的状态,则通过静音来控制
  265. {
  266. audio_conf->codec_conf.codec_opts->start(&audio_conf->codec_conf);
  267. audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_SET_MUTE, 1);
  268. }
  269. else
  270. {
  271. luat_audio_pm_request(multimedia_id,LUAT_AUDIO_PM_STANDBY); //默认进入standby模式
  272. }
  273. audio_conf->sleep_mode = LUAT_AUDIO_PM_STANDBY;
  274. return 0;
  275. }
  276. return 0;
  277. }
  278. LUAT_WEAK int luat_audio_set_bus_type(uint8_t multimedia_id,uint8_t bus_type){
  279. luat_audio_conf_t* audio_conf = luat_audio_get_config(multimedia_id);
  280. if (audio_conf){
  281. if (bus_type == LUAT_AUDIO_BUS_I2S){
  282. audio_conf->codec_conf.multimedia_id = multimedia_id;
  283. audio_conf->bus_type = LUAT_AUDIO_BUS_I2S;
  284. return 0;
  285. }
  286. }
  287. return -1;
  288. }
  289. LUAT_WEAK int luat_audio_pm_request(uint8_t multimedia_id,luat_audio_pm_mode_t mode){
  290. luat_audio_conf_t* audio_conf = luat_audio_get_config(multimedia_id);
  291. if (audio_conf!=NULL && audio_conf->bus_type == LUAT_AUDIO_BUS_I2S){
  292. if (!audio_conf->pa_is_control_enable)
  293. {
  294. if (mode)
  295. {
  296. luat_audio_mute(multimedia_id, 1);
  297. }
  298. else
  299. {
  300. luat_audio_mute(multimedia_id, 0);
  301. }
  302. audio_conf->sleep_mode = mode;
  303. return 0;
  304. }
  305. if (audio_conf->codec_conf.codec_opts->no_control) //codec没有寄存器的情况
  306. {
  307. switch(mode)
  308. {
  309. case LUAT_AUDIO_PM_RESUME:
  310. luat_audio_power_keep_ctrl_by_bsp(1);
  311. luat_audio_play_blank(multimedia_id, 1);
  312. luat_audio_power(multimedia_id,1);
  313. break;
  314. case LUAT_AUDIO_PM_STANDBY: //只关PA,codec不关,这样下次播放的时候不需要重新启动codec,节省启动时间
  315. luat_audio_power_keep_ctrl_by_bsp(1);
  316. luat_audio_play_blank(multimedia_id, 1);
  317. luat_audio_pa(multimedia_id,0,0);
  318. break;
  319. default:
  320. luat_audio_power_keep_ctrl_by_bsp(0);
  321. luat_audio_pa(multimedia_id,0,0);
  322. if (audio_conf->power_off_delay_time)
  323. luat_rtos_task_sleep(audio_conf->power_off_delay_time);
  324. luat_audio_power(multimedia_id,0);
  325. audio_conf->wakeup_ready = 0;
  326. audio_conf->pa_on_enable = 0;
  327. #ifdef __LUATOS__
  328. luat_audio_play_blank(multimedia_id, 0);
  329. #endif
  330. break;
  331. }
  332. audio_conf->sleep_mode = mode;
  333. }
  334. else //codec有寄存器的情况
  335. {
  336. if (audio_conf->sleep_mode == mode) return 0;
  337. switch (mode){
  338. case LUAT_AUDIO_PM_RESUME:
  339. luat_audio_power_keep_ctrl_by_bsp(1);
  340. if (!audio_conf->speech_uplink_type && !audio_conf->speech_downlink_type && !audio_conf->record_mode)
  341. {
  342. //LLOGD("audio pm !");
  343. luat_audio_play_blank(multimedia_id, 1);
  344. }
  345. if (LUAT_AUDIO_PM_POWER_OFF == audio_conf->sleep_mode) //之前已经强制断电过了,就必须重新初始化
  346. {
  347. luat_audio_init(multimedia_id, audio_conf->last_vol, audio_conf->last_mic_vol);
  348. }
  349. //LLOGD("audio pm %d,%d", audio_conf->last_vol, audio_conf->last_mic_vol);
  350. audio_conf->codec_conf.codec_opts->start(&audio_conf->codec_conf);
  351. audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_SET_VOICE_VOL, audio_conf->last_vol);
  352. audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_SET_MIC_VOL, audio_conf->last_mic_vol);
  353. audio_conf->last_wakeup_time_ms = luat_mcu_tick64_ms();
  354. audio_conf->sleep_mode = LUAT_AUDIO_PM_RESUME;
  355. break;
  356. case LUAT_AUDIO_PM_STANDBY:
  357. luat_audio_power_keep_ctrl_by_bsp(1);
  358. luat_audio_pa(multimedia_id,0,0);
  359. audio_conf->codec_conf.codec_opts->stop(&audio_conf->codec_conf);
  360. audio_conf->sleep_mode = LUAT_AUDIO_PM_STANDBY;
  361. #ifdef __LUATOS__
  362. luat_audio_play_blank(multimedia_id, 0);
  363. #endif
  364. break;
  365. case LUAT_AUDIO_PM_SHUTDOWN:
  366. luat_audio_pa(multimedia_id,0,0);
  367. if (audio_conf->power_off_delay_time)
  368. luat_rtos_task_sleep(audio_conf->power_off_delay_time);
  369. audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_MODE_PWRDOWN,0);
  370. luat_audio_power_keep_ctrl_by_bsp(0);
  371. audio_conf->wakeup_ready = 0;
  372. audio_conf->pa_on_enable = 0;
  373. audio_conf->sleep_mode = LUAT_AUDIO_PM_SHUTDOWN;
  374. #ifdef __LUATOS__
  375. luat_audio_play_blank(multimedia_id, 0);
  376. #endif
  377. break;
  378. case LUAT_AUDIO_PM_POWER_OFF:
  379. luat_audio_pa(multimedia_id,0,0);
  380. if (audio_conf->power_off_delay_time)
  381. luat_rtos_task_sleep(audio_conf->power_off_delay_time);
  382. luat_audio_power(multimedia_id,0);
  383. luat_audio_power_keep_ctrl_by_bsp(0);
  384. audio_conf->wakeup_ready = 0;
  385. audio_conf->pa_on_enable = 0;
  386. audio_conf->sleep_mode = LUAT_AUDIO_PM_POWER_OFF;
  387. #ifdef __LUATOS__
  388. luat_audio_play_blank(multimedia_id, 0);
  389. #endif
  390. break;
  391. default:
  392. return -1;
  393. }
  394. }
  395. return 0;
  396. }
  397. return -1;
  398. }
  399. LUAT_WEAK void luat_audio_power_keep_ctrl_by_bsp(uint8_t on_off)
  400. {
  401. ;
  402. }
  403. LUAT_WEAK void luat_audio_run_callback_in_task(void *api, uint8_t *data, uint32_t len)
  404. {
  405. ;
  406. }
  407. LUAT_WEAK void luat_audio_setup_record_callback(uint8_t multimedia_id, luat_audio_record_callback callback, void *param)
  408. {
  409. return ;
  410. }