良许Linux教程网 干货合集 Linux内核音频驱动

Linux内核音频驱动

本文以I2S接口为例介绍Linux内核音频相关知识。

一、名词介绍

下面是音频调试中常见的名词缩略语。

  1. AEC(Acoustic Echo Cancellor):声学回声消除。

  2. AGC(Automatic Gain Control):自动增益控制,用于调整麦克风的收音量。

  3. ALSA(Advanced Linux Sound Architecture):高级Linux声音架构。

  4. ANS(Automatic Noise Suppression):自动噪声抑制,可检测并消除背景固定频率的杂音。

  5. BCK(Bit Clock Line):位时钟线,对应数字音频的每一位数据。标准中称为SCK(Serial Clock),串行时钟。SCK的频率为采样频率的两倍乘以采样位数。

  6. DAI(Digital Audio Interface):数字音频接口。

  7. DAPM(Dynamic Audio Power Management):动态音频电源管理,可使基于Linux的移动设备上的音频子系统在任何时候都处于最低功耗状态。

  8. DRC(Dynamic Range Control):动态范围控制,用于将音频输出控制在一定范围内。

  9. EQ(Equaliser):均衡器,通过调整一个或多个频段的增益或衰减来调整音色。

  10. I2S(Inter-IC Sound):IC间音频传输,是一种传输数字音频数据的接口标准,以串行方式传输两组(左右声道)数据。

  11. LRCK(Left-Right Clock):左右通道时钟,用于切换左右声道数据,0代表左声道,1代表右声道。LRCK的频率等于采样频率。

  12. MCLK(Master Clock):主时钟,通常为LRCK的256倍。虽然不是I2S标准的一部分,但主要用于同步模拟/数字转换器的内部操作。

  13. Mono:单声道。

  14. OSS(Open Sound System):开放声音系统。

  15. PCM(Pulse Code Modulation):脉冲编码调制,I2S是PCM的一种。

  16. Ramp:音量逐渐增加或减小,以避免声音的突然变化,常用于音乐的暂停或恢复。

  17. Stereo:立体声。

  18. TDM(Time Division Multiplexing):时分复用。虽然I2S最多只能传输2声道数据,但TDM最多支持16个通道的传输。

二、I2S接口

I2S是飞利浦定义的数字音频传输标准,用于数字音频数据在系统内部器件之间传输。

I2S是PCM的一个分支,接口定义相同。I2S的采样率一般为44.1/48KHZ,PCM采样频率一般为8/16KHZ等。

I2S接口有4组信号:SCK(位时钟)、LRCK(帧时钟)、SDI/SDO(数据)。

在I2S总线上,只能同时存在一个主设备和发送设备,主设备可以是发送设备或接收设备。常见的I2S框图如下:

image-20240413221835691
image-20240413221835691

三、I2S协议

I2S接口常见的协议模式包括:I2S正常模式、I2S左对齐模式和I2S右对齐模式等。

1、I2S正常模式

I2S正常模式属于I2S左对齐中的一种特例,也叫Philips模式。下图是I2S正常模式的波形。

image-20240413221839025
image-20240413221839025

LRCK(i2s_LRCK_rx/i2s_LRCK_tx)信号变低表示左声道,变高表示右声道。

SD(i2s_sdo,i2s_sdi)信号首先传输MSB或LSB,并在LRCK改变后的一个SCLK时钟周期发送第一个bit。

SD信号宽度的范围是从16到32位。

2、I2S左对齐模式

下图是I2S左对齐模式的波形。

image-20240413221841704
image-20240413221841704

LRCK(i2s_LRCK_rx/i2s_LRCK_tx)信号变高表示左声道,变低表示右声道。

SD(i2s_sdo,i2s_sdi)信号首先传输MSB或LSB,并在LRCK改变的同时发送第一个bit。

SD信号宽度的范围是从16到32位。

3、I2S右对齐模式

下图是I2S右对齐模式的波形。

image-20240413221844502
image-20240413221844502

LRCK(i2s_LRCK_rx/i2s_LRCK_tx)信号变高表示左通道,变低表示右通道。

SD(i2s_sdo、i2s_sdi)信号首先传输MSB或LSB,与I2S正常或左对齐模式不同,其数据与LRCK信号的边缘处的最后一位对齐。

SD信号宽度范围从16到32位。

其他协议格式还有:PCM early mode、PCM late1 mode和PCM late2 mode等。

四、ALSA框架

Linux内核2.6之后,ALSA取代了OSS成为了Linux内核音频子系统是一部分。

ALSA系统包括:

1、alsa-driver:alsa系统驱动。

2、alsa-lib:alsa库,用户空间调用,和内核空间交互。

3、alsa-utils:命令行工具。

4、alsa-plugin:alsa插件。

5、alsa-tools:alsa工具。

ALSA框架组成如下:

image-20240413221853926
image-20240413221853926

alsa-driver中相关功能如下(以rockchip 4a单板为例):

1、Codec:音频芯片共有的部分,包括codec初始化函数,控制接口,寄存器缓存,控件,dapm部件,音频路由,偏置电压设置函数等描述信息。

