改进音频播放

当您尝试使用 USB API 直接访问 USB 音频外设时, 可能出现的任何问题。这些问题可能包括:安全问题、媒体受限 以及经过以下时间的闹钟、通知和铃声丢失 USB 设备。

如需改善音频播放,请改为配置混音器属性。

配置混音器属性

使用 AudioMixerAttributes API、 您可以通过 USB 使用首选的混音器属性配置您的应用。

当应用播放与编码格式、通道掩码和样本匹配时 首选混音器属性的百分比,则播放将附加到音频 输出流,其混音器配置了首选混音器属性。

您的应用能够以任何配置流式传输到硬件抽象 层 (HAL) 发出,并上传到设备,只要 USB 设备支持 配置。

AudioMixerAttributes 中允许的两种混音器行为是 DEFAULTBIT_PERFECT。当混音器行为为 DEFAULT 时,表示音频 来自不同来源的数据各不相同。

当混音器行为为 BIT_PERFECT 时,不进行混音、调整音量 或经过音频处理的效果会应用到播放中。数据会以 向下连接到 HAL,最后向下连接到 USB 设备。

借助“BIT_PERFECT”,你可以直接直播 数字 (DSD) 脉冲编码调制 (PCM)。 以下代码示例展示了如何完成此操作:

KotlinJava
val EXPECTED_FORMAT: AudioFormat = AudioFormat.Builder()
 
.setEncoding(AudioFormat.ENCODING_PCM_24BIT_PACKED)
 
.setChannelMask(AudioFormat.CHANNEL_OUT_STEREO)
 
.setSampleRate(44100)
 
.build()

fun startPlayback() {
 
// Query all supported mixer attributes
 
val mixerAttributesList: List<AudioMixerAttributes?> =
    mAudioManager
.getSupportedMixerAttributes(usbDevice)

 
// Find the wanted mixer attributes
 
val mixerAttributes = mixerAttributesList.stream()
   
.filter { mixerAttr: AudioMixerAttributes? ->
      EXPECTED
_FORMAT.equals(
        mixerAttr
!!.format
     
)
   
}
   
.findAny()
   
.orElse(null)

 
// Register a listener to mixer attributes changed
 
val listener = MyPreferredMixerAttributesChangedListener()
  mAudioManager
.addOnPreferredMixerAttributesChangedListener(
   
Executors.newSingleThreadExecutor(), listener
 
)

 
// Currently, only media usage over USB devices will be allowed
 
val attr: AudioAttributes = AudioAttributes.Builder()
   
.setUsage(AudioAttributes.USAGE_MEDIA).build()
 
// Set preferred mixer attributes
  mAudioManager
.setPreferredMixerAttributes(
    attr
, usbDevice, mixerAttributes
 
)

 
// Start playback, note the playback and the audio format must
 
// match what is set when calling `setPreferredMixerAttriutes`
 
// API.
 
val audioTrack = AudioTrack.Builder()
   
.setAudioAttributes(attr)
   
.setAudioFormat(mixerAttributes!!.format)
   
.build()

 
// Clear all preferred mixer attributes related stuff when
 
// playback task is completed
  mAudioManager
.clearPreferredMixerAttributes(attr, usbDevice)
  mAudioManager
.removeOnPreferredMixerAttributesChangedListener(listener)
}

private class MyPreferredMixerAttributesChangedListener :
 
AudioManager.OnPreferredMixerAttributesChangedListener {
 
override fun onPreferredMixerAttributesChanged(
    attributes
: AudioAttributes,
    device
: AudioDeviceInfo,
    mixerAttributes
: AudioMixerAttributes?,
 
) {
   
// Do something when preferred mixer attributes changed
 
}
}
final AudioFormat EXPECTED_FORMAT = new AudioFormat.Builder()
       
.setEncoding(AudioFormat.ENCODING_PCM_24BIT_PACKED)
       
.setChannelMask(AudioFormat.CHANNEL_OUT_STEREO)
       
.setSampleRate(44100)
       
.build();

void startPlayback() {
   
// Query all supported mixer attributes
   
List<AudioMixerAttributes> mixerAttributesList =
    mAudioManager
.getSupportedMixerAttributes(usbDevice);

   
// Find the wanted mixer attributes
   
AudioMixerAttributes mixerAttributes =
        mixerAttributesList
.stream()
           
.filter(mixerAttr -> EXPECTED_FORMAT.equals(mixerAttr.getFormat()))
           
.findAny()
           
.orElse(null);

   
// Register a listener to mixer attributes changed
   
MyPreferredMixerAttributesChangedListener listener =
       
new MyPreferredMixerAttributesChangedListener();
    mAudioManager
.addOnPreferredMixerAttributesChangedListener(
           
Executors.newSingleThreadExecutor(), listener);

   
// Currently, only media usage over USB devices will be allowed
   
AudioAttributes attr = new AudioAttributes.Builder()
           
.setUsage(AudioAttributes.USAGE_MEDIA).build();
   
// Set preferred mixer attributes
    mAudioManager
.setPreferredMixerAttributes(
            attr
, usbDevice, mixerAttributes);

   
// Start playback, note the playback and the audio format must
   
// match what is set when calling `setPreferredMixerAttriutes`
   
// API.
   
AudioTrack audioTrack = new AudioTrack.Builder()
           
.setAudioAttributes(attr)
           
.setAudioFormat(mixerAttributes.getFormat())
           
.build();

   
// Clear all preferred mixer attributes related stuff when
   
// playback task is completed
    mAudioManager
.clearPreferredMixerAttributes(attr, usbDevice);
    mAudioManager
.removeOnPreferredMixerAttributesChangedListener(
            listener
);
}

private class MyPreferredMixerAttributesChangedListener
       
implements AudioManager.OnPreferredMixerAttributesChangedListener {
   
@Override
   
public void onPreferredMixerAttributesChanged(
       
AudioAttributes attributes,
       
AudioDeviceInfo device,
       
AudioMixerAttributes mixerAttributes) {
       
// Do something when preferred mixer attributes changed
   
}
}