blob: abc5a915a6f41d00af855ae9880488e818ac10c5 [file] [log] [blame]
Shraddha Basantwanif627d802022-11-08 14:45:07 +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
Sneha Patile5849232023-12-01 17:52:28 +053017#include <limits.h>
Mikhail Naganov872d4a62023-03-09 18:19:01 -080018
Sneha Patile5849232023-12-01 17:52:28 +053019#define LOG_TAG "VtsHalBassBoostTest"
20#include <aidl/Vintf.h>
21#include <android-base/logging.h>
Shraddha Basantwanif627d802022-11-08 14:45:07 +053022#include "EffectHelper.h"
Sneha Patile5849232023-12-01 17:52:28 +053023#include "pffft.hpp"
Shraddha Basantwanif627d802022-11-08 14:45:07 +053024
25using namespace android;
26
Sneha Patile5849232023-12-01 17:52:28 +053027using aidl::android::hardware::audio::common::getChannelCount;
Shraddha Basantwanif627d802022-11-08 14:45:07 +053028using aidl::android::hardware::audio::effect::BassBoost;
29using aidl::android::hardware::audio::effect::Capability;
30using aidl::android::hardware::audio::effect::Descriptor;
Shunkai Yaof8be1ac2023-03-06 18:41:27 +000031using aidl::android::hardware::audio::effect::getEffectTypeUuidBassBoost;
Shraddha Basantwanif627d802022-11-08 14:45:07 +053032using aidl::android::hardware::audio::effect::IEffect;
33using aidl::android::hardware::audio::effect::IFactory;
Shraddha Basantwanif627d802022-11-08 14:45:07 +053034using aidl::android::hardware::audio::effect::Parameter;
Shunkai Yao0a0c45e2023-02-13 17:41:11 +000035using aidl::android::hardware::audio::effect::Range;
Jaideep Sharma74498412023-09-13 15:25:25 +053036using android::hardware::audio::common::testing::detail::TestExecutionTracer;
Shraddha Basantwanif627d802022-11-08 14:45:07 +053037
Sneha Patile5849232023-12-01 17:52:28 +053038// minimal HAL interface version to run bassboost data path test
39constexpr int32_t kMinDataTestHalVersion = 2;
40static const std::vector<int32_t> kLayouts = {AudioChannelLayout::LAYOUT_STEREO,
41 AudioChannelLayout::LAYOUT_MONO};
Shraddha Basantwanif627d802022-11-08 14:45:07 +053042/*
43 * Testing parameter range, assuming the parameter supported by effect is in this range.
44 * Parameter should be within the valid range defined in the documentation,
45 * for any supported value test expects EX_NONE from IEffect.setParameter(),
46 * otherwise expect EX_ILLEGAL_ARGUMENT.
47 */
48
Sneha Patile5849232023-12-01 17:52:28 +053049class BassBoostEffectHelper : public EffectHelper {
Shraddha Basantwanif627d802022-11-08 14:45:07 +053050 public:
Sneha Patile5849232023-12-01 17:52:28 +053051 void SetUpBassBoost(int32_t layout = AudioChannelLayout::LAYOUT_STEREO) {
Shraddha Basantwanif627d802022-11-08 14:45:07 +053052 ASSERT_NE(nullptr, mFactory);
Shunkai Yaocb0fc412022-12-15 20:34:32 +000053 ASSERT_NO_FATAL_FAILURE(create(mFactory, mEffect, mDescriptor));
Sneha Patile5849232023-12-01 17:52:28 +053054 setFrameCounts(layout);
55
56 AudioChannelLayout channelLayout =
57 AudioChannelLayout::make<AudioChannelLayout::layoutMask>(layout);
Shraddha Basantwanif627d802022-11-08 14:45:07 +053058
59 Parameter::Specific specific = getDefaultParamSpecific();
Shunkai Yao61f9dd22024-05-08 22:34:36 +000060 Parameter::Common common = createParamCommon(
Sneha Patile5849232023-12-01 17:52:28 +053061 0 /* session */, 1 /* ioHandle */, kSamplingFrequency /* iSampleRate */,
62 kSamplingFrequency /* oSampleRate */, mInputFrameCount /* iFrameCount */,
63 mOutputFrameCount /* oFrameCount */, channelLayout, channelLayout);
64 ASSERT_NO_FATAL_FAILURE(open(mEffect, common, specific, &mOpenEffectReturn, EX_NONE));
Shraddha Basantwanif627d802022-11-08 14:45:07 +053065 ASSERT_NE(nullptr, mEffect);
66 }
67
Sneha Patile5849232023-12-01 17:52:28 +053068 void TearDownBassBoost() {
Shraddha Basantwanif627d802022-11-08 14:45:07 +053069 ASSERT_NO_FATAL_FAILURE(close(mEffect));
70 ASSERT_NO_FATAL_FAILURE(destroy(mFactory, mEffect));
Sneha Patile5849232023-12-01 17:52:28 +053071 mOpenEffectReturn = IEffect::OpenEffectReturn{};
Shraddha Basantwanif627d802022-11-08 14:45:07 +053072 }
73
74 Parameter::Specific getDefaultParamSpecific() {
Sham Rathod8411fd22022-12-27 10:27:03 +053075 BassBoost bb = BassBoost::make<BassBoost::strengthPm>(0);
Shraddha Basantwanif627d802022-11-08 14:45:07 +053076 Parameter::Specific specific =
77 Parameter::Specific::make<Parameter::Specific::bassBoost>(bb);
78 return specific;
79 }
80
Sneha Patile5849232023-12-01 17:52:28 +053081 void setFrameCounts(int32_t inputBufferLayout) {
82 int channelCount = getChannelCount(
83 AudioChannelLayout::make<AudioChannelLayout::layoutMask>(inputBufferLayout));
84 mInputFrameCount = kInputSize / channelCount;
85 mOutputFrameCount = kInputSize / channelCount;
86 }
Shraddha Basantwanif627d802022-11-08 14:45:07 +053087
Sneha Patile5849232023-12-01 17:52:28 +053088 Parameter createBassBoostParam(int strength) {
89 return Parameter::make<Parameter::specific>(
90 Parameter::Specific::make<Parameter::Specific::bassBoost>(
91 BassBoost::make<BassBoost::strengthPm>(strength)));
92 }
Shraddha Basantwanif627d802022-11-08 14:45:07 +053093
Sneha Patile5849232023-12-01 17:52:28 +053094 bool isStrengthValid(int strength) {
95 auto bb = BassBoost::make<BassBoost::strengthPm>(strength);
96 return isParameterValid<BassBoost, Range::bassBoost>(bb, mDescriptor);
97 }
Shraddha Basantwanif627d802022-11-08 14:45:07 +053098
Sneha Patile5849232023-12-01 17:52:28 +053099 void setAndVerifyParameters(int strength, binder_exception_t expected) {
100 auto expectedParam = createBassBoostParam(strength);
101 EXPECT_STATUS(expected, mEffect->setParameter(expectedParam)) << expectedParam.toString();
Shraddha Basantwanif627d802022-11-08 14:45:07 +0530102
Sneha Patile5849232023-12-01 17:52:28 +0530103 if (expected == EX_NONE) {
104 auto bbId = BassBoost::Id::make<BassBoost::Id::commonTag>(
105 BassBoost::Tag(BassBoost::strengthPm));
106 auto id = Parameter::Id::make<Parameter::Id::bassBoostTag>(bbId);
107 // get parameter
108 Parameter getParam;
109 // if set success, then get should match
110 EXPECT_STATUS(expected, mEffect->getParameter(id, &getParam));
111 EXPECT_EQ(expectedParam, getParam) << "\nexpectedParam:" << expectedParam.toString()
112 << "\ngetParam:" << getParam.toString();
Shraddha Basantwanif627d802022-11-08 14:45:07 +0530113 }
114 }
115
Sneha Patile5849232023-12-01 17:52:28 +0530116 static constexpr int kSamplingFrequency = 44100;
117 static constexpr int kDurationMilliSec = 2000;
118 static constexpr int kInputSize = kSamplingFrequency * kDurationMilliSec / 1000;
119 long mInputFrameCount, mOutputFrameCount;
120 std::shared_ptr<IFactory> mFactory;
121 Descriptor mDescriptor;
122 std::shared_ptr<IEffect> mEffect;
123 IEffect::OpenEffectReturn mOpenEffectReturn;
124};
125
126/**
127 * Here we focus on specific parameter checking, general IEffect interfaces testing performed in
128 * VtsAudioEffectTargetTest.
129 */
130enum ParamName { PARAM_INSTANCE_NAME, PARAM_STRENGTH };
131using BassBoostParamTestParam = std::tuple<std::pair<std::shared_ptr<IFactory>, Descriptor>, int>;
132
133class BassBoostParamTest : public ::testing::TestWithParam<BassBoostParamTestParam>,
134 public BassBoostEffectHelper {
135 public:
136 BassBoostParamTest() : mParamStrength(std::get<PARAM_STRENGTH>(GetParam())) {
137 std::tie(mFactory, mDescriptor) = std::get<PARAM_INSTANCE_NAME>(GetParam());
Shraddha Basantwanif627d802022-11-08 14:45:07 +0530138 }
139
Sneha Patile5849232023-12-01 17:52:28 +0530140 void SetUp() override { ASSERT_NO_FATAL_FAILURE(SetUpBassBoost()); }
141 void TearDown() override { TearDownBassBoost(); }
142
143 int mParamStrength = 0;
Shraddha Basantwanif627d802022-11-08 14:45:07 +0530144};
145
146TEST_P(BassBoostParamTest, SetAndGetStrength) {
Sneha Patile5849232023-12-01 17:52:28 +0530147 if (isStrengthValid(mParamStrength)) {
148 ASSERT_NO_FATAL_FAILURE(setAndVerifyParameters(mParamStrength, EX_NONE));
149 } else {
150 ASSERT_NO_FATAL_FAILURE(setAndVerifyParameters(mParamStrength, EX_ILLEGAL_ARGUMENT));
151 }
152}
153
154enum DataParamName { DATA_INSTANCE_NAME, DATA_LAYOUT };
155
156using BassBoostDataTestParam =
157 std::tuple<std::pair<std::shared_ptr<IFactory>, Descriptor>, int32_t>;
158
159class BassBoostDataTest : public ::testing::TestWithParam<BassBoostDataTestParam>,
160 public BassBoostEffectHelper {
161 public:
162 BassBoostDataTest() : mChannelLayout(std::get<DATA_LAYOUT>(GetParam())) {
163 std::tie(mFactory, mDescriptor) = std::get<DATA_INSTANCE_NAME>(GetParam());
164 mStrengthValues = getTestValueSet<BassBoost, int, Range::bassBoost, BassBoost::strengthPm>(
165 {std::get<DATA_INSTANCE_NAME>(GetParam())}, expandTestValueBasic<int>);
166 }
167
168 void SetUp() override {
Shunkai Yao50e478b2024-03-14 01:54:58 +0000169 SKIP_TEST_IF_DATA_UNSUPPORTED(mDescriptor.common.flags);
Sneha Patile5849232023-12-01 17:52:28 +0530170 ASSERT_NO_FATAL_FAILURE(SetUpBassBoost(mChannelLayout));
171 if (int32_t version;
172 mEffect->getInterfaceVersion(&version).isOk() && version < kMinDataTestHalVersion) {
173 GTEST_SKIP() << "Skipping the data test for version: " << version << "\n";
174 }
175 }
176
Shunkai Yao50e478b2024-03-14 01:54:58 +0000177 void TearDown() override {
178 SKIP_TEST_IF_DATA_UNSUPPORTED(mDescriptor.common.flags);
179 TearDownBassBoost();
180 }
Sneha Patile5849232023-12-01 17:52:28 +0530181
182 // Find FFT bin indices for testFrequencies and get bin center frequencies
183 void roundToFreqCenteredToFftBin(std::vector<int>& testFrequencies,
184 std::vector<int>& binOffsets) {
185 for (size_t i = 0; i < testFrequencies.size(); i++) {
186 binOffsets[i] = std::round(testFrequencies[i] / kBinWidth);
187 testFrequencies[i] = std::round(binOffsets[i] * kBinWidth);
188 }
189 }
190
191 // Generate multitone input between -1 to +1 using testFrequencies
192 void generateMultiTone(const std::vector<int>& testFrequencies, std::vector<float>& input) {
193 for (auto i = 0; i < kInputSize; i++) {
194 input[i] = 0;
195
196 for (size_t j = 0; j < testFrequencies.size(); j++) {
197 input[i] += sin(2 * M_PI * testFrequencies[j] * i / kSamplingFrequency);
198 }
199 input[i] /= testFrequencies.size();
200 }
201 }
202
203 // Use FFT transform to convert the buffer to frequency domain
204 // Compute its magnitude at binOffsets
205 std::vector<float> calculateMagnitude(const std::vector<float>& buffer,
206 const std::vector<int>& binOffsets) {
207 std::vector<float> fftInput(kNPointFFT);
208 PFFFT_Setup* inputHandle = pffft_new_setup(kNPointFFT, PFFFT_REAL);
209 pffft_transform_ordered(inputHandle, buffer.data(), fftInput.data(), nullptr,
210 PFFFT_FORWARD);
211 pffft_destroy_setup(inputHandle);
212 std::vector<float> bufferMag(binOffsets.size());
213 for (size_t i = 0; i < binOffsets.size(); i++) {
214 size_t k = binOffsets[i];
215 bufferMag[i] = sqrt((fftInput[k * 2] * fftInput[k * 2]) +
216 (fftInput[k * 2 + 1] * fftInput[k * 2 + 1]));
217 }
218
219 return bufferMag;
220 }
221
222 // Calculate gain difference between low frequency and high frequency magnitude
223 float calculateGainDiff(const std::vector<float>& inputMag,
224 const std::vector<float>& outputMag) {
225 std::vector<float> gains(inputMag.size());
226
227 for (size_t i = 0; i < inputMag.size(); i++) {
228 gains[i] = 20 * log10(outputMag[i] / inputMag[i]);
229 }
230
231 return gains[0] - gains[1];
232 }
233
234 static constexpr int kNPointFFT = 32768;
235 static constexpr float kBinWidth = (float)kSamplingFrequency / kNPointFFT;
236 std::set<int> mStrengthValues;
237 int32_t mChannelLayout;
238};
239
240TEST_P(BassBoostDataTest, IncreasingStrength) {
241 // Frequencies to generate multitone input
242 std::vector<int> testFrequencies = {100, 1000};
243
244 // FFT bin indices for testFrequencies
245 std::vector<int> binOffsets(testFrequencies.size());
246
247 std::vector<float> input(kInputSize);
248 std::vector<float> baseOutput(kInputSize);
249
250 std::vector<float> inputMag(testFrequencies.size());
251 float prevGain = -100;
252
253 roundToFreqCenteredToFftBin(testFrequencies, binOffsets);
254
255 generateMultiTone(testFrequencies, input);
256
257 inputMag = calculateMagnitude(input, binOffsets);
258
259 if (isStrengthValid(0)) {
260 ASSERT_NO_FATAL_FAILURE(setAndVerifyParameters(0, EX_NONE));
261 } else {
262 GTEST_SKIP() << "Strength not supported, skipping the test\n";
263 }
264
265 ASSERT_NO_FATAL_FAILURE(
266 processAndWriteToOutput(input, baseOutput, mEffect, &mOpenEffectReturn));
267
268 std::vector<float> baseMag(testFrequencies.size());
269 baseMag = calculateMagnitude(baseOutput, binOffsets);
270 float baseDiff = calculateGainDiff(inputMag, baseMag);
271
272 for (int strength : mStrengthValues) {
273 // Skipping the further steps for invalid strength values
274 if (!isStrengthValid(strength)) {
275 continue;
276 }
277
278 ASSERT_NO_FATAL_FAILURE(setAndVerifyParameters(strength, EX_NONE));
279
280 std::vector<float> output(kInputSize);
281 std::vector<float> outputMag(testFrequencies.size());
282
283 ASSERT_NO_FATAL_FAILURE(
284 processAndWriteToOutput(input, output, mEffect, &mOpenEffectReturn));
285
286 outputMag = calculateMagnitude(output, binOffsets);
287 float diff = calculateGainDiff(inputMag, outputMag);
288
289 ASSERT_GT(diff, prevGain);
290 ASSERT_GT(diff, baseDiff);
291 prevGain = diff;
292 }
Shraddha Basantwanif627d802022-11-08 14:45:07 +0530293}
294
Shunkai Yao0a0c45e2023-02-13 17:41:11 +0000295std::vector<std::pair<std::shared_ptr<IFactory>, Descriptor>> kDescPair;
Shraddha Basantwanif627d802022-11-08 14:45:07 +0530296INSTANTIATE_TEST_SUITE_P(
297 BassBoostTest, BassBoostParamTest,
Sham Rathod8411fd22022-12-27 10:27:03 +0530298 ::testing::Combine(
Shunkai Yao0a0c45e2023-02-13 17:41:11 +0000299 testing::ValuesIn(kDescPair = EffectFactoryHelper::getAllEffectDescriptors(
Shunkai Yaof8be1ac2023-03-06 18:41:27 +0000300 IFactory::descriptor, getEffectTypeUuidBassBoost())),
Shunkai Yao0a0c45e2023-02-13 17:41:11 +0000301 testing::ValuesIn(EffectHelper::getTestValueSet<BassBoost, int, Range::bassBoost,
302 BassBoost::strengthPm>(
303 kDescPair, EffectHelper::expandTestValueBasic<int>))),
Shraddha Basantwanif627d802022-11-08 14:45:07 +0530304 [](const testing::TestParamInfo<BassBoostParamTest::ParamType>& info) {
Shunkai Yaocb0fc412022-12-15 20:34:32 +0000305 auto descriptor = std::get<PARAM_INSTANCE_NAME>(info.param).second;
Shraddha Basantwanif627d802022-11-08 14:45:07 +0530306 std::string strength = std::to_string(std::get<PARAM_STRENGTH>(info.param));
Jaideep Sharmae4c7a962023-06-14 19:14:44 +0530307 std::string name = getPrefix(descriptor) + "_strength_" + strength;
Shraddha Basantwanif627d802022-11-08 14:45:07 +0530308 std::replace_if(
309 name.begin(), name.end(), [](const char c) { return !std::isalnum(c); }, '_');
310 return name;
311 });
312
313GTEST_ALLOW_UNINSTANTIATED_PARAMETERIZED_TEST(BassBoostParamTest);
314
Sneha Patile5849232023-12-01 17:52:28 +0530315INSTANTIATE_TEST_SUITE_P(
316 BassBoostTest, BassBoostDataTest,
317 ::testing::Combine(testing::ValuesIn(EffectFactoryHelper::getAllEffectDescriptors(
318 IFactory::descriptor, getEffectTypeUuidBassBoost())),
319 testing::ValuesIn(kLayouts)),
320 [](const testing::TestParamInfo<BassBoostDataTest::ParamType>& info) {
321 auto descriptor = std::get<DATA_INSTANCE_NAME>(info.param).second;
322 std::string layout = std::to_string(std::get<DATA_LAYOUT>(info.param));
323 std::string name = getPrefix(descriptor) + "_layout_" + layout;
324 std::replace_if(
325 name.begin(), name.end(), [](const char c) { return !std::isalnum(c); }, '_');
326 return name;
327 });
328
329GTEST_ALLOW_UNINSTANTIATED_PARAMETERIZED_TEST(BassBoostDataTest);
330
Shraddha Basantwanif627d802022-11-08 14:45:07 +0530331int main(int argc, char** argv) {
332 ::testing::InitGoogleTest(&argc, argv);
Jaideep Sharma74498412023-09-13 15:25:25 +0530333 ::testing::UnitTest::GetInstance()->listeners().Append(new TestExecutionTracer());
Shraddha Basantwanif627d802022-11-08 14:45:07 +0530334 ABinderProcess_setThreadPoolMaxThreadCount(1);
335 ABinderProcess_startThreadPool();
336 return RUN_ALL_TESTS();
337}