Linux ALSA音频框架及RK3399 DTS音频配置

ALSA音频框架

Alsa是Advanced Linux Sound Architecture的缩写,即高级Linux声音架构,在Linux操作系统上提供了对音频和MIDI的支持。在Linux 2.6的内核版本后,Alsa目前已经成为了linux的主流音频体系结构。

除了 alsa-driver,ALSA 包含在用户空间的 alsa-lib 函数库,具有更加友好的编程接口,并且完全兼容于 OSS,开发者可以通过这些高级 API 使用驱动,不必直接与内核驱动 API 进行交互。

一、系统框架

  • User空间:主要由Alsa Libray API对应用程序提供统一的API接口,各个APP应用程序只要调用 alsa-lib 提供的 API接口来实现放音、录音、控制。现在提供了两套基本的库,tinyalsa是一个简化的alsa-lib库,现在Android的系统中主要使用它。

  • ALSA CORE:alsa 核心层,向上提供逻辑设备(PCM/CTL/MIDI/TIMER/…)系统调用,向下驱动硬件设备(Machine/I2S/DMA/CODEC)

  • ASOC Core:是 ALSA 的标准框架,是 ALSA-driver 的核心部分,提供了各种音频设备驱动的通用方法和数据结构,为 Audio driver提供 ALSA Driver API

  • Hardware Driver:音频硬件设备驱动,由三大部分组成,分别是 Machine、Platform、Codec,提供的 ALSA Driver API 和相应音频设备的初始化及工作流程,实现具体的功能组件,这也是驱动开发人员需要具体实现的部分。

二、ASoC 硬件驱动结构

ALSA driver--Asoc

ASoC--ALSA System on Chip , 是建立在标准ALSA驱动层上, 为了更好地支持嵌入式处理器和移动设备中的音频Codec的一套软件体系.

嵌入式设备的音频系统可以被划分为板载硬件(Machine)、Soc(Platform)、Codec三大部分。

image

image

1. ASoC Platform Driver

指某款 SoC 平台的音频模块,如 exynos、omap、qcom 等等。

包括 dma 和 cpu_dai 两部分:

  • dma:负责把 dma buffer 中的音频数据搬运到 I2S tx FIFO。音频 dma 驱动通过snd_soc_register_platform() 来注册。
  • cpu dai:指 SoC 的 I2S、PCM 总线控制器,负责把音频数据从 I2S tx FIFO 搬运到 CODEC(这是回放的情形,录制则方向相反)。cpu_dai 通过 snd_soc_register_dai() 来注册。

2. ASoC Machine Driver

作为链结 Platform 和 Codec 的载体,通过配置 dai_link 把 cpu_dai、codec_dai、modem_dai 各个音频接口给链结成一条条音频链路,然后注册 snd_soc_card。

  • snd_soc_dai_link:音频链路描述及板级操作函数,它指定链路用到的 codec、codec_dai、cpu_dai、platform,这四者就构成了一条音频数据链路用于多媒体声音的回放和录制。
  • snd_soc_dai_driver:音频数据接口描述及操作函数,根据 codec 端和 soc 端,分为 codec_dai 和 cpu_dai。
    • linux 4.4 内核中支持两种方式创建dai_driver, 一种是通用的simple-audio-card 架构(简单通用的 machine driver), 一种是传统的编写自定义的 machine driver 来创建。
  • snd_soc_codec_driver:音频编解码芯片描述及操作函数,如控件/微件/音频路由的描述信息、时钟配置、IO 控制等
  • snd_soc_platform_driver:音频 dma 设备描述及操作函数

1)simple-audio-card

简单通用的 machine driver, 是一个为了简化音频框架,在alsa上面的一个封装。如果 simple-audio-card 框架足够满足需求, 建议优先使用 simple-audio-card 框架。

simple-audio-card的框架主要配置说明

1
2
3
4
5
6
7
8
9
status:声卡目前的状态,目前是未激活;
compatible:设备文件中的的名字,系统靠这个去匹配驱动代码中的simple-audio-card层的驱动程序;
simple-audio-card,name:声卡在系统中的名字;
simple-audio-card,cpu {
      sound-dai:soc端的dai 配置,就是rk3399的spdif或i2s接口的配置;
}
simple-audio-card,codec {
      sound-dai:codec端的dai配置,就是soc外界codec的接口的配置,这里是虚拟声卡;
}

