blob: 356b765f91c0b9539283ccb6aa05c29ccdfe23b9 [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 android.annotation.SystemApi;
import android.os.Parcel;
import android.os.Parcelable;
import java.lang.annotation.Retention;
import java.lang.annotation.RetentionPolicy;
import java.util.Arrays;
import java.util.Objects;
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 implements Parcelable {
/**
* 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;
/**
* Encapsulation format is PCM, which can be used by other formats that can be wrapped in
* a PCM frame, such as DSD(Direct Stream Digital).
*/
public static final int AUDIO_ENCAPSULATION_TYPE_PCM = 2;
/** @hide */
@IntDef({
AUDIO_ENCAPSULATION_TYPE_NONE,
AUDIO_ENCAPSULATION_TYPE_IEC61937,
AUDIO_ENCAPSULATION_TYPE_PCM,
})
@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;
/**
* @hide
* Constructor from format, sampling rates, channel masks, channel index masks and
* encapsulation type.
* @param format the audio format
* @param samplingRates the supported sampling rates
* @param channelMasks the supported channel masks
* @param channelIndexMasks the supported channel index masks
* @param encapsulationType the encapsulation type of the encoding format
*/
@SystemApi
public 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
* @see #AUDIO_ENCAPSULATION_TYPE_PCM
*/
public @EncapsulationType int getEncapsulationType() {
return mEncapsulationType;
}
@Override
public int hashCode() {
return Objects.hash(mFormat, Arrays.hashCode(mSamplingRates),
Arrays.hashCode(mChannelMasks), Arrays.hashCode(mChannelIndexMasks),
mEncapsulationType);
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
AudioProfile that = (AudioProfile) o;
return ((mFormat == that.mFormat)
&& (hasIdenticalElements(mSamplingRates, that.mSamplingRates))
&& (hasIdenticalElements(mChannelMasks, that.mChannelMasks))
&& (hasIdenticalElements(mChannelIndexMasks, that.mChannelIndexMasks))
&& (mEncapsulationType == that.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(", encapsulation type=" + mEncapsulationType);
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(", "));
}
private static boolean hasIdenticalElements(int[] array1, int[] array2) {
int[] sortedArray1 = Arrays.copyOf(array1, array1.length);
Arrays.sort(sortedArray1);
int[] sortedArray2 = Arrays.copyOf(array2, array2.length);
Arrays.sort(sortedArray2);
return Arrays.equals(sortedArray1, sortedArray2);
}
@Override
public int describeContents() {
return 0;
}
@Override
public void writeToParcel(@NonNull Parcel dest, int flags) {
dest.writeInt(mFormat);
dest.writeIntArray(mSamplingRates);
dest.writeIntArray(mChannelMasks);
dest.writeIntArray(mChannelIndexMasks);
dest.writeInt(mEncapsulationType);
}
private AudioProfile(@NonNull Parcel in) {
mFormat = in.readInt();
mSamplingRates = in.createIntArray();
mChannelMasks = in.createIntArray();
mChannelIndexMasks = in.createIntArray();
mEncapsulationType = in.readInt();
}
public static final @NonNull Parcelable.Creator<AudioProfile> CREATOR =
new Parcelable.Creator<AudioProfile>() {
/**
* Rebuilds an AudioProfile previously stored with writeToParcel().
* @param p Parcel object to read the AudioProfile from
* @return a new AudioProfile created from the data in the parcel
*/
public AudioProfile createFromParcel(Parcel p) {
return new AudioProfile(p);
}
public AudioProfile[] newArray(int size) {
return new AudioProfile[size];
}
};
}