Для воспроизведения звука на Android можно использовать следующие механизмы:
Каждый из этих механизмов позволяет воспроизводить аудио в Android. Для воспроизведения радио или с устройств ввода этих возможностей может быть недостаточно, хотя каждый из них можно объединить с захватом звука или классом MediaRecorder
для первоначального захвата звука и его последующего воспроизведения с Android. В частности, для системных приложений следующая информация может быть использована для подключения устройства ввода к выходному микшеру в AAOS.
HwAudioSource player
HwAudioSource
подключает устройство-источник звука напрямую к микшеру Android.
Мотивации
При использовании аудиопатча «устройство-устройство» или аппаратного аудиопатча с Android может возникнуть ряд ограничений. Ни один из вариантов не может получать события клавиш медиа, такие как PLAY , PAUSE и STOP , и, поскольку они обходят аудиостек Android, для микширования патча с другим аудио Android требуется аппаратное обеспечение.
Использовать HwAudioSource
HwAudioSource
— это новый тип проигрывателя, разработанный в виде программного патча. Он позволяет приложениям, использующим этот проигрыватель, получать события нажатия клавиш медиа, а также микшировать и маршрутизировать выходной поток средствами Android.
mHwAudioSource = new HwAudioSource.Builder()
.setAudioDeviceInfo(AudioDeviceInfo: info)
.setAudioAttributes(new AudioAttributes.Builder()
.setUsage(AudioAttributes.USAGE_MEDIA)
.build())
.build();
mHwAudioSource.play();
mHwAudioSource.stop();
Изменения в аудио HAL
При использовании этого нового плеера учтите следующие ожидания для аудио HAL. Например, device/generic/car/emulator/audio/driver/audio_hw.c
.
adev_create_audio_patch
ожидает запрос на установление аудиосоединения от устройства к микшеру.adev_open_input_stream
ожидает, чтоaudio_source
будетAUDIO_SOURCE_FM_TUNER
.in_read
заполняет аудиобуфер данными радиовещательной трансляции.
Мы рекомендуем вам настроить тюнер с типом AUDIO_DEVICE_IN_FM_TUNER
в audio_policy_configuration.xml
:
<devicePort
tagName="Tuner_source"
type="AUDIO_DEVICE_IN_FM_TUNER"
role="source"
address="tuner0">
<profile
name=""
format="AUDIO_FORMAT_PCM_16_BIT"
samplingRates="48000"
channelMasks="AUDIO_CHANNEL_IN_STEREO"/>
</devicePort>
С помощью этой конфигурации устройства вы можете облегчить поиск устройства ввода FM-радио, используя AudioManager.getDevices(AudioManager.GET_DEVICES_INPUTS
совместно с AudioDeviceInfo.TYPE_FM_TUNER
.
Создание аудиопатчей
Вы можете создать аудиопатч между двумя аудиопортами: портом микширования или портом устройства. Обычно аудиопатч от порта микширования к порту устройства предназначен для воспроизведения, а обратный — для захвата.
Например, аудиопатч, направляющий аудиосэмплы из источника FM_TUNER
напрямую в медиа-приемник, обходит программный микшер. В этом случае необходимо использовать аппаратный микшер для микширования аудиосэмплов из Android и FM_TUNER
для медиа-приемника. При создании аудиопатча напрямую из источника FM_TUNER
в медиа-приемник:
Регулировка громкости применяется к медиаприемнику и должна влиять как на звук Android, так и на звук
FM_TUNER
.Пользователи могут переключаться между Android и аудио
FM_TUNER
с помощью простого переключения приложений (явный выбор источника мультимедиа не требуется).
В автомобильных реализациях также может потребоваться создать аудиопатч между двумя портами устройств. Для этого необходимо сначала объявить порты устройств и возможные маршруты в audio_policy_configuration.xml
, а затем связать микспорты с портами устройств.
Пример конфигурации
См. этот пример конфигурации, device/generic/car/emulator/audio/audio_policy_configuration.xml
.
<audioPolicyConfiguration>
<modules>
<module name="primary" halVersion="3.0">
<attachedDevices>
<item>bus0_media_out</item>
<item>bus1_audio_patch_test_in</item>
</attachedDevices>
<mixPorts>
<mixPort name="mixport_bus0_media_out" role="source"
flags="AUDIO_OUTPUT_FLAG_PRIMARY">
<profile name="" format="AUDIO_FORMAT_PCM_16_BIT"
samplingRates="48000"
channelMasks="AUDIO_CHANNEL_OUT_STEREO"/>
</mixPort>
<mixPort name="mixport_audio_patch_in" role="sink">
<profile name="" format="AUDIO_FORMAT_PCM_16_BIT"
samplingRates="48000"
channelMasks="AUDIO_CHANNEL_IN_STEREO"/>
</mixPort>
</mixPorts>
<devicePorts>
<devicePort tagName="bus0_media_out" role="sink" type="AUDIO_DEVICE_OUT_BUS"
address="bus0_media_out">
<profile balance="" format="AUDIO_FORMAT_PCM_16_BIT"
samplingRates="48000" channelMasks="AUDIO_CHANNEL_OUT_STEREO"/>
<gains>
<gain name="" mode="AUDIO_GAIN_MODE_JOINT"
minValueMB="-8400" maxValueMB="4000" defaultValueMB="0" stepValueMB="100"/>
</gains>
</devicePort>
<devicePort tagName="bus1_audio_patch_test_in" type="AUDIO_DEVICE_IN_BUS" role="source"
address="bus1_audio_patch_test_in">
<profile name="" format="AUDIO_FORMAT_PCM_16_BIT"
samplingRates="48000" channelMasks="AUDIO_CHANNEL_IN_STEREO"/>
<gains>
<gain name="" mode="AUDIO_GAIN_MODE_JOINT"
minValueMB="-8400" maxValueMB="4000" defaultValueMB="0" stepValueMB="100"/>
</gains>
</devicePort>
</devicePorts>
<routes>
<route type="mix" sink="bus0_media_out" sources="mixport_bus0_media_out,bus1_audio_patch_test_in"/>
<route type="mix" sink="mixport_audio_patch_in" sources="bus1_audio_patch_test_in"/>
</routes>
</module>
</modules>
</audioPolicyConfiguration>
API аудиодрайвера
Вы можете использовать функцию getExternalSources()
для получения списка доступных источников (идентифицированных по адресу), а затем создать аудиосоединения между этими источниками и портами приёмника по использованию аудио. Соответствующие точки входа в Audio HAL указаны в IDevice.hal
:
Interface IDevice {
...
/
* Creates an audio patch between several source and sink ports. The handle
* is allocated by the HAL and must be unique for this audio HAL module.
*
* @param sources patch sources.
* @param sinks patch sinks.
* @return retval operation completion status.
* @return patch created patch handle.
*/
createAudioPatch(vec<AudioPortConfig> sources, vec<AudioPortConfig> sinks)
generates (Result retval, AudioPatchHandle patch);
* Release an audio patch.
*
* @param patch patch handle.
* @return retval operation completion status.
*/
releaseAudioPatch(AudioPatchHandle patch) generates (Result retval);
...
}
Радиотюнер
При создании радиоприложения мы рекомендуем использовать HwAudioSource
, поскольку он отвечает как за создание патча, так и за медиа-сессию для обработки ключевых событий медиа. Можно создать несколько аудиоисточников для одних и тех же источников и аудиоатрибутов. Можно использовать один источник для обычного использования радио, а второй — для объявлений о дорожном движении.
При записи FM_TUNER
в Android 11 разрешение на запись было изменено на android.permission.CAPTURE_AUDIO_OUTPUT
. Оно больше не проходит проверку разрешения OP_RECORD_AUDIO
, которая применяется только к микрофонам. Это не должно повлиять на приложения, поскольку FM_TUNER
уже требует разрешения SYSTEM_API
для доступа.
Подробную информацию о создании приложения радио см. в разделе Реализация радио .