这里先说下rk3288 HDMIin音频通路:

hdmiin:
HDMIIn声音直接通过codec输出到喇叭、耳机,不需要送到主控进行处理:
TC358749XBG-> alc5651 i2s2 -> alc5651 dac -> hp/lineout

android读取音频声道数 android 音频输出通道_hdmiin

alsa HAL 层

RK平台 android 5.1 BOX MID的SDK之后统一使用这个目录下面的代码
\hardware\rockchip\audio\tinyalsa_hal

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

1、音频编解码器 Codec 负责处理音频信息,包括 ADC,DAC,Mixer,DSP,输入输出以及音量控制等所有
与音频相关的功能。
Codec 与处理器之间通过 I2C 总线和数字音频接口 DAI 进行通信。
I2C 总线 - 实现对 Codec 寄存器数据的读写。

2、DAI –(digital audio interface)实现音频数据在 CPU 和 Codec 间的通信,包括 I2S、PCM 和 AC97
等。
蓝牙立体声音乐播放走的是蓝牙跟 CPU 直接的 UART 接口。
在只有一组 I2S 的主控上面,蓝牙通话 SCO 暂时是通过 Codec 中继,即通过 codec 来路由选择是
否走通话,走的是 I2S 接口 。如果主控有两组 I2S,那么可以将 BT PCM 直接接到主控的 PCM 接
口上。
S/DIF(Sony/Philips Digital Interface Format),通过光纤或同轴电缆传输音频,保证音频质量。

Codec 内部通路举例 RK616

同时音频驱动有三部分:
Machine driver
sound/soc/rockchip/rk_rk616.c
其中的 Machine 驱动负责 Platform 和 Codec 之间的耦合以及部分和设备或板子特定的代码采样率
时钟配置

Platform driver
sound/soc/ rockchip /rk30_i2s.c
I2S 控制器驱动 采样率时钟 DMA 等配置

Codec driver
sound/soc/codecs/rk616_codec.c
codec 的寄存器通路的配置

android audio route通路介绍

android定义了很多的音频devices:在frameworks/base/media/java/android/media/AudioSystem.java文件中

349     // output devices, be sure to update AudioManager.java also
350     public static final int DEVICE_OUT_EARPIECE = 0x1;
351     public static final int DEVICE_OUT_SPEAKER = 0x2;
352     public static final int DEVICE_OUT_WIRED_HEADSET = 0x4;
353     public static final int DEVICE_OUT_WIRED_HEADPHONE = 0x8;
354     public static final int DEVICE_OUT_BLUETOOTH_SCO = 0x10;
355     public static final int DEVICE_OUT_BLUETOOTH_SCO_HEADSET = 0x20;
356     public static final int DEVICE_OUT_BLUETOOTH_SCO_CARKIT = 0x40;
357     public static final int DEVICE_OUT_BLUETOOTH_A2DP = 0x80;
358     public static final int DEVICE_OUT_BLUETOOTH_A2DP_HEADPHONES = 0x100;
359     public static final int DEVICE_OUT_BLUETOOTH_A2DP_SPEAKER = 0x200;
360     public static final int DEVICE_OUT_AUX_DIGITAL = 0x400;
361     public static final int DEVICE_OUT_HDMI = DEVICE_OUT_AUX_DIGITAL;
362     public static final int DEVICE_OUT_ANLG_DOCK_HEADSET = 0x800;
363     public static final int DEVICE_OUT_DGTL_DOCK_HEADSET = 0x1000;
364     public static final int DEVICE_OUT_USB_ACCESSORY = 0x2000;
365     public static final int DEVICE_OUT_USB_DEVICE = 0x4000;
366     public static final int DEVICE_OUT_REMOTE_SUBMIX = 0x8000;
367     public static final int DEVICE_OUT_TELEPHONY_TX = 0x10000;
368     public static final int DEVICE_OUT_LINE = 0x20000;
369     public static final int DEVICE_OUT_HDMI_ARC = 0x40000;
370     public static final int DEVICE_OUT_SPDIF = 0x80000;
371     public static final int DEVICE_OUT_FM = 0x100000;
372     public static final int DEVICE_OUT_AUX_LINE = 0x200000;
373     public static final int DEVICE_OUT_SPEAKER_SAFE = 0x400000;
374     public static final int DEVICE_OUT_IP = 0x800000;
375     public static final int DEVICE_OUT_BUS = 0x1000000;

