luat_multimedia_audio.c 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447
  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. audio_conf->power_pin = pin;
  125. audio_conf->power_on_level = level;
  126. audio_conf->power_off_delay_time = dac_off_delay_time;
  127. }else{
  128. audio_conf->power_pin = LUAT_GPIO_NONE;
  129. }
  130. }
  131. }
  132. static LUAT_RT_RET_TYPE pa_delay_timer_cb(LUAT_RT_CB_PARAM){
  133. uint8_t multimedia_id = (uint8_t)param;
  134. luat_audio_pa(multimedia_id,1, 0);
  135. }
  136. LUAT_WEAK void luat_audio_pa(uint8_t multimedia_id,uint8_t on, uint32_t delay){
  137. luat_audio_conf_t* audio_conf = luat_audio_get_config(multimedia_id);
  138. if (audio_conf){
  139. if (!audio_conf->pa_is_control_enable) return;
  140. if (audio_conf->pa_delay_timer!=NULL&&delay>0){
  141. luat_rtos_timer_start(audio_conf->pa_delay_timer,delay,0,pa_delay_timer_cb,(void*)multimedia_id);
  142. }
  143. else{
  144. #ifdef LUAT_USE_DRV_GPIO
  145. luat_drv_gpio_set(audio_conf->pa_pin, on?audio_conf->pa_on_level:!audio_conf->pa_on_level);
  146. #else
  147. luat_gpio_set(audio_conf->pa_pin, on?audio_conf->pa_on_level:!audio_conf->pa_on_level);
  148. #endif
  149. //LLOGD("PA %d,%d,%d", audio_conf->pa_pin, audio_conf->pa_on_level, on);
  150. if (on) audio_conf->pa_on_enable = 1;
  151. }
  152. }
  153. }
  154. LUAT_WEAK void luat_audio_power(uint8_t multimedia_id,uint8_t on){
  155. luat_audio_conf_t* audio_conf = luat_audio_get_config(multimedia_id);
  156. if (audio_conf){
  157. if (audio_conf->power_pin == LUAT_GPIO_NONE) return;
  158. #ifdef LUAT_USE_DRV_GPIO
  159. luat_drv_gpio_set(audio_conf->power_pin, on?audio_conf->power_on_level:!audio_conf->power_on_level);
  160. #else
  161. luat_gpio_set(audio_conf->power_pin, on?audio_conf->power_on_level:!audio_conf->power_on_level);
  162. #endif
  163. }
  164. }
  165. LUAT_WEAK uint16_t luat_audio_vol(uint8_t multimedia_id, uint16_t vol){
  166. luat_audio_conf_t* audio_conf = luat_audio_get_config(multimedia_id);
  167. if (audio_conf == NULL || vol < 0 || vol > 1000) return -1;
  168. if (audio_conf->codec_conf.codec_opts->no_control) {
  169. audio_conf->soft_vol = vol;
  170. return vol;
  171. }
  172. audio_conf->soft_vol = vol<=100?100:vol;
  173. if (audio_conf->bus_type == LUAT_AUDIO_BUS_I2S){
  174. uint8_t sleep_mode = audio_conf->sleep_mode;
  175. audio_conf->last_vol = vol;
  176. if (sleep_mode && audio_conf->codec_conf.codec_opts->no_control!=1) luat_audio_pm_request(multimedia_id,LUAT_AUDIO_PM_RESUME);
  177. if (vol <= 100){
  178. audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_SET_VOICE_VOL,vol);
  179. vol = audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_GET_VOICE_VOL,0);
  180. }else{
  181. audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_SET_VOICE_VOL,100);
  182. }
  183. if (sleep_mode && audio_conf->codec_conf.codec_opts->no_control!=1) luat_audio_pm_request(multimedia_id,sleep_mode);
  184. return vol;
  185. }
  186. return -1;
  187. }
  188. LUAT_WEAK uint8_t luat_audio_mic_vol(uint8_t multimedia_id, uint16_t vol){
  189. if(vol < 0 || vol > 100) return -1;
  190. luat_audio_conf_t* audio_conf = luat_audio_get_config(multimedia_id);
  191. if (audio_conf){
  192. if (audio_conf->bus_type == LUAT_AUDIO_BUS_I2S){
  193. if (audio_conf->codec_conf.codec_opts->no_control) return -1;
  194. uint8_t sleep_mode = audio_conf->sleep_mode;
  195. if (sleep_mode) luat_audio_pm_request(multimedia_id,LUAT_AUDIO_PM_RESUME);
  196. audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_SET_MIC_VOL,vol);
  197. vol = audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_GET_MIC_VOL,0);
  198. if (sleep_mode) luat_audio_pm_request(multimedia_id,sleep_mode);
  199. audio_conf->last_mic_vol = vol;
  200. return vol;
  201. }
  202. }
  203. return -1;
  204. }
  205. //通用方式待实现
  206. LUAT_WEAK int luat_audio_play_blank(uint8_t multimedia_id, uint8_t on_off){
  207. return -1;
  208. }
  209. LUAT_WEAK uint8_t luat_audio_mute(uint8_t multimedia_id, uint8_t on){
  210. luat_audio_conf_t* audio_conf = luat_audio_get_config(multimedia_id);
  211. if (audio_conf){
  212. if (audio_conf->bus_type == LUAT_AUDIO_BUS_I2S)
  213. {
  214. if (audio_conf->codec_conf.codec_opts->no_control)
  215. {
  216. luat_audio_play_blank(multimedia_id, 1);
  217. }
  218. else
  219. {
  220. audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_SET_MUTE,on);
  221. return audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_GET_MUTE,0);
  222. }
  223. }
  224. }
  225. return -1;
  226. }
  227. LUAT_WEAK int luat_audio_setup_codec(uint8_t multimedia_id, const luat_audio_codec_conf_t *codec_conf){
  228. luat_audio_conf_t* audio_conf = luat_audio_get_config(multimedia_id);
  229. if (audio_conf){
  230. if (audio_conf->bus_type == LUAT_AUDIO_BUS_I2S){
  231. audio_conf->codec_conf= *codec_conf;
  232. return 0;
  233. }
  234. }
  235. return -1;
  236. }
  237. LUAT_WEAK int luat_audio_init(uint8_t multimedia_id, uint16_t init_vol, uint16_t init_mic_vol){
  238. luat_audio_conf_t* audio_conf = luat_audio_get_config(multimedia_id);
  239. if (audio_conf == NULL) return -1;
  240. audio_conf->last_wakeup_time_ms = luat_mcu_tick64_ms();
  241. audio_conf->last_vol = init_vol;
  242. audio_conf->last_mic_vol = init_mic_vol;
  243. if (audio_conf->bus_type == LUAT_AUDIO_BUS_I2S){
  244. if (audio_conf->codec_conf.codec_opts->no_control)
  245. {
  246. audio_conf->sleep_mode = LUAT_AUDIO_PM_SHUTDOWN;
  247. return 0;
  248. }
  249. luat_audio_power_keep_ctrl_by_bsp(1);
  250. int result = audio_conf->codec_conf.codec_opts->init(&audio_conf->codec_conf, LUAT_CODEC_MODE_SLAVE);
  251. if (result) return result;
  252. LLOGD("codec init %s ",audio_conf->codec_conf.codec_opts->name);
  253. result = audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_SET_RATE, LUAT_I2S_HZ_16k);
  254. if (result) return result;
  255. result = audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_SET_BITS, LUAT_I2S_BITS_16);
  256. if (result) return result;
  257. result = audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_SET_FORMAT,LUAT_CODEC_FORMAT_I2S);
  258. if (result) return result;
  259. result = audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_SET_VOICE_VOL, init_vol);
  260. if (result) return result;
  261. result = audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_SET_MIC_VOL, init_mic_vol);
  262. if (result) return result;
  263. if (!audio_conf->pa_is_control_enable) //PA无法控制的状态,则通过静音来控制
  264. {
  265. audio_conf->codec_conf.codec_opts->start(&audio_conf->codec_conf);
  266. audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_SET_MUTE, 1);
  267. }
  268. else
  269. {
  270. luat_audio_pm_request(multimedia_id,LUAT_AUDIO_PM_STANDBY); //默认进入standby模式
  271. }
  272. audio_conf->sleep_mode = LUAT_AUDIO_PM_STANDBY;
  273. return 0;
  274. }
  275. return 0;
  276. }
  277. LUAT_WEAK int luat_audio_set_bus_type(uint8_t multimedia_id,uint8_t bus_type){
  278. luat_audio_conf_t* audio_conf = luat_audio_get_config(multimedia_id);
  279. if (audio_conf){
  280. if (bus_type == LUAT_AUDIO_BUS_I2S){
  281. audio_conf->codec_conf.multimedia_id = multimedia_id;
  282. audio_conf->bus_type = LUAT_AUDIO_BUS_I2S;
  283. return 0;
  284. }
  285. }
  286. return -1;
  287. }
  288. LUAT_WEAK int luat_audio_pm_request(uint8_t multimedia_id,luat_audio_pm_mode_t mode){
  289. luat_audio_conf_t* audio_conf = luat_audio_get_config(multimedia_id);
  290. if (audio_conf!=NULL && audio_conf->bus_type == LUAT_AUDIO_BUS_I2S){
  291. if (!audio_conf->pa_is_control_enable)
  292. {
  293. if (mode)
  294. {
  295. luat_audio_mute(multimedia_id, 1);
  296. }
  297. else
  298. {
  299. luat_audio_mute(multimedia_id, 0);
  300. }
  301. audio_conf->sleep_mode = mode;
  302. return 0;
  303. }
  304. if (audio_conf->codec_conf.codec_opts->no_control) //codec没有寄存器的情况
  305. {
  306. switch(mode)
  307. {
  308. case LUAT_AUDIO_PM_RESUME:
  309. luat_audio_power_keep_ctrl_by_bsp(1);
  310. luat_audio_play_blank(multimedia_id, 1);
  311. luat_audio_power(multimedia_id,1);
  312. break;
  313. case LUAT_AUDIO_PM_STANDBY: //只关PA,codec不关,这样下次播放的时候不需要重新启动codec,节省启动时间
  314. luat_audio_power_keep_ctrl_by_bsp(1);
  315. luat_audio_play_blank(multimedia_id, 1);
  316. luat_audio_pa(multimedia_id,0,0);
  317. break;
  318. default:
  319. luat_audio_power_keep_ctrl_by_bsp(0);
  320. luat_audio_pa(multimedia_id,0,0);
  321. if (audio_conf->power_off_delay_time)
  322. luat_rtos_task_sleep(audio_conf->power_off_delay_time);
  323. luat_audio_power(multimedia_id,0);
  324. audio_conf->wakeup_ready = 0;
  325. audio_conf->pa_on_enable = 0;
  326. #ifdef __LUATOS__
  327. luat_audio_play_blank(multimedia_id, 0);
  328. #endif
  329. break;
  330. }
  331. audio_conf->sleep_mode = mode;
  332. }
  333. else //codec有寄存器的情况
  334. {
  335. if (audio_conf->sleep_mode == mode) return 0;
  336. switch (mode){
  337. case LUAT_AUDIO_PM_RESUME:
  338. luat_audio_power_keep_ctrl_by_bsp(1);
  339. if (!audio_conf->speech_uplink_type && !audio_conf->speech_downlink_type && !audio_conf->record_mode)
  340. {
  341. //LLOGD("audio pm !");
  342. luat_audio_play_blank(multimedia_id, 1);
  343. }
  344. if (LUAT_AUDIO_PM_POWER_OFF == audio_conf->sleep_mode) //之前已经强制断电过了,就必须重新初始化
  345. {
  346. luat_audio_init(multimedia_id, audio_conf->last_vol, audio_conf->last_mic_vol);
  347. }
  348. //LLOGD("audio pm %d,%d", audio_conf->last_vol, audio_conf->last_mic_vol);
  349. audio_conf->codec_conf.codec_opts->start(&audio_conf->codec_conf);
  350. audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_SET_VOICE_VOL, audio_conf->last_vol);
  351. audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_SET_MIC_VOL, audio_conf->last_mic_vol);
  352. audio_conf->last_wakeup_time_ms = luat_mcu_tick64_ms();
  353. audio_conf->sleep_mode = LUAT_AUDIO_PM_RESUME;
  354. break;
  355. case LUAT_AUDIO_PM_STANDBY:
  356. luat_audio_power_keep_ctrl_by_bsp(1);
  357. luat_audio_pa(multimedia_id,0,0);
  358. audio_conf->codec_conf.codec_opts->stop(&audio_conf->codec_conf);
  359. audio_conf->sleep_mode = LUAT_AUDIO_PM_STANDBY;
  360. #ifdef __LUATOS__
  361. luat_audio_play_blank(multimedia_id, 0);
  362. #endif
  363. break;
  364. case LUAT_AUDIO_PM_SHUTDOWN:
  365. luat_audio_pa(multimedia_id,0,0);
  366. if (audio_conf->power_off_delay_time)
  367. luat_rtos_task_sleep(audio_conf->power_off_delay_time);
  368. audio_conf->codec_conf.codec_opts->control(&audio_conf->codec_conf,LUAT_CODEC_MODE_PWRDOWN,0);
  369. luat_audio_power_keep_ctrl_by_bsp(0);
  370. audio_conf->wakeup_ready = 0;
  371. audio_conf->pa_on_enable = 0;
  372. audio_conf->sleep_mode = LUAT_AUDIO_PM_SHUTDOWN;
  373. #ifdef __LUATOS__
  374. luat_audio_play_blank(multimedia_id, 0);
  375. #endif
  376. break;
  377. case LUAT_AUDIO_PM_POWER_OFF:
  378. luat_audio_pa(multimedia_id,0,0);
  379. if (audio_conf->power_off_delay_time)
  380. luat_rtos_task_sleep(audio_conf->power_off_delay_time);
  381. luat_audio_power(multimedia_id,0);
  382. luat_audio_power_keep_ctrl_by_bsp(0);
  383. audio_conf->wakeup_ready = 0;
  384. audio_conf->pa_on_enable = 0;
  385. audio_conf->sleep_mode = LUAT_AUDIO_PM_POWER_OFF;
  386. #ifdef __LUATOS__
  387. luat_audio_play_blank(multimedia_id, 0);
  388. #endif
  389. break;
  390. default:
  391. return -1;
  392. }
  393. }
  394. return 0;
  395. }
  396. return -1;
  397. }
  398. LUAT_WEAK void luat_audio_power_keep_ctrl_by_bsp(uint8_t on_off)
  399. {
  400. ;
  401. }
  402. LUAT_WEAK void luat_audio_run_callback_in_task(void *api, uint8_t *data, uint32_t len)
  403. {
  404. ;
  405. }
  406. LUAT_WEAK void luat_audio_setup_record_callback(uint8_t multimedia_id, luat_audio_record_callback callback, void *param)
  407. {
  408. return ;
  409. }