/* * Copyright (C) 2022 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 #define LOG_TAG "VtsHalLoudnessEnhancerTest" #include #include "EffectHelper.h" using namespace android; using aidl::android::hardware::audio::effect::Descriptor; using aidl::android::hardware::audio::effect::getEffectTypeUuidLoudnessEnhancer; using aidl::android::hardware::audio::effect::IEffect; using aidl::android::hardware::audio::effect::IFactory; using aidl::android::hardware::audio::effect::LoudnessEnhancer; using aidl::android::hardware::audio::effect::Parameter; using android::hardware::audio::common::testing::detail::TestExecutionTracer; static constexpr float kMaxAudioSample = 1; static constexpr int kZeroGain = 0; static constexpr int kMaxGain = std::numeric_limits::max(); static constexpr int kMinGain = std::numeric_limits::min(); static constexpr float kAbsError = 0.0001; // Every int 32 bit value is a valid gain, so testing the corner cases and one regular value. // TODO : Update the test values once range/capability is updated by implementation. static const std::vector kGainMbValues = {kMinGain, -100, -50, kZeroGain, 50, 100, kMaxGain}; class LoudnessEnhancerEffectHelper : public EffectHelper { public: void SetUpLoudnessEnhancer() { ASSERT_NE(nullptr, mFactory); ASSERT_NO_FATAL_FAILURE(create(mFactory, mEffect, mDescriptor)); Parameter::Specific specific = getDefaultParamSpecific(); Parameter::Common common = createParamCommon( 0 /* session */, 1 /* ioHandle */, 44100 /* iSampleRate */, 44100 /* oSampleRate */, kInputFrameCount /* iFrameCount */, kOutputFrameCount /* oFrameCount */); ASSERT_NO_FATAL_FAILURE(open(mEffect, common, specific, &mOpenEffectReturn, EX_NONE)); ASSERT_NE(nullptr, mEffect); mVersion = EffectFactoryHelper::getHalVersion(mFactory); } void TearDownLoudnessEnhancer() { ASSERT_NO_FATAL_FAILURE(close(mEffect)); ASSERT_NO_FATAL_FAILURE(destroy(mFactory, mEffect)); mOpenEffectReturn = IEffect::OpenEffectReturn{}; } Parameter::Specific getDefaultParamSpecific() { LoudnessEnhancer le = LoudnessEnhancer::make(0); Parameter::Specific specific = Parameter::Specific::make(le); return specific; } Parameter createLoudnessParam(int gainMb) { LoudnessEnhancer le; le.set(gainMb); Parameter param; Parameter::Specific specific; specific.set(le); param.set(specific); return param; } binder_exception_t isGainValid(int gainMb) { LoudnessEnhancer le; le.set(gainMb); if (isParameterValid(le, mDescriptor)) { return EX_NONE; } else { return EX_ILLEGAL_ARGUMENT; } } void setParameters(int gain, binder_exception_t expected) { // set parameter auto param = createLoudnessParam(gain); EXPECT_STATUS(expected, mEffect->setParameter(param)) << param.toString(); } void validateParameters(int gain) { // get parameter LoudnessEnhancer::Id leId; Parameter getParam; Parameter::Id id; LoudnessEnhancer::Tag tag(LoudnessEnhancer::gainMb); leId.set(tag); id.set(leId); EXPECT_STATUS(EX_NONE, mEffect->getParameter(id, &getParam)); auto expectedParam = createLoudnessParam(gain); EXPECT_EQ(expectedParam, getParam) << "\nexpectedParam:" << expectedParam.toString() << "\ngetParam:" << getParam.toString(); } static const long kInputFrameCount = 0x100, kOutputFrameCount = 0x100; IEffect::OpenEffectReturn mOpenEffectReturn; std::shared_ptr mFactory; std::shared_ptr mEffect; Descriptor mDescriptor; int mVersion = 0; }; /** * Here we focus on specific parameter checking, general IEffect interfaces testing performed in * VtsAudioEffectTargetTest. */ enum ParamName { PARAM_INSTANCE_NAME, PARAM_GAIN_MB }; using LoudnessEnhancerParamTestParam = std::tuple, Descriptor>, int>; class LoudnessEnhancerParamTest : public ::testing::TestWithParam, public LoudnessEnhancerEffectHelper { public: LoudnessEnhancerParamTest() : mParamGainMb(std::get(GetParam())) { std::tie(mFactory, mDescriptor) = std::get(GetParam()); } void SetUp() override { SetUpLoudnessEnhancer(); } void TearDown() override { TearDownLoudnessEnhancer(); } int mParamGainMb = 0; }; TEST_P(LoudnessEnhancerParamTest, SetAndGetGainMb) { binder_exception_t expected = isGainValid(mParamGainMb); setParameters(mParamGainMb, expected); if (expected == EX_NONE) { validateParameters(mParamGainMb); } } using LoudnessEnhancerDataTestParam = std::pair, Descriptor>; class LoudnessEnhancerDataTest : public ::testing::TestWithParam, public LoudnessEnhancerEffectHelper { public: LoudnessEnhancerDataTest() { std::tie(mFactory, mDescriptor) = GetParam(); generateInputBuffer(); mOutputBuffer.resize(kBufferSize); } void SetUp() override { SKIP_TEST_IF_DATA_UNSUPPORTED(mDescriptor.common.flags); SetUpLoudnessEnhancer(); // Creating AidlMessageQueues mStatusMQ = std::make_unique(mOpenEffectReturn.statusMQ); mInputMQ = std::make_unique(mOpenEffectReturn.inputDataMQ); mOutputMQ = std::make_unique(mOpenEffectReturn.outputDataMQ); } void TearDown() override { SKIP_TEST_IF_DATA_UNSUPPORTED(mDescriptor.common.flags); TearDownLoudnessEnhancer(); } // Fill inputBuffer with random values between -kMaxAudioSample to kMaxAudioSample void generateInputBuffer() { for (size_t i = 0; i < kBufferSize; i++) { mInputBuffer.push_back(((static_cast(std::rand()) / RAND_MAX) * 2 - 1) * kMaxAudioSample); } } // Add gains to the mInputBuffer and store processed output to mOutputBuffer void processAndWriteToOutput() { // Check AidlMessageQueues are not null ASSERT_TRUE(mStatusMQ->isValid()); ASSERT_TRUE(mInputMQ->isValid()); ASSERT_TRUE(mOutputMQ->isValid()); // Enabling the process ASSERT_NO_FATAL_FAILURE(command(mEffect, CommandId::START)); ASSERT_NO_FATAL_FAILURE(expectState(mEffect, State::PROCESSING)); // Write from buffer to message queues and calling process EXPECT_NO_FATAL_FAILURE( EffectHelper::writeToFmq(mStatusMQ, mInputMQ, mInputBuffer, mVersion)); // Read the updated message queues into buffer EXPECT_NO_FATAL_FAILURE(EffectHelper::readFromFmq(mStatusMQ, 1, mOutputMQ, mOutputBuffer.size(), mOutputBuffer)); // Disable the process ASSERT_NO_FATAL_FAILURE(command(mEffect, CommandId::STOP)); } void assertGreaterGain(const std::vector& first, const std::vector& second) { for (size_t i = 0; i < first.size(); i++) { if (first[i] != 0) { ASSERT_GT(abs(first[i]), abs(second[i])); } else { ASSERT_EQ(first[i], second[i]); } } } void assertSequentialGains(const std::vector& gainValues, bool isIncreasing) { std::vector baseOutput(kBufferSize); // Process a reference output buffer with 0 gain which gives compressed input values binder_exception_t expected; expected = isGainValid(kZeroGain); ASSERT_EQ(expected, EX_NONE); setParameters(kZeroGain, expected); ASSERT_NO_FATAL_FAILURE(processAndWriteToOutput()); baseOutput = mOutputBuffer; // Compare the outputs for increasing gain for (int gain : gainValues) { // Setting the parameters binder_exception_t expected = isGainValid(gain); if (expected != EX_NONE) { GTEST_SKIP() << "Gains not supported."; } setParameters(gain, expected); ASSERT_NO_FATAL_FAILURE(processAndWriteToOutput()); // Compare the mOutputBuffer values with baseOutput and update it if (isIncreasing) { ASSERT_NO_FATAL_FAILURE(assertGreaterGain(mOutputBuffer, baseOutput)); } else { ASSERT_NO_FATAL_FAILURE(assertGreaterGain(baseOutput, mOutputBuffer)); } baseOutput = mOutputBuffer; } } std::unique_ptr mStatusMQ; std::unique_ptr mInputMQ; std::unique_ptr mOutputMQ; std::vector mInputBuffer; std::vector mOutputBuffer; static constexpr float kBufferSize = 128; }; TEST_P(LoudnessEnhancerDataTest, IncreasingGains) { static const std::vector kIncreasingGains = {50, 100}; assertSequentialGains(kIncreasingGains, true /*isIncreasing*/); } TEST_P(LoudnessEnhancerDataTest, DecreasingGains) { static const std::vector kDecreasingGains = {-50, -100}; assertSequentialGains(kDecreasingGains, false /*isIncreasing*/); } TEST_P(LoudnessEnhancerDataTest, MinimumGain) { // Setting the parameters binder_exception_t expected = isGainValid(kMinGain); if (expected != EX_NONE) { GTEST_SKIP() << "Minimum integer value not supported"; } setParameters(kMinGain, expected); ASSERT_NO_FATAL_FAILURE(processAndWriteToOutput()); // Validate that mOutputBuffer has 0 values for INT_MIN gain for (size_t i = 0; i < mOutputBuffer.size(); i++) { ASSERT_FLOAT_EQ(mOutputBuffer[i], 0); } } TEST_P(LoudnessEnhancerDataTest, MaximumGain) { // Setting the parameters binder_exception_t expected = isGainValid(kMaxGain); if (expected != EX_NONE) { GTEST_SKIP() << "Maximum integer value not supported"; } setParameters(kMaxGain, expected); ASSERT_NO_FATAL_FAILURE(processAndWriteToOutput()); // Validate that mOutputBuffer reaches to kMaxAudioSample for INT_MAX gain for (size_t i = 0; i < mOutputBuffer.size(); i++) { if (mInputBuffer[i] != 0) { EXPECT_NEAR(kMaxAudioSample, abs(mOutputBuffer[i]), kAbsError); } else { ASSERT_EQ(mOutputBuffer[i], mInputBuffer[i]); } } } INSTANTIATE_TEST_SUITE_P( LoudnessEnhancerTest, LoudnessEnhancerParamTest, ::testing::Combine(testing::ValuesIn(EffectFactoryHelper::getAllEffectDescriptors( IFactory::descriptor, getEffectTypeUuidLoudnessEnhancer())), testing::ValuesIn(kGainMbValues)), [](const testing::TestParamInfo& info) { auto descriptor = std::get(info.param).second; std::string gainMb = std::to_string(std::get(info.param)); std::string name = getPrefix(descriptor) + "_gainMb_" + gainMb; std::replace_if( name.begin(), name.end(), [](const char c) { return !std::isalnum(c); }, '_'); return name; }); GTEST_ALLOW_UNINSTANTIATED_PARAMETERIZED_TEST(LoudnessEnhancerParamTest); INSTANTIATE_TEST_SUITE_P( LoudnessEnhancerTest, LoudnessEnhancerDataTest, testing::ValuesIn(EffectFactoryHelper::getAllEffectDescriptors( IFactory::descriptor, getEffectTypeUuidLoudnessEnhancer())), [](const testing::TestParamInfo& info) { auto descriptor = info.param; std::string name = getPrefix(descriptor.second); std::replace_if( name.begin(), name.end(), [](const char c) { return !std::isalnum(c); }, '_'); return name; }); GTEST_ALLOW_UNINSTANTIATED_PARAMETERIZED_TEST(LoudnessEnhancerDataTest); int main(int argc, char** argv) { ::testing::InitGoogleTest(&argc, argv); ::testing::UnitTest::GetInstance()->listeners().Append(new TestExecutionTracer()); ABinderProcess_setThreadPoolMaxThreadCount(1); ABinderProcess_startThreadPool(); return RUN_ALL_TESTS(); }