Loading media/libeffects/preprocessing/tests/Android.bp +29 −0 Original line number Diff line number Diff line Loading @@ -10,6 +10,35 @@ package { ], } cc_test { name: "EffectPreprocessingTest", vendor: true, gtest: true, host_supported: true, test_suites: ["device-tests"], srcs: [ "EffectPreprocessingTest.cpp", "EffectTestHelper.cpp", ], static_libs: [ "libaudiopreprocessing", "libaudioutils", "webrtc_audio_processing", ], shared_libs: [ "liblog", ], header_libs: [ "libaudioeffects", "libhardware_headers", ], target: { darwin: { enabled: false, }, }, } cc_test { name: "AudioPreProcessingTest", vendor: true, Loading media/libeffects/preprocessing/tests/EffectPreprocessingTest.cpp 0 → 100644 +332 −0 Original line number Diff line number Diff line /* * Copyright 2021 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 "EffectTestHelper.h" #include <getopt.h> #include <stddef.h> #include <stdint.h> #include <tuple> #include <vector> #include <audio_effects/effect_aec.h> #include <audio_effects/effect_agc.h> #include <audio_effects/effect_agc2.h> #include <audio_effects/effect_ns.h> #include <log/log.h> constexpr effect_uuid_t kAGCUuid = { 0xaa8130e0, 0x66fc, 0x11e0, 0xbad0, {0x00, 0x02, 0xa5, 0xd5, 0xc5, 0x1b}}; constexpr effect_uuid_t kAGC2Uuid = { 0x89f38e65, 0xd4d2, 0x4d64, 0xad0e, {0x2b, 0x3e, 0x79, 0x9e, 0xa8, 0x86}}; constexpr effect_uuid_t kAECUuid = { 0xbb392ec0, 0x8d4d, 0x11e0, 0xa896, {0x00, 0x02, 0xa5, 0xd5, 0xc5, 0x1b}}; constexpr effect_uuid_t kNSUuid = { 0xc06c8400, 0x8e06, 0x11e0, 0x9cb6, {0x00, 0x02, 0xa5, 0xd5, 0xc5, 0x1b}}; static bool isAGCEffect(const effect_uuid_t* uuid) { return uuid == &kAGCUuid; } static bool isAGC2Effect(const effect_uuid_t* uuid) { return uuid == &kAGC2Uuid; } static bool isAECEffect(const effect_uuid_t* uuid) { return uuid == &kAECUuid; } static bool isNSEffect(const effect_uuid_t* uuid) { return uuid == &kNSUuid; } constexpr int kAGCTargetLevels[] = {0, -300, -500, -1000, -3100}; constexpr int kAGCCompLevels[] = {0, -300, -500, -1000, -9000}; constexpr size_t kAGC2FixedDigitalGains[] = {0, 3, 10, 20, 49}; constexpr size_t kAGC2AdaptGigitalLevelEstimators[] = {0, 1}; constexpr size_t kAGC2ExtraSaturationMargins[] = {0, 3, 10, 20, 100}; constexpr size_t kAECEchoDelays[] = {0, 250, 500}; constexpr size_t kNSLevels[] = {0, 1, 2, 3}; struct AGCParams { int targetLevel; int compLevel; }; struct AGC2Params { size_t fixedDigitalGain; size_t adaptDigiLevelEstimator; size_t extraSaturationMargin; }; struct AECParams { size_t echoDelay; }; struct NSParams { size_t level; }; struct PreProcParams { const effect_uuid_t* uuid; union { AGCParams agcParams; AGC2Params agc2Params; AECParams aecParams; NSParams nsParams; }; }; // Create a list of pre-processing parameters to be used for testing static const std::vector<PreProcParams> kPreProcParams = [] { std::vector<PreProcParams> result; for (const auto targetLevel : kAGCTargetLevels) { for (const auto compLevel : kAGCCompLevels) { AGCParams agcParams = {.targetLevel = targetLevel, .compLevel = compLevel}; PreProcParams params = {.uuid = &kAGCUuid, .agcParams = agcParams}; result.push_back(params); } } for (const auto fixedDigitalGain : kAGC2FixedDigitalGains) { for (const auto adaptDigiLevelEstimator : kAGC2AdaptGigitalLevelEstimators) { for (const auto extraSaturationMargin : kAGC2ExtraSaturationMargins) { AGC2Params agc2Params = {.fixedDigitalGain = fixedDigitalGain, .adaptDigiLevelEstimator = adaptDigiLevelEstimator, .extraSaturationMargin = extraSaturationMargin}; PreProcParams params = {.uuid = &kAGC2Uuid, .agc2Params = agc2Params}; result.push_back(params); } } } for (const auto echoDelay : kAECEchoDelays) { AECParams aecParams = {.echoDelay = echoDelay}; PreProcParams params = {.uuid = &kAECUuid, .aecParams = aecParams}; result.push_back(params); } for (const auto level : kNSLevels) { NSParams nsParams = {.level = level}; PreProcParams params = {.uuid = &kNSUuid, .nsParams = nsParams}; result.push_back(params); } return result; }(); static const size_t kNumPreProcParams = std::size(kPreProcParams); void setPreProcParams(const effect_uuid_t* uuid, EffectTestHelper& effect, size_t paramIdx) { const PreProcParams* params = &kPreProcParams[paramIdx]; if (isAGCEffect(uuid)) { const AGCParams* agcParams = ¶ms->agcParams; ASSERT_NO_FATAL_FAILURE(effect.setParam(AGC_PARAM_TARGET_LEVEL, agcParams->targetLevel)); ASSERT_NO_FATAL_FAILURE(effect.setParam(AGC_PARAM_COMP_GAIN, agcParams->compLevel)); } else if (isAGC2Effect(uuid)) { const AGC2Params* agc2Params = ¶ms->agc2Params; ASSERT_NO_FATAL_FAILURE( effect.setParam(AGC2_PARAM_FIXED_DIGITAL_GAIN, agc2Params->fixedDigitalGain)); ASSERT_NO_FATAL_FAILURE(effect.setParam(AGC2_PARAM_ADAPT_DIGI_LEVEL_ESTIMATOR, agc2Params->adaptDigiLevelEstimator)); ASSERT_NO_FATAL_FAILURE(effect.setParam(AGC2_PARAM_ADAPT_DIGI_EXTRA_SATURATION_MARGIN, agc2Params->extraSaturationMargin)); } else if (isAECEffect(uuid)) { const AECParams* aecParams = ¶ms->aecParams; ASSERT_NO_FATAL_FAILURE(effect.setParam(AEC_PARAM_ECHO_DELAY, aecParams->echoDelay)); } else if (isNSEffect(uuid)) { const NSParams* nsParams = ¶ms->nsParams; ASSERT_NO_FATAL_FAILURE(effect.setParam(NS_PARAM_LEVEL, nsParams->level)); } } typedef std::tuple<int, int, int, int> SingleEffectTestParam; class SingleEffectTest : public ::testing::TestWithParam<SingleEffectTestParam> { public: SingleEffectTest() : mSampleRate(EffectTestHelper::kSampleRates[std::get<1>(GetParam())]), mFrameCount(mSampleRate * EffectTestHelper::kTenMilliSecVal), mLoopCount(EffectTestHelper::kLoopCounts[std::get<2>(GetParam())]), mTotalFrameCount(mFrameCount * mLoopCount), mChMask(EffectTestHelper::kChMasks[std::get<0>(GetParam())]), mChannelCount(audio_channel_count_from_in_mask(mChMask)), mParamIdx(std::get<3>(GetParam())), mUuid(kPreProcParams[mParamIdx].uuid){}; const size_t mSampleRate; const size_t mFrameCount; const size_t mLoopCount; const size_t mTotalFrameCount; const size_t mChMask; const size_t mChannelCount; const size_t mParamIdx; const effect_uuid_t* mUuid; }; // Tests applying a single effect TEST_P(SingleEffectTest, SimpleProcess) { SCOPED_TRACE(testing::Message() << " chMask: " << mChMask << " sampleRate: " << mSampleRate << " loopCount: " << mLoopCount << " paramIdx " << mParamIdx); EffectTestHelper effect(mUuid, mChMask, mSampleRate, mLoopCount); ASSERT_NO_FATAL_FAILURE(effect.createEffect()); ASSERT_NO_FATAL_FAILURE(effect.setConfig(isAECEffect(mUuid))); ASSERT_NO_FATAL_FAILURE(setPreProcParams(mUuid, effect, mParamIdx)); // Initialize input buffer with deterministic pseudo-random values std::vector<int16_t> input(mTotalFrameCount * mChannelCount); std::vector<int16_t> output(mTotalFrameCount * mChannelCount); std::vector<int16_t> farInput(mTotalFrameCount * mChannelCount); std::minstd_rand gen(mChMask); std::uniform_int_distribution<int16_t> dis(INT16_MIN, INT16_MAX); for (auto& in : input) { in = dis(gen); } if (isAECEffect(mUuid)) { for (auto& farIn : farInput) { farIn = dis(gen); } } ASSERT_NO_FATAL_FAILURE(effect.process(input.data(), output.data(), isAECEffect(mUuid))); if (isAECEffect(mUuid)) { ASSERT_NO_FATAL_FAILURE(effect.process_reverse(farInput.data(), output.data())); } ASSERT_NO_FATAL_FAILURE(effect.releaseEffect()); } INSTANTIATE_TEST_SUITE_P( PreProcTestAll, SingleEffectTest, ::testing::Combine(::testing::Range(0, (int)EffectTestHelper::kNumChMasks), ::testing::Range(0, (int)EffectTestHelper::kNumSampleRates), ::testing::Range(0, (int)EffectTestHelper::kNumLoopCounts), ::testing::Range(0, (int)kNumPreProcParams))); typedef std::tuple<int, int, int> SingleEffectComparisonTestParam; class SingleEffectComparisonTest : public ::testing::TestWithParam<SingleEffectComparisonTestParam> { public: SingleEffectComparisonTest() : mSampleRate(EffectTestHelper::kSampleRates[std::get<0>(GetParam())]), mFrameCount(mSampleRate * EffectTestHelper::kTenMilliSecVal), mLoopCount(EffectTestHelper::kLoopCounts[std::get<1>(GetParam())]), mTotalFrameCount(mFrameCount * mLoopCount), mParamIdx(std::get<2>(GetParam())), mUuid(kPreProcParams[mParamIdx].uuid){}; const size_t mSampleRate; const size_t mFrameCount; const size_t mLoopCount; const size_t mTotalFrameCount; const size_t mParamIdx; const effect_uuid_t* mUuid; }; // Compares first channel in multi-channel output to mono output when same effect is applied TEST_P(SingleEffectComparisonTest, SimpleProcess) { SCOPED_TRACE(testing::Message() << " sampleRate: " << mSampleRate << " loopCount: " << mLoopCount << " paramIdx " << mParamIdx); // Initialize mono input buffer with deterministic pseudo-random values std::vector<int16_t> monoInput(mTotalFrameCount); std::vector<int16_t> monoFarInput(mTotalFrameCount); std::minstd_rand gen(mSampleRate); std::uniform_int_distribution<int16_t> dis(INT16_MIN, INT16_MAX); for (auto& in : monoInput) { in = dis(gen); } if (isAECEffect(mUuid)) { for (auto& farIn : monoFarInput) { farIn = dis(gen); } } // Apply effect on mono channel EffectTestHelper monoEffect(mUuid, AUDIO_CHANNEL_INDEX_MASK_1, mSampleRate, mLoopCount); ASSERT_NO_FATAL_FAILURE(monoEffect.createEffect()); ASSERT_NO_FATAL_FAILURE(monoEffect.setConfig(isAECEffect(mUuid))); ASSERT_NO_FATAL_FAILURE(setPreProcParams(mUuid, monoEffect, mParamIdx)); std::vector<int16_t> monoOutput(mTotalFrameCount); ASSERT_NO_FATAL_FAILURE( monoEffect.process(monoInput.data(), monoOutput.data(), isAECEffect(mUuid))); if (isAECEffect(mUuid)) { ASSERT_NO_FATAL_FAILURE(monoEffect.process_reverse(monoFarInput.data(), monoOutput.data())); } ASSERT_NO_FATAL_FAILURE(monoEffect.releaseEffect()); for (size_t chMask : EffectTestHelper::kChMasks) { size_t channelCount = audio_channel_count_from_in_mask(chMask); EffectTestHelper testEffect(mUuid, chMask, mSampleRate, mLoopCount); ASSERT_NO_FATAL_FAILURE(testEffect.createEffect()); ASSERT_NO_FATAL_FAILURE(testEffect.setConfig(isAECEffect(mUuid))); ASSERT_NO_FATAL_FAILURE(setPreProcParams(mUuid, testEffect, mParamIdx)); std::vector<int16_t> testInput(mTotalFrameCount * channelCount); std::vector<int16_t> testFarInput(mTotalFrameCount * channelCount); // Repeat mono channel data to all the channels // adjust_channels() zero fills channels > 2, hence can't be used here for (size_t i = 0; i < mTotalFrameCount; ++i) { auto* fpInput = &testInput[i * channelCount]; std::fill(fpInput, fpInput + channelCount, monoInput[i]); } if (isAECEffect(mUuid)) { for (size_t i = 0; i < mTotalFrameCount; ++i) { auto* fpFarInput = &testFarInput[i * channelCount]; std::fill(fpFarInput, fpFarInput + channelCount, monoFarInput[i]); } } std::vector<int16_t> testOutput(mTotalFrameCount * channelCount); ASSERT_NO_FATAL_FAILURE( testEffect.process(testInput.data(), testOutput.data(), isAECEffect(mUuid))); if (isAECEffect(mUuid)) { ASSERT_NO_FATAL_FAILURE( testEffect.process_reverse(testFarInput.data(), testOutput.data())); } ASSERT_NO_FATAL_FAILURE(testEffect.releaseEffect()); // Adjust the test output to mono channel std::vector<int16_t> monoTestOutput(mTotalFrameCount); adjust_channels(testOutput.data(), channelCount, monoTestOutput.data(), FCC_1, sizeof(int16_t), mTotalFrameCount * sizeof(int16_t) * channelCount); ASSERT_EQ(0, memcmp(monoOutput.data(), monoTestOutput.data(), mTotalFrameCount * sizeof(int16_t))) << "Mono channel output does not match with reference output \n"; } } INSTANTIATE_TEST_SUITE_P( PreProcTestAll, SingleEffectComparisonTest, ::testing::Combine(::testing::Range(0, (int)EffectTestHelper::kNumSampleRates), ::testing::Range(0, (int)EffectTestHelper::kNumLoopCounts), ::testing::Range(0, (int)kNumPreProcParams))); int main(int argc, char** argv) { ::testing::InitGoogleTest(&argc, argv); int status = RUN_ALL_TESTS(); ALOGV("Test result = %d", status); return status; } media/libeffects/preprocessing/tests/EffectTestHelper.cpp 0 → 100644 +99 −0 Original line number Diff line number Diff line /* * Copyright 2021 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 "EffectTestHelper.h" extern audio_effect_library_t AUDIO_EFFECT_LIBRARY_INFO_SYM; void EffectTestHelper::createEffect() { int status = AUDIO_EFFECT_LIBRARY_INFO_SYM.create_effect(mUuid, 1, 1, &mEffectHandle); ASSERT_EQ(status, 0) << "create_effect returned an error " << status; } void EffectTestHelper::releaseEffect() { int status = AUDIO_EFFECT_LIBRARY_INFO_SYM.release_effect(mEffectHandle); ASSERT_EQ(status, 0) << "release_effect returned an error " << status; } void EffectTestHelper::setConfig(bool configReverse) { effect_config_t config{}; config.inputCfg.samplingRate = config.outputCfg.samplingRate = mSampleRate; config.inputCfg.channels = config.outputCfg.channels = mChMask; config.inputCfg.format = config.outputCfg.format = AUDIO_FORMAT_PCM_16_BIT; int reply = 0; uint32_t replySize = sizeof(reply); int status = (*mEffectHandle) ->command(mEffectHandle, EFFECT_CMD_SET_CONFIG, sizeof(effect_config_t), &config, &replySize, &reply); ASSERT_EQ(status, 0) << "set_config returned an error " << status; ASSERT_EQ(reply, 0) << "set_config reply non zero " << reply; if (configReverse) { int status = (*mEffectHandle) ->command(mEffectHandle, EFFECT_CMD_SET_CONFIG_REVERSE, sizeof(effect_config_t), &config, &replySize, &reply); ASSERT_EQ(status, 0) << "set_config_reverse returned an error " << status; ASSERT_EQ(reply, 0) << "set_config_reverse reply non zero " << reply; } status = (*mEffectHandle) ->command(mEffectHandle, EFFECT_CMD_ENABLE, 0, nullptr, &replySize, &reply); ASSERT_EQ(status, 0) << "cmd_enable returned an error " << status; ASSERT_EQ(reply, 0) << "cmd_enable reply non zero " << reply; } void EffectTestHelper::setParam(uint32_t type, uint32_t value) { int reply = 0; uint32_t replySize = sizeof(reply); uint32_t paramData[2] = {type, value}; auto effectParam = (effect_param_t*)malloc(sizeof(effect_param_t) + sizeof(paramData)); memcpy(&effectParam->data[0], ¶mData[0], sizeof(paramData)); effectParam->psize = sizeof(paramData[0]); effectParam->vsize = sizeof(paramData[1]); int status = (*mEffectHandle) ->command(mEffectHandle, EFFECT_CMD_SET_PARAM, sizeof(effect_param_t) + sizeof(paramData), effectParam, &replySize, &reply); free(effectParam); ASSERT_EQ(status, 0) << "set_param returned an error " << status; ASSERT_EQ(reply, 0) << "set_param reply non zero " << reply; } void EffectTestHelper::process(int16_t* input, int16_t* output, bool setAecEchoDelay) { audio_buffer_t inBuffer = {.frameCount = mFrameCount, .s16 = input}; audio_buffer_t outBuffer = {.frameCount = mFrameCount, .s16 = output}; for (size_t i = 0; i < mLoopCount; i++) { if (setAecEchoDelay) ASSERT_NO_FATAL_FAILURE(setParam(AEC_PARAM_ECHO_DELAY, kAECDelay)); int status = (*mEffectHandle)->process(mEffectHandle, &inBuffer, &outBuffer); ASSERT_EQ(status, 0) << "process returned an error " << status; inBuffer.s16 += mFrameCount * mChannelCount; outBuffer.s16 += mFrameCount * mChannelCount; } } void EffectTestHelper::process_reverse(int16_t* farInput, int16_t* output) { audio_buffer_t farInBuffer = {.frameCount = mFrameCount, .s16 = farInput}; audio_buffer_t outBuffer = {.frameCount = mFrameCount, .s16 = output}; for (size_t i = 0; i < mLoopCount; i++) { int status = (*mEffectHandle)->process_reverse(mEffectHandle, &farInBuffer, &outBuffer); ASSERT_EQ(status, 0) << "process returned an error " << status; farInBuffer.s16 += mFrameCount * mChannelCount; outBuffer.s16 += mFrameCount * mChannelCount; } } media/libeffects/preprocessing/tests/EffectTestHelper.h 0 → 100644 +109 −0 Original line number Diff line number Diff line /* * Copyright 2021 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. */ #pragma once #include <array> #include <audio_effects/effect_aec.h> #include <audio_utils/channels.h> #include <audio_utils/primitives.h> #include <climits> #include <cstdlib> #include <gtest/gtest.h> #include <hardware/audio_effect.h> #include <log/log.h> #include <random> #include <stdint.h> #include <system/audio.h> #include <vector> template <typename T> static float computeSnr(const T* ref, const T* tst, size_t count) { double signal{}; double noise{}; for (size_t i = 0; i < count; ++i) { const double value(ref[i]); const double diff(tst[i] - value); signal += value * value; noise += diff * diff; } // Initialized to large value to handle // cases where ref and tst match exactly float snr = FLT_MAX; if (signal > 0.0f && noise > 0.0f) { snr = 10.f * log(signal / noise); } return snr; } class EffectTestHelper { public: EffectTestHelper(const effect_uuid_t* uuid, size_t chMask, size_t sampleRate, size_t loopCount) : mUuid(uuid), mChMask(chMask), mChannelCount(audio_channel_count_from_in_mask(mChMask)), mSampleRate(sampleRate), mFrameCount(mSampleRate * kTenMilliSecVal), mLoopCount(loopCount) {} void createEffect(); void releaseEffect(); void setConfig(bool configReverse); void setParam(uint32_t type, uint32_t val); void process(int16_t* input, int16_t* output, bool setAecEchoDelay); void process_reverse(int16_t* farInput, int16_t* output); // Corresponds to SNR for 1 bit difference between two int16_t signals static constexpr float kSNRThreshold = 90.308998; static constexpr audio_channel_mask_t kChMasks[] = { AUDIO_CHANNEL_IN_MONO, AUDIO_CHANNEL_IN_STEREO, AUDIO_CHANNEL_IN_FRONT_BACK, AUDIO_CHANNEL_IN_6, AUDIO_CHANNEL_IN_2POINT0POINT2, AUDIO_CHANNEL_IN_2POINT1POINT2, AUDIO_CHANNEL_IN_3POINT0POINT2, AUDIO_CHANNEL_IN_3POINT1POINT2, AUDIO_CHANNEL_IN_5POINT1, AUDIO_CHANNEL_IN_VOICE_UPLINK_MONO, AUDIO_CHANNEL_IN_VOICE_DNLINK_MONO, AUDIO_CHANNEL_IN_VOICE_CALL_MONO, }; static constexpr float kTenMilliSecVal = 0.01; static constexpr size_t kNumChMasks = std::size(kChMasks); static constexpr size_t kSampleRates[] = {8000, 16000, 24000, 32000, 48000}; static constexpr size_t kNumSampleRates = std::size(kSampleRates); static constexpr size_t kLoopCounts[] = {1, 4}; static constexpr size_t kNumLoopCounts = std::size(kLoopCounts); static constexpr size_t kAECDelay = 0; private: const effect_uuid_t* mUuid; const size_t mChMask; const size_t mChannelCount; const size_t mSampleRate; const size_t mFrameCount; const size_t mLoopCount; effect_handle_t mEffectHandle{}; }; Loading
media/libeffects/preprocessing/tests/Android.bp +29 −0 Original line number Diff line number Diff line Loading @@ -10,6 +10,35 @@ package { ], } cc_test { name: "EffectPreprocessingTest", vendor: true, gtest: true, host_supported: true, test_suites: ["device-tests"], srcs: [ "EffectPreprocessingTest.cpp", "EffectTestHelper.cpp", ], static_libs: [ "libaudiopreprocessing", "libaudioutils", "webrtc_audio_processing", ], shared_libs: [ "liblog", ], header_libs: [ "libaudioeffects", "libhardware_headers", ], target: { darwin: { enabled: false, }, }, } cc_test { name: "AudioPreProcessingTest", vendor: true, Loading
media/libeffects/preprocessing/tests/EffectPreprocessingTest.cpp 0 → 100644 +332 −0 Original line number Diff line number Diff line /* * Copyright 2021 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 "EffectTestHelper.h" #include <getopt.h> #include <stddef.h> #include <stdint.h> #include <tuple> #include <vector> #include <audio_effects/effect_aec.h> #include <audio_effects/effect_agc.h> #include <audio_effects/effect_agc2.h> #include <audio_effects/effect_ns.h> #include <log/log.h> constexpr effect_uuid_t kAGCUuid = { 0xaa8130e0, 0x66fc, 0x11e0, 0xbad0, {0x00, 0x02, 0xa5, 0xd5, 0xc5, 0x1b}}; constexpr effect_uuid_t kAGC2Uuid = { 0x89f38e65, 0xd4d2, 0x4d64, 0xad0e, {0x2b, 0x3e, 0x79, 0x9e, 0xa8, 0x86}}; constexpr effect_uuid_t kAECUuid = { 0xbb392ec0, 0x8d4d, 0x11e0, 0xa896, {0x00, 0x02, 0xa5, 0xd5, 0xc5, 0x1b}}; constexpr effect_uuid_t kNSUuid = { 0xc06c8400, 0x8e06, 0x11e0, 0x9cb6, {0x00, 0x02, 0xa5, 0xd5, 0xc5, 0x1b}}; static bool isAGCEffect(const effect_uuid_t* uuid) { return uuid == &kAGCUuid; } static bool isAGC2Effect(const effect_uuid_t* uuid) { return uuid == &kAGC2Uuid; } static bool isAECEffect(const effect_uuid_t* uuid) { return uuid == &kAECUuid; } static bool isNSEffect(const effect_uuid_t* uuid) { return uuid == &kNSUuid; } constexpr int kAGCTargetLevels[] = {0, -300, -500, -1000, -3100}; constexpr int kAGCCompLevels[] = {0, -300, -500, -1000, -9000}; constexpr size_t kAGC2FixedDigitalGains[] = {0, 3, 10, 20, 49}; constexpr size_t kAGC2AdaptGigitalLevelEstimators[] = {0, 1}; constexpr size_t kAGC2ExtraSaturationMargins[] = {0, 3, 10, 20, 100}; constexpr size_t kAECEchoDelays[] = {0, 250, 500}; constexpr size_t kNSLevels[] = {0, 1, 2, 3}; struct AGCParams { int targetLevel; int compLevel; }; struct AGC2Params { size_t fixedDigitalGain; size_t adaptDigiLevelEstimator; size_t extraSaturationMargin; }; struct AECParams { size_t echoDelay; }; struct NSParams { size_t level; }; struct PreProcParams { const effect_uuid_t* uuid; union { AGCParams agcParams; AGC2Params agc2Params; AECParams aecParams; NSParams nsParams; }; }; // Create a list of pre-processing parameters to be used for testing static const std::vector<PreProcParams> kPreProcParams = [] { std::vector<PreProcParams> result; for (const auto targetLevel : kAGCTargetLevels) { for (const auto compLevel : kAGCCompLevels) { AGCParams agcParams = {.targetLevel = targetLevel, .compLevel = compLevel}; PreProcParams params = {.uuid = &kAGCUuid, .agcParams = agcParams}; result.push_back(params); } } for (const auto fixedDigitalGain : kAGC2FixedDigitalGains) { for (const auto adaptDigiLevelEstimator : kAGC2AdaptGigitalLevelEstimators) { for (const auto extraSaturationMargin : kAGC2ExtraSaturationMargins) { AGC2Params agc2Params = {.fixedDigitalGain = fixedDigitalGain, .adaptDigiLevelEstimator = adaptDigiLevelEstimator, .extraSaturationMargin = extraSaturationMargin}; PreProcParams params = {.uuid = &kAGC2Uuid, .agc2Params = agc2Params}; result.push_back(params); } } } for (const auto echoDelay : kAECEchoDelays) { AECParams aecParams = {.echoDelay = echoDelay}; PreProcParams params = {.uuid = &kAECUuid, .aecParams = aecParams}; result.push_back(params); } for (const auto level : kNSLevels) { NSParams nsParams = {.level = level}; PreProcParams params = {.uuid = &kNSUuid, .nsParams = nsParams}; result.push_back(params); } return result; }(); static const size_t kNumPreProcParams = std::size(kPreProcParams); void setPreProcParams(const effect_uuid_t* uuid, EffectTestHelper& effect, size_t paramIdx) { const PreProcParams* params = &kPreProcParams[paramIdx]; if (isAGCEffect(uuid)) { const AGCParams* agcParams = ¶ms->agcParams; ASSERT_NO_FATAL_FAILURE(effect.setParam(AGC_PARAM_TARGET_LEVEL, agcParams->targetLevel)); ASSERT_NO_FATAL_FAILURE(effect.setParam(AGC_PARAM_COMP_GAIN, agcParams->compLevel)); } else if (isAGC2Effect(uuid)) { const AGC2Params* agc2Params = ¶ms->agc2Params; ASSERT_NO_FATAL_FAILURE( effect.setParam(AGC2_PARAM_FIXED_DIGITAL_GAIN, agc2Params->fixedDigitalGain)); ASSERT_NO_FATAL_FAILURE(effect.setParam(AGC2_PARAM_ADAPT_DIGI_LEVEL_ESTIMATOR, agc2Params->adaptDigiLevelEstimator)); ASSERT_NO_FATAL_FAILURE(effect.setParam(AGC2_PARAM_ADAPT_DIGI_EXTRA_SATURATION_MARGIN, agc2Params->extraSaturationMargin)); } else if (isAECEffect(uuid)) { const AECParams* aecParams = ¶ms->aecParams; ASSERT_NO_FATAL_FAILURE(effect.setParam(AEC_PARAM_ECHO_DELAY, aecParams->echoDelay)); } else if (isNSEffect(uuid)) { const NSParams* nsParams = ¶ms->nsParams; ASSERT_NO_FATAL_FAILURE(effect.setParam(NS_PARAM_LEVEL, nsParams->level)); } } typedef std::tuple<int, int, int, int> SingleEffectTestParam; class SingleEffectTest : public ::testing::TestWithParam<SingleEffectTestParam> { public: SingleEffectTest() : mSampleRate(EffectTestHelper::kSampleRates[std::get<1>(GetParam())]), mFrameCount(mSampleRate * EffectTestHelper::kTenMilliSecVal), mLoopCount(EffectTestHelper::kLoopCounts[std::get<2>(GetParam())]), mTotalFrameCount(mFrameCount * mLoopCount), mChMask(EffectTestHelper::kChMasks[std::get<0>(GetParam())]), mChannelCount(audio_channel_count_from_in_mask(mChMask)), mParamIdx(std::get<3>(GetParam())), mUuid(kPreProcParams[mParamIdx].uuid){}; const size_t mSampleRate; const size_t mFrameCount; const size_t mLoopCount; const size_t mTotalFrameCount; const size_t mChMask; const size_t mChannelCount; const size_t mParamIdx; const effect_uuid_t* mUuid; }; // Tests applying a single effect TEST_P(SingleEffectTest, SimpleProcess) { SCOPED_TRACE(testing::Message() << " chMask: " << mChMask << " sampleRate: " << mSampleRate << " loopCount: " << mLoopCount << " paramIdx " << mParamIdx); EffectTestHelper effect(mUuid, mChMask, mSampleRate, mLoopCount); ASSERT_NO_FATAL_FAILURE(effect.createEffect()); ASSERT_NO_FATAL_FAILURE(effect.setConfig(isAECEffect(mUuid))); ASSERT_NO_FATAL_FAILURE(setPreProcParams(mUuid, effect, mParamIdx)); // Initialize input buffer with deterministic pseudo-random values std::vector<int16_t> input(mTotalFrameCount * mChannelCount); std::vector<int16_t> output(mTotalFrameCount * mChannelCount); std::vector<int16_t> farInput(mTotalFrameCount * mChannelCount); std::minstd_rand gen(mChMask); std::uniform_int_distribution<int16_t> dis(INT16_MIN, INT16_MAX); for (auto& in : input) { in = dis(gen); } if (isAECEffect(mUuid)) { for (auto& farIn : farInput) { farIn = dis(gen); } } ASSERT_NO_FATAL_FAILURE(effect.process(input.data(), output.data(), isAECEffect(mUuid))); if (isAECEffect(mUuid)) { ASSERT_NO_FATAL_FAILURE(effect.process_reverse(farInput.data(), output.data())); } ASSERT_NO_FATAL_FAILURE(effect.releaseEffect()); } INSTANTIATE_TEST_SUITE_P( PreProcTestAll, SingleEffectTest, ::testing::Combine(::testing::Range(0, (int)EffectTestHelper::kNumChMasks), ::testing::Range(0, (int)EffectTestHelper::kNumSampleRates), ::testing::Range(0, (int)EffectTestHelper::kNumLoopCounts), ::testing::Range(0, (int)kNumPreProcParams))); typedef std::tuple<int, int, int> SingleEffectComparisonTestParam; class SingleEffectComparisonTest : public ::testing::TestWithParam<SingleEffectComparisonTestParam> { public: SingleEffectComparisonTest() : mSampleRate(EffectTestHelper::kSampleRates[std::get<0>(GetParam())]), mFrameCount(mSampleRate * EffectTestHelper::kTenMilliSecVal), mLoopCount(EffectTestHelper::kLoopCounts[std::get<1>(GetParam())]), mTotalFrameCount(mFrameCount * mLoopCount), mParamIdx(std::get<2>(GetParam())), mUuid(kPreProcParams[mParamIdx].uuid){}; const size_t mSampleRate; const size_t mFrameCount; const size_t mLoopCount; const size_t mTotalFrameCount; const size_t mParamIdx; const effect_uuid_t* mUuid; }; // Compares first channel in multi-channel output to mono output when same effect is applied TEST_P(SingleEffectComparisonTest, SimpleProcess) { SCOPED_TRACE(testing::Message() << " sampleRate: " << mSampleRate << " loopCount: " << mLoopCount << " paramIdx " << mParamIdx); // Initialize mono input buffer with deterministic pseudo-random values std::vector<int16_t> monoInput(mTotalFrameCount); std::vector<int16_t> monoFarInput(mTotalFrameCount); std::minstd_rand gen(mSampleRate); std::uniform_int_distribution<int16_t> dis(INT16_MIN, INT16_MAX); for (auto& in : monoInput) { in = dis(gen); } if (isAECEffect(mUuid)) { for (auto& farIn : monoFarInput) { farIn = dis(gen); } } // Apply effect on mono channel EffectTestHelper monoEffect(mUuid, AUDIO_CHANNEL_INDEX_MASK_1, mSampleRate, mLoopCount); ASSERT_NO_FATAL_FAILURE(monoEffect.createEffect()); ASSERT_NO_FATAL_FAILURE(monoEffect.setConfig(isAECEffect(mUuid))); ASSERT_NO_FATAL_FAILURE(setPreProcParams(mUuid, monoEffect, mParamIdx)); std::vector<int16_t> monoOutput(mTotalFrameCount); ASSERT_NO_FATAL_FAILURE( monoEffect.process(monoInput.data(), monoOutput.data(), isAECEffect(mUuid))); if (isAECEffect(mUuid)) { ASSERT_NO_FATAL_FAILURE(monoEffect.process_reverse(monoFarInput.data(), monoOutput.data())); } ASSERT_NO_FATAL_FAILURE(monoEffect.releaseEffect()); for (size_t chMask : EffectTestHelper::kChMasks) { size_t channelCount = audio_channel_count_from_in_mask(chMask); EffectTestHelper testEffect(mUuid, chMask, mSampleRate, mLoopCount); ASSERT_NO_FATAL_FAILURE(testEffect.createEffect()); ASSERT_NO_FATAL_FAILURE(testEffect.setConfig(isAECEffect(mUuid))); ASSERT_NO_FATAL_FAILURE(setPreProcParams(mUuid, testEffect, mParamIdx)); std::vector<int16_t> testInput(mTotalFrameCount * channelCount); std::vector<int16_t> testFarInput(mTotalFrameCount * channelCount); // Repeat mono channel data to all the channels // adjust_channels() zero fills channels > 2, hence can't be used here for (size_t i = 0; i < mTotalFrameCount; ++i) { auto* fpInput = &testInput[i * channelCount]; std::fill(fpInput, fpInput + channelCount, monoInput[i]); } if (isAECEffect(mUuid)) { for (size_t i = 0; i < mTotalFrameCount; ++i) { auto* fpFarInput = &testFarInput[i * channelCount]; std::fill(fpFarInput, fpFarInput + channelCount, monoFarInput[i]); } } std::vector<int16_t> testOutput(mTotalFrameCount * channelCount); ASSERT_NO_FATAL_FAILURE( testEffect.process(testInput.data(), testOutput.data(), isAECEffect(mUuid))); if (isAECEffect(mUuid)) { ASSERT_NO_FATAL_FAILURE( testEffect.process_reverse(testFarInput.data(), testOutput.data())); } ASSERT_NO_FATAL_FAILURE(testEffect.releaseEffect()); // Adjust the test output to mono channel std::vector<int16_t> monoTestOutput(mTotalFrameCount); adjust_channels(testOutput.data(), channelCount, monoTestOutput.data(), FCC_1, sizeof(int16_t), mTotalFrameCount * sizeof(int16_t) * channelCount); ASSERT_EQ(0, memcmp(monoOutput.data(), monoTestOutput.data(), mTotalFrameCount * sizeof(int16_t))) << "Mono channel output does not match with reference output \n"; } } INSTANTIATE_TEST_SUITE_P( PreProcTestAll, SingleEffectComparisonTest, ::testing::Combine(::testing::Range(0, (int)EffectTestHelper::kNumSampleRates), ::testing::Range(0, (int)EffectTestHelper::kNumLoopCounts), ::testing::Range(0, (int)kNumPreProcParams))); int main(int argc, char** argv) { ::testing::InitGoogleTest(&argc, argv); int status = RUN_ALL_TESTS(); ALOGV("Test result = %d", status); return status; }
media/libeffects/preprocessing/tests/EffectTestHelper.cpp 0 → 100644 +99 −0 Original line number Diff line number Diff line /* * Copyright 2021 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 "EffectTestHelper.h" extern audio_effect_library_t AUDIO_EFFECT_LIBRARY_INFO_SYM; void EffectTestHelper::createEffect() { int status = AUDIO_EFFECT_LIBRARY_INFO_SYM.create_effect(mUuid, 1, 1, &mEffectHandle); ASSERT_EQ(status, 0) << "create_effect returned an error " << status; } void EffectTestHelper::releaseEffect() { int status = AUDIO_EFFECT_LIBRARY_INFO_SYM.release_effect(mEffectHandle); ASSERT_EQ(status, 0) << "release_effect returned an error " << status; } void EffectTestHelper::setConfig(bool configReverse) { effect_config_t config{}; config.inputCfg.samplingRate = config.outputCfg.samplingRate = mSampleRate; config.inputCfg.channels = config.outputCfg.channels = mChMask; config.inputCfg.format = config.outputCfg.format = AUDIO_FORMAT_PCM_16_BIT; int reply = 0; uint32_t replySize = sizeof(reply); int status = (*mEffectHandle) ->command(mEffectHandle, EFFECT_CMD_SET_CONFIG, sizeof(effect_config_t), &config, &replySize, &reply); ASSERT_EQ(status, 0) << "set_config returned an error " << status; ASSERT_EQ(reply, 0) << "set_config reply non zero " << reply; if (configReverse) { int status = (*mEffectHandle) ->command(mEffectHandle, EFFECT_CMD_SET_CONFIG_REVERSE, sizeof(effect_config_t), &config, &replySize, &reply); ASSERT_EQ(status, 0) << "set_config_reverse returned an error " << status; ASSERT_EQ(reply, 0) << "set_config_reverse reply non zero " << reply; } status = (*mEffectHandle) ->command(mEffectHandle, EFFECT_CMD_ENABLE, 0, nullptr, &replySize, &reply); ASSERT_EQ(status, 0) << "cmd_enable returned an error " << status; ASSERT_EQ(reply, 0) << "cmd_enable reply non zero " << reply; } void EffectTestHelper::setParam(uint32_t type, uint32_t value) { int reply = 0; uint32_t replySize = sizeof(reply); uint32_t paramData[2] = {type, value}; auto effectParam = (effect_param_t*)malloc(sizeof(effect_param_t) + sizeof(paramData)); memcpy(&effectParam->data[0], ¶mData[0], sizeof(paramData)); effectParam->psize = sizeof(paramData[0]); effectParam->vsize = sizeof(paramData[1]); int status = (*mEffectHandle) ->command(mEffectHandle, EFFECT_CMD_SET_PARAM, sizeof(effect_param_t) + sizeof(paramData), effectParam, &replySize, &reply); free(effectParam); ASSERT_EQ(status, 0) << "set_param returned an error " << status; ASSERT_EQ(reply, 0) << "set_param reply non zero " << reply; } void EffectTestHelper::process(int16_t* input, int16_t* output, bool setAecEchoDelay) { audio_buffer_t inBuffer = {.frameCount = mFrameCount, .s16 = input}; audio_buffer_t outBuffer = {.frameCount = mFrameCount, .s16 = output}; for (size_t i = 0; i < mLoopCount; i++) { if (setAecEchoDelay) ASSERT_NO_FATAL_FAILURE(setParam(AEC_PARAM_ECHO_DELAY, kAECDelay)); int status = (*mEffectHandle)->process(mEffectHandle, &inBuffer, &outBuffer); ASSERT_EQ(status, 0) << "process returned an error " << status; inBuffer.s16 += mFrameCount * mChannelCount; outBuffer.s16 += mFrameCount * mChannelCount; } } void EffectTestHelper::process_reverse(int16_t* farInput, int16_t* output) { audio_buffer_t farInBuffer = {.frameCount = mFrameCount, .s16 = farInput}; audio_buffer_t outBuffer = {.frameCount = mFrameCount, .s16 = output}; for (size_t i = 0; i < mLoopCount; i++) { int status = (*mEffectHandle)->process_reverse(mEffectHandle, &farInBuffer, &outBuffer); ASSERT_EQ(status, 0) << "process returned an error " << status; farInBuffer.s16 += mFrameCount * mChannelCount; outBuffer.s16 += mFrameCount * mChannelCount; } }
media/libeffects/preprocessing/tests/EffectTestHelper.h 0 → 100644 +109 −0 Original line number Diff line number Diff line /* * Copyright 2021 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. */ #pragma once #include <array> #include <audio_effects/effect_aec.h> #include <audio_utils/channels.h> #include <audio_utils/primitives.h> #include <climits> #include <cstdlib> #include <gtest/gtest.h> #include <hardware/audio_effect.h> #include <log/log.h> #include <random> #include <stdint.h> #include <system/audio.h> #include <vector> template <typename T> static float computeSnr(const T* ref, const T* tst, size_t count) { double signal{}; double noise{}; for (size_t i = 0; i < count; ++i) { const double value(ref[i]); const double diff(tst[i] - value); signal += value * value; noise += diff * diff; } // Initialized to large value to handle // cases where ref and tst match exactly float snr = FLT_MAX; if (signal > 0.0f && noise > 0.0f) { snr = 10.f * log(signal / noise); } return snr; } class EffectTestHelper { public: EffectTestHelper(const effect_uuid_t* uuid, size_t chMask, size_t sampleRate, size_t loopCount) : mUuid(uuid), mChMask(chMask), mChannelCount(audio_channel_count_from_in_mask(mChMask)), mSampleRate(sampleRate), mFrameCount(mSampleRate * kTenMilliSecVal), mLoopCount(loopCount) {} void createEffect(); void releaseEffect(); void setConfig(bool configReverse); void setParam(uint32_t type, uint32_t val); void process(int16_t* input, int16_t* output, bool setAecEchoDelay); void process_reverse(int16_t* farInput, int16_t* output); // Corresponds to SNR for 1 bit difference between two int16_t signals static constexpr float kSNRThreshold = 90.308998; static constexpr audio_channel_mask_t kChMasks[] = { AUDIO_CHANNEL_IN_MONO, AUDIO_CHANNEL_IN_STEREO, AUDIO_CHANNEL_IN_FRONT_BACK, AUDIO_CHANNEL_IN_6, AUDIO_CHANNEL_IN_2POINT0POINT2, AUDIO_CHANNEL_IN_2POINT1POINT2, AUDIO_CHANNEL_IN_3POINT0POINT2, AUDIO_CHANNEL_IN_3POINT1POINT2, AUDIO_CHANNEL_IN_5POINT1, AUDIO_CHANNEL_IN_VOICE_UPLINK_MONO, AUDIO_CHANNEL_IN_VOICE_DNLINK_MONO, AUDIO_CHANNEL_IN_VOICE_CALL_MONO, }; static constexpr float kTenMilliSecVal = 0.01; static constexpr size_t kNumChMasks = std::size(kChMasks); static constexpr size_t kSampleRates[] = {8000, 16000, 24000, 32000, 48000}; static constexpr size_t kNumSampleRates = std::size(kSampleRates); static constexpr size_t kLoopCounts[] = {1, 4}; static constexpr size_t kNumLoopCounts = std::size(kLoopCounts); static constexpr size_t kAECDelay = 0; private: const effect_uuid_t* mUuid; const size_t mChMask; const size_t mChannelCount; const size_t mSampleRate; const size_t mFrameCount; const size_t mLoopCount; effect_handle_t mEffectHandle{}; };