Gdy próbujesz uzyskać bezpośredni dostęp do urządzenia peryferyjnego USB audio za pomocą interfejsów USB API, pojawiają się problemy. Należą do nich: problemy z bezpieczeństwem, ograniczenie multimediów odtwarzania z innych aplikacji oraz utraty alarmów, powiadomień i dzwonków przez Urządzenia USB.
Aby ulepszyć odtwarzanie dźwięku, skonfiguruj atrybuty miksera.
Skonfiguruj atrybuty miksera
Za pomocą
interfejsy API AudioMixerAttributes
,
możesz skonfigurować aplikację, używając preferowanych atrybutów miksera przez USB.
Gdy odtwarzanie w aplikacji pasuje do formatu kodowania, maski kanału i próbki preferowanej wartości atrybutów miksera, odtwarzanie jest dołączane do dźwięku strumień wyjściowy, którego mikser jest skonfigurowany za pomocą preferowanych atrybutów miksera.
Aplikacja może przesyłać strumieniowo dane w dowolnej konfiguracji do abstrakcji sprzętowej lub warstwy (HAL) i na urządzeniu, o ile urządzenie USB obsługuje konfiguracji.
2 dozwolone zachowania miksera w AudioMixerAttributes
to DEFAULT
oraz
BIT_PERFECT
Gdy mikser działa na poziomie DEFAULT
, oznacza to, że dźwięk
dane z różnych źródeł są pomieszane.
Gdy mikser działa w trybie BIT_PERFECT
, nie jest miksowane, nie ma regulacji głośności,
lub efektu przetworzonego dźwięku. Dane są wysyłane jako:
do HAL, a na końcu do urządzenia USB.
BIT_PERFECT
umożliwia bezpośrednią transmisję
cyfrowych (DSD) w przypadku urządzeń z Androidem za pomocą technologii PCM (PCM).
Oto przykładowy kod, który pokazuje, jak to zrobić:
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
}
}