news 2026/4/16 13:59:32

Android音频系列(09)-AudioPolicyManager代码解析

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Android音频系列(09)-AudioPolicyManager代码解析

目录

1. 简述

2 类结构说明

3. 初始化流程

4. 设备连接管理

5. 输出路由策略

1) 获取输出 (getOutputForAttr)

2) Engine 选设备逻辑 (getDevicesForStrategyInt)

6. 输入路由策略

1) getInputForAttr

2) getDeviceForInputSource

7. 动态策略与 Mix

8. 总结


深入剖析 Android AudioPolicyManager (后面简称APM) 及其策略引擎 (Engine) 的内部运作机制、核心数据结构与关键流程。

本文基于 Android 源码,重点关注设备管理、路由策略选择及其与 Engine 的交互。

1. 简述

AudioPolicyManager (APM) 是 Android 音频系统的策略中心:

  • 设备管理:维护系统所有可用输入输出设备的状态。
  • 策略路由:决定特定音频流(Stream)或属性(Attributes)应路由到哪个设备。
  • 资源控制:管理音频输入输出并发、音量计算。

Engine 是 APM 的策略决策模块(Strategy Decision Module):

  • 定义产品策略 (Product Strategy):将 Audio Attributes 映射到 Strategy。
  • 设备选择算法:根据当前 Use Case(如通话、媒体、铃声)和强制配置(Force Use),为每个 Strategy 计算最佳输出设备。

这种分离设计使得 APM 框架可以保持通用,而具体的路由策略(如插入蓝牙耳机等等)可以通过替换 Engine 实现或配置。

2 类结构说明

APM 管理着由于音频策略配置文件audio_policy_configuration.xml解析出的对象模型。

图1 APM与Engine简单图

如图1,APM与Engine是一种策略模式,更详细的关系如图2所示。

图2 APM与Engine

HwModule: 对应 HAL 模块(如 primary、 a2dp、usb)。包含其支持的 IOProfile。

IOProfile: 描述了 HAL 能提供的输入输出能力(采样率、格式、通道、支持的设备)。它是静态配置的。

SwAudioOutputDescriptor: 描述了一个实际打开的输出流(Stream)。它关联到一个具体的 IOProfile 和当前路由的 DeviceDescriptor。

DeviceDescriptor: 运行时设备对象,包含地址、类型、能力等。

3. 初始化流程

初始化阶段,APM 会加载 XML 配置文件,构建 HwModule 和 IOProfile 及其支持的 DeviceDescriptor 集合。

1)Factory 创建: AudioPolicyService 调用 createAudioPolicyManager。

2) Config 解析: APM 构造函数中解析 audio_policy_configuration.xml,填充 mHwModules。

3)Engine 初始化: Engine 加载自己的策略配置(如有),建立 Attributes 到 Strategy 的映射表。

4)打开 Output: APM 遍历 HwModule,预先打开那些 Attached(非动态)的 Output Profile(如 Speaker/Earpiece 对应的 Output)。

4. 设备连接管理

当设备插入(如有线耳机、蓝牙耳机)时,流程如图3所示:

图3

checkOutputsForDevice 逻辑如下:

当新设备连接时,APM 会检查所有活动输出,询问 Engine:“这个 Output 现在应该去哪个设备?”

如果 Engine 返回的新设备就是刚连接的设备(或者包含它),APM 就会执行路由切换(Tear down patch > Create new patch)。

5. 输出路由策略

这是 APM 最核心的部分:App 请求播放 -> 决定使用哪个 Output -> 路由到哪个 Device。

1) 获取输出 (getOutputForAttr)

当 App 创建 AudioTrack 时, AudioTrack调用AudioFlinger::createTrack, 然后会调用getOutputForAttr函数:

status_t AudioFlinger::createTrack(const media::CreateTrackRequest& _input, media::CreateTrackResponse& _output) { …… // Aptiv Audio lStatus = AudioSystem::getOutputForAttr(&localAttr, &output.outputId, sessionId, &streamType,adjAttributionSource, &input.config, input.flags, &output.selectedDeviceId, &portId, &secondaryOutputs); …… }

后续流程如图4所示:

图4 getOutputForAttr流程

2) Engine 选设备逻辑 (getDevicesForStrategyInt)