Rockchip 4A单板,Codec选用的是ES8316芯片,该芯片的dts配置如下:

image-20240413221857777
image-20240413221857777

2、Codec DAI:codec上的音频接口驱动描述,包括:时钟配置,格式配置,能力描述等。

Codec DAI相关实现如下:

image-20240413221902085
image-20240413221902085

3、CPU DAI:指 SoC 的 I2S、PCM 总线控制器,负责把音频数据从 I2S tx FIFO 搬运到 codec。

RK3399 CPU DAI的dts相关配置如下:

image-20240413221906189
image-20240413221906189

4、DAI Link:音频数据链路,它指定链路用到的codec、codec_dai、cpu_dai和platform。

Linux 4.4 内核支持两种方式创建声卡:一种是通用的simple-card framework;一种是传统的编写自定义的 machine driver 来创建。

Simple card是简单通用的machine driver,如果 simple-card框架足够满足需求,可优先选择simple card 框架。

DAI Link的dts相关配置如下:

image-20240413221910906
image-20240413221910906

DAI Link的相关实现如下:

image-20240413221914967
image-20240413221914967

5、DAPM:动态电源管理,是基于kcontrol改进过后的相应框架,增加了相应的电源管理机制, widget是DAPM的基本单位。

Codec(es8316)中kcontrol、dapm widget和dapm routes的相关实现如下:

image-20240413221918261
image-20240413221918261

6、DMA:负责把 dma buffer 中的音频数据搬运到 I2S tx FIFO。

五、调试命令

ALSA是Linux处理音频的基本接口,但ALSA只提供基层的接口,操作较为繁复,一般情况下可以直接使用其附带提供的 utils 工具集,utils 工具集是一些封装好了的功能模块,直接以命令的方式提供,用户只需要敲入相关命令和参数即可实现音频操作功能。

1、查看声卡信息

root@xiaotianbsp:~# cat /proc/asound/cards
 0 [rockchipes8316c]: rockchip_es8316 - rockchip,es8316-codec
                      rockchip,es8316-codec
 1 [HDMICODEC      ]: HDMI-CODEC - HDMI-CODEC
                      HDMI-CODEC
root@xiaotianbsp:~# ls -l /dev/snd/
total 0
drwxr-xr-x  2 root root       80 Aug 16 14:43 by-path
crw-rw----+ 1 root audio 116,  2 Aug 16 14:43 controlC0
crw-rw----+ 1 root audio 116,  5 Aug 16 14:43 controlC1
crw-rw----+ 1 root audio 116,  4 Aug 16 14:43 pcmC0D0c
crw-rw----+ 1 root audio 116,  3 Aug 16 14:43 pcmC0D0p
crw-rw----+ 1 root audio 116,  6 Aug 16 14:43 pcmC1D0p
crw-rw----+ 1 root audio 116,  1 Aug 16 14:43 seq
crw-rw----+ 1 root audio 116, 33 Aug 16 14:43 timer

controlCx:控制接口,提供灵活的方式管理注册的声卡和对存在的声卡进行查询。

pcmCxDxc:PCM接口,对应录音设备。

pcmCxDxp:PCM接口,对应放音设备。

timer:支持声音的同步事件提供声卡上的定时器。

seq:音序器接口,一个比原始MIDI接口高级的MIDI编程和声音同步高层接口。

2、查看声卡采集、播放PCM信息

root@xiaotianbsp:~# cat /proc/asound/pcm
00-00: ff880000.i2s-ES8316 HiFi ES8316 HiFi-0 :  : playback 1 : capture 1
01-00: ff8a0000.i2s-i2s-hifi i2s-hifi-0 :  : playback 1

3、查看ALSA驱动版本

root@xiaotianbsp:~# cat /proc/asound/version
Advanced Linux Sound Architecture Driver Version k4.4.154-90-rockchip-ga14f6502e045.

4、查看声卡0的信息

root@xiaotianbsp:~# cat /proc/asound/card0/pcm0p/sub0/status
closed

5、查看寄存器

## regmap名称
root@xiaotianbsp:~# cat /sys/kernel/debug/regmap/1-0011/name
es8316
root@xiaotianbsp:~# cat /sys/kernel/debug/regmap/ff880000.i2s/name
rockchip-i2s

## rk3399 i2s0控制器寄存器
root@xiaotianbsp:~# cat /sys/kernel/debug/regmap/ff880000.i2s/registers
00: 0000000f
04: 0000000f
08: 00033f3f
0c: 00000000
10: 000f0010
14: 01f00000
18: XXXXXXXX
1c: 00000000
20: XXXXXXXX
28: 00000000

## es8316寄存器
root@xiaotianbsp:~# cat /sys/kernel/debug/regmap/1-0011/registers
00: c0
01: f3
02: 08
03: 20
04: 11
05: 00
06: 11
07: 00
08: 00
09: 04
0a: 0c
0b: 0c
0c: ff
0d: 3f
0e: ff
0f: ff
10: 11
11: fc
12: 28
13: 00
14: 00
15: 33
16: 00
17: 00
18: 00

6、查看时钟

