阅读目录

  • 1. 架构
  • 2. Audio HAL
  • 3. Native Audio
  • 4. Java Audio

开始这篇文章之前,需要先了解<Linux音频编程>

回到顶部

1. 架构

在Android中,audio以分层的方式实现,从上到下依次为

- 应用框架: 提供android.media API

音频管理器: AudioManager
音频采集: MediaRecoder, AudioRecord
音频播放: SoundPool, MediaPlayer, AudioTrack
音频编解码: MediaCodec

- JNI: 通过调用libmedia库实现android.media所需的接口,在libandroid_runtime.so中体现

- HAL层: 实现audio_hw_device和audio_policy_hal, 实现与ALSA的音频接口, 实现音频路径的创建和连接

Android session 音频 android audio_输入流

回到顶部

2. Audio HAL

Audio HAL架构使用的是比较复杂,混合了HIDL和Legacy HAL,笔者看的也是一头雾水
可参考<Android O Treble架构下Hal进程启动及HIDL服务注册过程>

2.1 HAL接口

Audio HAL提供的接口(以2.0为例)主要包括

Android session 音频 android audio_Android_02

// 音频设备
IDevice.hal
  |- initCheck()
  |- setMasterVolume(float): 设置除voice call外其他音频活动的音量
  |- getMasterVolume():      获取主音量
  |- setMicMute(bool):       设置麦克风静音状态
  |- getMicMute():           获取麦克风静音状态
  |- setMasterMute(bool):    设置静音状态
  |- getMasterMute(bool):    获取静音状态
  |- getInputBufferSize(AudioConfig): 获取音频输入缓冲区大小
  |- openOutputStream(*):    创建和打开音频硬件输出流
  |- openInputStream(*):     创建和打开音频硬件输入流
  |- supportsAudioPatches(): 判断HAL是否支持AudioPatch
  |- createAudioPatch(*):    为SRC和SINK创建AudioPatch
  |- releaseAudioPatch(*):   释放一个AudioPatch
  |- getAudioPort(*):        获取指定音频端口属性
  |- setAudioPortConfig(*):  配置音频端口
  |- getHwAvSync():          获取设备的硬件同步源
  |- setScreenState(bool):   设置屏幕状态
  |- getParameters(vec<string>): 获取厂商定义的参数值
  |- setParameters(vec<ParameterValue>): 设置厂商定义的参数值

// 音频代理
IDevicesFactory.hal
  |- openDevice(Device):     打开一个音频设备

// 主音频设备
IPrimaryDevice.hal
  |- setMasterVolume(float): 设置voice call音量
  |- setMode(AudioMode):     设置音频模式
  |- getBtScoNrecEnabled():  获取蓝牙ECNR使能状态
  |- setBtScoNrecEnabled():  设置蓝牙ECNR使能状态
  |- getBtScoWidebandEnabled():     获取蓝牙Wideband使能状态
  |- setBtScoWidebandEnabled(bool): 设置蓝牙Wideband使能状态
  |- getTtyMode():           获取当前TTY模式
  |- setTtyMode():           设置当前TTY模式
  |- getHacEnabled():        获取HearingAid使能状态
  |- setHacEnabled():        设置HearingAid使能状态

// 音频流
IStream.hal
  |- getFrameSize():         获取帧大小
  |- getFrameCount():        获取缓冲区帧数
  |- getBufferSize():        获取流的缓冲区大小
  |- getSampleRate():        获取采样率(Hz)
  |- getSupportedSampleRates():  获取流支持的支持的采样率(Hz)
  |- setSampleRate(uint32_t):    设置流的采样率
  |- getChannelMask():       获取流的channel mask
  |- getSupportedChannelMasks(): 获取流支持的channel mask
  |- setChannelMask():       获取流的channel mask
  |- getFormat():            获取流的音频格式
  |- getSupportedFormats():  获取流支持的音频格式
  |- setFormat():            设置流的音频格式
  |- getAudioProperties():   获取流参数
  |- addEffect():            添加音效到流
  |- removeEffect(uint64_t): 从流上停止某音效
  |- standby(uint64_t):      让硬件输入输出进入standby模式
  |- getDevice():            获取流连接的设备
  |- setDevice():            连接设置到流
  |- setConnectedState():    通知设备连接状态
  |- setHwAvSync(AudioHwSync):   设置硬件同步源
  |- getParameters(vec<string>): 获取厂商参数
  |- setParameters(vec<ParameterValue>):  设置厂商参数
  |- start():                开始流操作(mmap模式)
  |- stop():                 停止流操作
  |- createMmapBuffer():     获取audio mmap缓冲区信息
  |- getMmapPosition():      读取audio mmap缓冲区读写位置
  |- close():                关闭和释放流

