| /* |
| * Copyright (C) 2019 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. |
| */ |
| |
| #include <map> |
| #include <set> |
| |
| #include <media/TypeConverter.h> |
| #include <system/audio.h> |
| #include <utils/Log.h> |
| #include <utils/String8.h> |
| |
| #include "AudioPolicyTestClient.h" |
| |
| namespace android { |
| |
| class AudioPolicyManagerTestClient : public AudioPolicyTestClient { |
| public: |
| // AudioPolicyClientInterface implementation |
| audio_module_handle_t loadHwModule(const char* name) override { |
| if (!mAllowedModuleNames.empty() && !mAllowedModuleNames.count(name)) { |
| return AUDIO_MODULE_HANDLE_NONE; |
| } |
| return mNextModuleHandle++; |
| } |
| |
| status_t openOutput(audio_module_handle_t module, |
| audio_io_handle_t *output, |
| audio_config_t *halConfig, |
| audio_config_base_t *mixerConfig, |
| const sp<DeviceDescriptorBase>& /*device*/, |
| uint32_t * /*latencyMs*/, |
| audio_output_flags_t flags, |
| audio_attributes_t /*attributes*/) override { |
| if (module >= mNextModuleHandle) { |
| ALOGE("%s: Module handle %d has not been allocated yet (next is %d)", |
| __func__, module, mNextModuleHandle); |
| return BAD_VALUE; |
| } |
| *output = mNextIoHandle++; |
| mOpenedOutputs[*output] = flags; |
| ALOGD("%s: opened output %d: HAL(%s %s %d) Mixer(%s %s %d) %s", __func__, *output, |
| audio_channel_out_mask_to_string(halConfig->channel_mask), |
| audio_format_to_string(halConfig->format), halConfig->sample_rate, |
| audio_channel_out_mask_to_string(mixerConfig->channel_mask), |
| audio_format_to_string(mixerConfig->format), mixerConfig->sample_rate, |
| android::toString(flags).c_str()); |
| return NO_ERROR; |
| } |
| |
| audio_io_handle_t openDuplicateOutput(audio_io_handle_t /*output1*/, |
| audio_io_handle_t /*output2*/) override { |
| audio_io_handle_t id = mNextIoHandle++; |
| return id; |
| } |
| |
| status_t closeOutput(audio_io_handle_t output) override { |
| if (auto iter = mOpenedOutputs.find(output); iter != mOpenedOutputs.end()) { |
| mOpenedOutputs.erase(iter); |
| return NO_ERROR; |
| } else { |
| ALOGE("%s: Unknown output %d", __func__, output); |
| return BAD_VALUE; |
| } |
| } |
| |
| status_t openInput(audio_module_handle_t module, |
| audio_io_handle_t *input, |
| audio_config_t * /*config*/, |
| audio_devices_t * /*device*/, |
| const String8 & /*address*/, |
| audio_source_t /*source*/, |
| audio_input_flags_t /*flags*/) override { |
| if (module >= mNextModuleHandle) { |
| ALOGE("%s: Module handle %d has not been allocated yet (next is %d)", |
| __func__, module, mNextModuleHandle); |
| return BAD_VALUE; |
| } |
| *input = mNextIoHandle++; |
| mOpenedInputs.insert(*input); |
| ALOGD("%s: opened input %d", __func__, *input); |
| return NO_ERROR; |
| } |
| |
| status_t closeInput(audio_io_handle_t input) override { |
| if (mOpenedInputs.erase(input) != 1) { |
| if (input >= mNextIoHandle) { |
| ALOGE("%s: I/O handle %d has not been allocated yet (next is %d)", |
| __func__, input, mNextIoHandle); |
| } else { |
| ALOGE("%s: Attempt to close input %d twice", __func__, input); |
| } |
| return BAD_VALUE; |
| } |
| ALOGD("%s: closed input %d", __func__, input); |
| return NO_ERROR; |
| } |
| |
| status_t createAudioPatch(const struct audio_patch *patch, |
| audio_patch_handle_t *handle, |
| int /*delayMs*/) override { |
| auto iter = mActivePatches.find(*handle); |
| if (iter != mActivePatches.end()) { |
| mActivePatches.erase(*handle); |
| } |
| *handle = mNextPatchHandle++; |
| mActivePatches.insert(std::make_pair(*handle, *patch)); |
| return NO_ERROR; |
| } |
| |
| status_t releaseAudioPatch(audio_patch_handle_t handle, |
| int /*delayMs*/) override { |
| if (mActivePatches.erase(handle) != 1) { |
| if (handle >= mNextPatchHandle) { |
| ALOGE("%s: Patch handle %d has not been allocated yet (next is %d)", |
| __func__, handle, mNextPatchHandle); |
| } else { |
| ALOGE("%s: Attempt to release patch %d twice", __func__, handle); |
| } |
| return BAD_VALUE; |
| } |
| return NO_ERROR; |
| } |
| |
| void onAudioPortListUpdate() override { |
| ++mAudioPortListUpdateCount; |
| } |
| |
| status_t setDeviceConnectedState(const struct audio_port_v7 *port, |
| media::DeviceConnectedState state) override { |
| if (state == media::DeviceConnectedState::CONNECTED) { |
| mConnectedDevicePorts.push_back(*port); |
| } else if (state == media::DeviceConnectedState::DISCONNECTED){ |
| mDisconnectedDevicePorts.push_back(*port); |
| } |
| return NO_ERROR; |
| } |
| |
| // Helper methods for tests |
| size_t getActivePatchesCount() const { return mActivePatches.size(); } |
| |
| const struct audio_patch *getLastAddedPatch() const { |
| if (mActivePatches.empty()) { |
| return nullptr; |
| } |
| auto it = --mActivePatches.end(); |
| return &it->second; |
| }; |
| |
| size_t getOpenedInputsCount() const { return mOpenedInputs.size(); } |
| |
| audio_module_handle_t peekNextModuleHandle() const { return mNextModuleHandle; } |
| |
| void swapAllowedModuleNames(std::set<std::string>&& names = {}) { |
| mAllowedModuleNames.swap(names); |
| } |
| |
| size_t getAudioPortListUpdateCount() const { return mAudioPortListUpdateCount; } |
| |
| void onRoutingUpdated() override { |
| mRoutingUpdatedUpdateCount++; |
| } |
| |
| void resetRoutingUpdatedCounter() { |
| mRoutingUpdatedUpdateCount = 0; |
| } |
| |
| size_t getRoutingUpdatedCounter() const { |
| return mRoutingUpdatedUpdateCount; |
| } |
| |
| void onVolumeRangeInitRequest() override { |
| |
| } |
| |
| status_t updateSecondaryOutputs( |
| const TrackSecondaryOutputsMap& trackSecondaryOutputs __unused) override { |
| return NO_ERROR; |
| } |
| |
| size_t getConnectedDevicePortCount() const { |
| return mConnectedDevicePorts.size(); |
| } |
| |
| const struct audio_port_v7 *getLastConnectedDevicePort() const { |
| if (mConnectedDevicePorts.empty()) { |
| return nullptr; |
| } |
| auto it = --mConnectedDevicePorts.end(); |
| return &(*it); |
| } |
| |
| size_t getDisconnectedDevicePortCount() const { |
| return mDisconnectedDevicePorts.size(); |
| } |
| |
| const struct audio_port_v7 *getLastDisconnectedDevicePort() const { |
| if (mDisconnectedDevicePorts.empty()) { |
| return nullptr; |
| } |
| auto it = --mDisconnectedDevicePorts.end(); |
| return &(*it); |
| } |
| |
| String8 getParameters(audio_io_handle_t /* ioHandle */, const String8& /* keys*/ ) override { |
| AudioParameter mAudioParameters; |
| std::string formats; |
| for (const auto& f : mSupportedFormats) { |
| if (!formats.empty()) formats += AUDIO_PARAMETER_VALUE_LIST_SEPARATOR; |
| formats += audio_format_to_string(f); |
| } |
| mAudioParameters.add( |
| String8(AudioParameter::keyStreamSupportedFormats), |
| String8(formats.c_str())); |
| mAudioParameters.addInt(String8(AudioParameter::keyStreamSupportedSamplingRates), 48000); |
| std::string channelMasks; |
| for (const auto& cm : mSupportedChannelMasks) { |
| if (!audio_channel_mask_is_valid(cm)) { |
| continue; |
| } |
| if (!channelMasks.empty()) channelMasks += AUDIO_PARAMETER_VALUE_LIST_SEPARATOR; |
| channelMasks += audio_channel_mask_to_string(cm); |
| } |
| mAudioParameters.add( |
| String8(AudioParameter::keyStreamSupportedChannels), String8(channelMasks.c_str())); |
| return mAudioParameters.toString(); |
| } |
| |
| status_t getAudioMixPort(const struct audio_port_v7 *devicePort __unused, |
| struct audio_port_v7 *mixPort) override { |
| mixPort->num_audio_profiles = 0; |
| for (auto format : mSupportedFormats) { |
| const int i = mixPort->num_audio_profiles; |
| mixPort->audio_profiles[i].format = format; |
| mixPort->audio_profiles[i].num_sample_rates = 1; |
| mixPort->audio_profiles[i].sample_rates[0] = 48000; |
| mixPort->audio_profiles[i].num_channel_masks = 0; |
| for (const auto& cm : mSupportedChannelMasks) { |
| if (audio_channel_mask_is_valid(cm)) { |
| mixPort->audio_profiles[i].channel_masks[ |
| mixPort->audio_profiles[i].num_channel_masks++] = cm; |
| } |
| } |
| mixPort->num_audio_profiles++; |
| } |
| return NO_ERROR; |
| } |
| |
| status_t setTracksInternalMute( |
| const std::vector<media::TrackInternalMuteInfo>& tracksInternalMute) override { |
| for (const auto& trackInternalMute : tracksInternalMute) { |
| mTracksInternalMute[(audio_port_handle_t)trackInternalMute.portId] = |
| trackInternalMute.muted; |
| } |
| return NO_ERROR; |
| } |
| |
| void addSupportedFormat(audio_format_t format) { |
| mSupportedFormats.insert(format); |
| } |
| |
| void addSupportedChannelMask(audio_channel_mask_t channelMask) { |
| mSupportedChannelMasks.insert(channelMask); |
| } |
| |
| bool getTrackInternalMute(audio_port_handle_t portId) { |
| auto it = mTracksInternalMute.find(portId); |
| return it == mTracksInternalMute.end() ? false : it->second; |
| } |
| |
| std::optional<audio_output_flags_t> getOpenOutputFlags(audio_io_handle_t output) const { |
| if (auto iter = mOpenedOutputs.find(output); iter != mOpenedOutputs.end()) { |
| return iter->second; |
| } |
| return std::nullopt; |
| } |
| |
| private: |
| audio_module_handle_t mNextModuleHandle = AUDIO_MODULE_HANDLE_NONE + 1; |
| audio_io_handle_t mNextIoHandle = AUDIO_IO_HANDLE_NONE + 1; |
| audio_patch_handle_t mNextPatchHandle = AUDIO_PATCH_HANDLE_NONE + 1; |
| std::map<audio_patch_handle_t, struct audio_patch> mActivePatches; |
| std::set<std::string> mAllowedModuleNames; |
| size_t mAudioPortListUpdateCount = 0; |
| size_t mRoutingUpdatedUpdateCount = 0; |
| std::vector<struct audio_port_v7> mConnectedDevicePorts; |
| std::vector<struct audio_port_v7> mDisconnectedDevicePorts; |
| std::set<audio_format_t> mSupportedFormats; |
| std::set<audio_channel_mask_t> mSupportedChannelMasks; |
| std::map<audio_port_handle_t, bool> mTracksInternalMute; |
| std::set<audio_io_handle_t> mOpenedInputs; |
| std::map<audio_io_handle_t, audio_output_flags_t> mOpenedOutputs; |
| }; |
| |
| } // namespace android |