在 Engine.cpp 中,getDevicesForStrategyInt 是决策核心。以下是 STRATEGY_MEDIA 为例简单说明下:

  • 强制配置检查: getForceUse(FOR_MEDIA)。如果是 FORCE_SPEAKER,直接选 Speaker。
  • 外部设备优先: 检查 A2DP, Wired Headset, USB Device。getLastRemovableMediaDevices() 会返回最后连接的外部设备。
  • 默认设备: 如果没有外部设备,退回默认(Speaker 或 Earpiece)。
  • 特殊规则:

STRATEGY_PHONE (通话):优先蓝牙 SCO,然后 Earpiece。

STRATEGY_SONIFICATION (铃声):同时从 Speaker 和 Headset 出声(Dual Routing)。

6. 输入路由策略

输入路由即决定录音来源选择哪个麦克风或输入设备。

1) getInputForAttr

图5 getInputForAttr流程

2) getDeviceForInputSource

Engine::getDeviceForInputSource 处理输入源映射:

VOICE_COMMUNICATION: 如果在通话中,可能强制使用与输出设备匹配的输入(如蓝牙耳机麦克风)。

DEFAULT/MIC: 优先蓝牙 A2DP/SCO(如果支持),其次有线耳机麦克风,最后是内置 Mic。

7. 动态策略与 Mix

除了传统的基于 Stream Type 的策略,Android 还支持 Dynamic Policy (Audio Mix),常用于:

车机系统(Car Audio):将特定 uid 的导航声音路由到特定 BUS 设备。

投屏/录屏:捕获特定内容的音频。

相关类:AudioPolicyMix。Engine 在计算设备时,会优先检查是否有 AudioPolicyMix 匹配当前 Attributes,如果有,则强制路由到对应的 Mix 输出,绕过默认策略。

8. 总结

Android 音频策略框架是一个分层清晰的系统:

1) 静态配置层 (HwModule/IOProfile):定义硬件能力底座。

2) 动态状态层 (DeviceDescriptor/SwAudioOutputDescriptor):维护运行时状态。

3) 策略决策层 (Engine):getDevicesForStrategyInt 是“大脑”,根据优先级规则输出目标设备。

4) 执行层 (APM):根据 Engine 的决策,操作 AudioPolicyClient 完成物理连接。

理解 Engine 的 getDevicesForStrategyInt 和 APM 的 setDeviceConnectionState 是掌握 Android 音频路由切换的关键。

更多的内容,敬请关注,待分解。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/16 12:57:15

AutoGen Studio作品分享:Qwen3-4B驱动的AI研发助手完整工作流截图集

AutoGen Studio作品分享:Qwen3-4B驱动的AI研发助手完整工作流截图集 1. 什么是AutoGen Studio AutoGen Studio是一个面向开发者的低代码交互界面,它不追求炫酷的UI动效,而是把重心放在“让AI代理真正能干活”这件事上。你不需要写一整套Age…

作者头像 李华
网站建设 2026/4/16 12:57:19

小团队高效开发秘籍:利用镜像加速Qwen2.5-7B迭代

小团队高效开发秘籍:利用镜像加速Qwen2.5-7B迭代 在AI应用快速落地的今天,小团队常面临一个现实困境:想用大模型做业务创新,却被“部署难、调参繁、迭代慢”三座大山压得喘不过气。模型下载动辄几十GB、环境配置一错再错、微调一次…

作者头像 李华
网站建设 2026/4/16 12:57:19

基于 PLC 的全自动洗衣机控制设计实战

基于plc的全自动洗衣机控制设计 采用博途1200仿真,提供程序,设计报告,接线图。 实现要求(详见上方演示视频):PLC投入运行,系统处于初始状态,准备好启动。 启动时开始进水,水满(即水位到达高水位…

作者头像 李华
网站建设 2026/4/16 12:55:39

加载示例音频快速测试,Emotion2Vec+上手无压力

加载示例音频快速测试,Emotion2Vec上手无压力 1. 为什么语音情感识别值得你花5分钟试试? 你有没有过这样的经历:客服电话里对方语气冷淡,你却不确定是自己理解错了,还是对方真的不耐烦?又或者&#xff0c…

作者头像 李华
网站建设 2026/4/16 12:55:35

小显存福音!Z-Image Turbo显存优化使用指南

小显存福音!Z-Image Turbo显存优化使用指南 1. 引言:为什么小显存用户终于能畅快绘图了? 你是不是也经历过这些时刻? ——刚点下“生成”,显存占用就飙到98%,进度条卡在30%不动; ——想试试10…

作者头像 李华