// 音频输入流
IStreamIn.hal
  |- getAudioSource():       获取输入流的source描述
  |- setGain():              设置音频驱动的输入增益
  |- prepareForReading():    设置必需的传输层以从驱动接收音频缓冲区
  |- getInputFramesLost():   获取丢失的输入帧的数量
  |- getCapturePosition():   获取接收到的音频帧数与时钟时间。

// 音频输出流
IStreamOut.hal
  |- getLatency():           获取硬件传输延迟(毫秒)
  |- setVolume(float, float):    设置音量, 用于混音后
  |- prepareForWriting():    设置必需的传输层将音频缓冲区传递给驱动
  |- getRenderPosition():    获取音频DSP写入DAC的音频帧数
  |- getNextWriteTimestamp():    获取下一次写入音频驱动时间(微秒)
  |- setCallback():          设置回调接口, 用于非阻塞模式
  |- clearCallback():        清楚回调
  |- supportsPauseAndResume():   HAL是否支持暂停和恢复流
  |- pause():                暂停流
  |- resume():               恢复流
  |- supportsDrain():        ???
  |- drain():                ???
  |- flush():                刷新流
  |- getPresentationPosition():  获取音频帧数

// 音频输入流回调
IStreamOutCallback.hal
  |- onWriteReady():         非阻塞写入已完成
  |- onDrainReady():         Drain(?)完成
  |- onError():              出错

Android session 音频 android audio_Android_02

2.2 HIDL服务

Android session 音频 android audio_Android_02

/*
 * Code:   hardware/interfaces/audio/common/all-versions/default/service/service.cpp
 * Output: /vendor/bin/hw/android.hardware.audio@2.0-service
 */
main()
   // 连接至vndservicemanager
   android::ProcessState::initWithDriver("/dev/vndbinder") 
   android::ProcessState::self()->startThreadPool()

   registerPassthroughServiceImplementation<audio::V4_0::IDevicesFactory>()
   registerPassthroughServiceImplementation<audio::V2_0::IDevicesFactory>()

   registerPassthroughServiceImplementation<audio::effect::V4_0::IEffectsFactory>()
   registerPassthroughServiceImplementation<audio::effect::V2_0::IEffectsFactory>()

   registerPassthroughServiceImplementation<soundtrigger::V2_1::ISoundTriggerHw>()
   registerPassthroughServiceImplementation<soundtrigger::V2_0::ISoundTriggerHw>()

   registerPassthroughServiceImplementation<bluetooth::a2dp::V1_0::IBluetoothAudioOffload>()

   android::hardware::joinRpcThreadpool()

# cat /vendor/etc/init/android.hardware.audio@2.0-service.rc
service vendor.audio-hal-2-0 /vendor/bin/hw/android.hardware.audio@2.0-service
    class hal
    user audioserver
    oneshot
    interface android.hardware.audio@4.0::IDevicesFactory default
    interface android.hardware.audio@2.0::IDevicesFactory default

# /vendor/etc/vintf/manifest.xml
<manifest version="1.0" type="device" target-level="3">
...
    <hal format="hidl">
        <name>android.hardware.audio</name>
        <transport>hwbinder</transport>
        <version>4.0</version>
        <interface>
            <name>IDevicesFactory</name>
            <instance>default</instance>
        </interface>
    </hal>
...
</manifest>

Android session 音频 android audio_Android_02

2.3 libaudiohal

libaudiohal封装了audio HIDL的接口,以libaudiohal.so的形式供AudioFlinger使用,而libaudiohal又使用了libaudiohal@4.0和libaudiohal@2.0两个库

Android session 音频 android audio_Android_02

# tree frameworks/av/media/libaudiohal
.
+--- 2.0
|   +--- Android.bp   -- libaudiohal@2.0
+--- 4.0
|   +--- Android.bp   -- libaudiohal@4.0
+--- Android.bp       -- libaudiohal

/*
 * DevicesFactoryHalInterface提供了create和openDevice两个方法
 */