3. ASoC Codec Driver

Codec 字面意思是编解码器,可以是用于回放或录制音频的。对于回放来说,userspace 送过来的音频数据是经过采样量化的数字信号,在 codec 经过 DAC 转换成模拟信号然后输出到外放或耳机,这样就可以听到声音了(对于录制则相反)。

Codec 芯片里面的功能部件很多,常见的有 AIF、DAC、ADC、Mixer、PGA、Line-in、Line-out,有些高端的 codec 芯片还有 EQ、DSP、SRC、DRC、AGC、Echo-Canceller、Noise-Suppression 等部件。

image

Codec驱动主要提供以下特性:

  • Codec DAI 和 PCM的配置信息;
  • Codec的IO控制方式(I2C,SPI等);
  • Mixer和其他的音频控件;
  • Codec的ALSA音频操作接口;
  • DAPM描述信息;
  • DAPM事件处理程序;
  • DAC数字静音控制

1)dummy_codec

ASoC Codec Driver之一,dummy_codec 是虚拟声卡 ,在soc外部没有外接codec的情况下,为了匹配声卡驱动框架,虚拟的一个设备,类似于占位符之类的东西的作用。

三、DTS实例

以rk3399为处理器,使用Simple-audio-card 作为machine driver,使用PCM5102芯片作为回放Codec,使用ICS-43432作为录制Codec,以此为基础配置设备树。

  • PCM5102使用PCM5102 driver,绑定到I2S0。
  • ICS-43432使用dummy_codec driver,绑定到I2S1。
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
/ {
    ...
    pcm5102a:  pcm5102a {
        compatible = "ti,pcm5102a";
        #sound-dai-cells = <0>;
        status = "okay";
    };

    pcm5102-sound {
        status = "okay";
        compatible = "simple-audio-card";
        simple-audio-card,format = "i2s";
        simple-audio-card,name = "ti,pcm5102-codec";
        simple-audio-card,mclk-fs = <256>;
        // simple-audio-card,widgets =
        //  "Microphone", "Mic Jack",
        //  "Headphone", "Headphone Jack";
        // simple-audio-card,routing =
        //  "Mic Jack", "MICBIAS1",
        //  "IN1P", "Mic Jack",
        //  "Headphone Jack", "HPOL",
        //  "Headphone Jack", "HPOR";

        simple-audio-card,cpu {
            sound-dai = <&i2s0>;
        };
        simple-audio-card,codec {
            sound-dai = <&pcm5102a>;
            #sound-dai-cells = <0>;
        };
    };
   
    dummy_codec: dummy-codec {
        compatible = "rockchip,dummy-codec";
        #sound-dai-cells = <0>;
        capture-volume = <0>;
    };
   
    i2s-dmic-array {
        status = "okay";
        compatible = "simple-audio-card";
        simple-audio-card,format = "i2s";
        simple-audio-card,name = "rockchip,i2s-dmic-array";
        simple-audio-card,mclk-fs = <256>;
        simple-audio-card,cpu {
            sound-dai = <&i2s1>;
        };
        simple-audio-card,codec {
            sound-dai = <&dummy_codec>;
            #sound-dai-cells = <0>;
        };
    };
   
};
   
&i2s0 {
    rockchip,i2s-broken-burst-len;
    rockchip,capture-channels = <8>;
    rockchip,playback-channels = <8>;
    #sound-dai-cells = <0>;
    status = "okay";
};

&i2s1 {
    rockchip,i2s-broken-burst-len;
    rockchip,playback-channels = <2>;
    rockchip,capture-channels = <2>;
    assigned-clocks = <&cru SCLK_I2S1_DIV>, <&cru SCLK_I2S_8CH>;
    assigned-clock-parents = <&cru PLL_GPLL>, <&cru SCLK_I2S1_8CH>;
    #sound-dai-cells = <0>;
    status = "okay";
};