Sham Rathod | 1b6c1f0 | 2022-11-22 17:39:22 +0530 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (C) 2022 The Android Open Source Project |
| 3 | * |
| 4 | * Licensed under the Apache License, Version 2.0 (the "License"); |
| 5 | * you may not use this file except in compliance with the License. |
| 6 | * You may obtain a copy of the License at |
| 7 | * |
| 8 | * http://www.apache.org/licenses/LICENSE-2.0 |
| 9 | * |
| 10 | * Unless required by applicable law or agreed to in writing, software |
| 11 | * distributed under the License is distributed on an "AS IS" BASIS, |
| 12 | * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| 13 | * See the License for the specific language governing permissions and |
| 14 | * limitations under the License. |
| 15 | */ |
| 16 | |
Mikhail Naganov | 872d4a6 | 2023-03-09 18:19:01 -0800 | [diff] [blame] | 17 | #define LOG_TAG "VtsHalVolumeTest" |
| 18 | #include <android-base/logging.h> |
| 19 | |
Sham Rathod | 1b6c1f0 | 2022-11-22 17:39:22 +0530 | [diff] [blame] | 20 | #include "EffectHelper.h" |
| 21 | |
| 22 | using namespace android; |
| 23 | |
Sneha Patil | 93e4eb5 | 2024-02-13 14:09:48 +0530 | [diff] [blame] | 24 | using aidl::android::hardware::audio::common::getChannelCount; |
Sham Rathod | 1b6c1f0 | 2022-11-22 17:39:22 +0530 | [diff] [blame] | 25 | using aidl::android::hardware::audio::effect::Descriptor; |
Shunkai Yao | f8be1ac | 2023-03-06 18:41:27 +0000 | [diff] [blame] | 26 | using aidl::android::hardware::audio::effect::getEffectTypeUuidVolume; |
Sham Rathod | 1b6c1f0 | 2022-11-22 17:39:22 +0530 | [diff] [blame] | 27 | using aidl::android::hardware::audio::effect::IEffect; |
| 28 | using aidl::android::hardware::audio::effect::IFactory; |
Sham Rathod | 1b6c1f0 | 2022-11-22 17:39:22 +0530 | [diff] [blame] | 29 | using aidl::android::hardware::audio::effect::Parameter; |
| 30 | using aidl::android::hardware::audio::effect::Volume; |
Jaideep Sharma | 7449841 | 2023-09-13 15:25:25 +0530 | [diff] [blame] | 31 | using android::hardware::audio::common::testing::detail::TestExecutionTracer; |
Sham Rathod | 1b6c1f0 | 2022-11-22 17:39:22 +0530 | [diff] [blame] | 32 | |
Sneha Patil | 93e4eb5 | 2024-02-13 14:09:48 +0530 | [diff] [blame] | 33 | class VolumeControlHelper : public EffectHelper { |
| 34 | public: |
| 35 | void SetUpVolumeControl() { |
| 36 | ASSERT_NE(nullptr, mFactory); |
| 37 | ASSERT_NO_FATAL_FAILURE(create(mFactory, mEffect, mDescriptor)); |
| 38 | initFrameCount(); |
| 39 | Parameter::Specific specific = getDefaultParamSpecific(); |
| 40 | Parameter::Common common = EffectHelper::createParamCommon( |
| 41 | 0 /* session */, 1 /* ioHandle */, kSamplingFrequency /* iSampleRate */, |
| 42 | kSamplingFrequency /* oSampleRate */, mInputFrameCount /* iFrameCount */, |
| 43 | mInputFrameCount /* oFrameCount */); |
| 44 | ASSERT_NO_FATAL_FAILURE(open(mEffect, common, specific, &mOpenEffectReturn, EX_NONE)); |
| 45 | ASSERT_NE(nullptr, mEffect); |
| 46 | } |
| 47 | |
| 48 | void TearDownVolumeControl() { |
| 49 | ASSERT_NO_FATAL_FAILURE(close(mEffect)); |
| 50 | ASSERT_NO_FATAL_FAILURE(destroy(mFactory, mEffect)); |
| 51 | mOpenEffectReturn = IEffect::OpenEffectReturn{}; |
| 52 | } |
| 53 | |
| 54 | Parameter::Specific getDefaultParamSpecific() { |
| 55 | Volume vol = Volume::make<Volume::levelDb>(kMinLevel); |
| 56 | Parameter::Specific specific = Parameter::Specific::make<Parameter::Specific::volume>(vol); |
| 57 | return specific; |
| 58 | } |
| 59 | |
| 60 | Parameter createVolumeParam(int param, Volume::Tag volTag) { |
| 61 | return Parameter::make<Parameter::specific>( |
| 62 | Parameter::Specific::make<Parameter::Specific::volume>( |
| 63 | (volTag == Volume::mute) ? Volume::make<Volume::mute>(param) |
| 64 | : Volume::make<Volume::levelDb>(param))); |
| 65 | } |
| 66 | |
| 67 | void initFrameCount() { |
| 68 | int channelCount = getChannelCount( |
| 69 | AudioChannelLayout::make<AudioChannelLayout::layoutMask>(kDefaultChannelLayout)); |
| 70 | mInputFrameCount = kBufferSize / channelCount; |
| 71 | mOutputFrameCount = kBufferSize / channelCount; |
| 72 | } |
| 73 | |
| 74 | bool isLevelValid(int level) { |
| 75 | auto vol = Volume::make<Volume::levelDb>(level); |
| 76 | return isParameterValid<Volume, Range::volume>(vol, mDescriptor); |
| 77 | } |
| 78 | |
| 79 | void setAndVerifyParameters(Volume::Tag volTag, int param, binder_exception_t expected) { |
| 80 | auto expectedParam = createVolumeParam(param, volTag); |
| 81 | EXPECT_STATUS(expected, mEffect->setParameter(expectedParam)) << expectedParam.toString(); |
| 82 | |
| 83 | if (expected == EX_NONE) { |
| 84 | Volume::Id volId = Volume::Id::make<Volume::Id::commonTag>(volTag); |
| 85 | |
| 86 | auto id = Parameter::Id::make<Parameter::Id::volumeTag>(volId); |
| 87 | // get parameter |
| 88 | Parameter getParam; |
| 89 | // if set success, then get should match |
| 90 | EXPECT_STATUS(expected, mEffect->getParameter(id, &getParam)); |
| 91 | EXPECT_EQ(expectedParam, getParam) << "\nexpectedParam:" << expectedParam.toString() |
| 92 | << "\ngetParam:" << getParam.toString(); |
| 93 | } |
| 94 | } |
| 95 | |
| 96 | static constexpr int kSamplingFrequency = 44100; |
| 97 | static constexpr int kDurationMilliSec = 2000; |
| 98 | static constexpr int kBufferSize = kSamplingFrequency * kDurationMilliSec / 1000; |
| 99 | static constexpr int kMinLevel = -96; |
| 100 | static constexpr int kDefaultChannelLayout = AudioChannelLayout::LAYOUT_STEREO; |
| 101 | long mInputFrameCount, mOutputFrameCount; |
| 102 | std::shared_ptr<IFactory> mFactory; |
| 103 | std::shared_ptr<IEffect> mEffect; |
| 104 | IEffect::OpenEffectReturn mOpenEffectReturn; |
| 105 | Descriptor mDescriptor; |
| 106 | }; |
Sham Rathod | 1b6c1f0 | 2022-11-22 17:39:22 +0530 | [diff] [blame] | 107 | /** |
| 108 | * Here we focus on specific parameter checking, general IEffect interfaces testing performed in |
| 109 | * VtsAudioEffectTargetTest. |
| 110 | */ |
Sham Rathod | 85793d8 | 2022-12-22 19:09:10 +0530 | [diff] [blame] | 111 | enum ParamName { PARAM_INSTANCE_NAME, PARAM_LEVEL, PARAM_MUTE }; |
Sham Rathod | 1b6c1f0 | 2022-11-22 17:39:22 +0530 | [diff] [blame] | 112 | using VolumeParamTestParam = |
| 113 | std::tuple<std::pair<std::shared_ptr<IFactory>, Descriptor>, int, bool>; |
| 114 | |
Sneha Patil | 93e4eb5 | 2024-02-13 14:09:48 +0530 | [diff] [blame] | 115 | class VolumeParamTest : public ::testing::TestWithParam<VolumeParamTestParam>, |
| 116 | public VolumeControlHelper { |
Sham Rathod | 1b6c1f0 | 2022-11-22 17:39:22 +0530 | [diff] [blame] | 117 | public: |
| 118 | VolumeParamTest() |
Sham Rathod | 85793d8 | 2022-12-22 19:09:10 +0530 | [diff] [blame] | 119 | : mParamLevel(std::get<PARAM_LEVEL>(GetParam())), |
Sham Rathod | 1b6c1f0 | 2022-11-22 17:39:22 +0530 | [diff] [blame] | 120 | mParamMute(std::get<PARAM_MUTE>(GetParam())) { |
| 121 | std::tie(mFactory, mDescriptor) = std::get<PARAM_INSTANCE_NAME>(GetParam()); |
| 122 | } |
| 123 | |
Sneha Patil | 93e4eb5 | 2024-02-13 14:09:48 +0530 | [diff] [blame] | 124 | void SetUp() override { ASSERT_NO_FATAL_FAILURE(SetUpVolumeControl()); } |
| 125 | void TearDown() override { TearDownVolumeControl(); } |
Sham Rathod | 1b6c1f0 | 2022-11-22 17:39:22 +0530 | [diff] [blame] | 126 | |
Sham Rathod | 1b6c1f0 | 2022-11-22 17:39:22 +0530 | [diff] [blame] | 127 | int mParamLevel = 0; |
| 128 | bool mParamMute = false; |
Sham Rathod | 1b6c1f0 | 2022-11-22 17:39:22 +0530 | [diff] [blame] | 129 | }; |
| 130 | |
Sneha Patil | 93e4eb5 | 2024-02-13 14:09:48 +0530 | [diff] [blame] | 131 | TEST_P(VolumeParamTest, SetAndGetParams) { |
| 132 | ASSERT_NO_FATAL_FAILURE( |
| 133 | setAndVerifyParameters(Volume::levelDb, mParamLevel, |
| 134 | isLevelValid(mParamLevel) ? EX_NONE : EX_ILLEGAL_ARGUMENT)); |
| 135 | ASSERT_NO_FATAL_FAILURE(setAndVerifyParameters(Volume::mute, mParamMute, EX_NONE)); |
Sham Rathod | 1b6c1f0 | 2022-11-22 17:39:22 +0530 | [diff] [blame] | 136 | } |
| 137 | |
Sneha Patil | 93e4eb5 | 2024-02-13 14:09:48 +0530 | [diff] [blame] | 138 | using VolumeDataTestParam = std::pair<std::shared_ptr<IFactory>, Descriptor>; |
| 139 | |
| 140 | class VolumeDataTest : public ::testing::TestWithParam<VolumeDataTestParam>, |
| 141 | public VolumeControlHelper { |
| 142 | public: |
| 143 | VolumeDataTest() { |
| 144 | std::tie(mFactory, mDescriptor) = GetParam(); |
| 145 | mInput.resize(kBufferSize); |
| 146 | mInputMag.resize(mTestFrequencies.size()); |
| 147 | mBinOffsets.resize(mTestFrequencies.size()); |
| 148 | roundToFreqCenteredToFftBin(mTestFrequencies, mBinOffsets, kBinWidth); |
| 149 | generateMultiTone(mTestFrequencies, mInput, kSamplingFrequency); |
| 150 | mInputMag = calculateMagnitude(mInput, mBinOffsets, kNPointFFT); |
| 151 | } |
| 152 | |
| 153 | std::vector<int> calculatePercentageDiff(const std::vector<float>& outputMag) { |
| 154 | std::vector<int> percentages(mTestFrequencies.size()); |
| 155 | |
| 156 | for (size_t i = 0; i < mInputMag.size(); i++) { |
| 157 | float diff = mInputMag[i] - outputMag[i]; |
| 158 | percentages[i] = std::round(diff / mInputMag[i] * 100); |
| 159 | } |
| 160 | return percentages; |
| 161 | } |
| 162 | |
| 163 | // Convert Decibel value to Percentage |
| 164 | int percentageDb(float level) { return std::round((1 - (pow(10, level / 20))) * 100); } |
| 165 | |
| 166 | void SetUp() override { ASSERT_NO_FATAL_FAILURE(SetUpVolumeControl()); } |
| 167 | void TearDown() override { TearDownVolumeControl(); } |
| 168 | |
| 169 | static constexpr int kMaxAudioSample = 1; |
| 170 | static constexpr int kTransitionDuration = 300; |
| 171 | static constexpr int kNPointFFT = 32768; |
| 172 | static constexpr float kBinWidth = (float)kSamplingFrequency / kNPointFFT; |
| 173 | static constexpr size_t offset = kSamplingFrequency * kTransitionDuration / 1000; |
| 174 | static constexpr float kBaseLevel = 0; |
| 175 | std::vector<int> mTestFrequencies = {100, 1000}; |
| 176 | std::vector<float> mInput; |
| 177 | std::vector<float> mInputMag; |
| 178 | std::vector<int> mBinOffsets; |
| 179 | }; |
| 180 | |
| 181 | TEST_P(VolumeDataTest, ApplyLevelMuteUnmute) { |
| 182 | std::vector<float> output(kBufferSize); |
| 183 | std::vector<int> diffs(mTestFrequencies.size()); |
| 184 | std::vector<float> outputMag(mTestFrequencies.size()); |
| 185 | |
| 186 | if (!isLevelValid(kBaseLevel)) { |
| 187 | GTEST_SKIP() << "Volume Level not supported, skipping the test\n"; |
| 188 | } |
| 189 | |
| 190 | // Apply Volume Level |
| 191 | |
| 192 | ASSERT_NO_FATAL_FAILURE(setAndVerifyParameters(Volume::levelDb, kBaseLevel, EX_NONE)); |
| 193 | ASSERT_NO_FATAL_FAILURE(processAndWriteToOutput(mInput, output, mEffect, &mOpenEffectReturn)); |
| 194 | |
| 195 | outputMag = calculateMagnitude(output, mBinOffsets, kNPointFFT); |
| 196 | diffs = calculatePercentageDiff(outputMag); |
| 197 | |
| 198 | for (size_t i = 0; i < diffs.size(); i++) { |
| 199 | ASSERT_EQ(diffs[i], percentageDb(kBaseLevel)); |
| 200 | } |
| 201 | |
| 202 | // Apply Mute |
| 203 | |
| 204 | ASSERT_NO_FATAL_FAILURE(setAndVerifyParameters(Volume::mute, true /*mute*/, EX_NONE)); |
| 205 | ASSERT_NO_FATAL_FAILURE(processAndWriteToOutput(mInput, output, mEffect, &mOpenEffectReturn)); |
| 206 | |
| 207 | std::vector<float> subOutputMute(output.begin() + offset, output.end()); |
| 208 | outputMag = calculateMagnitude(subOutputMute, mBinOffsets, kNPointFFT); |
| 209 | diffs = calculatePercentageDiff(outputMag); |
| 210 | |
| 211 | for (size_t i = 0; i < diffs.size(); i++) { |
| 212 | ASSERT_EQ(diffs[i], percentageDb(kMinLevel /*Mute*/)); |
| 213 | } |
| 214 | |
| 215 | // Verifying Fade out |
| 216 | outputMag = calculateMagnitude(output, mBinOffsets, kNPointFFT); |
| 217 | diffs = calculatePercentageDiff(outputMag); |
| 218 | |
| 219 | for (size_t i = 0; i < diffs.size(); i++) { |
| 220 | ASSERT_LT(diffs[i], percentageDb(kMinLevel /*Mute*/)); |
| 221 | } |
| 222 | |
| 223 | // Apply Unmute |
| 224 | |
| 225 | ASSERT_NO_FATAL_FAILURE(setAndVerifyParameters(Volume::mute, false /*unmute*/, EX_NONE)); |
| 226 | ASSERT_NO_FATAL_FAILURE(processAndWriteToOutput(mInput, output, mEffect, &mOpenEffectReturn)); |
| 227 | |
| 228 | std::vector<float> subOutputUnmute(output.begin() + offset, output.end()); |
| 229 | |
| 230 | outputMag = calculateMagnitude(subOutputUnmute, mBinOffsets, kNPointFFT); |
| 231 | diffs = calculatePercentageDiff(outputMag); |
| 232 | |
| 233 | for (size_t i = 0; i < diffs.size(); i++) { |
| 234 | ASSERT_EQ(diffs[i], percentageDb(kBaseLevel)); |
| 235 | } |
| 236 | |
| 237 | // Verifying Fade in |
| 238 | outputMag = calculateMagnitude(output, mBinOffsets, kNPointFFT); |
| 239 | diffs = calculatePercentageDiff(outputMag); |
| 240 | |
| 241 | for (size_t i = 0; i < diffs.size(); i++) { |
| 242 | ASSERT_GT(diffs[i], percentageDb(kBaseLevel)); |
| 243 | } |
| 244 | } |
| 245 | |
| 246 | TEST_P(VolumeDataTest, DecreasingLevels) { |
| 247 | std::vector<int> decreasingLevels = {-24, -48, -96}; |
| 248 | std::vector<float> baseOutput(kBufferSize); |
| 249 | std::vector<int> baseDiffs(mTestFrequencies.size()); |
| 250 | std::vector<float> outputMag(mTestFrequencies.size()); |
| 251 | |
| 252 | if (!isLevelValid(kBaseLevel)) { |
| 253 | GTEST_SKIP() << "Volume Level not supported, skipping the test\n"; |
| 254 | } |
| 255 | |
| 256 | ASSERT_NO_FATAL_FAILURE(setAndVerifyParameters(Volume::levelDb, kBaseLevel, EX_NONE)); |
| 257 | ASSERT_NO_FATAL_FAILURE( |
| 258 | processAndWriteToOutput(mInput, baseOutput, mEffect, &mOpenEffectReturn)); |
| 259 | |
| 260 | outputMag = calculateMagnitude(baseOutput, mBinOffsets, kNPointFFT); |
| 261 | baseDiffs = calculatePercentageDiff(outputMag); |
| 262 | |
| 263 | for (int level : decreasingLevels) { |
| 264 | std::vector<float> output(kBufferSize); |
| 265 | std::vector<int> diffs(mTestFrequencies.size()); |
| 266 | |
| 267 | // Skipping the further steps for unnsupported level values |
| 268 | if (!isLevelValid(level)) { |
| 269 | continue; |
| 270 | } |
| 271 | ASSERT_NO_FATAL_FAILURE(setAndVerifyParameters(Volume::levelDb, level, EX_NONE)); |
| 272 | ASSERT_NO_FATAL_FAILURE( |
| 273 | processAndWriteToOutput(mInput, output, mEffect, &mOpenEffectReturn)); |
| 274 | |
| 275 | outputMag = calculateMagnitude(output, mBinOffsets, kNPointFFT); |
| 276 | diffs = calculatePercentageDiff(outputMag); |
| 277 | |
| 278 | // Decrease in volume level results in greater magnitude difference |
| 279 | for (size_t i = 0; i < diffs.size(); i++) { |
| 280 | ASSERT_GT(diffs[i], baseDiffs[i]); |
| 281 | } |
| 282 | |
| 283 | baseDiffs = diffs; |
| 284 | } |
Sham Rathod | 1b6c1f0 | 2022-11-22 17:39:22 +0530 | [diff] [blame] | 285 | } |
| 286 | |
Shunkai Yao | 0a0c45e | 2023-02-13 17:41:11 +0000 | [diff] [blame] | 287 | std::vector<std::pair<std::shared_ptr<IFactory>, Descriptor>> kDescPair; |
Sham Rathod | 1b6c1f0 | 2022-11-22 17:39:22 +0530 | [diff] [blame] | 288 | INSTANTIATE_TEST_SUITE_P( |
| 289 | VolumeTest, VolumeParamTest, |
Sham Rathod | 85793d8 | 2022-12-22 19:09:10 +0530 | [diff] [blame] | 290 | ::testing::Combine( |
Shunkai Yao | 0a0c45e | 2023-02-13 17:41:11 +0000 | [diff] [blame] | 291 | testing::ValuesIn(kDescPair = EffectFactoryHelper::getAllEffectDescriptors( |
Shunkai Yao | f8be1ac | 2023-03-06 18:41:27 +0000 | [diff] [blame] | 292 | IFactory::descriptor, getEffectTypeUuidVolume())), |
Shunkai Yao | 0a0c45e | 2023-02-13 17:41:11 +0000 | [diff] [blame] | 293 | testing::ValuesIn( |
| 294 | EffectHelper::getTestValueSet<Volume, int, Range::volume, Volume::levelDb>( |
| 295 | kDescPair, EffectHelper::expandTestValueBasic<int>)), |
Sham Rathod | 85793d8 | 2022-12-22 19:09:10 +0530 | [diff] [blame] | 296 | testing::Bool() /* mute */), |
Sham Rathod | 1b6c1f0 | 2022-11-22 17:39:22 +0530 | [diff] [blame] | 297 | [](const testing::TestParamInfo<VolumeParamTest::ParamType>& info) { |
| 298 | auto descriptor = std::get<PARAM_INSTANCE_NAME>(info.param).second; |
Sham Rathod | 85793d8 | 2022-12-22 19:09:10 +0530 | [diff] [blame] | 299 | std::string level = std::to_string(std::get<PARAM_LEVEL>(info.param)); |
Sham Rathod | 1b6c1f0 | 2022-11-22 17:39:22 +0530 | [diff] [blame] | 300 | std::string mute = std::to_string(std::get<PARAM_MUTE>(info.param)); |
Shunkai Yao | 9e60e63 | 2023-06-23 04:34:50 +0000 | [diff] [blame] | 301 | std::string name = getPrefix(descriptor) + "_level" + level + "_mute" + mute; |
Sham Rathod | 1b6c1f0 | 2022-11-22 17:39:22 +0530 | [diff] [blame] | 302 | std::replace_if( |
| 303 | name.begin(), name.end(), [](const char c) { return !std::isalnum(c); }, '_'); |
| 304 | return name; |
| 305 | }); |
| 306 | |
| 307 | GTEST_ALLOW_UNINSTANTIATED_PARAMETERIZED_TEST(VolumeParamTest); |
| 308 | |
Sneha Patil | 93e4eb5 | 2024-02-13 14:09:48 +0530 | [diff] [blame] | 309 | INSTANTIATE_TEST_SUITE_P(VolumeTest, VolumeDataTest, |
| 310 | testing::ValuesIn(EffectFactoryHelper::getAllEffectDescriptors( |
| 311 | IFactory::descriptor, getEffectTypeUuidVolume())), |
| 312 | [](const testing::TestParamInfo<VolumeDataTest::ParamType>& info) { |
| 313 | auto descriptor = info.param; |
| 314 | std::string name = getPrefix(descriptor.second); |
| 315 | std::replace_if( |
| 316 | name.begin(), name.end(), |
| 317 | [](const char c) { return !std::isalnum(c); }, '_'); |
| 318 | return name; |
| 319 | }); |
| 320 | |
| 321 | GTEST_ALLOW_UNINSTANTIATED_PARAMETERIZED_TEST(VolumeDataTest); |
| 322 | |
Sham Rathod | 1b6c1f0 | 2022-11-22 17:39:22 +0530 | [diff] [blame] | 323 | int main(int argc, char** argv) { |
| 324 | ::testing::InitGoogleTest(&argc, argv); |
Jaideep Sharma | 7449841 | 2023-09-13 15:25:25 +0530 | [diff] [blame] | 325 | ::testing::UnitTest::GetInstance()->listeners().Append(new TestExecutionTracer()); |
Sham Rathod | 1b6c1f0 | 2022-11-22 17:39:22 +0530 | [diff] [blame] | 326 | ABinderProcess_setThreadPoolMaxThreadCount(1); |
| 327 | ABinderProcess_startThreadPool(); |
| 328 | return RUN_ALL_TESTS(); |
| 329 | } |