QCC51xx学习笔记:理解CVC Audio Chain

Posted NiceBT

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了QCC51xx学习笔记:理解CVC Audio Chain相关的知识,希望对你有一定的参考价值。

为了方便大家学习,现与我爱蓝牙网联合推出【QCC300x/CSR867x/QCC30xx/QCC51xx开发板】

技术交流QQ群号:743434463
开发板会员QQ群号:725398389(凭订单号入群,赠独家学习资料)
——————————正文分割线———————————–

1. 引言

最近有项目需要定制修改QCC512x的cvc audio chain。自ADK6起,cvc音频链路开始通过kymera audio chain机制生成,可在QACT中查看生成的链路视图:

2. audio chain基本构成

在"chain.h"中,描述了chain的数据结构:

  • operators:描述了chain中包含的capability及其配置。每个capability以role作为指代,用于区分chain中相同的capability。
  • path & nodes:描述了音频流通过operators的路径配置。

下文给出包含3个path、3个operator的chain:

蓝色箭头路径起始于一个外部输入sink,经过operatorA->operatorB,到达operatorC后路径结束,对应的代码描述如下:


    blue_role,          // a value uniquely identifying this path within the context of the chain
    path_with_input,    // this path forms and input of the chain, but terminates internaly within the chain
    3,                  // the path has 3 nodes
    ->                 // address of an array containing the following 3 node structures:
        Operator A role, left input terminal, left output terminal,
        Operator B role, left input terminal, left output terminal,
        Operator C role, left input terminal, ignored value
    

红色箭头路径代表OperatorC本身可以作为一个source,作为其他路径的输入。

3. 构建cvc audio chain

3.1. 准备chain config

以1-mic headset cvc为例,其chain的配置代码如下:

static const chain_config_t audio_voice_hfp_config_1mic_nb_hs =
    MAKE_CHAIN_CONFIG_WITH_PATHS(chain_id_cvc_common, audio_ucid_hfp_cvc_headset, ops_nb_1mic_hs, paths);

3.1.1. 配置operator

#define CVC_OPS_NB \\
   MAKE_OPERATOR_CONFIG_PRIORITY_MEDIUM(capability_id_none, receive_role), \\
   MAKE_OPERATOR_CONFIG_PRIORITY_HIGH(capability_id_none, send_role), \\
   MAKE_OPERATOR_CONFIG_PRIORITY_MEDIUM(capability_id_cvc_receive_nb, cvc_receive_role), \\
   MAKE_OPERATOR_CONFIG(capability_id_none, rate_adjustment_send_role)

static operator_config_t ops_nb_1mic_hs[] =

   CVC_OPS_NB,
   MAKE_OPERATOR_CONFIG_PRIORITY_LOWEST(capability_id_cvc_hs_1mic_nb, cvc_send_role)
;
  • chain包含了5个operators
  • 其中receive_role、send_role、rate_adjustment_send_role没有指定capability,将在代码中动态配置
  • 其余cvc_receive_role、cvc_send_role指定了cvc capability

3.1.2. 配置path

static const operator_path_t paths[] =

   path_receive, path_with_in_and_out, ARRAY_DIM((receive)), receive,
   path_send_mic1, path_with_in_and_out, ARRAY_DIM((send_mic1)), send_mic1,
   path_send_mic2, path_with_input, ARRAY_DIM((send_mic2)), send_mic2,
   path_aec_ref, path_with_input, ARRAY_DIM((aec_ref)), aec_ref
;

static const operator_path_node_t receive[] =

   receive_role, 0, 0,
   cvc_receive_role, 0, 0,
;

static const operator_path_node_t send_mic1[] =

   cvc_send_role, 1, 0,
   rate_adjustment_send_role, 0, 0,
   send_role, 0, 0
;

static const operator_path_node_t send_mic2[] =

   cvc_send_role, 2, UNCONNECTED,
;

static const operator_path_node_t aec_ref[] =

   cvc_send_role, 0, UNCONNECTED
;

上述代码包含4条路径和4个节点,画出路径后,可以看到cvc的基本框架:

  • 蓝色路径:接收到手机的SCO通道数据,调用SCO decode解码->调用cvc receive nb处理->spk输出
  • 绿色路径:接收到mic1的ADC数据,调用cvc 1mic nb处理->调用采样率匹配处理->通过SCO通道输出给手机
  • 红色路径:接收到回声消除参考输入数据,作为cvc 1mic nb处理的另一个输入,不需要输出。