DevicesFactoryHalInterface::create()
    /*
     * 这里仅仅分析了V2_0版本, V4_0类似
     * 提供了两种方式来访问audio hardware
     *   - HIDL:   即DevicesFactoryHalHidl, 用于primary, usb和remote_submix
     *   - Legacy: 即DevicesFactoryHalLocal, 用于a2dp
     */
    new DevicesFactoryHalHybrid()
        new DevicesFactoryHalLocal()
        new DevicesFactoryHalHidl()
            hardware::audio::V2_0::IDevicesFactory::getService()

DevicesFactoryHalInterface::openDevice(char *name, DeviceHalInterface *device)
    DevicesFactoryHalHybrid::openDevice(name, device)
        // 针对hearing_aid和a2dp设备
        DevicesFactoryHalLocal::openDevice(name, device)
            load_audio_interface(name, audio_hw_device_t **dev)
            new DeviceHalLocal(dev)
        // 针对其他设备, 包括primary, usb, remote_submix设备
        DevicesFactoryHalHidl::openDevice(name, device)
            // 获取HIDL接口名
            nameFromHal(name, IDevicesFactory::Device &)
            IDevicesFactory::openDevice()
                new DeviceHalHidl(IDevice)

/* 这里暂不做分析 */
EffectsFactoryHalInterface::create()

Android session 音频 android audio_Android_02

回到顶部

3. Native Audio

3.1 介绍

Native Audio服务在Android N之前存在于mediaserver中,Android N之后以audioserver形式存在

audioserver启动了两个Native binder服务

 

- AudioFlinger:      音频系统策略的执行者, 负责音频流设备的管理及音频流数据的处理传输
- AudioPolicyService: 音频系统策略的制定者, 负责音频设备切换的策略抉择、音量调节策略等

值得注意的AudioFlinger和AudioPolicyService提供了binder服务,然后Java层并不是直接使用这些服务,而是Native层将这些binder服务封装为C++ android::media::*接口,然后通过JNI的方式提供给Jave层使用

Android session 音频 android audio_Android_02

// android::media::*  <===> frameworks/av/media/libaudioclient/
// JNI                <===> frameworks/base/core/jni
-----------------------------------------------------
| android::media::* | JNI                           |
-----------------------------------------------------
| AudioSystem.cpp   | android_media_AudioSystem.cpp |
| AudioRecord.cpp   | android_media_AudioRecord.cpp |
| AudioTrack.cpp    | android_media_AudioTrack.cpp  |
-----------------------------------------------------

Android session 音频 android audio_Android_02

audioserver的启动的详细过程如下

Android session 音频 android audio_Android_02

/*
 * frameworks/av/media/audioserver/audioserver.rc
 */
# cat audioserver.rc
service audioserver /system/bin/audioserver
    class core
    user audioserver
    onrestart restart vendor.audio-hal-2-0
    onrestart restart audio-hal-2-0
/*
 * Code:   frameworks/av/media/audioserver/main_audioserver.cpp
 * Output: /system/bin/audioserver
 */
main()
    AudioFlinger::instantiate()
        BinderService::instantiate()
            BinderService::publish()
                IServiceManager sm = defaultServiceManager()
                sm::addService("media.audio_flinger", new AudioFlinger())
                    AudioFlinger::onFirstRef()
                        new PatchPanel(this)
                        gAudioFlinger = this;

    AudioPolicyService::instantiate()
                sm::addService("media.audio_policy", new AudioPolicyService())
                    AudioPolicyService::onFirstRef()
                        // Tone播放线程
                        new AudioCommandThread("ApmTone", this)
                            AudioCommandThread::onFirstRef()
                                Thread::run()
                                    AudioCommandThread::threadLoop()
                        // Audio命令线程
                        new AudioCommandThread("ApmAudio", this)
                        // 输出命令线程
                        new AudioCommandThread("ApmOutput", this)
                        new AudioPolicyClient(this)
                        createAudioPolicyManager()
                            new AudioPolicyManager(mAudioPolicyClient)
                                AudioPolicyManager::AudioPolicyManager()
                                    /*
                                     * 当定义了USE_XML_AUDIO_POLICY_CONF = 1
                                     * 加载/odm/etc/audio_policy_configuration.xml
                                     *     /vendor/etc/audio_policy_configuration.xml
                                     *     /system/etc/audio_policy_configuration.xml
                                     * 否则加载
                                     *     /system/etc/audio_policy.conf
                                     *     /vendor/etc/audio_policy.conf
                                     */
                                    AudioPolicyManager::loadConfig()
                                        deserializeAudioPolicyXmlConfig()
                                    // FIXME: Do a lot of things
                                    AudioPolicyManager::initialize()
                        new AudioPolicyEffects()
                        new UidPolicy(this)
                        UidPolicy::registerSelf()
    // Oboe Service
    AAudioService::instantiate()
    SoundTriggerHwService::instantiate()

