blob: 0b68b02789ece85130cd823706f87318b4404826 [file] [log] [blame]
Sham Rathod1b6c1f02022-11-22 17:39:22 +05301/*
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 Naganov872d4a62023-03-09 18:19:01 -080017#define LOG_TAG "VtsHalVolumeTest"
18#include <android-base/logging.h>
19
Sham Rathod1b6c1f02022-11-22 17:39:22 +053020#include "EffectHelper.h"
21
22using namespace android;
23
Sneha Patil93e4eb52024-02-13 14:09:48 +053024using aidl::android::hardware::audio::common::getChannelCount;
Sham Rathod1b6c1f02022-11-22 17:39:22 +053025using aidl::android::hardware::audio::effect::Descriptor;
Shunkai Yaof8be1ac2023-03-06 18:41:27 +000026using aidl::android::hardware::audio::effect::getEffectTypeUuidVolume;
Sham Rathod1b6c1f02022-11-22 17:39:22 +053027using aidl::android::hardware::audio::effect::IEffect;
28using aidl::android::hardware::audio::effect::IFactory;
Sham Rathod1b6c1f02022-11-22 17:39:22 +053029using aidl::android::hardware::audio::effect::Parameter;
30using aidl::android::hardware::audio::effect::Volume;
Jaideep Sharma74498412023-09-13 15:25:25 +053031using android::hardware::audio::common::testing::detail::TestExecutionTracer;
Sham Rathod1b6c1f02022-11-22 17:39:22 +053032
Sneha Patil93e4eb52024-02-13 14:09:48 +053033class 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();
Shunkai Yao61f9dd22024-05-08 22:34:36 +000040 Parameter::Common common = createParamCommon(
Sneha Patil93e4eb52024-02-13 14:09:48 +053041 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(
Aayush Sonifbb5cd32024-11-21 10:51:46 +000069 AudioChannelLayout::make<AudioChannelLayout::layoutMask>(mChannelLayout));
Sneha Patil93e4eb52024-02-13 14:09:48 +053070 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
Aayush Sonifbb5cd32024-11-21 10:51:46 +000096 static constexpr int kDurationMilliSec = 1440;
Sneha Patil93e4eb52024-02-13 14:09:48 +053097 static constexpr int kBufferSize = kSamplingFrequency * kDurationMilliSec / 1000;
98 static constexpr int kMinLevel = -96;
Aayush Sonifbb5cd32024-11-21 10:51:46 +000099 static constexpr int mChannelLayout = kDefaultChannelLayout;
Sneha Patil93e4eb52024-02-13 14:09:48 +0530100 long mInputFrameCount, mOutputFrameCount;
101 std::shared_ptr<IFactory> mFactory;
102 std::shared_ptr<IEffect> mEffect;
103 IEffect::OpenEffectReturn mOpenEffectReturn;
104 Descriptor mDescriptor;
105};
Sham Rathod1b6c1f02022-11-22 17:39:22 +0530106/**
107 * Here we focus on specific parameter checking, general IEffect interfaces testing performed in
108 * VtsAudioEffectTargetTest.
109 */
Sneha Patil6a717472024-10-24 11:07:37 +0000110enum VolumeLevelParamName { PARAM_INSTANCE_NAME, PARAM_LEVEL };
111using VolumeLevelTestParam = std::tuple<std::pair<std::shared_ptr<IFactory>, Descriptor>, int>;
Sham Rathod1b6c1f02022-11-22 17:39:22 +0530112
Sneha Patil6a717472024-10-24 11:07:37 +0000113class VolumeLevelParamTest : public ::testing::TestWithParam<VolumeLevelTestParam>,
114 public VolumeControlHelper {
Sham Rathod1b6c1f02022-11-22 17:39:22 +0530115 public:
Sneha Patil6a717472024-10-24 11:07:37 +0000116 VolumeLevelParamTest() : mParamLevel(std::get<PARAM_LEVEL>(GetParam())) {
Sham Rathod1b6c1f02022-11-22 17:39:22 +0530117 std::tie(mFactory, mDescriptor) = std::get<PARAM_INSTANCE_NAME>(GetParam());
118 }
119
Sneha Patil93e4eb52024-02-13 14:09:48 +0530120 void SetUp() override { ASSERT_NO_FATAL_FAILURE(SetUpVolumeControl()); }
121 void TearDown() override { TearDownVolumeControl(); }
Sham Rathod1b6c1f02022-11-22 17:39:22 +0530122
Sham Rathod1b6c1f02022-11-22 17:39:22 +0530123 int mParamLevel = 0;
Sham Rathod1b6c1f02022-11-22 17:39:22 +0530124};
125
Sneha Patil6a717472024-10-24 11:07:37 +0000126TEST_P(VolumeLevelParamTest, SetAndGetParams) {
Sneha Patil93e4eb52024-02-13 14:09:48 +0530127 ASSERT_NO_FATAL_FAILURE(
128 setAndVerifyParameters(Volume::levelDb, mParamLevel,
129 isLevelValid(mParamLevel) ? EX_NONE : EX_ILLEGAL_ARGUMENT));
Sneha Patil6a717472024-10-24 11:07:37 +0000130}
131
132enum VolumeMuteParamName { MUTE_PARAM_INSTANCE_NAME, PARAM_MUTE };
133using VolumeMuteTestParam = std::tuple<std::pair<std::shared_ptr<IFactory>, Descriptor>, bool>;
134
135class VolumeMuteParamTest : public ::testing::TestWithParam<VolumeMuteTestParam>,
136 public VolumeControlHelper {
137 public:
138 VolumeMuteParamTest() : mParamMute(std::get<PARAM_MUTE>(GetParam())) {
139 std::tie(mFactory, mDescriptor) = std::get<MUTE_PARAM_INSTANCE_NAME>(GetParam());
140 }
141
142 void SetUp() override { ASSERT_NO_FATAL_FAILURE(SetUpVolumeControl()); }
143 void TearDown() override { TearDownVolumeControl(); }
144
145 bool mParamMute = false;
146};
147
148TEST_P(VolumeMuteParamTest, SetAndGetParams) {
Sneha Patil93e4eb52024-02-13 14:09:48 +0530149 ASSERT_NO_FATAL_FAILURE(setAndVerifyParameters(Volume::mute, mParamMute, EX_NONE));
Sham Rathod1b6c1f02022-11-22 17:39:22 +0530150}
151
Sneha Patil93e4eb52024-02-13 14:09:48 +0530152using VolumeDataTestParam = std::pair<std::shared_ptr<IFactory>, Descriptor>;
153
154class VolumeDataTest : public ::testing::TestWithParam<VolumeDataTestParam>,
155 public VolumeControlHelper {
156 public:
Shunkai Yao40d9d9e2024-03-25 17:52:34 +0000157 VolumeDataTest()
158 : kVsrApiLevel(
159 android::base::GetIntProperty("ro.vendor.api_level", __ANDROID_API_FUTURE__)) {
Sneha Patil93e4eb52024-02-13 14:09:48 +0530160 std::tie(mFactory, mDescriptor) = GetParam();
161 mInput.resize(kBufferSize);
162 mInputMag.resize(mTestFrequencies.size());
163 mBinOffsets.resize(mTestFrequencies.size());
164 roundToFreqCenteredToFftBin(mTestFrequencies, mBinOffsets, kBinWidth);
Sneha Patil93e4eb52024-02-13 14:09:48 +0530165 }
166
167 std::vector<int> calculatePercentageDiff(const std::vector<float>& outputMag) {
168 std::vector<int> percentages(mTestFrequencies.size());
169
170 for (size_t i = 0; i < mInputMag.size(); i++) {
171 float diff = mInputMag[i] - outputMag[i];
172 percentages[i] = std::round(diff / mInputMag[i] * 100);
173 }
174 return percentages;
175 }
176
177 // Convert Decibel value to Percentage
178 int percentageDb(float level) { return std::round((1 - (pow(10, level / 20))) * 100); }
179
Shunkai Yaof0cb5ec2024-03-14 01:54:58 +0000180 void SetUp() override {
181 SKIP_TEST_IF_DATA_UNSUPPORTED(mDescriptor.common.flags);
Shunkai Yao40d9d9e2024-03-25 17:52:34 +0000182 // Skips test fixture if api_level <= 34 (__ANDROID_API_U__).
183 if (kVsrApiLevel <= __ANDROID_API_U__) GTEST_SKIP();
Aayush Sonifbb5cd32024-11-21 10:51:46 +0000184 ASSERT_NO_FATAL_FAILURE(generateSineWave(mTestFrequencies, mInput, 1.0, kSamplingFrequency,
185 mChannelLayout));
186 ASSERT_NO_FATAL_FAILURE(
187 calculateAndVerifyMagnitude(mInputMag, mChannelLayout, mInput, mBinOffsets));
188
Shunkai Yaof0cb5ec2024-03-14 01:54:58 +0000189 ASSERT_NO_FATAL_FAILURE(SetUpVolumeControl());
190 }
191 void TearDown() override {
192 SKIP_TEST_IF_DATA_UNSUPPORTED(mDescriptor.common.flags);
Shunkai Yao40d9d9e2024-03-25 17:52:34 +0000193 if (kVsrApiLevel <= __ANDROID_API_U__) GTEST_SKIP();
Shunkai Yaof0cb5ec2024-03-14 01:54:58 +0000194 TearDownVolumeControl();
195 }
Sneha Patil93e4eb52024-02-13 14:09:48 +0530196
Shunkai Yao40d9d9e2024-03-25 17:52:34 +0000197 const int kVsrApiLevel;
Sneha Patil93e4eb52024-02-13 14:09:48 +0530198 static constexpr int kMaxAudioSample = 1;
199 static constexpr int kTransitionDuration = 300;
Sneha Patil93e4eb52024-02-13 14:09:48 +0530200 static constexpr float kBinWidth = (float)kSamplingFrequency / kNPointFFT;
201 static constexpr size_t offset = kSamplingFrequency * kTransitionDuration / 1000;
202 static constexpr float kBaseLevel = 0;
203 std::vector<int> mTestFrequencies = {100, 1000};
204 std::vector<float> mInput;
205 std::vector<float> mInputMag;
206 std::vector<int> mBinOffsets;
207};
208
209TEST_P(VolumeDataTest, ApplyLevelMuteUnmute) {
210 std::vector<float> output(kBufferSize);
211 std::vector<int> diffs(mTestFrequencies.size());
212 std::vector<float> outputMag(mTestFrequencies.size());
213
214 if (!isLevelValid(kBaseLevel)) {
215 GTEST_SKIP() << "Volume Level not supported, skipping the test\n";
216 }
217
218 // Apply Volume Level
219
220 ASSERT_NO_FATAL_FAILURE(setAndVerifyParameters(Volume::levelDb, kBaseLevel, EX_NONE));
221 ASSERT_NO_FATAL_FAILURE(processAndWriteToOutput(mInput, output, mEffect, &mOpenEffectReturn));
222
Aayush Sonifbb5cd32024-11-21 10:51:46 +0000223 ASSERT_NO_FATAL_FAILURE(
224 calculateAndVerifyMagnitude(outputMag, mChannelLayout, output, mBinOffsets));
225
Sneha Patil93e4eb52024-02-13 14:09:48 +0530226 diffs = calculatePercentageDiff(outputMag);
227
228 for (size_t i = 0; i < diffs.size(); i++) {
229 ASSERT_EQ(diffs[i], percentageDb(kBaseLevel));
230 }
231
232 // Apply Mute
233
234 ASSERT_NO_FATAL_FAILURE(setAndVerifyParameters(Volume::mute, true /*mute*/, EX_NONE));
235 ASSERT_NO_FATAL_FAILURE(processAndWriteToOutput(mInput, output, mEffect, &mOpenEffectReturn));
236
237 std::vector<float> subOutputMute(output.begin() + offset, output.end());
Aayush Sonifbb5cd32024-11-21 10:51:46 +0000238
239 ASSERT_NO_FATAL_FAILURE(
240 calculateAndVerifyMagnitude(outputMag, mChannelLayout, subOutputMute, mBinOffsets));
241
Sneha Patil93e4eb52024-02-13 14:09:48 +0530242 diffs = calculatePercentageDiff(outputMag);
243
244 for (size_t i = 0; i < diffs.size(); i++) {
245 ASSERT_EQ(diffs[i], percentageDb(kMinLevel /*Mute*/));
246 }
247
248 // Verifying Fade out
Aayush Sonifbb5cd32024-11-21 10:51:46 +0000249 ASSERT_NO_FATAL_FAILURE(
250 calculateAndVerifyMagnitude(outputMag, mChannelLayout, output, mBinOffsets));
251
Sneha Patil93e4eb52024-02-13 14:09:48 +0530252 diffs = calculatePercentageDiff(outputMag);
253
254 for (size_t i = 0; i < diffs.size(); i++) {
255 ASSERT_LT(diffs[i], percentageDb(kMinLevel /*Mute*/));
256 }
257
258 // Apply Unmute
259
260 ASSERT_NO_FATAL_FAILURE(setAndVerifyParameters(Volume::mute, false /*unmute*/, EX_NONE));
261 ASSERT_NO_FATAL_FAILURE(processAndWriteToOutput(mInput, output, mEffect, &mOpenEffectReturn));
262
263 std::vector<float> subOutputUnmute(output.begin() + offset, output.end());
264
Aayush Sonifbb5cd32024-11-21 10:51:46 +0000265 ASSERT_NO_FATAL_FAILURE(
266 calculateAndVerifyMagnitude(outputMag, mChannelLayout, subOutputUnmute, mBinOffsets));
267
Sneha Patil93e4eb52024-02-13 14:09:48 +0530268 diffs = calculatePercentageDiff(outputMag);
269
270 for (size_t i = 0; i < diffs.size(); i++) {
271 ASSERT_EQ(diffs[i], percentageDb(kBaseLevel));
272 }
273
274 // Verifying Fade in
Aayush Sonifbb5cd32024-11-21 10:51:46 +0000275 ASSERT_NO_FATAL_FAILURE(
276 calculateAndVerifyMagnitude(outputMag, mChannelLayout, output, mBinOffsets));
277
Sneha Patil93e4eb52024-02-13 14:09:48 +0530278 diffs = calculatePercentageDiff(outputMag);
279
280 for (size_t i = 0; i < diffs.size(); i++) {
281 ASSERT_GT(diffs[i], percentageDb(kBaseLevel));
282 }
283}
284
285TEST_P(VolumeDataTest, DecreasingLevels) {
286 std::vector<int> decreasingLevels = {-24, -48, -96};
287 std::vector<float> baseOutput(kBufferSize);
288 std::vector<int> baseDiffs(mTestFrequencies.size());
289 std::vector<float> outputMag(mTestFrequencies.size());
290
291 if (!isLevelValid(kBaseLevel)) {
292 GTEST_SKIP() << "Volume Level not supported, skipping the test\n";
293 }
294
295 ASSERT_NO_FATAL_FAILURE(setAndVerifyParameters(Volume::levelDb, kBaseLevel, EX_NONE));
296 ASSERT_NO_FATAL_FAILURE(
297 processAndWriteToOutput(mInput, baseOutput, mEffect, &mOpenEffectReturn));
298
Aayush Sonifbb5cd32024-11-21 10:51:46 +0000299 ASSERT_NO_FATAL_FAILURE(
300 calculateAndVerifyMagnitude(outputMag, mChannelLayout, baseOutput, mBinOffsets));
301
Sneha Patil93e4eb52024-02-13 14:09:48 +0530302 baseDiffs = calculatePercentageDiff(outputMag);
303
304 for (int level : decreasingLevels) {
305 std::vector<float> output(kBufferSize);
306 std::vector<int> diffs(mTestFrequencies.size());
307
308 // Skipping the further steps for unnsupported level values
309 if (!isLevelValid(level)) {
310 continue;
311 }
312 ASSERT_NO_FATAL_FAILURE(setAndVerifyParameters(Volume::levelDb, level, EX_NONE));
313 ASSERT_NO_FATAL_FAILURE(
314 processAndWriteToOutput(mInput, output, mEffect, &mOpenEffectReturn));
315
Aayush Sonifbb5cd32024-11-21 10:51:46 +0000316 ASSERT_NO_FATAL_FAILURE(
317 calculateAndVerifyMagnitude(outputMag, mChannelLayout, output, mBinOffsets));
318
Sneha Patil93e4eb52024-02-13 14:09:48 +0530319 diffs = calculatePercentageDiff(outputMag);
320
321 // Decrease in volume level results in greater magnitude difference
322 for (size_t i = 0; i < diffs.size(); i++) {
323 ASSERT_GT(diffs[i], baseDiffs[i]);
324 }
325
326 baseDiffs = diffs;
327 }
Sham Rathod1b6c1f02022-11-22 17:39:22 +0530328}
329
Shunkai Yao0a0c45e2023-02-13 17:41:11 +0000330std::vector<std::pair<std::shared_ptr<IFactory>, Descriptor>> kDescPair;
Sham Rathod1b6c1f02022-11-22 17:39:22 +0530331INSTANTIATE_TEST_SUITE_P(
Sneha Patil6a717472024-10-24 11:07:37 +0000332 VolumeTest, VolumeLevelParamTest,
Sham Rathod85793d82022-12-22 19:09:10 +0530333 ::testing::Combine(
Shunkai Yao0a0c45e2023-02-13 17:41:11 +0000334 testing::ValuesIn(kDescPair = EffectFactoryHelper::getAllEffectDescriptors(
Shunkai Yaof8be1ac2023-03-06 18:41:27 +0000335 IFactory::descriptor, getEffectTypeUuidVolume())),
Shunkai Yao0a0c45e2023-02-13 17:41:11 +0000336 testing::ValuesIn(
337 EffectHelper::getTestValueSet<Volume, int, Range::volume, Volume::levelDb>(
Sneha Patil6a717472024-10-24 11:07:37 +0000338 kDescPair, EffectHelper::expandTestValueBasic<int>))),
339 [](const testing::TestParamInfo<VolumeLevelParamTest::ParamType>& info) {
Sham Rathod1b6c1f02022-11-22 17:39:22 +0530340 auto descriptor = std::get<PARAM_INSTANCE_NAME>(info.param).second;
Sham Rathod85793d82022-12-22 19:09:10 +0530341 std::string level = std::to_string(std::get<PARAM_LEVEL>(info.param));
Sneha Patil6a717472024-10-24 11:07:37 +0000342 std::string name = getPrefix(descriptor) + "_level" + level;
Sham Rathod1b6c1f02022-11-22 17:39:22 +0530343 std::replace_if(
344 name.begin(), name.end(), [](const char c) { return !std::isalnum(c); }, '_');
345 return name;
346 });
347
Sneha Patil6a717472024-10-24 11:07:37 +0000348GTEST_ALLOW_UNINSTANTIATED_PARAMETERIZED_TEST(VolumeLevelParamTest);
349
350INSTANTIATE_TEST_SUITE_P(
351 VolumeTest, VolumeMuteParamTest,
352 ::testing::Combine(
353 testing::ValuesIn(kDescPair = EffectFactoryHelper::getAllEffectDescriptors(
354 IFactory::descriptor, getEffectTypeUuidVolume())),
355 testing::Bool() /* mute */),
356 [](const testing::TestParamInfo<VolumeMuteParamTest::ParamType>& info) {
357 auto descriptor = std::get<MUTE_PARAM_INSTANCE_NAME>(info.param).second;
358 std::string mute = std::to_string(std::get<PARAM_MUTE>(info.param));
359 std::string name = getPrefix(descriptor) + "_mute" + mute;
360 std::replace_if(
361 name.begin(), name.end(), [](const char c) { return !std::isalnum(c); }, '_');
362 return name;
363 });
364
365GTEST_ALLOW_UNINSTANTIATED_PARAMETERIZED_TEST(VolumeMuteParamTest);
Sham Rathod1b6c1f02022-11-22 17:39:22 +0530366
Sneha Patil93e4eb52024-02-13 14:09:48 +0530367INSTANTIATE_TEST_SUITE_P(VolumeTest, VolumeDataTest,
368 testing::ValuesIn(EffectFactoryHelper::getAllEffectDescriptors(
369 IFactory::descriptor, getEffectTypeUuidVolume())),
370 [](const testing::TestParamInfo<VolumeDataTest::ParamType>& info) {
371 auto descriptor = info.param;
372 std::string name = getPrefix(descriptor.second);
373 std::replace_if(
374 name.begin(), name.end(),
375 [](const char c) { return !std::isalnum(c); }, '_');
376 return name;
377 });
378
379GTEST_ALLOW_UNINSTANTIATED_PARAMETERIZED_TEST(VolumeDataTest);
380
Sham Rathod1b6c1f02022-11-22 17:39:22 +0530381int main(int argc, char** argv) {
382 ::testing::InitGoogleTest(&argc, argv);
Jaideep Sharma74498412023-09-13 15:25:25 +0530383 ::testing::UnitTest::GetInstance()->listeners().Append(new TestExecutionTracer());
Sham Rathod1b6c1f02022-11-22 17:39:22 +0530384 ABinderProcess_setThreadPoolMaxThreadCount(1);
385 ABinderProcess_startThreadPool();
386 return RUN_ALL_TESTS();
387}