## 采样率44.1KHZ,mclk = 256 * sample = 256*44.1*1000 = 11289600 HZ
root@xiaotianbsp:~# cat /sys/kernel/debug/clk/clk_summary | grep i2s0
          clk_i2s0_div                    1            1   800000000          0 0
             clk_i2s0_frac                1            1    11289600          0 0
                clk_i2s0_mux              1            1    11289600          0 0
                   clk_i2s0               1            2    11289600          0 0
                hclk_i2s0                 1            2   100000000          0 0

7、amixer命令

root@xiaotianbsp:~# amixer -h
Usage: amixer  [command]

Available options:
  -h,--help       this help
  -c,--card N     select the card
  -D,--device N   select the device, default 'default'
  -d,--debug      debug mode
  -n,--nocheck    do not perform range checking
  -v,--version    print version of this program
  -q,--quiet      be quiet
  -i,--inactive   show also inactive controls
  -a,--abstract L select abstraction level (none or basic)
  -s,--stdin      Read and execute commands from stdin sequentially
  -R,--raw-volume Use the raw value (default)
  -M,--mapped-volume Use the mapped volume

Available commands:
  scontrols       show all mixer simple controls
  scontents       show contents of all mixer simple controls (default command)
  sset sID P      set contents for one mixer simple control
  sget sID        get contents for one mixer simple control
  controls        show all controls for given card
  contents        show contents of all controls for given card
  cset cID P      set control contents for one control
  cget cID        get control contents for one control

7.1查看card0内容

root@xiaotianbsp:~# amixer -c 0 contents
numid=16,iface=MIXER,name='ADC Double FS Mode'
  ; type=BOOLEAN,access=rw------,values=1
  : values=off
numid=14,iface=MIXER,name='ADC Soft Ramp'
  ; type=BOOLEAN,access=rw------,values=1
  : values=on
numid=13,iface=MIXER,name='ADC Capture Volume'
  ; type=INTEGER,access=rw---R--,values=1,min=0,max=192,step=0
  : values=192
  | dBscale-min=-96.00dB,step=0.50dB,mute=1
numid=15,iface=MIXER,name='Capture Polarity'
  ; type=ENUMERATED,access=rw------,values=1,items=2
  ; Item #0 'Normal'
  ; Item #1 'Invert'
  : values=0

7.2查看和配置某个numid命令

## 查看
root@xiaotianbsp:~# amixer -c 0 cget numid=16
numid=16,iface=MIXER,name='ADC Double FS Mode'
  ; type=BOOLEAN,access=rw------,values=1
  : values=off
  
## 配置
root@xiaotianbsp:~# amixer -c 0 cset numid=16 on
numid=16,iface=MIXER,name='ADC Double FS Mode'
  ; type=BOOLEAN,access=rw------,values=1
  : values=on

8、设备名

root@xiaotianbsp:~# aplay -l
**** List of PLAYBACK Hardware Devices ****
card 0: rockchipes8316c [rockchip,es8316-codec], device 0: ff880000.i2s-ES8316 HiFi ES8316 HiFi-0 []
  Subdevices: 1/1
  Subdevice #0: subdevice #0
card 1: HDMICODEC [HDMI-CODEC], device 0: ff8a0000.i2s-i2s-hifi i2s-hifi-0 []
  Subdevices: 1/1
  Subdevice #0: subdevice #0
root@linaro-alip:~# arecord -l
**** List of CAPTURE Hardware Devices ****
card 0: rockchipes8316c [rockchip,es8316-codec], device 0: ff880000.i2s-ES8316 HiFi ES8316 HiFi-0 []
  Subdevices: 1/1
  Subdevice #0: subdevice #0

9、放音

aplay -D hw:0,0 -r 44100 -c 2 -f s16_le play.wav

10、录音

arecord -D hw:0,0 -r 16000 -c 1 -f s16_le record.wav

-D:播放设备,aplay -l 或arecord -l列出的设备名

-r:采样率

-c:音频文件通道

-f:采样格式,常用的有S16_LE ,S24_LE ,S32_LE, cd (可忽略)

也可参考tinyalsa命令,如:tinymix、tinypcminfo、tinyplay和tinycap的用法。

11、mp3转wav

lame test.mp3 test.wav --decode

以上就是良许教程网为各位朋友分享的Linu系统相关内容。想要了解更多Linux相关知识记得关注公众号“良许Linux”,或扫描下方二维码进行关注,更多干货等着你 !

137e00002230ad9f26e78-265x300
本文由 良许Linux教程网 发布,可自由转载、引用,但需署名作者且注明文章出处。如转载至微信公众号,请在文末添加作者公众号二维码。
良许

作者: 良许

良许,世界500强企业Linux开发工程师,公众号【良许Linux】的作者,全网拥有超30W粉丝。个人标签:创业者,CSDN学院讲师,副业达人,流量玩家,摄影爱好者。
上一篇
下一篇

发表评论

联系我们

联系我们

公众号:良许Linux

在线咨询: QQ交谈

邮箱: yychuyu@163.com

关注微信
微信扫一扫关注我们

微信扫一扫关注我们

关注微博
返回顶部