Android session 音频 android audio_Android_02

3.2 AudioFlinger

AudioFlinger服务通过binder服务方式提供对外提供了服务

Android session 音频 android audio_Android_02

// 接口定义
frameworks/av/include/media/IAudioFlinger.h 

// 实现AudioFlinger本地端
libaudioflinger <==> frameworks/av/services/audioflinger/* 

// 实现AudioFlinger远程端
libaudioclient  <==> frameworks/av/media/libaudioclient/*

Android session 音频 android audio_Android_02

AudioFlinger主要提供了如下接口

Android session 音频 android audio_输入流_14

3.3 AudioPolicyService

Android session 音频 android audio_Android_02

// 接口定义
frameworks/av/include/media/IAudioPolicyService.h

// 实现AudioPolicyService本地端
libaudiopolicyservice <==> frameworks/av/services/audiopolicy/* 

// 实现AudioPolicyService远程端
libaudioclient        <==> frameworks/av/media/libaudioclient/*

Android session 音频 android audio_Android_02

回到顶部

4. Java Audio

在Java层把Audio从功能上分为三类接口

AudioSystem: API为AudioManager,负责的是Audio系统的综合管理功能
AudioTrack: 负责音频数据的输出,即播放
AudioRecorder: 负责音频数据的输出和输入,即录制

4.1 AudioService

AudioService由SystemServer启动,继承自IAudioService.Stub(通过IAudioService.aidl自动生成的),AudioService位于IAudioService的Bn端;

AudioManager拥有IAudioService的Bp端,是AudioService在客户端的一个代理,几乎所有客户端对AudioManager进行的请求,最终都会交由AudioService实现

AudioService的功能实现依赖AudioSystem类,AudioSystem是Java层到native层的代理;AudioService将通过AudioSystem与AudioPolicyService以及AudioFlinger进行交互

IAudioService  -   frameworks/base/media/java/android/media/IAudioService.aidl)

Android session 音频 android audio_Android_02

SystemServer::startOtherServices()
    SystemServiceManager::startService(AudioService.Lifecycle.class)
        new AudioService()
        AudioService::onStart()
            publishBinderService(Context.AUDIO_SERVICE, new AudioService());
                // 向servicemanager注册服务
                ServiceManager.addService( , , , )
        AudioService::onBootPhase(SystemService.PHASE_ACTIVITY_MANAGER_READY)
            AudioService::systemReady()
                AudioHandler::handleMessage(MSG_SYSTEM_READY)
                    AudioService::onSystemReady()
                        AudioHandler::onLoadSoundEffects()
                        // Bluetooth releated config
                        AudioService::onIndicateSystemReady()
                            // 位于android_media_AudioSystem.cpp
                            android_media_AudioSystem_systemReady()
                                // 位于AudioSystem.cpp
                                AudioSystem::systemReady()

Android session 音频 android audio_Android_02

AudioService主要是作为AudioManager的后端服务存在

Android session 音频 android audio_Android_02

@SystemService(Context.AUDIO_SERVICE)
public class AudioManager {
    ...
    private static IAudioService getService()
    {
        if (sService != null) {
            return sService;
        }
        IBinder b = ServiceManager.getService(Context.AUDIO_SERVICE);
        sService = IAudioService.Stub.asInterface(b);
        return sService;
    }
    ...
}

Android session 音频 android audio_Android_02

AudioService/AudioManager的功能主要包括下面三个方面

- 音量控制
- 音频IO设备的管理
- 音频焦点机制

4.2 AudioTrack

Java层的AudioTrack通过JNI的方式使用android_media_AudioTrack.cpp封装的接口,然后调用到libaudioclient中的的android::media::AudioTrack

 

111

4.3 AudioRecord

Java层的AudioRecord通过JNI的方式使用android_media_AudioRecord.cpp封装的接口

OpenSL ES 和 AAudio

https://www.jianshu.com/p/a676d4d959ae

https://source.android.google.cn/devices/audio/implement

https://www.jianshu.com/p/cf98b3cc6767