Melhorar a reprodução de áudio

Quando você tenta acessar diretamente o periférico de áudio USB usando as APIs USB, que problemas possam surgir. Esses problemas podem incluir: problemas de segurança, limitação de mídia reprodução de outros apps e perda de alarmes, notificações e toques Dispositivos USB.

Para melhorar a reprodução de áudio, configure os atributos de mixer.

Configurar atributos do mixer

Ao usar o método APIs do AudioMixerAttributes, você pode configurar seu aplicativo com os atributos preferidos de mixer por USB.

Quando a reprodução do app corresponde ao formato de codificação, à máscara de canal e à amostra dos atributos preferidos do mixer, a reprodução é anexada ao áudio stream de saída cujo mixer está configurado com os atributos de mixer preferidos.

Seu app pode fazer streaming em qualquer configuração para a abstração de hardware (HAL) e ao dispositivo, contanto que o dispositivo USB seja compatível configuração do Terraform.

Os dois comportamentos permitidos do mixer em AudioMixerAttributes são DEFAULT e BIT_PERFECT Quando o comportamento do mixer é DEFAULT, isso indica que o áudio os dados de diferentes fontes são misturados.

Quando o comportamento do mixer for BIT_PERFECT, não haverá mixagem de áudio, ajuste de volume ou efeito de áudio processado é aplicado à reprodução. Os dados são enviados como até a HAL, depois o dispositivo USB.

O uso de BIT_PERFECT permite que você direcione o stream modulação por código de pulso (PCM, na sigla em inglês) digital (DSD) em dispositivos com Android. O exemplo de código a seguir mostra como fazer isso:

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
   
}
}