/* * Copyright (C) 2018 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 com.android.car.audio; import android.annotation.NonNull; import android.annotation.Nullable; import android.annotation.UserIdInt; import android.car.media.CarAudioManager; import android.content.Context; import android.media.AudioDevicePort; import android.os.UserHandle; import android.util.Log; import android.util.SparseArray; import com.android.car.CarLog; import com.android.internal.util.Preconditions; import java.io.PrintWriter; import java.util.ArrayList; import java.util.Arrays; import java.util.HashMap; import java.util.List; import java.util.Map; /** * A class encapsulates a volume group in car. * * Volume in a car is controlled by group. A group holds one or more car audio contexts. * Call {@link CarAudioManager#getVolumeGroupCount()} to get the count of {@link CarVolumeGroup} * supported in a car. */ /* package */ final class CarVolumeGroup { private CarAudioSettings mSettingsManager; private final int mZoneId; private final int mId; private final SparseArray mContextToAddress = new SparseArray<>(); private final Map mAddressToCarAudioDeviceInfo = new HashMap<>(); private final Object mLock = new Object(); private int mDefaultGain = Integer.MIN_VALUE; private int mMaxGain = Integer.MIN_VALUE; private int mMinGain = Integer.MAX_VALUE; private int mStepSize = 0; private int mStoredGainIndex; private int mCurrentGainIndex = -1; private @UserIdInt int mUserId = UserHandle.USER_CURRENT; /** * Constructs a {@link CarVolumeGroup} instance * @param Settings {@link CarAudioSettings} instance * @param zoneId Audio zone this volume group belongs to * @param id ID of this volume group */ CarVolumeGroup(CarAudioSettings settings, int zoneId, int id) { mSettingsManager = settings; mZoneId = zoneId; mId = id; mStoredGainIndex = mSettingsManager.getStoredVolumeGainIndexForUser(mUserId, mZoneId, mId); } /** * Constructs a {@link CarVolumeGroup} instance * @param context {@link Context} instance * @param zoneId Audio zone this volume group belongs to * @param id ID of this volume group * @param contexts Pre-populated array of car contexts, for legacy car_volume_groups.xml only * @deprecated In favor of {@link #CarVolumeGroup(Context, int, int)} */ @Deprecated CarVolumeGroup(CarAudioSettings settings, int zoneId, int id, @NonNull int[] contexts) { this(settings, zoneId, id); // Deal with the pre-populated car audio contexts for (int audioContext : contexts) { mContextToAddress.put(audioContext, null); } } /** * @param address Physical address for the audio device * @return {@link CarAudioDeviceInfo} associated with a given address */ CarAudioDeviceInfo getCarAudioDeviceInfoForAddress(String address) { return mAddressToCarAudioDeviceInfo.get(address); } /** * @return Array of car audio contexts {@link CarAudioContext} in this {@link CarVolumeGroup} */ int[] getContexts() { final int[] carAudioContexts = new int[mContextToAddress.size()]; for (int i = 0; i < carAudioContexts.length; i++) { carAudioContexts[i] = mContextToAddress.keyAt(i); } return carAudioContexts; } /** * Returns the devices address for the given context * or {@code null} if the context does not exist in the volume group */ @Nullable String getAddressForContext(int audioContext) { return mContextToAddress.get(audioContext); } /** * @param address Physical address for the audio device * @return Array of car audio contexts {@link CarAudioContext} assigned to a given address */ int[] getContextsForAddress(@NonNull String address) { List carAudioContexts = new ArrayList<>(); for (int i = 0; i < mContextToAddress.size(); i++) { String value = mContextToAddress.valueAt(i); if (address.equals(value)) { carAudioContexts.add(mContextToAddress.keyAt(i)); } } return carAudioContexts.stream().mapToInt(i -> i).toArray(); } /** * @return Array of addresses in this {@link CarVolumeGroup} */ List getAddresses() { return new ArrayList<>(mAddressToCarAudioDeviceInfo.keySet()); } /** * Binds the context number to physical address and audio device port information. * Because this may change the groups min/max values, thus invalidating an index computed from * a gain before this call, all calls to this function must happen at startup before any * set/getGainIndex calls. * * @param carAudioContext Context to bind audio to {@link CarAudioContext} * @param info {@link CarAudioDeviceInfo} instance relates to the physical address */ void bind(int carAudioContext, CarAudioDeviceInfo info) { Preconditions.checkArgument(mContextToAddress.get(carAudioContext) == null, String.format("Context %s has already been bound to %s", CarAudioContext.toString(carAudioContext), mContextToAddress.get(carAudioContext))); synchronized (mLock) { if (mAddressToCarAudioDeviceInfo.size() == 0) { mStepSize = info.getStepValue(); } else { Preconditions.checkArgument( info.getStepValue() == mStepSize, "Gain controls within one group must have same step value"); } mAddressToCarAudioDeviceInfo.put(info.getAddress(), info); mContextToAddress.put(carAudioContext, info.getAddress()); if (info.getDefaultGain() > mDefaultGain) { // We're arbitrarily selecting the highest // device default gain as the group's default. mDefaultGain = info.getDefaultGain(); } if (info.getMaxGain() > mMaxGain) { mMaxGain = info.getMaxGain(); } if (info.getMinGain() < mMinGain) { mMinGain = info.getMinGain(); } updateCurrentGainIndexLocked(); } } /** * Update the user with the a new user * @param userId new user * @note also reloads the store gain index for the user */ private void updateUserIdLocked(@UserIdInt int userId) { mUserId = userId; mStoredGainIndex = getCurrentGainIndexForUserLocked(); } private int getCurrentGainIndexForUserLocked() { int gainIndexForUser = mSettingsManager.getStoredVolumeGainIndexForUser(mUserId, mZoneId, mId); Log.i(CarLog.TAG_AUDIO, "updateUserId userId " + mUserId + " gainIndexForUser " + gainIndexForUser); return gainIndexForUser; } /** * Update the current gain index based on the stored gain index */ private void updateCurrentGainIndexLocked() { synchronized (mLock) { if (mStoredGainIndex < getIndexForGainLocked(mMinGain) || mStoredGainIndex > getIndexForGainLocked(mMaxGain)) { // We expected to load a value from last boot, but if we didn't (perhaps this is the // first boot ever?), then use the highest "default" we've seen to initialize // ourselves. mCurrentGainIndex = getIndexForGainLocked(mDefaultGain); } else { // Just use the gain index we stored last time the gain was // set (presumably during our last boot cycle). mCurrentGainIndex = mStoredGainIndex; } } } private int getDefaultGainIndex() { synchronized (mLock) { return getIndexForGainLocked(mDefaultGain); } } int getMaxGainIndex() { synchronized (mLock) { return getIndexForGainLocked(mMaxGain); } } int getMinGainIndex() { synchronized (mLock) { return getIndexForGainLocked(mMinGain); } } int getCurrentGainIndex() { synchronized (mLock) { return mCurrentGainIndex; } } /** * Sets the gain on this group, gain will be set on all devices within volume group. * @param gainIndex The gain index */ void setCurrentGainIndex(int gainIndex) { synchronized (mLock) { int gainInMillibels = getGainForIndexLocked(gainIndex); Preconditions.checkArgument( gainInMillibels >= mMinGain && gainInMillibels <= mMaxGain, "Gain out of range (" + mMinGain + ":" + mMaxGain + ") " + gainInMillibels + "index " + gainIndex); for (String address : mAddressToCarAudioDeviceInfo.keySet()) { CarAudioDeviceInfo info = mAddressToCarAudioDeviceInfo.get(address); info.setCurrentGain(gainInMillibels); } mCurrentGainIndex = gainIndex; storeGainIndexForUserLocked(mCurrentGainIndex, mUserId); } } private void storeGainIndexForUserLocked(int gainIndex, @UserIdInt int userId) { mSettingsManager.storeVolumeGainIndexForUser(userId, mZoneId, mId, gainIndex); } // Given a group level gain index, return the computed gain in millibells // TODO (randolphs) If we ever want to add index to gain curves other than lock-stepped // linear, this would be the place to do it. private int getGainForIndexLocked(int gainIndex) { return mMinGain + gainIndex * mStepSize; } // TODO (randolphs) if we ever went to a non-linear index to gain curve mapping, we'd need to // revisit this as it assumes (at the least) that getGainForIndex is reversible. Luckily, // this is an internal implementation details we could factor out if/when necessary. private int getIndexForGainLocked(int gainInMillibel) { return (gainInMillibel - mMinGain) / mStepSize; } /** * Gets {@link AudioDevicePort} from a context number */ @Nullable AudioDevicePort getAudioDevicePortForContext(int carAudioContext) { final String address = mContextToAddress.get(carAudioContext); if (address == null || mAddressToCarAudioDeviceInfo.get(address) == null) { return null; } return mAddressToCarAudioDeviceInfo.get(address).getAudioDevicePort(); } @Override public String toString() { return "CarVolumeGroup id: " + mId + " currentGainIndex: " + mCurrentGainIndex + " contexts: " + Arrays.toString(getContexts()) + " addresses: " + String.join(", ", getAddresses()); } /** Writes to dumpsys output */ void dump(String indent, PrintWriter writer) { synchronized (mLock) { writer.printf("%sCarVolumeGroup(%d)\n", indent, mId); writer.printf("%sUserId(%d)\n", indent, mUserId); writer.printf("%sGain values (min / max / default/ current): %d %d %d %d\n", indent, mMinGain, mMaxGain, mDefaultGain, getGainForIndexLocked(mCurrentGainIndex)); writer.printf("%sGain indexes (min / max / default / current): %d %d %d %d\n", indent, getMinGainIndex(), getMaxGainIndex(), getDefaultGainIndex(), mCurrentGainIndex); for (int i = 0; i < mContextToAddress.size(); i++) { writer.printf("%sContext: %s -> Address: %s\n", indent, CarAudioContext.toString(mContextToAddress.keyAt(i)), mContextToAddress.valueAt(i)); } mAddressToCarAudioDeviceInfo.keySet().stream() .map(mAddressToCarAudioDeviceInfo::get) .forEach((info -> info.dump(indent, writer))); // Empty line for comfortable reading writer.println(); } } /** * Load volumes for new user * @param userId new user to load */ void loadVolumesForUser(@UserIdInt int userId) { synchronized (mLock) { //Update the volume for the new user updateUserIdLocked(userId); //Update the current gain index updateCurrentGainIndexLocked(); //Reset devices with current gain index } setCurrentGainIndex(getCurrentGainIndex()); } }