KT25-1015_AC695x_SDK310/apps/soundbox/aec/br23/audio_aec_demo.c

485 lines
14 KiB
C
Raw Normal View History

2025-10-15 07:18:43 +00:00
/*
***************************************************************************
* AUDIO AEC DEMO
* File : audio_aec_demo.c
* By : GZR
* Notes : 1.
* (1)free_ram: (mem_stats:physics memory size xxxx bytes)
* (2)mux_rx_bulk:(MUX_RX_BULK_MAX)
* 2.demo默认将输入数据copy到输出
* audio_aec_run()
* 3.mic ENC开发
* #define TCFG_AUDIO_DUAL_MIC_ENABLE ENABLE_THIS_MOUDLE
* 4.使
* 5.EQ处理AEC_UL_EQ_EN
* 6.使240MHzAEC_CLK来修改
***************************************************************************
*/
#include "aec_user.h"
#include "system/includes.h"
#include "media/includes.h"
#include "media/audio_eq_drc_apply.h"
#include "circular_buf.h"
#include "debug.h"
#if defined(TCFG_CVP_DEVELOP_ENABLE) && (TCFG_CVP_DEVELOP_ENABLE == 1)
#define AEC_MALLOC_ENABLE 0 /*是否使用动态内存*/
#define AEC_EXTERN_REFERENCE_ENABLE 0 /*是否使用外部取的参考dac数据*/
#define AEC_CLK (160 * 1000000L) /*模块运行时钟(MaxFre:240MHz)*/
#define AEC_FRAME_POINTS 256 /*AEC处理帧长跟mic采样长度关联*/
#define AEC_FRAME_SIZE (AEC_FRAME_POINTS << 1)
#if TCFG_AUDIO_DUAL_MIC_ENABLE /*ac695不支持双mic*/
#define AEC_MIC_NUM 2 /*双mic*/
#else
#define AEC_MIC_NUM 1 /*单mic*/
#endif/*TCFG_AUDIO_DUAL_MIC_ENABLE*/
#ifdef AUDIO_PCM_DEBUG
/*AEC串口数据导出*/
const u8 CONST_AEC_EXPORT = 1;
#else
const u8 CONST_AEC_EXPORT = 0;
#endif/*AUDIO_PCM_DEBUG*/
/*上行数据eq*/
#define AEC_UL_EQ_EN 0//TCFG_AEC_UL_EQ_ENABLE
extern int audio_dac_read_reset(void);
extern int audio_dac_read(s16 points_offset, void *data, int len);
extern void esco_enc_resume(void);
extern int aec_uart_init();
extern int aec_uart_fill(u8 ch, void *buf, u16 size);
extern void aec_uart_write(void);
extern int aec_uart_close(void);
/*
*buf复用使用说明
*(1)apps/earphone/log_config/lib_btctrler_config.c
* const int CONFIG_ESCO_MUX_RX_BULK_ENABLE = 1;
*(2)MUX_RX_BULK_TEST_DEMO是使用范例
*(3)zalloc_mux_rx_bulk/free_mux_rx_bulk用法和普通malloc/free一样
*/
//#define MUX_RX_BULK_TEST_DEMO /*蓝牙收数buf复用使用范例*/
#define MUX_RX_BULK_MAX (9 * 1024 - 100) /*蓝牙收数buf最多可以申请到7k*/
extern void *zalloc_mux_rx_bulk(int size);
extern void free_mux_rx_bulk(void *buf);
int *mux_rx_bulk_test = NULL;
/*AEC输入buf复用mic_adc采样buf*/
#define MIC_BULK_MAX 3
struct mic_bulk {
struct list_head entry;
s16 *addr;
u16 len;
u16 used;
};
struct audio_aec_hdl {
u8 start;
volatile u8 busy;
u8 output_buf[1000]; /*aec模块输出缓存*/
cbuffer_t output_cbuf;
s16 *mic; /*主mic数据地址*/
s16 *mic_ref; /*参考mic数据地址*/
s16 *free_ram; /*当前可用内存*/
s16 spk_ref[AEC_FRAME_POINTS]; /*扬声器参考数据*/
s16 out[AEC_FRAME_POINTS]; /*运算输出地址*/
OS_SEM sem;
/*数据复用相关数据结构*/
struct mic_bulk in_bulk[MIC_BULK_MAX];
struct mic_bulk inref_bulk[MIC_BULK_MAX];
struct list_head in_head;
struct list_head inref_head;
#if AEC_UL_EQ_EN
struct audio_eq *ul_eq;
#endif/*AEC_UL_EQ_EN*/
struct aec_s_attr attr;
#if AEC_EXTERN_REFERENCE_ENABLE
u8 extern_ref_buf[1000]; /*aec外部参考数据缓存*/
cbuffer_t extern_ref_cbuf;
#endif // AEC_EXTERN_REFERENCE_ENABLE
};
struct audio_aec_hdl *aec_hdl = NULL;
#if (AEC_MALLOC_ENABLE == 0)
struct audio_aec_hdl aec_hdl_mem AT(.aec_mem);
#endif /*AEC_MALLOC_ENABLE*/
extern void esco_adc_mic_en();
void audio_aec_ref_start(u8 en)
{
if (aec_hdl) {
if (en != aec_hdl->attr.fm_tx_start) {
aec_hdl->attr.fm_tx_start = en;
esco_adc_mic_en();
y_printf("fm_tx_start:%d\n", en);
}
}
}
int audio_aec_output_read(s16 *buf, u16 len)
{
local_irq_disable();
if (!aec_hdl || !aec_hdl->start) {
printf("audio_aec close now");
local_irq_enable();
return -EINVAL;
}
u16 rlen = cbuf_read(&aec_hdl->output_cbuf, buf, len);
if (rlen == 0) {
//putchar('N');
}
local_irq_enable();
return rlen;
}
static int audio_aec_output(s16 *buf, u16 len)
{
u16 wlen = 0;
if (aec_hdl && aec_hdl->start) {
wlen = cbuf_write(&aec_hdl->output_cbuf, buf, len);
if (wlen != len) {
printf("aec_out_full:%d,%d\n", len, wlen);
}
esco_enc_resume();
}
return wlen;
}
#if AEC_UL_EQ_EN
static int ul_eq_output(void *priv, void *data, u32 len)
{
return 0;
}
#endif/*AEC_UL_EQ_EN*/
/*
*使:physics memory size xxxx bytes
*3k bytes的剩余空间给到系统调度开销
*/
static void sys_memory_trace(void)
{
static int cnt = 0;
if (cnt++ > 200) {
cnt = 0;
mem_stats();
}
}
/*
*********************************************************************
* Audio AEC RUN
* Description: AEC数据处理核心
* Arguments : in mic数据
* inref mic数据(mic降噪有用)
* ref speaker参考数据
* out
* Return :
* Note(s) : AEC_core
*********************************************************************
*/
static int audio_aec_run(s16 *in, s16 *inref, s16 *ref, s16 *out, u16 points)
{
int out_size = 0;
putchar('.');
memcpy(out, in, (points << 1));
//memcpy(out, inref, (points << 1));
out_size = points << 1;
#if AEC_UL_EQ_EN
if (aec_hdl->ul_eq) {
//putchar('Q');
audio_eq_run(aec_hdl->ul_eq, out, out_size);
}
#endif/*AEC_UL_EQ_EN*/
sys_memory_trace();
return out_size;
}
/*
*********************************************************************
* Audio AEC Task
* Description: AEC任务
* Arguments : priv
* Return : None.
* Note(s) : None.
*********************************************************************
*/
static void audio_aec_task(void *priv)
{
printf("==Audio AEC Task==\n");
struct mic_bulk *bulk = NULL;
struct mic_bulk *bulk_ref = NULL;
u8 pend = 1;
while (1) {
if (pend) {
os_sem_pend(&aec_hdl->sem, 0);
}
pend = 1;
if (aec_hdl->start) {
if (!list_empty(&aec_hdl->in_head)) {
aec_hdl->busy = 1;
local_irq_disable();
/*1.获取主mic数据*/
bulk = list_first_entry(&aec_hdl->in_head, struct mic_bulk, entry);
list_del(&bulk->entry);
aec_hdl->mic = bulk->addr;
#if (AEC_MIC_NUM == 2)
/*2.获取参考mic数据*/
bulk_ref = list_first_entry(&aec_hdl->inref_head, struct mic_bulk, entry);
list_del(&bulk_ref->entry);
aec_hdl->mic_ref = bulk_ref->addr;
#endif/*Dual_Microphone*/
local_irq_enable();
#if AEC_EXTERN_REFERENCE_ENABLE
/*3.获取外部参考数据*/
//audio_dac_read(60, aec_hdl->spk_ref, AEC_FRAME_SIZE);
cbuf_read(&aec_hdl->extern_ref_cbuf, aec_hdl->spk_ref, AEC_FRAME_SIZE);
#else
/*3.获取speaker参考数据*/
audio_dac_read(60, aec_hdl->spk_ref, AEC_FRAME_SIZE);
#endif //AEC_EXTERN_REFERENCE_ENABLE
/*4.算法处理*/
int out_len = audio_aec_run(aec_hdl->mic, aec_hdl->mic_ref, aec_hdl->spk_ref, aec_hdl->out, AEC_FRAME_POINTS);
/*5.结果输出*/
audio_aec_output(aec_hdl->out, out_len);
/*6.数据导出*/
if (CONST_AEC_EXPORT) {
aec_uart_fill(0, aec_hdl->mic, 512); //主mic数据
// aec_uart_fill(1, aec_hdl->mic_ref, 512); //副mic数据
aec_uart_fill(1, aec_hdl->spk_ref, 512); //扬声器数据
aec_uart_fill(2, aec_hdl->out, out_len); //算法运算结果
aec_uart_write();
}
bulk->used = 0;
#if (AEC_MIC_NUM == 2)
bulk_ref->used = 0;
#endif/*Dual_Microphone*/
aec_hdl->busy = 0;
pend = 0;
}
}
}
}
/*
*********************************************************************
* Audio AEC Init
* Description: AEC模块
* Arguments : sample_rate (8000/16000)
* Return : 0
* Note(s) : None.
*********************************************************************
*/
int audio_aec_init(u16 sample_rate)
{
printf("audio_aec_init,sr = %d\n", sample_rate);
mem_stats();
if (aec_hdl) {
return -1;
}
#if AEC_MALLOC_ENABLE
aec_hdl = zalloc(sizeof(struct audio_aec_hdl));
#else /*AEC_MALLOC_ENABLE == 0*/
memset(&aec_hdl_mem, 0, sizeof(aec_hdl_mem));
aec_hdl = &aec_hdl_mem;
#endif /*AEC_MALLOC_ENABLE*/
printf("aec_hdl size:%d\n", sizeof(struct audio_aec_hdl));
clk_set("sys", AEC_CLK);
INIT_LIST_HEAD(&aec_hdl->in_head);
INIT_LIST_HEAD(&aec_hdl->inref_head);
cbuf_init(&aec_hdl->output_cbuf, aec_hdl->output_buf, sizeof(aec_hdl->output_buf));
#if AEC_EXTERN_REFERENCE_ENABLE
cbuf_init(&aec_hdl->extern_ref_cbuf, aec_hdl->extern_ref_buf, sizeof(aec_hdl->extern_ref_buf));
#endif // AEC_EXTERN_REFERENCE_ENABLE
if (CONST_AEC_EXPORT) {
aec_uart_init();
printf("aec_uart_init \n");
}
#if AEC_UL_EQ_EN
struct audio_eq_param ul_eq_param = {0};
ul_eq_param.sr = sample_rate;
ul_eq_param.channels = 1;
ul_eq_param.online_en = 1;
ul_eq_param.mode_en = 0;
ul_eq_param.remain_en = 0;
ul_eq_param.max_nsection = EQ_SECTION_MAX;
ul_eq_param.cb = aec_ul_eq_filter;
ul_eq_param.eq_name = aec_eq_mode;
aec_hdl->ul_eq = audio_dec_eq_open(&ul_eq_param);
/* audio_eq_open(&aec_hdl->ul_eq, &ul_eq_param); */
/* audio_eq_set_samplerate(&aec_hdl->ul_eq, sample_rate); */
/* audio_eq_set_output_handle(&aec_hdl->ul_eq, ul_eq_output, NULL); */
/* audio_eq_start(&aec_hdl->ul_eq); */
#endif/*AEC_UL_EQ_EN*/
os_sem_create(&aec_hdl->sem, 0);
task_create(audio_aec_task, NULL, "aec");
audio_dac_read_reset();
aec_hdl->start = 1;
mem_stats();
#if 0
aec_hdl->free_ram = malloc(1024 * 63);
mem_stats();
#endif
#ifdef MUX_RX_BULK_TEST_DEMO
mux_rx_bulk_test = zalloc_mux_rx_bulk(MUX_RX_BULK_MAX);
if (mux_rx_bulk_test) {
printf("mux_rx_bulk_test:0x%x\n", mux_rx_bulk_test);
free_mux_rx_bulk(mux_rx_bulk_test);
mux_rx_bulk_test = NULL;
}
#endif/*MUX_RX_BULK_TEST_DEMO*/
printf("audio_aec_init succ\n");
return 0;
}
/*
*********************************************************************
* Audio AEC Close
* Description: AEC模块
* Arguments : None.
* Return : None.
* Note(s) : None.
*********************************************************************
*/
void audio_aec_close(void)
{
printf("audio_aec_close succ\n");
if (aec_hdl) {
aec_hdl->start = 0;
while (aec_hdl->busy) {
os_time_dly(2);
}
task_kill("aec");
if (CONST_AEC_EXPORT) {
aec_uart_close();
}
#if AEC_UL_EQ_EN
if (aec_hdl->ul_eq) {
audio_dec_eq_close(aec_hdl->ul_eq);
}
#endif/*AEC_UL_EQ_EN*/
local_irq_disable();
if (aec_hdl->free_ram) {
free(aec_hdl->free_ram);
}
#if AEC_MALLOC_ENABLE
free(aec_hdl);
#endif /*AEC_MALLOC_ENABLE*/
aec_hdl = NULL;
local_irq_enable();
printf("audio_aec_close succ\n");
}
}
/*
*********************************************************************
* Audio AEC Input
* Description: AEC源数据输入
* Arguments : buf
* len
* Return : None.
* Note(s) : 256
*********************************************************************
*/
void audio_aec_inbuf(s16 *buf, u16 len)
{
if (aec_hdl && aec_hdl->start) {
int i = 0;
for (i = 0; i < MIC_BULK_MAX; i++) {
if (aec_hdl->in_bulk[i].used == 0) {
break;
}
}
if (i < MIC_BULK_MAX) {
aec_hdl->in_bulk[i].addr = buf;
aec_hdl->in_bulk[i].used = 0x55;
aec_hdl->in_bulk[i].len = len;
list_add_tail(&aec_hdl->in_bulk[i].entry, &aec_hdl->in_head);
} else {
printf(">>>aec_in_full\n");
/*align reset*/
struct mic_bulk *bulk;
list_for_each_entry(bulk, &aec_hdl->in_head, entry) {
bulk->used = 0;
__list_del_entry(&bulk->entry);
}
return;
}
os_sem_set(&aec_hdl->sem, 0);
os_sem_post(&aec_hdl->sem);
}
}
void audio_aec_inbuf_ref(s16 *buf, u16 len)
{
if (aec_hdl && aec_hdl->start) {
int i = 0;
for (i = 0; i < MIC_BULK_MAX; i++) {
if (aec_hdl->inref_bulk[i].used == 0) {
break;
}
}
if (i < MIC_BULK_MAX) {
aec_hdl->inref_bulk[i].addr = buf;
aec_hdl->inref_bulk[i].used = 0x55;
aec_hdl->inref_bulk[i].len = len;
list_add_tail(&aec_hdl->inref_bulk[i].entry, &aec_hdl->inref_head);
} else {
printf(">>>aec_inref_full\n");
/*align reset*/
struct mic_bulk *bulk;
list_for_each_entry(bulk, &aec_hdl->inref_head, entry) {
bulk->used = 0;
__list_del_entry(&bulk->entry);
}
return;
}
}
}
/*
*********************************************************************
* Audio AEC Reference
* Description: AEC模块参考数据输入
* Arguments : buf
* len
* Return : None.
* Note(s) : DAC
*********************************************************************
*/
void audio_aec_refbuf(s16 *buf, u16 len)
{
#if AEC_EXTERN_REFERENCE_ENABLE
u16 wlen = 0;
if (aec_hdl && aec_hdl->start) {
wlen = cbuf_write(&aec_hdl->extern_ref_cbuf, buf, len);
}
#endif // AEC_EXTERN_REFERENCE_ENABLE
}
void aec_cfg_fill(AEC_CONFIG *cfg)
{
}
void audio_aec_dlyest_en(void)
{
}
#endif /*TCFG_CVP_DEVELOP_ENABLE*/