blob: ae8d0a5ad4ab10a7b662f231250c661e7f5340b4 [file] [log] [blame]
/*
* Copyright (C) 2020 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package android.media;
import android.annotation.IntDef;
import android.annotation.NonNull;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.util.Arrays;
import java.util.stream.Collectors;
/**
* An AudioProfile is specific to an audio format and lists supported sampling rates and
* channel masks for that format. An {@link AudioDeviceInfo} has a list of supported AudioProfiles.
* There can be multiple profiles whose encoding format is the same. This usually happens when
* an encoding format is only supported when it is encapsulated by some particular encapsulation
* types. If there are multiple encapsulation types that can carry this encoding format, they will
* be reported in different audio profiles. The application can choose any of the encapsulation
* types.
*/
public class AudioProfile {
/**
* No encapsulation type is specified.
*/
public static final int AUDIO_ENCAPSULATION_TYPE_NONE = 0;
/**
* Encapsulation format is defined in standard IEC 61937.
*/
public static final int AUDIO_ENCAPSULATION_TYPE_IEC61937 = 1;
/** @hide */
@IntDef({
AUDIO_ENCAPSULATION_TYPE_NONE,
AUDIO_ENCAPSULATION_TYPE_IEC61937,
})
@Retention(RetentionPolicy.SOURCE)
public @interface EncapsulationType {}
private final int mFormat;
private final int[] mSamplingRates;
private final int[] mChannelMasks;
private final int[] mChannelIndexMasks;
private final int mEncapsulationType;
AudioProfile(int format, @NonNull int[] samplingRates, @NonNull int[] channelMasks,
@NonNull int[] channelIndexMasks,
int encapsulationType) {
mFormat = format;
mSamplingRates = samplingRates;
mChannelMasks = channelMasks;
mChannelIndexMasks = channelIndexMasks;
mEncapsulationType = encapsulationType;
}
/**
* @return the encoding format for this AudioProfile.
*/
public @AudioFormat.Encoding int getFormat() {
return mFormat;
}
/**
* @return an array of channel position masks that are associated with the encoding format.
*/
public @NonNull int[] getChannelMasks() {
return mChannelMasks;
}
/**
* @return an array of channel index masks that are associated with the encoding format.
*/
public @NonNull int[] getChannelIndexMasks() {
return mChannelIndexMasks;
}
/**
* @return an array of sample rates that are associated with the encoding format.
*/
public @NonNull int[] getSampleRates() {
return mSamplingRates;
}
/**
* The encapsulation type indicates what encapsulation type is required when the framework is
* using this format when playing to a device exposing this audio profile.
* When encapsulation is required, only playback with {@link android.media.AudioTrack} API is
* supported. But playback with {@link android.media.MediaPlayer} is not.
* When an encapsulation type is required, the {@link AudioFormat} encoding selected when
* creating the {@link AudioTrack} must match the encapsulation type, e.g
* AudioFormat.ENCODING_IEC61937 for AUDIO_ENCAPSULATION_TYPE_IEC61937.
*
* @return an integer representing the encapsulation type
*
* @see #AUDIO_ENCAPSULATION_TYPE_NONE
* @see #AUDIO_ENCAPSULATION_TYPE_IEC61937
*/
public @EncapsulationType int getEncapsulationType() {
return mEncapsulationType;
}
@Override
public String toString() {
StringBuilder sb = new StringBuilder("{");
sb.append(AudioFormat.toLogFriendlyEncoding(mFormat));
if (mSamplingRates != null && mSamplingRates.length > 0) {
sb.append(", sampling rates=").append(Arrays.toString(mSamplingRates));
}
if (mChannelMasks != null && mChannelMasks.length > 0) {
sb.append(", channel masks=").append(toHexString(mChannelMasks));
}
if (mChannelIndexMasks != null && mChannelIndexMasks.length > 0) {
sb.append(", channel index masks=").append(Arrays.toString(mChannelIndexMasks));
}
sb.append("}");
return sb.toString();
}
private static String toHexString(int[] ints) {
if (ints == null || ints.length == 0) {
return "";
}
return Arrays.stream(ints).mapToObj(anInt -> String.format("0x%02X", anInt))
.collect(Collectors.joining(", "));
}
}