目录
一、AudioPolicyService初始化
1.1 AudioPolicyService::onFirstRef()函数
1.2 AudioCommandThread线程
1.3 AudioPolicyManager初始化
1.3.1 解析audio_policy_configuration.xml配置文件
1.3.2 解析audio_policy_engine_configuration.xml配置文件
1.3.3 AudioPolicyManager::onNewAudioModulesAvailableInt()函数
1.3.4 AudioPolicyManager::updateDevicesAndOutputs()函数
二、AudioFlinger初始化
2.1 AudioFlinger::loadHwModule()函数
2.2 AudioFlinger::openOutput()函数
三、AudioService初始化
本文我会分别介绍AudioFlinger、AudioPolicyService和AudioService这三个模块的初始化过程。AudioHAL进程的核心源码AOSP里面没有,所以这里不做介绍。想了解AudioHAL进程的启动流程,可以从/hardware/interfaces/audio/common/all-versions/default/service/service.cpp文件的main()函数入手,进行分析。
大家知道分析代码的执行流程,一般画时序图比较合适,但是我在分析的过程中发现,当跟进一个复杂模块的函数调用流程时,画出来的时序图会非常大,完全看不清楚,所以,这里我会使用伪代码的方式来说明调用流程。
AudioPolicyService和AudioFlinger都是运行在AudioServer进程中的模块,它们的入口是在/frameworks/av/media/audioserver/main_audioserver.cpp文件中的main()函数中,由init进程通过解析audioserver.rc文件启动。main()函数干的事也比较简单,就是创建AudioPolicyService和AudioFlinger对象,并将它们两个作为Binder服务,添加进ServiceManager中。
由于AudioPolicyService和AudioFlinger都继承了RefBase类,所以它们的初始化函数有两个:构造函数和onFirstRef()函数。不了解Android RefBase类和强弱指针机制的朋友,可以在网上搜一下,写这部分的文章很多。
AudioPolicyService的构造函数撒也没干,所以它的初始化入口就只剩下了一个onFirstRef()函数。下面我们看看onFirstRef()函数干了什么事。
总体来说onFirstRef()函数主要干了三件事:
调用流程如下:
AudioPolicyService.cpp->onFirstRef()
|-->new AudioCommandThread("ApmAudio")
|-->new AudioCommandThread("ApmOutput")
|-->AudioPolicyService.cpp->loadAudioPolicyManager()//加载"libaudiopolicymanagercustom.so",默认没有这个库
|-->AudioPolicyService.cpp->createAudioPolicyManager()//创建并初始化AudioPolicyManager对象
|-->AudioPolicyManager.cpp->canBeSpatializedInt()//判断当前系统中是否支持空间音频。只有存在AUDIO_OUTPUT_FLAG_SPATIALIZER的流才代表支持。如果存在,就创建Spatializer对象。同时,空间音频功能目前只支持AUDIO_USAGE_MEDIA和AUDIO_USAGE_GAME这两个场景。
|-->AudioSystem.cpp->audioPolicyReady()
|-->AudioFlinger.h->audioPolicyReady()//通知AudioFlinger,自己已经初始化完成
AudioCommandThread是AudioPolicyService定义的内部类,继承了Android标准的循环线程类Thread.h。所以,当它的run()函数被调用后,线程就会循环一直调用threadLoop()函数。我们来看一下AudioCommandThread类的初始化函数AudioCommandThread::onFirstRef()函数。在这个函数中它调用了父类Thread.h的run()函数,所以当AudioCommandThread对象被创建时,就会启动此线程。
在线程运行函数AudioCommandThread::threadLoop()中,它会一直遍历自己的消息队列Vector < sp
/**
*向AudioCommandThread的消息队列中添加新消息
**/
status_t AudioPolicyService::AudioCommandThread::sendCommand(sp& command, int delayMs){
insertCommand_l(command, delayMs);//插入新的待处理消息
mWaitWorkCV.signal();//唤醒当前线程
}
/**
*AudioCommandThread线程循环运行函数
*/
bool AudioPolicyService::AudioCommandThread::threadLoop(){
if (waitTime == -1) {
mWaitWorkCV.wait(mLock);//让当前线程睡眠
} else {
mWaitWorkCV.waitRelative(mLock, waitTime);//让当前线程睡眠指定的时间
}
}
在AudioPolicyService.cpp::createAudioPolicyManager()函数中,会干两件事:new AudioPolicyManager对象,然后调用它的initialize()函数。
我们先来看AudioPolicyManager的构造函数,它会调用loadConfig()函数,该函数会通过Serializer.cpp工具来解析audio_policy_configuration.xml配置文件,然后将解析到的配置信息保存到AudioPolicyManager类的AudioPolicyConfig对象中。代码调用流程如下:
AudioPolicyService.cpp->createAudioPolicyManager()
|-->AudioPolicyManager.cpp->AudioPolicyManager()
|-->AudioPolicyManager.cpp->loadConfig()
|-->AudioPolicyManager.cpp->deserializeAudioPolicyXmlConfig()
|-->Serializer.cpp->deserializeAudioPolicyFile()
audio_policy_configuration.xml配置文件中定义的各种节点,与AudioPolicyManager中的数据结构类型之间的映射关系如下:
从上面的数据结构映射关系,结合上一篇Android13音频子系统分析(一)---整体架构中对AudioPolicyService模块的分析,可以看出, audio_policy_configuration.xml文件的作用就是配置当前平台支持哪些硬件设备,默认已加载了哪些硬件设备,并定义平台支持哪些OutStream和InputStream,以及这些IOStream能够与哪些硬件设备进行连接。
AudioPolicyManager::initialize()函数是AudioPolicyManager初始化的核心,它会创建并初始化Engine对象,Engine的父类EngineBase在其构造函数中会解析audio_policy_engine_configuration.xml配置文件,然后根据从配置文件中获取到的信息,通知AudioFlinger加载HwModule、创建PlaybackThread。以下是代码调用流程:
AudioPolicyManager.cpp->initialize()
|-->Engine.cpp->Engine()
|-->EngineBase.cpp->loadAudioPolicyEngineConfig()//解析audio_policy_engine_configuration.xml配置文件,将解析的音量配置保存到mVolumeGroups对象中,解析的上层场景配置保存到mProductStrategies对象中。
|-->Engine.cpp->updateDeviceSelectionCache()//更新每个strategy对应的devices,保存到mDevicesForStrategies Map集合中。由于此时AudioPolicyManager的mAvailableOutputDevices集合在onNewAudioModulesAvailableInt()函数中才会初始化,所以此时调用此函数是无法初始化每个strategy对应的devices。
|-->AudioPolicyManager.cpp->onNewAudioModulesAvailableInt()//通知AudioFlinger打开HwModule,并创建PlaybackThread。然后向mHwModules列表中添加通过AudioFlinger成功打开的HwModule对象。
|-->AudioPolicyManager.cpp->updateDevicesAndOutputs()//更新每个strategy对应的devices
|-->Engine.cpp->updateDeviceSelectionCache()
在EngineBase.cpp->loadAudioPolicyEngineConfig()函数中,会解析audio_policy_engine_configuration.xml配置文件。包含三个配置文件:audio_policy_engine_product_strategies.xml、audio_policy_engine_stream_volumes.xml、audio_policy_engine_default_stream_volumes.xml。这些文件的作用是配置不同播放场景下应该选择的设备路由策略,每种场景对应的音量等级,以及该场景下对应的输出设备的音量曲线。
当解析完这些配置文件后,会把其中的数据保存到EngineBase类的两个成员变量中:mProductStrategies集合和mVolumeGroups集合。
下面我介绍一下这些配置文件中涉及到的AudioAttributes、StreamType、ProductStrategy、VolumeGroup、deviceCategory、VolumeCurve这些数据结构的作用,以及它们之间的关联关系。
下图表示的是AudioAttributes、StreamType、ProductStrategy、VolumeGroup、deviceCategory、VolumeCurve之间的关联关系:
从图中可以看出:
即:同一个硬件播放设备,在不同的StreamType场景下,会使用不同的音量曲线。
当解析完配置文件后,AudioPolicyManager.cpp的initialize()函数就会调用onNewAudioModulesAvailableInt()函数。此函数会根据解析到的配置信息,通知AudioFlinger加载AudioModule,打开PlaybackThread。下面以播放场景(OutputProfile)举例列出其代码调用流程,录制场景(InputProfile)的逻辑类似。
AudioPolicyManager.cpp->onNewAudioModulesAvailableInt()
|-->AudioPolicyClientImpl.cpp->AudioPolicyService::AudioPolicyClient::loadHwModule()
|-->AudioFlinger.cpp->loadHwModule()//通知AudioFlinger加载AudioModule
|-->AudioOutputDescriptor.cpp->SwAudioOutputDescriptor::open()
|-->AudioPolicyClientImpl.cpp->AudioPolicyService::AudioPolicyClient::openOutput()
|-->AudioFlinger.cpp->openOutput()//通知AudioFlinger,创建PlaybackThread,并打开StreamOut。
|-->AudioPolicyManager.cpp->setEngineDeviceConnectionState()//遍历当前IOProfile对应的所有可用的devices,然后逐个进行设置
|-->EngineBase.cpp->setDeviceConnectionState()
|-->LastRemovableMediaDevices.cpp->setRemovableMediaDevices()//向mMediaDevices列表的头部添加此device,这个列表中保存的是可使用的有线外设和无线蓝牙外设,内置的设备如speaker不会被保存。
|-->AudioPolicyManager.cpp->addOutput()//在成员变量mOutputs集合中保存SwAudioOutputDescriptor对象
|-->AudioPolicyManager.cpp->applyStreamVolumes()//设置当前outputStream在不同StreamType场景下应该使用的音量大小
|-->AudioPolicyManager.cpp->setOutputDevices()//设置当前OutputStream支持的播放设备类型。将其保存到SwAudioOutputDescriptor的mDevices集合中。
|-->AudioPolicyManager.cpp->installPatch()//通过AudioFlinger创建一个AudioPatch,source是当前的OutputStream,sink目前支持的播放设备。
|-->AudioPolicyClientImpl.cpp->AudioPolicyService::AudioPolicyClient::createAudioPatch()
|-->PatchPanel.cpp->AudioFlinger::createAudioPatch()
针对onNewAudioModulesAvailableInt()函数的实现逻辑,我觉得有以下几点需要说明一下:
updateDevicesAndOutputs()函数的代码调用流程如下:
AudioPolicyManager.cpp->updateDevicesAndOutputs()
|-->Engine.cpp->updateDeviceSelectionCache()//遍历所有的ProductStrategy,找到它们目前应该选择的Device。然后保存到Engine::mDevicesForStrategies集合中。
|-->Engine.cpp->getDevicesForProductStrategy()
|-->Engine.cpp->remapStrategyFromContext()//调整设备选择策略。比如当处于通话或VoIP状态时,STRATEGY_MEDIA和STRATEGY_SONIFICATION都改为使用STRATEGY_PHONE对应的播放设备。也就是说通话时音乐播放的设备会改为与通话设备相同,比如听筒。
|-->Engine.cpp->filterOutputDevicesForStrategy()//过滤掉一些可用的Device。比如通话场景下去掉A2DP设备。
|-->Engine.cpp->getPreferredAvailableDevicesForProductStrategy()//查找上层AudioService设置的PreferredAvailableDevices。
|-->EngineBase.cpp->getDevicesForRoleAndStrategy()//从EngineBase::ProductStrategyDevicesRoleMap集合中找到该strategy对应的所有devices,这个集合里的数据由AudioService设置。
|-->Engine.cpp->getDevicesForStrategyInt()//只有当AudioService没有设置此ProductStrategy对应的PreferredDevices时,才会调用它。
其中比较关键的是Engine::getDevicesForProductStrategy()函数。它定义了根据不同的场景如何选择合适播放设备的策略。整体的策略是:
可以看出,上层AudioService设置下来的PreferredDevices,优先级会高于Engine模块自己的策略。Engine::getDevicesForStrategyInt()函数中实现的主要策略如下:
以上就是AudioPolicyService的整个初始化流程,可以看出,AudioPolicyService模块的核心逻辑全部是在AudioPolicyManager子模块中实现的。总结一下:AudioPolicyService在初始化时,干的事就是通过解析配置文件,制定设备路由策略、加载音量曲线,然后根据配置文件中的信息通知AudioFlinger加载HwModule、创建Thread来打开IOStream。
AudioFlinger在其构造函数和onFirstRef()函数中,除了创建DevicesFactoryHalInterface和EffectsFactoryHalInterface对象,用于获取AudioHAL层的接口。就没有干其他重要的事情了。所以,AudioFlinger的初始化是比较简单的。但是,上面我们在分析AudioPolicyService初始化时,可以看到AudioPolicyService有通知AudioFlinger创建HwModule和PlaybackThread,它们对应的两个函数入口分别是AudioFlinger::loadHwModule()函数、AudioFlinger::openOutput()函数。
loadHwModule()函数会根据AudioPolicyManager传递过来的name字符串,通知AudioHAL加载对应名称的so库,在HAL层,每个so库代表的就是不同的HwModule,也就是libaudiohal中描述的DeviceHalHidl对象。然后会将DeviceHalHidl对象保存到AudioHwDevice中,最后将AudioHwDevice对象保存AudioFlinger::mAudioHwDevs集合中,供后续的业务逻辑代码查找使用。代码调用流程如下:
AudioFlinger.cpp->loadHwModule()//AudioPolicyManager初始化时,会在onNewAudioModulesAvailableInt()函数中调用此接口
|-->AudioFlinger.cpp->loadHwModule_l()
|-->DevicesFactoryHalHidl.cpp->openDevice()//通知AudioHAL层打开虚拟device。AudioHAL会根据传入的name字符串,加载对应的so库。
|-->DevicesFactory.cpp->openDevice()//通过Binder跨进程调用,接下来的代码都运行在AudioHAL进程中
|-->DevicesFactory.cpp->loadAudioInterface()//AUDIO_HARDWARE_MODULE_ID在/hardware/libhardware/include/hardware/audio.h文件中定义,value为"audio"。
|-->hardware.c->hw_get_module_by_class()//拼接对应so库的名称。so库的代码实现由SoC厂商提供。
|-->hardware.c->load()//通过dlopen()加载这个so库。然后在so库中找到名称为HAL_MODULE_INFO_SYM的hw_module结构体。
|-->audio.h->audio_hw_device_open()//调用加载的so库中,定义的audio_module.common.methods.open()函数。
其中,DevicesFactory.cpp源文件位于hardware/interfaces/audio/core/all-versions/default/目录中。
hardware.c源文件位于hardware/libhardware/目录中。
audio.h源文件位于hardware/libhardware/include/hardware/目录中。
AUDIO_HARDWARE_MODULE_ID在/hardware/libhardware/include/hardware/audio.h文件中定义,value为"audio"。
openOutput()函数会通过AudioHwDevice对象通知AudioHAL打开OutStream,对应的是libaudiohal中的StreamOutHalHidl,这个对象会被保存到AudioStreamOut中供AudioFlinger使用。最后,会根据不同的flag类型创建不同的PlaybackThread子类,并把它们统一保存到AudioFlinger::mPlaybackThreads集合中。代码调用流程如下:
AudioFlinger.cpp->openOutput()//AudioPolicyManager初始化时,会在onNewAudioModulesAvailableInt()函数中调用此接口
|-->AudioFlinger.cpp->openOutput_l()//根据不同的flag创建不同的PlaybackThread对象。
|-->AudioHwDevice.cpp->openOutputStream()//通知AudioHAL打开StreamOut,然后创建一个对应的AudioStreamOut对象。
在PlaybackThread的onFirstRef()函数中,会调用父类Thread.h的run()函数。所以,当PlaybackThread的对象被创建时,它就会被为作为循环线程启动。然后此线程就会循环调用PlaybackThread::threadLoop()函数。所以,可以把threadLoop()函数看做是PlaybackThread的入口函数,它也是PlaybackThread的核心,主要的播放流程都在这里面实现的。当我分析到播放流程时,会再深入分析下此函数。
AudioService运行在SystemServer进程中,由SystemServer.java负责创建,然后将其作为Binder服务添加到ServiceManager中,供AudioManager调用。它的初始化函数入口有两个:构造函数、systemReady()函数。AudioService的初始化工作内容,除了创建各个子模块比如AudioDeviceBroker、PlaybackActivityMonitor等,创建自己的循环消息处理线程以外,最主要的工作就是进行音量初始化。
AudioService会从系统属性中读取每个StreamType配置的最大音量等级,从SettingProvider数据库中读取已保存的当前正在使用的音量等级,然后通知AudioPolicyService去设置每个StreamType对应的最大音量等级,以及设置该StreamType包含的每类Device所对应使用的音量等级。
在详细分析整个初始化流程之前,我先介绍下AudioService.VolumeStreamState和AudioService.VolumeGroupState这两个内部类。它们是音量控制逻辑的核心地方。
从上面这个类图中可以看出,这两个AudioService内部类的成员变量和对外提供的接口函数都很相似,有些函数连名称都是一样的,这就导致我们在走读跟踪代码时,容易看错,比如错把VolumeGroupState的applyAllVolumes()函数当成了VolumeStreamState的。大家在看代码时也要注意一下。一个简单的办法是看这个函数实现中用的哪把线程锁,比如VolumeStreamState.applyAllVolumes()肯定用的是VolumeStreamState.class,VolumeGroupState.applyAllVolumes()肯定用的是VolumeGroupState.class。这样从名称上就可以直接区别了。
那么这两个内部类有什么不同呢?VolumeStreamState代表的是一个StreamType所对应的音量等级。包括最小音量等级mIndexMin、最大音量等级mIndexMax、每种Device对应正在使用的音量等级mIndexMap,mIndexMap的key是Device类型,比如DEVICE_OUT_SPEAKER。也就是说VolumeStreamState类是基于StreamType来进行音量控制。需要注意的是mIndexMin、mIndexMax、mIndexMap这三个成员变量中保存的音量等级大小都会X10,放大十倍。因为音量等级是int型的,放大十倍来保存数值的目的是为了精确计算,更平滑的控制音量大小。VolumeStreamState最后在通知AudioPolicyService进行音量等级设置时,会加上5再除以10,类似四舍五入。而三个比较重要的数组AudioService.MIN_STREAM_VOLUME[]、AudioService.MAX_STREAM_VOLUME[]、AudioSystem.DEFAULT_STREAM_VOLUME[]中保存的数值是没有经过放大的,大家在修改代码时需要注意。VolumeGroupState中保存的音量等级数值也没有经过10倍放大。
VolumeGroupState代表的是一个VolumeGroup所对应的音量等级。因为在AudioPolicyService的配置中,一个VolumeGroup可以被多个不同的StreamType复用,也就是1:n的关系。所以理论上VolumeGroupState列表会比VolumeStreamState列表的个数少一些。但是实际配置文件中,StreamType和VolumeGroup是1:1的关系,所以两者其实都是基于StreamType在设置音量。区别只是VolumeGroupState提供了根据AudioAttributes进行音量设置的接口。主要是通过AudioManager的setVolumeIndexForAttributes()函数、getVolumeIndexForAttributes()函数。我全局搜索了这两个函数,发现并没有地方调用它们。所以,VolumeGroupState类在实际运行时,并没有被真正用到。
以下是音量初始化的代码流程:
AudioService.java->AudioService()
|-->AudioProductStrategy.java->getAudioProductStrategies()
|-->AudioProductStrategy.java->initializeAudioProductStrategies()
|-->AudioSystem.cpp->listAudioProductStrategies()//调用AudioPolicyService的listAudioProductStrategies函数,获取ProductStrategies数据。从而读取各个StreamType对应的最小和最大音量等级
|-->从系统属性中获取每种StreamType配置的最大音量等级和开机默认音量等级,分别保存到MAX_STREAM_VOLUME[]数组和DEFAULT_STREAM_VOLUME[]数组中。
|-->AudioService.java->createAudioSystemThread()//创建并启动名为AudioService的循环消息处理线程
|-->AudioService.java->updateStreamVolumeAlias()//根据平台不同,选择Stream别名的定义策略。比如TV电视会STREAM_VOLUME_ALIAS_TELEVISION,手机会使用STREAM_VOLUME_ALIAS_VOICE。
|-->AudioService.java->onInitStreamsAndVolumes()//内部线程在处理MSG_INIT_STREAMS_VOLUMES消息时调用
|-->AudioService.java->createStreamStates()//根据StreamType的数量,创建对应的VolumeStreamState对象,保存到mStreamStates数组中。
|-->new VolumeStreamState()
|-->AudioSystem.cpp->initStreamVolume()//通知AudioPolicyService设置最大和最小音量等级
|-->VolumeStreamState.readSettings()//从SettingProvider中读取当前StreamType对应的各个device的默认音量等级,如果SettingProvider中没有配置,就使用AudioSystem.DEFAULT_STREAM_VOLUME[]数组中的配置。
|-->AudioService.java->checkAllFixedVolumeDevices()//使用固定音量的Device,将其音量设置为最大。
|-->AudioService.java->checkAllAliasStreamVolumes()
|-->VolumeStreamState.setAllIndexes()//拷贝当前StreamType对应别名的音量
|-->VolumeStreamState.applyAllVolumes()//设置当前streamType所有对应device的音量等级到AudioPolicyService中
|-->VolumeStreamState.setStreamVolumeIndex()
|-->AudioSystem.cpp->setStreamVolumeIndexAS(mStreamType, index, device)
|-->AudioService.java->initVolumeGroupStates()//初始化VolumeGroupState对象
最后,介绍一下StreamVolumeAlias,它的作用让一种StreamType可以复用另外一种StreamType的音量等级数据。通过一个数组来表示它们复用的关联关系。因为有了这个功能,导致AudioService的音量设置相关代码中,可以大量看到转换StreamType的地方,增加了一点点阅读代码的难度。以下是手机采用的StreamType复用的例子,可以看出,STREAM_NOTIFICATION和STREAM_SYSTEM都是复用的STREAM_RING的音量等级数据。