news 2026/4/23 15:31:24

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 18:31:38

AI显微镜-Swin2SR效果展示:模糊马赛克图秒变4096px高清作品集

AI显微镜-Swin2SR效果展示:模糊马赛克图秒变4096px高清作品集 1. 什么是AI显微镜——Swin2SR 你有没有遇到过这样的情况:一张刚生成的AI绘画草稿只有512512,放大后全是马赛克;十年前拍的老照片发虚泛黄,连人脸轮廓都…

作者头像 李华
网站建设 2026/4/23 12:42:21

BAAI/bge-m3能否用于抄袭检测?学术文本比对实战案例

BAAI/bge-m3能否用于抄袭检测?学术文本比对实战案例 1. 为什么传统查重工具不够用了? 你有没有遇到过这样的情况:学生交来的论文,文字几乎没重复,但核心观点、论证逻辑、甚至段落结构都和某篇已发表文献高度一致&…

作者头像 李华
网站建设 2026/4/23 14:46:07

医生级AI助手来了!MedGemma 1.5实战:症状自查与病理分析指南

医生级AI助手来了!MedGemma 1.5实战:症状自查与病理分析指南 1. 为什么你需要一个“本地医生”?——从隐私焦虑到真实需求 你有没有过这样的经历:深夜刷到一篇关于某种症状的科普文章,心跳加速,立刻打开搜…

作者头像 李华
网站建设 2026/4/23 14:10:13

CSDN开发者专属:教你打造自己的AI助手模型

CSDN开发者专属:教你打造自己的AI助手模型 你有没有想过,让一个大模型真正“认得”你?不是泛泛而谈的“我是Qwen”,而是清清楚楚地说出:“我由CSDN迪菲赫尔曼开发和维护”。这不是科幻设定,而是今天就能在…

作者头像 李华
网站建设 2026/4/23 14:09:42

Phi-3-mini-4k-instruct参数详解:Ollama中temperature/top_p/num_ctx调优指南

Phi-3-mini-4k-instruct参数详解:Ollama中temperature/top_p/num_ctx调优指南 1. 为什么你需要关注这三个参数 你刚在Ollama里拉取了phi3:mini,输入“你好”就得到了一句礼貌又简洁的回复——看起来一切顺利。但当你试着让它写一封专业邮件、解一道数学…

作者头像 李华
网站建设 2026/4/23 14:08:55

Llama-3.2-3B惊艳效果:Ollama本地运行3B模型生成带格式表格数据

Llama-3.2-3B惊艳效果:Ollama本地运行3B模型生成带格式表格数据 1. 为什么3B模型也能“扛大活”?——轻量不等于妥协 很多人一听到“3B参数”,第一反应是:“这能干啥?怕不是连长句子都理不清。” 但Llama-3.2-3B彻底…

作者头像 李华