3.2. 准备filter config

对于receive_role、send_role、rate_adjustment_send_role来说,虽然两者在chain config中没有指定capability,但是代码中可利用filtering机制动态确定:

static operator_config_t hfp_nb[] =

    MAKE_OPERATOR_CONFIG_PRIORITY_MEDIUM(capability_id_sco_receive, audio_voice_receive_role), \\
    MAKE_OPERATOR_CONFIG_PRIORITY_HIGH(capability_id_sco_send, audio_voice_send_role)
;

static operator_config_t* getOperatorFilter(bool wideband)

    return (wideband ? hfp_wb : hfp_nb);


static operator_filters_t* getFilters(bool wideband)

    operator_filters_t* filters = (operator_filters_t*)calloc(1,sizeof(operator_filters_t));

    filters->num_operator_filters = 2;
    filters->operator_filters = getOperatorFilter(wideband);

    return filters;


operator_filters_t* filter = getFilters(AudioVoiceCommonIsWideband(plugin->encoder));

3.3. 创建并运行chain

在准备好chain config和filter config后,即可调用初始的api生成audio chain:

可以看到cvc的代码中也体现了这个流程:

// 创建cvc的基本chain,包含ChainCreateWIthFilter()、ChainConfigure()、ChainConnect()
createOperators(ctx, filters);

// 连接sco_receive端口的输出、source sync、volume、AEC这三个capability与cvc基本chain,包含ChainConnectInput() & ChainConnectOutput()
if(connectSource(ctx))

    uint16 sample_rate = GetLinkEncodingTypeSampleRate(ctx->encoder);

    AudioHardwareSetMicUse(audio_hw_voice_call);
    PanicFalse(AudioHardwareConnectInput(sample_rate));

    // 连接mic硬件接口的输出与基本chain的输入
    connectMicrophones(ctx);
    
    // 连接cvc基本chian的输出与sco_send端口的输入
    if(connectSink(ctx))
    
        // 连接AEC的输出与基本chain的输入
        connectAecReference(ctx);
        
        // 启动chain的运行,包含ChainStart()
        startChain(task, ctx);
        return;
    

将上述代码用graphviz工具描述,得出完整的cvc链路如下:

4. 尝试一点小改动

为了检验对上述知识的理解是否准确,我在volume和AEC之间插入一个splitter模块来做一个简单的测验。

首先增加一个rx_splitter_role:

typedef enum _audio_mixer_speaker_roles

    media_volume_role,
    speaker_peq_role,
    stereo_to_mono_role,
    crossover_role,
    master_volume_role,
    compander_role,
    post_processing_role,
    rx_splitter_role
 audio_mixer_speaker_roles_t;

然后新建一个operator对应splitter capability,并将splitter插入到既有的链路中:

static const operator_config_t speaker_ops_low_power[] =

    MAKE_OPERATOR_CONFIG(capability_id_volume, media_volume_role),    
    MAKE_OPERATOR_CONFIG(capability_id_splitter, rx_splitter_role)
;

static const operator_path_node_t left_low_power[] =

    media_volume_role,    VOLUME_INPUT_MAIN1, VOLUME_OUTPUT1,
    rx_splitter_role,    0, 0
;

进入通话模式后,可以看到splitter处在volume和AEC之间,与我们的预期相符:

5. 总结

通过对cvc audio chain的学习,我认为它相较于旧版867x的音频框架有几个明显的优点:

  • 去掉了dsp工程,整个音频链路由代码动态生成
  • 算法模块可封装成capability,方便第三方集成和调用
  • 加入了任务调度,算法模块运行在线程中,可根据需求设定优先级
  • dsp运行时可以动态调整链路上的算法模块,更换或移除个别算法模块不影响整个框架运行

在熟悉了cvc audio chain后,可尝试集成第三方语音处理算法或增加更多输入输出通道,使我们有机会为客户提供更高附加值的技术服务。

以上是关于QCC51xx学习笔记:理解CVC Audio Chain的主要内容,如果未能解决你的问题,请参考以下文章

QCC51xx学习笔记:理解CVC Audio Chain

QCC3003项目实战:BlueMotor6 AGHFP CVC 蓝牙对讲耳机

QCC3003项目实战:BlueMotor6 AGHFP CVC 蓝牙对讲耳机

QCC51xx学习笔记:KSE简介和跑通官方仿真例程

QCC51xx学习笔记:KSE简介和跑通官方仿真例程

QCC30xx学习笔记:earbuds 2.0例程简介