HAL层通过一定转换跟android上层对应

在hardware/rockchip/audio/tinyalsa_hal/alsa_route.c 中通过route_set_controls(route)中通过 get_route_config(int route)将以上的各个route值转换为codec_config中的xxx.h配置文件。

文件:./hardware/rockchip/audio/tinyalsa_hal/alsa_route.c
282 const struct config_route *get_route_config(unsigned route)  //即在这里获取音频路由设备
283 {
284     ALOGV("get_route_config() route %d", route);
285
286
287     if (!route_table) {
288         ALOGE("get_route_config() route_table is NULL!");
289         return NULL;
290     }
291     switch (route) {
292     case SPEAKER_NORMAL_ROUTE:
293         return &(route_table->speaker_normal);
		......

HAL层中定义的 route table,再通过route_set_controls将这些table配置,最终设置到codec的寄存器

文件:./hardware/rockchip/audio/tinyalsa_hal/codec_config/rt5651_config.h


	1110 const struct config_route_table rt5651_config_table = {
	1111     //speaker
	1112     .speaker_normal = {
	1113         .sound_card = 0,
	1114         .devices = DEVICES_0,
	1115         .controls = rt5651_speaker_normal_controls,
	1116         .controls_count = sizeof(rt5651_speaker_normal_controls) / sizeof(struct config_control),
	1117     },
		......
		
	  48 const struct config_control rt5651_speaker_normal_controls[] = {
	
	  56
	  57         {
	  58             .ctl_name = "DAC MIXL INF1 Switch",
	  59             .int_val = {on},
	  60         },
	  61         {
	  62             .ctl_name = "DAC MIXR INF1 Switch",
	  63             .int_val = {on},
	  64         },
	  65         {
	  66             .ctl_name = "Stereo DAC MIXL DAC L1 Switch",
	  67             .int_val = {on},
	  68         },
	  69         {
	  70             .ctl_name = "Stereo DAC MIXR DAC R1 Switch",
	  71             .int_val = {on},
	  72         },
	  73         {
	  74             .ctl_name = "OUT MIXL DAC L1 Switch",
	  75             .int_val = {on},
	  76         },
	  77         {
	  78             .ctl_name = "OUT MIXR DAC R1 Switch",
	  79             .int_val = {on},
	  80         },
	  81

HAL层会根据sound card name选择使用哪个table,如果没有匹配默认使用default_config.h

文件:./hardware/rockchip/audio/tinyalsa_hal/alsa_route.c

 89 /**
 90  * @brief route_init
 91  *
 92  * @returns
 93  */
 94 int route_init(void)
 95 {
 96     char soundCardID[20] = "";
 97     static FILE * fp;
 98     unsigned i, config_count = sizeof(sound_card_config_list) / sizeof(struct alsa_sound_card_config);
 99     size_t read_size;
100
101     ALOGV("route_init()");
102
103     fp = fopen("/proc/asound/card0/id", "rt"); //id是第0个声卡,串口执行cat /proc/asound/card0/id可以查看                                         
104     if (!fp) {
105         ALOGE("Open sound card0 id error!");
106     } else {
107         read_size = fread(soundCardID, sizeof(char), sizeof(soundCardID), fp);
108         fclose(fp);
109
110         if (soundCardID[read_size - 1] == '\n') {
111             read_size--;
112             soundCardID[read_size] = '\0';
113         }
114
115         ALOGV("Sound card0 is %s", soundCardID);
116
117         for (i = 0; i < config_count; i++) {
118             if (!(sound_card_config_list + i) || !sound_card_config_list[i].sound_card_name ||
119                 !sound_card_config_list[i].route_table)
120                 continue;
121
122             if (strncmp(sound_card_config_list[i].sound_card_name, soundCardID, //根据sound card name选择使用哪个table
123                 read_size) == 0) {
124                 route_table = sound_card_config_list[i].route_table;
125                 ALOGD("Get route table for sound card0 %s", soundCardID);
126             }
127         }
128     }
129
130     if (!route_table) {
131         route_table = &default_config_table; //如果没有匹配默认使用default_config.h
132         ALOGD("Can not get config table for sound card0 %s, so get default config table.", soundCardID);
133     }
134
135     for (i = PCM_DEVICE0_PLAYBACK; i < PCM_MAX; i++)
136          mPcm[i] = NULL;
137
138     return 0;
139 }

HAL层声卡名称sound card name和配置表列表table的声明在 ./tinyalsa_hal/codec_config/config_list.h 文件中。Audio将获取config_route_table并根据声卡0(识别到的声卡列表的第一个)和sound_card_name的名称设置路由route。
rk3288:/ # cat /proc/asound/card0/id
realtekrt5651co

文件:./hardware/rockchip/audio/tinyalsa_hal/codec_config/config_list.h

 71 struct alsa_sound_card_config sound_card_config_list[] = {
 72     {
 73         .sound_card_name = "RKRK616",
 74         .route_table = &rk616_config_table,
 75     },
 76     {
 77         .sound_card_name = "RK29RT3224",
 78         .route_table = &rt3224_config_table,
 79     },
 80     {
 81         .sound_card_name = "rockchiprt5640c",
 82         .route_table = &rt5640_config_table,
 83     },
 84     {
 85         .sound_card_name = "rockchipes8396c",
 86         .route_table = &es8396_config_table,
 87     },
 88     {
 89         .sound_card_name = "RK29RT3261",
 90         .route_table = &rt3261_config_table,
 91     },
 92     {
 93         .sound_card_name = "RK29WM8960",
 94         .route_table = &wm8960_config_table,
 95     },
 96     {
 97         .sound_card_name = "RKRT3224",
 98         .route_table = &rt3224_config_table,
 99     },
100     {
101         .sound_card_name = "RKRT3261",
102         .route_table = &rt3261_config_table,
103     },
104     {
105         .sound_card_name = "RKWM8960",
106         .route_table = &wm8960_config_table,
107     },
108     {
109         .sound_card_name = "RKRT5616",
110         .route_table = &rt5616_config_table,
111     },
112     {
113         .sound_card_name = "realtekrt5651co",
114         .route_table = &rt5651_config_table,
115     },
		......

.route_table的引用对应在xxxx_config.h文件中有定义,例如.route_table = &rt5651_config_table在rt5651_config.h中。

文件:/hardware/rockchip/audio/tinyalsa_hal/codec_config/rt5651_config.h
1143 const struct config_route_table rt5651_config_table = {
1144     //speaker
1145     .speaker_normal = {
1146         .sound_card = 0,
1147         .devices = DEVICES_0,
1148         .controls = rt5651_speaker_normal_controls,
1149         .controls_count = sizeof(rt5651_speaker_normal_controls) / sizeof(struct config_control),
1150     },
1151     .speaker_incall = {
1152         .sound_card = 0,
1153         .devices = DEVICES_0,
1154         .controls = rt5651_speaker_incall_controls,
1155         .controls_count = sizeof(rt5651_speaker_incall_controls) / sizeof(struct config_control),
1156     },
		......

然后再在.controls当中配置ctl_name和int_val,例如rt5651_speaker_normal_controls的配置,对应在kernel层会获取到这里的设置。

const struct config_control rt5651_speaker_normal_controls[] = {
   99     {
 100         .ctl_name = "RT5651 ASRC Switch",
 101         .int_val = {1},
 102     },
 103
 104     {
 105         .ctl_name = "DAC MIXL INF1 Switch",
 106         .int_val = {on},
 107     },
 108     {
 109         .ctl_name = "DAC MIXR INF1 Switch",
 110         .int_val = {on},
 111     },
 112     {
 113         .ctl_name = "Stereo DAC MIXL DAC L1 Switch",
 114         .int_val = {on},
 115     },
 116     {
 117         .ctl_name = "Stereo DAC MIXR DAC R1 Switch",
 118         .int_val = {on},
 119     },
 120     {
 121         .ctl_name = "OUT MIXL DAC L1 Switch",
 122         .int_val = {on},
 123     },
 124     {
 125         .ctl_name = "OUT MIXR DAC R1 Switch",
 126         .int_val = {on},
 127     },

 128     {
 129         .ctl_name = "HPOVOL L Switch",
 130         .int_val = {on},
 131     },
 132     {
 133         .ctl_name = "HPOVOL R Switch",
 134         .int_val = {on},
 135     },
 136     {
 137         .ctl_name = "HPO MIX HPVOL Switch",
 138         .int_val = {on},
 139     },
 140     {
 141         .ctl_name = "HPO MIX HPVOL Switch",
 142         .int_val = {on},
 143     },
 144     {
 145         .ctl_name = "HPO L Playback Switch",
 146         .int_val = {on},
 147     },
 148     {
 149         .ctl_name = "HPO R Playback Switch",
 150         .int_val = {on},
 151     },

 156     {    .ctl_name = "LOUT MIX DAC L1 Switch",    .int_val = {on},    },
 157     {    .ctl_name = "LOUT MIX DAC R1 Switch",    .int_val = {on},    },
 158
 159     //   SPEAKER_NORMAL_ROUTE  HEADSET_NORMAL_ROUTE
 160     {    .ctl_name =   "LOUT MIX OUTVOL L Switch",    .int_val = {on},    },
 161     {    .ctl_name =   "LOUT MIX OUTVOL R Switch",    .int_val = {on},    },
 162
 163     {    .ctl_name = "LOUT L Playback Switch",    .int_val = {on},    },
 164     {    .ctl_name = "LOUT R Playback Switch",    .int_val = {on},    },
 165
 166     //{    .ctl_name = "OUT Playback Volume",    .int_val = {26, 26},    }, // 31 31
 167
 168     {    .ctl_name = "DAC1 Playback Volume",    .int_val = {128, 128},    }, // {136, 136} //这里是音量设置

};

音量值调节:

hardware/rockchip/audio/tinyalsa_hal/codec_config/rt5651_config.h

const struct config_control rt5651_bluetooth_voip_controls[] = {

	 // init IN2p/n for line in input.
	 const struct config_control  rt5651_line_in_capture_controls[] = {

	  // try run it first, to fix conflict
	 {    .ctl_name = "RECMIXL INL1 Switch",    .int_val = {on},    }, //Replace mic: conf for L.in.L used
	 {    .ctl_name = "RECMIXR INR1 Switch",    .int_val = {on},    }, //Replace mic: conf for L.in.R used
	 {    .ctl_name = "RECMIXL BST2 Switch",    .int_val = {off},    }, // close conf for micL used
	 {    .ctl_name = "RECMIXR BST2 Switch",    .int_val = {off},    }, // close conf for micR used
	 {    .ctl_name = "RECMIXL BST1 Switch",    .int_val = {off},    },
     {    .ctl_name = "RECMIXR BST1 Switch",    .int_val = {off},    },
     {    .ctl_name = "RECMIXL BST3 Switch",    .int_val = {off},    },
     {    .ctl_name = "RECMIXR BST3 Switch",    .int_val = {off},    },
     {    .ctl_name = "Stereo1 ADC L1 Mux",    .str_val = "ADC",    },
     {    .ctl_name = "Stereo1 ADC R1 Mux",    .str_val = "ADC",    },
     {    .ctl_name = "Stereo1 ADC MIXL ADC1 Switch",    .int_val = {on},    },
     {    .ctl_name = "Stereo1 ADC MIXR ADC1 Switch",    .int_val = {on},    },
     {    .ctl_name = "ADC Capture Switch",    .int_val = {on, on},    },
     {    .ctl_name = "IN2 Boost",    .int_val = {2},    },
     {    .ctl_name = "ADC Capture Volume",    .int_val = {47, 47},    },
    // if you feel vol is low/high, can adjust it from 37 to 80. [63 is copied from rt5616]
    {    .ctl_name = "ADC Capture Volume",    .int_val = {63, 63},    }, // {57, 57} {47, 47}   //这里调节音量值设置

音频打开调用流程:

####  route_pcm_open() 
####	-> get_route_config(route)
####				-> route_set_controls()
####					-> route_pcm_close()
####						-> route_set_controls()
								 ->

如何debug

1、通过看 kernel log 确认 alsa sound card 有没有注册。

rk3288_mid:/ $ dmesg | grep ALSA -EC5                                          
[    1.551902] I : [File] : drivers/gpu/arm/mali400/mali/linux/mali_kernel_linux.c; [Line] : 417; [Func] : mali_module_init(); svn_rev_string_from_arm of this mali_ko is '-cf8da18', rk_ko_ver is '5', built at '11:19:29', on 'Jan 15 2020'.
[    1.553256] Mali: Mali device driver loaded
[    1.554211] of_get_named_gpiod_flags: parsed 'gpios' property of node '/rockchip-key/power-key[0]' - status (0)
[    1.554274] of_get_named_gpiod_flags: parsed 'gpios' property of node '/rockchip-key/power-key[0]' - status (0)
[    1.554741] input: rk29-keypad as /devices/platform/rockchip-key/input/input2
[    1.556689] ALSA device list:
[    1.556778]   #0: realtek,rt5651-codec
[    1.556977]   #1: rk,hdmiin-tc358749x-codec
[    1.557280]   #2: rockchip,hdmi
[    1.557312]   #3: rockchip-cdndp-sound
[    1.565498] mmc_host mmc2: Bus speed (slot 0) = 150000000Hz (slot req 150000000Hz, actual 150000000HZ div = 0)

2、确认 route 是否正常
声卡注册好之后确认 alsa route 是否正确
通过命令 logcat -s alsa_route

rk3288_mid:/ $ logcat -s alsa_route
--------- beginning of system
--------- beginning of main
01-01 01:19:51.274   255   324 D alsa_route: route_set_controls() set route 24
01-01 01:38:30.866   255   324 D alsa_route: route_info->sound_card 0, route_info->devices 0  
01-01 01:38:30.866   255   324 D alsa_route: route_set_controls() set route 0
01-01 01:38:34.105   255   324 D alsa_route: route_set_controls() set route 24
01-01 01:38:35.655   255   324 D alsa_route: route_info->sound_card 0, route_info->devices 0  
01-01 01:38:35.656   255   324 D alsa_route: route_set_controls() set route 0
01-01 01:39:23.035   255   324 D alsa_route: route_set_controls() set route 24

上面命令跑了两个路由 route 0 route 24。route的宏定义在alsa_audio.h中。

文件:hardware/rockchip/audio$ vi tinyalsa_hal/alsa_audio.h
 62 typedef enum _AudioRoute {
 63     SPEAKER_NORMAL_ROUTE = 0,
 64     SPEAKER_INCALL_ROUTE, // 1
 65     SPEAKER_RINGTONE_ROUTE,
 66     SPEAKER_VOIP_ROUTE,
 67
 68     EARPIECE_NORMAL_ROUTE, // 4
 69     EARPIECE_INCALL_ROUTE,
 70     EARPIECE_RINGTONE_ROUTE,
 71     EARPIECE_VOIP_ROUTE,
 72
 73     HEADPHONE_NORMAL_ROUTE, // 8
 74     HEADPHONE_INCALL_ROUTE,
 75     HEADPHONE_RINGTONE_ROUTE,
 76     SPEAKER_HEADPHONE_NORMAL_ROUTE,
 77     SPEAKER_HEADPHONE_RINGTONE_ROUTE,
 78     HEADPHONE_VOIP_ROUTE,
 79
 80     HEADSET_NORMAL_ROUTE, // 14
 81     HEADSET_INCALL_ROUTE,
 82     HEADSET_RINGTONE_ROUTE,
 83     HEADSET_VOIP_ROUTE,
 84
 85     BLUETOOTH_NORMAL_ROUTE, // 18
 86     BLUETOOTH_INCALL_ROUTE,
 87     BLUETOOTH_VOIP_ROUTE,
......
}AudioRoute;