xref: /aosp_15_r20/hardware/interfaces/audio/aidl/vts/EffectHelper.h (revision 4d7e907c777eeecc4c5bd7cf640a754fac206ff7)
1*4d7e907cSAndroid Build Coastguard Worker /*
2*4d7e907cSAndroid Build Coastguard Worker  * Copyright (C) 2022 The Android Open Source Project
3*4d7e907cSAndroid Build Coastguard Worker  *
4*4d7e907cSAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*4d7e907cSAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*4d7e907cSAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*4d7e907cSAndroid Build Coastguard Worker  *
8*4d7e907cSAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*4d7e907cSAndroid Build Coastguard Worker  *
10*4d7e907cSAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*4d7e907cSAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*4d7e907cSAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*4d7e907cSAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*4d7e907cSAndroid Build Coastguard Worker  * limitations under the License.
15*4d7e907cSAndroid Build Coastguard Worker  */
16*4d7e907cSAndroid Build Coastguard Worker 
17*4d7e907cSAndroid Build Coastguard Worker #pragma once
18*4d7e907cSAndroid Build Coastguard Worker 
19*4d7e907cSAndroid Build Coastguard Worker #include <algorithm>
20*4d7e907cSAndroid Build Coastguard Worker #include <memory>
21*4d7e907cSAndroid Build Coastguard Worker #include <optional>
22*4d7e907cSAndroid Build Coastguard Worker #include <string>
23*4d7e907cSAndroid Build Coastguard Worker #include <type_traits>
24*4d7e907cSAndroid Build Coastguard Worker #include <unordered_map>
25*4d7e907cSAndroid Build Coastguard Worker #include <utility>
26*4d7e907cSAndroid Build Coastguard Worker #include <vector>
27*4d7e907cSAndroid Build Coastguard Worker 
28*4d7e907cSAndroid Build Coastguard Worker #include <Utils.h>
29*4d7e907cSAndroid Build Coastguard Worker #include <aidl/android/hardware/audio/effect/IEffect.h>
30*4d7e907cSAndroid Build Coastguard Worker #include <aidl/android/hardware/audio/effect/IFactory.h>
31*4d7e907cSAndroid Build Coastguard Worker #include <aidl/android/media/audio/common/AudioChannelLayout.h>
32*4d7e907cSAndroid Build Coastguard Worker #include <android/binder_auto_utils.h>
33*4d7e907cSAndroid Build Coastguard Worker #include <fmq/AidlMessageQueue.h>
34*4d7e907cSAndroid Build Coastguard Worker #include <gtest/gtest.h>
35*4d7e907cSAndroid Build Coastguard Worker #include <system/audio_aidl_utils.h>
36*4d7e907cSAndroid Build Coastguard Worker #include <system/audio_effects/aidl_effects_utils.h>
37*4d7e907cSAndroid Build Coastguard Worker #include <system/audio_effects/effect_uuid.h>
38*4d7e907cSAndroid Build Coastguard Worker 
39*4d7e907cSAndroid Build Coastguard Worker #include "EffectFactoryHelper.h"
40*4d7e907cSAndroid Build Coastguard Worker #include "TestUtils.h"
41*4d7e907cSAndroid Build Coastguard Worker #include "pffft.hpp"
42*4d7e907cSAndroid Build Coastguard Worker 
43*4d7e907cSAndroid Build Coastguard Worker using namespace android;
44*4d7e907cSAndroid Build Coastguard Worker using aidl::android::hardware::audio::effect::CommandId;
45*4d7e907cSAndroid Build Coastguard Worker using aidl::android::hardware::audio::effect::Descriptor;
46*4d7e907cSAndroid Build Coastguard Worker using aidl::android::hardware::audio::effect::getEffectTypeUuidSpatializer;
47*4d7e907cSAndroid Build Coastguard Worker using aidl::android::hardware::audio::effect::getRange;
48*4d7e907cSAndroid Build Coastguard Worker using aidl::android::hardware::audio::effect::IEffect;
49*4d7e907cSAndroid Build Coastguard Worker using aidl::android::hardware::audio::effect::isRangeValid;
50*4d7e907cSAndroid Build Coastguard Worker using aidl::android::hardware::audio::effect::kEffectTypeUuidSpatializer;
51*4d7e907cSAndroid Build Coastguard Worker using aidl::android::hardware::audio::effect::kEventFlagDataMqNotEmpty;
52*4d7e907cSAndroid Build Coastguard Worker using aidl::android::hardware::audio::effect::kEventFlagDataMqUpdate;
53*4d7e907cSAndroid Build Coastguard Worker using aidl::android::hardware::audio::effect::kEventFlagNotEmpty;
54*4d7e907cSAndroid Build Coastguard Worker using aidl::android::hardware::audio::effect::kReopenSupportedVersion;
55*4d7e907cSAndroid Build Coastguard Worker using aidl::android::hardware::audio::effect::Parameter;
56*4d7e907cSAndroid Build Coastguard Worker using aidl::android::hardware::audio::effect::Range;
57*4d7e907cSAndroid Build Coastguard Worker using aidl::android::hardware::audio::effect::Spatializer;
58*4d7e907cSAndroid Build Coastguard Worker using aidl::android::hardware::audio::effect::State;
59*4d7e907cSAndroid Build Coastguard Worker using aidl::android::hardware::common::fmq::SynchronizedReadWrite;
60*4d7e907cSAndroid Build Coastguard Worker using aidl::android::media::audio::common::AudioChannelLayout;
61*4d7e907cSAndroid Build Coastguard Worker using aidl::android::media::audio::common::AudioFormatDescription;
62*4d7e907cSAndroid Build Coastguard Worker using aidl::android::media::audio::common::AudioFormatType;
63*4d7e907cSAndroid Build Coastguard Worker using aidl::android::media::audio::common::AudioUuid;
64*4d7e907cSAndroid Build Coastguard Worker using aidl::android::media::audio::common::PcmType;
65*4d7e907cSAndroid Build Coastguard Worker using ::android::audio::utils::toString;
66*4d7e907cSAndroid Build Coastguard Worker using ::android::hardware::EventFlag;
67*4d7e907cSAndroid Build Coastguard Worker 
68*4d7e907cSAndroid Build Coastguard Worker const AudioFormatDescription kDefaultFormatDescription = {
69*4d7e907cSAndroid Build Coastguard Worker         .type = AudioFormatType::PCM, .pcm = PcmType::FLOAT_32_BIT, .encoding = ""};
70*4d7e907cSAndroid Build Coastguard Worker 
71*4d7e907cSAndroid Build Coastguard Worker typedef ::android::AidlMessageQueue<IEffect::Status,
72*4d7e907cSAndroid Build Coastguard Worker                                     ::aidl::android::hardware::common::fmq::SynchronizedReadWrite>
73*4d7e907cSAndroid Build Coastguard Worker         StatusMQ;
74*4d7e907cSAndroid Build Coastguard Worker typedef ::android::AidlMessageQueue<float,
75*4d7e907cSAndroid Build Coastguard Worker                                     ::aidl::android::hardware::common::fmq::SynchronizedReadWrite>
76*4d7e907cSAndroid Build Coastguard Worker         DataMQ;
77*4d7e907cSAndroid Build Coastguard Worker 
getPrefix(Descriptor & descriptor)78*4d7e907cSAndroid Build Coastguard Worker static inline std::string getPrefix(Descriptor& descriptor) {
79*4d7e907cSAndroid Build Coastguard Worker     std::string prefix = "Implementor_" + descriptor.common.implementor + "_name_" +
80*4d7e907cSAndroid Build Coastguard Worker                          descriptor.common.name + "_UUID_" + toString(descriptor.common.id.uuid);
81*4d7e907cSAndroid Build Coastguard Worker     std::replace_if(
82*4d7e907cSAndroid Build Coastguard Worker             prefix.begin(), prefix.end(), [](const char c) { return !std::isalnum(c); }, '_');
83*4d7e907cSAndroid Build Coastguard Worker     return prefix;
84*4d7e907cSAndroid Build Coastguard Worker }
85*4d7e907cSAndroid Build Coastguard Worker 
86*4d7e907cSAndroid Build Coastguard Worker static constexpr float kMaxAudioSampleValue = 1;
87*4d7e907cSAndroid Build Coastguard Worker static constexpr int kSamplingFrequency = 44100;
88*4d7e907cSAndroid Build Coastguard Worker 
89*4d7e907cSAndroid Build Coastguard Worker class EffectHelper {
90*4d7e907cSAndroid Build Coastguard Worker   public:
91*4d7e907cSAndroid Build Coastguard Worker     void create(std::shared_ptr<IFactory> factory, std::shared_ptr<IEffect>& effect,
92*4d7e907cSAndroid Build Coastguard Worker                 Descriptor& desc, binder_status_t status = EX_NONE) {
93*4d7e907cSAndroid Build Coastguard Worker         ASSERT_NE(factory, nullptr);
94*4d7e907cSAndroid Build Coastguard Worker         auto& id = desc.common.id;
95*4d7e907cSAndroid Build Coastguard Worker         ASSERT_STATUS(status, factory->createEffect(id.uuid, &effect));
96*4d7e907cSAndroid Build Coastguard Worker         if (status == EX_NONE) {
97*4d7e907cSAndroid Build Coastguard Worker             ASSERT_NE(effect, nullptr) << toString(id.uuid);
98*4d7e907cSAndroid Build Coastguard Worker             ASSERT_NO_FATAL_FAILURE(expectState(effect, State::INIT));
99*4d7e907cSAndroid Build Coastguard Worker         }
100*4d7e907cSAndroid Build Coastguard Worker         mIsSpatializer = id.type == getEffectTypeUuidSpatializer();
101*4d7e907cSAndroid Build Coastguard Worker         mDescriptor = desc;
102*4d7e907cSAndroid Build Coastguard Worker     }
103*4d7e907cSAndroid Build Coastguard Worker 
destroyIgnoreRet(std::shared_ptr<IFactory> factory,std::shared_ptr<IEffect> effect)104*4d7e907cSAndroid Build Coastguard Worker     static void destroyIgnoreRet(std::shared_ptr<IFactory> factory,
105*4d7e907cSAndroid Build Coastguard Worker                                  std::shared_ptr<IEffect> effect) {
106*4d7e907cSAndroid Build Coastguard Worker         if (factory && effect) {
107*4d7e907cSAndroid Build Coastguard Worker             factory->destroyEffect(effect);
108*4d7e907cSAndroid Build Coastguard Worker         }
109*4d7e907cSAndroid Build Coastguard Worker     }
110*4d7e907cSAndroid Build Coastguard Worker 
111*4d7e907cSAndroid Build Coastguard Worker     static void destroy(std::shared_ptr<IFactory> factory, std::shared_ptr<IEffect> effect,
112*4d7e907cSAndroid Build Coastguard Worker                         binder_status_t status = EX_NONE) {
113*4d7e907cSAndroid Build Coastguard Worker         ASSERT_NE(factory, nullptr);
114*4d7e907cSAndroid Build Coastguard Worker         ASSERT_NE(effect, nullptr);
115*4d7e907cSAndroid Build Coastguard Worker         ASSERT_STATUS(status, factory->destroyEffect(effect));
116*4d7e907cSAndroid Build Coastguard Worker     }
117*4d7e907cSAndroid Build Coastguard Worker 
118*4d7e907cSAndroid Build Coastguard Worker     void open(std::shared_ptr<IEffect> effect, const Parameter::Common& common,
119*4d7e907cSAndroid Build Coastguard Worker               const std::optional<Parameter::Specific>& specific, IEffect::OpenEffectReturn* ret,
120*4d7e907cSAndroid Build Coastguard Worker               binder_status_t status = EX_NONE) {
121*4d7e907cSAndroid Build Coastguard Worker         ASSERT_NE(effect, nullptr);
122*4d7e907cSAndroid Build Coastguard Worker         ASSERT_STATUS(status, effect->open(common, specific, ret));
123*4d7e907cSAndroid Build Coastguard Worker         if (status != EX_NONE) {
124*4d7e907cSAndroid Build Coastguard Worker             return;
125*4d7e907cSAndroid Build Coastguard Worker         }
126*4d7e907cSAndroid Build Coastguard Worker 
127*4d7e907cSAndroid Build Coastguard Worker         ASSERT_TRUE(expectState(effect, State::IDLE));
128*4d7e907cSAndroid Build Coastguard Worker         updateFrameSize(common);
129*4d7e907cSAndroid Build Coastguard Worker     }
130*4d7e907cSAndroid Build Coastguard Worker 
131*4d7e907cSAndroid Build Coastguard Worker     void open(std::shared_ptr<IEffect> effect, int session = 0, binder_status_t status = EX_NONE) {
132*4d7e907cSAndroid Build Coastguard Worker         ASSERT_NE(effect, nullptr);
133*4d7e907cSAndroid Build Coastguard Worker         Parameter::Common common = createParamCommon(session);
134*4d7e907cSAndroid Build Coastguard Worker         IEffect::OpenEffectReturn ret;
135*4d7e907cSAndroid Build Coastguard Worker         ASSERT_NO_FATAL_FAILURE(open(effect, common, std::nullopt /* specific */, &ret, status));
136*4d7e907cSAndroid Build Coastguard Worker     }
137*4d7e907cSAndroid Build Coastguard Worker 
138*4d7e907cSAndroid Build Coastguard Worker     void reopen(std::shared_ptr<IEffect> effect, const Parameter::Common& common,
139*4d7e907cSAndroid Build Coastguard Worker                 IEffect::OpenEffectReturn* ret, binder_status_t status = EX_NONE) {
140*4d7e907cSAndroid Build Coastguard Worker         ASSERT_NE(effect, nullptr);
141*4d7e907cSAndroid Build Coastguard Worker         ASSERT_STATUS(status, effect->reopen(ret));
142*4d7e907cSAndroid Build Coastguard Worker         if (status != EX_NONE) {
143*4d7e907cSAndroid Build Coastguard Worker             return;
144*4d7e907cSAndroid Build Coastguard Worker         }
145*4d7e907cSAndroid Build Coastguard Worker         updateFrameSize(common);
146*4d7e907cSAndroid Build Coastguard Worker     }
147*4d7e907cSAndroid Build Coastguard Worker 
closeIgnoreRet(std::shared_ptr<IEffect> effect)148*4d7e907cSAndroid Build Coastguard Worker     static void closeIgnoreRet(std::shared_ptr<IEffect> effect) {
149*4d7e907cSAndroid Build Coastguard Worker         if (effect) {
150*4d7e907cSAndroid Build Coastguard Worker             effect->close();
151*4d7e907cSAndroid Build Coastguard Worker         }
152*4d7e907cSAndroid Build Coastguard Worker     }
153*4d7e907cSAndroid Build Coastguard Worker 
154*4d7e907cSAndroid Build Coastguard Worker     static void close(std::shared_ptr<IEffect> effect, binder_status_t status = EX_NONE) {
155*4d7e907cSAndroid Build Coastguard Worker         if (effect) {
156*4d7e907cSAndroid Build Coastguard Worker             ASSERT_STATUS(status, effect->close());
157*4d7e907cSAndroid Build Coastguard Worker             if (status == EX_NONE) {
158*4d7e907cSAndroid Build Coastguard Worker                 ASSERT_TRUE(expectState(effect, State::INIT));
159*4d7e907cSAndroid Build Coastguard Worker             }
160*4d7e907cSAndroid Build Coastguard Worker         }
161*4d7e907cSAndroid Build Coastguard Worker     }
162*4d7e907cSAndroid Build Coastguard Worker 
163*4d7e907cSAndroid Build Coastguard Worker     static void getDescriptor(std::shared_ptr<IEffect> effect, Descriptor& desc,
164*4d7e907cSAndroid Build Coastguard Worker                               binder_status_t status = EX_NONE) {
165*4d7e907cSAndroid Build Coastguard Worker         ASSERT_NE(effect, nullptr);
166*4d7e907cSAndroid Build Coastguard Worker         ASSERT_STATUS(status, effect->getDescriptor(&desc));
167*4d7e907cSAndroid Build Coastguard Worker     }
168*4d7e907cSAndroid Build Coastguard Worker 
expectState(std::shared_ptr<IEffect> effect,State expectState)169*4d7e907cSAndroid Build Coastguard Worker     static bool expectState(std::shared_ptr<IEffect> effect, State expectState) {
170*4d7e907cSAndroid Build Coastguard Worker         if (effect == nullptr) return false;
171*4d7e907cSAndroid Build Coastguard Worker 
172*4d7e907cSAndroid Build Coastguard Worker         if (State state; EX_NONE != effect->getState(&state).getStatus() || expectState != state) {
173*4d7e907cSAndroid Build Coastguard Worker             return false;
174*4d7e907cSAndroid Build Coastguard Worker         }
175*4d7e907cSAndroid Build Coastguard Worker 
176*4d7e907cSAndroid Build Coastguard Worker         return true;
177*4d7e907cSAndroid Build Coastguard Worker     }
178*4d7e907cSAndroid Build Coastguard Worker 
commandIgnoreRet(std::shared_ptr<IEffect> effect,CommandId command)179*4d7e907cSAndroid Build Coastguard Worker     static void commandIgnoreRet(std::shared_ptr<IEffect> effect, CommandId command) {
180*4d7e907cSAndroid Build Coastguard Worker         if (effect) {
181*4d7e907cSAndroid Build Coastguard Worker             effect->command(command);
182*4d7e907cSAndroid Build Coastguard Worker         }
183*4d7e907cSAndroid Build Coastguard Worker     }
184*4d7e907cSAndroid Build Coastguard Worker 
185*4d7e907cSAndroid Build Coastguard Worker     static void command(std::shared_ptr<IEffect> effect, CommandId command,
186*4d7e907cSAndroid Build Coastguard Worker                         binder_status_t status = EX_NONE) {
187*4d7e907cSAndroid Build Coastguard Worker         ASSERT_NE(effect, nullptr);
188*4d7e907cSAndroid Build Coastguard Worker         ASSERT_STATUS(status, effect->command(command));
189*4d7e907cSAndroid Build Coastguard Worker         if (status != EX_NONE) {
190*4d7e907cSAndroid Build Coastguard Worker             return;
191*4d7e907cSAndroid Build Coastguard Worker         }
192*4d7e907cSAndroid Build Coastguard Worker 
193*4d7e907cSAndroid Build Coastguard Worker         switch (command) {
194*4d7e907cSAndroid Build Coastguard Worker             case CommandId::START:
195*4d7e907cSAndroid Build Coastguard Worker                 ASSERT_TRUE(expectState(effect, State::PROCESSING));
196*4d7e907cSAndroid Build Coastguard Worker                 break;
197*4d7e907cSAndroid Build Coastguard Worker             case CommandId::STOP:
198*4d7e907cSAndroid Build Coastguard Worker                 ASSERT_TRUE(expectState(effect, State::IDLE) ||
199*4d7e907cSAndroid Build Coastguard Worker                             expectState(effect, State::DRAINING));
200*4d7e907cSAndroid Build Coastguard Worker                 break;
201*4d7e907cSAndroid Build Coastguard Worker             case CommandId::RESET:
202*4d7e907cSAndroid Build Coastguard Worker                 ASSERT_TRUE(expectState(effect, State::IDLE));
203*4d7e907cSAndroid Build Coastguard Worker                 break;
204*4d7e907cSAndroid Build Coastguard Worker             default:
205*4d7e907cSAndroid Build Coastguard Worker                 return;
206*4d7e907cSAndroid Build Coastguard Worker         }
207*4d7e907cSAndroid Build Coastguard Worker     }
208*4d7e907cSAndroid Build Coastguard Worker 
writeToFmq(std::unique_ptr<StatusMQ> & statusMq,std::unique_ptr<DataMQ> & dataMq,const std::vector<float> & buffer,int version)209*4d7e907cSAndroid Build Coastguard Worker     static void writeToFmq(std::unique_ptr<StatusMQ>& statusMq, std::unique_ptr<DataMQ>& dataMq,
210*4d7e907cSAndroid Build Coastguard Worker                            const std::vector<float>& buffer, int version) {
211*4d7e907cSAndroid Build Coastguard Worker         const size_t available = dataMq->availableToWrite();
212*4d7e907cSAndroid Build Coastguard Worker         ASSERT_NE(0Ul, available);
213*4d7e907cSAndroid Build Coastguard Worker         auto bufferFloats = buffer.size();
214*4d7e907cSAndroid Build Coastguard Worker         auto floatsToWrite = std::min(available, bufferFloats);
215*4d7e907cSAndroid Build Coastguard Worker         ASSERT_TRUE(dataMq->write(buffer.data(), floatsToWrite));
216*4d7e907cSAndroid Build Coastguard Worker 
217*4d7e907cSAndroid Build Coastguard Worker         EventFlag* efGroup;
218*4d7e907cSAndroid Build Coastguard Worker         ASSERT_EQ(::android::OK,
219*4d7e907cSAndroid Build Coastguard Worker                   EventFlag::createEventFlag(statusMq->getEventFlagWord(), &efGroup));
220*4d7e907cSAndroid Build Coastguard Worker         ASSERT_NE(nullptr, efGroup);
221*4d7e907cSAndroid Build Coastguard Worker         efGroup->wake(version >= kReopenSupportedVersion ? kEventFlagDataMqNotEmpty
222*4d7e907cSAndroid Build Coastguard Worker                                                          : kEventFlagNotEmpty);
223*4d7e907cSAndroid Build Coastguard Worker         ASSERT_EQ(::android::OK, EventFlag::deleteEventFlag(&efGroup));
224*4d7e907cSAndroid Build Coastguard Worker     }
225*4d7e907cSAndroid Build Coastguard Worker 
226*4d7e907cSAndroid Build Coastguard Worker     static void readFromFmq(std::unique_ptr<StatusMQ>& statusMq, size_t statusNum,
227*4d7e907cSAndroid Build Coastguard Worker                             std::unique_ptr<DataMQ>& dataMq, size_t expectFloats,
228*4d7e907cSAndroid Build Coastguard Worker                             std::vector<float>& buffer,
229*4d7e907cSAndroid Build Coastguard Worker                             std::optional<int> expectStatus = STATUS_OK) {
230*4d7e907cSAndroid Build Coastguard Worker         if (0 == statusNum) {
231*4d7e907cSAndroid Build Coastguard Worker             ASSERT_EQ(0ul, statusMq->availableToRead());
232*4d7e907cSAndroid Build Coastguard Worker             return;
233*4d7e907cSAndroid Build Coastguard Worker         }
234*4d7e907cSAndroid Build Coastguard Worker         IEffect::Status status{};
235*4d7e907cSAndroid Build Coastguard Worker         ASSERT_TRUE(statusMq->readBlocking(&status, statusNum));
236*4d7e907cSAndroid Build Coastguard Worker         if (expectStatus.has_value()) {
237*4d7e907cSAndroid Build Coastguard Worker             ASSERT_EQ(expectStatus.value(), status.status);
238*4d7e907cSAndroid Build Coastguard Worker         }
239*4d7e907cSAndroid Build Coastguard Worker 
240*4d7e907cSAndroid Build Coastguard Worker         ASSERT_EQ(expectFloats, (unsigned)status.fmqProduced);
241*4d7e907cSAndroid Build Coastguard Worker         ASSERT_EQ(expectFloats, dataMq->availableToRead());
242*4d7e907cSAndroid Build Coastguard Worker         if (expectFloats != 0) {
243*4d7e907cSAndroid Build Coastguard Worker             ASSERT_TRUE(dataMq->read(buffer.data(), expectFloats));
244*4d7e907cSAndroid Build Coastguard Worker         }
245*4d7e907cSAndroid Build Coastguard Worker     }
246*4d7e907cSAndroid Build Coastguard Worker 
expectDataMqUpdateEventFlag(std::unique_ptr<StatusMQ> & statusMq)247*4d7e907cSAndroid Build Coastguard Worker     static void expectDataMqUpdateEventFlag(std::unique_ptr<StatusMQ>& statusMq) {
248*4d7e907cSAndroid Build Coastguard Worker         EventFlag* efGroup;
249*4d7e907cSAndroid Build Coastguard Worker         ASSERT_EQ(::android::OK,
250*4d7e907cSAndroid Build Coastguard Worker                   EventFlag::createEventFlag(statusMq->getEventFlagWord(), &efGroup));
251*4d7e907cSAndroid Build Coastguard Worker         ASSERT_NE(nullptr, efGroup);
252*4d7e907cSAndroid Build Coastguard Worker         uint32_t efState = 0;
253*4d7e907cSAndroid Build Coastguard Worker         EXPECT_EQ(::android::OK, efGroup->wait(kEventFlagDataMqUpdate, &efState, 1'000'000 /*1ms*/,
254*4d7e907cSAndroid Build Coastguard Worker                                                true /* retry */));
255*4d7e907cSAndroid Build Coastguard Worker         EXPECT_TRUE(efState & kEventFlagDataMqUpdate);
256*4d7e907cSAndroid Build Coastguard Worker     }
257*4d7e907cSAndroid Build Coastguard Worker 
258*4d7e907cSAndroid Build Coastguard Worker     Parameter::Common createParamCommon(int session = 0, int ioHandle = -1, int iSampleRate = 48000,
259*4d7e907cSAndroid Build Coastguard Worker                                         int oSampleRate = 48000, long iFrameCount = 0x100,
260*4d7e907cSAndroid Build Coastguard Worker                                         long oFrameCount = 0x100) {
261*4d7e907cSAndroid Build Coastguard Worker         AudioChannelLayout inputLayout = AudioChannelLayout::make<AudioChannelLayout::layoutMask>(
262*4d7e907cSAndroid Build Coastguard Worker                 AudioChannelLayout::LAYOUT_STEREO);
263*4d7e907cSAndroid Build Coastguard Worker         AudioChannelLayout outputLayout = inputLayout;
264*4d7e907cSAndroid Build Coastguard Worker 
265*4d7e907cSAndroid Build Coastguard Worker         // query supported input layout and use it as the default parameter in common
266*4d7e907cSAndroid Build Coastguard Worker         if (mIsSpatializer && isRangeValid<Range::spatializer>(Spatializer::supportedChannelLayout,
267*4d7e907cSAndroid Build Coastguard Worker                                                                mDescriptor.capability)) {
268*4d7e907cSAndroid Build Coastguard Worker             const auto layoutRange = getRange<Range::spatializer, Range::SpatializerRange>(
269*4d7e907cSAndroid Build Coastguard Worker                     mDescriptor.capability, Spatializer::supportedChannelLayout);
270*4d7e907cSAndroid Build Coastguard Worker             if (std::vector<AudioChannelLayout> layouts;
271*4d7e907cSAndroid Build Coastguard Worker                 layoutRange &&
272*4d7e907cSAndroid Build Coastguard Worker                 0 != (layouts = layoutRange->min.get<Spatializer::supportedChannelLayout>())
273*4d7e907cSAndroid Build Coastguard Worker                                 .size()) {
274*4d7e907cSAndroid Build Coastguard Worker                 inputLayout = layouts[0];
275*4d7e907cSAndroid Build Coastguard Worker             }
276*4d7e907cSAndroid Build Coastguard Worker         }
277*4d7e907cSAndroid Build Coastguard Worker 
278*4d7e907cSAndroid Build Coastguard Worker         return createParamCommon(session, ioHandle, iSampleRate, oSampleRate, iFrameCount,
279*4d7e907cSAndroid Build Coastguard Worker                                  oFrameCount, inputLayout, outputLayout);
280*4d7e907cSAndroid Build Coastguard Worker     }
281*4d7e907cSAndroid Build Coastguard Worker 
createParamCommon(int session,int ioHandle,int iSampleRate,int oSampleRate,long iFrameCount,long oFrameCount,AudioChannelLayout inputChannelLayout,AudioChannelLayout outputChannelLayout)282*4d7e907cSAndroid Build Coastguard Worker     static Parameter::Common createParamCommon(int session, int ioHandle, int iSampleRate,
283*4d7e907cSAndroid Build Coastguard Worker                                                int oSampleRate, long iFrameCount, long oFrameCount,
284*4d7e907cSAndroid Build Coastguard Worker                                                AudioChannelLayout inputChannelLayout,
285*4d7e907cSAndroid Build Coastguard Worker                                                AudioChannelLayout outputChannelLayout) {
286*4d7e907cSAndroid Build Coastguard Worker         Parameter::Common common;
287*4d7e907cSAndroid Build Coastguard Worker         common.session = session;
288*4d7e907cSAndroid Build Coastguard Worker         common.ioHandle = ioHandle;
289*4d7e907cSAndroid Build Coastguard Worker 
290*4d7e907cSAndroid Build Coastguard Worker         auto& input = common.input;
291*4d7e907cSAndroid Build Coastguard Worker         auto& output = common.output;
292*4d7e907cSAndroid Build Coastguard Worker         input.base.sampleRate = iSampleRate;
293*4d7e907cSAndroid Build Coastguard Worker         input.base.channelMask = inputChannelLayout;
294*4d7e907cSAndroid Build Coastguard Worker         input.base.format = kDefaultFormatDescription;
295*4d7e907cSAndroid Build Coastguard Worker         input.frameCount = iFrameCount;
296*4d7e907cSAndroid Build Coastguard Worker         output.base.sampleRate = oSampleRate;
297*4d7e907cSAndroid Build Coastguard Worker         output.base.channelMask = outputChannelLayout;
298*4d7e907cSAndroid Build Coastguard Worker         output.base.format = kDefaultFormatDescription;
299*4d7e907cSAndroid Build Coastguard Worker         output.frameCount = oFrameCount;
300*4d7e907cSAndroid Build Coastguard Worker         return common;
301*4d7e907cSAndroid Build Coastguard Worker     }
302*4d7e907cSAndroid Build Coastguard Worker 
303*4d7e907cSAndroid Build Coastguard Worker     typedef ::android::AidlMessageQueue<
304*4d7e907cSAndroid Build Coastguard Worker             IEffect::Status, ::aidl::android::hardware::common::fmq::SynchronizedReadWrite>
305*4d7e907cSAndroid Build Coastguard Worker             StatusMQ;
306*4d7e907cSAndroid Build Coastguard Worker     typedef ::android::AidlMessageQueue<
307*4d7e907cSAndroid Build Coastguard Worker             float, ::aidl::android::hardware::common::fmq::SynchronizedReadWrite>
308*4d7e907cSAndroid Build Coastguard Worker             DataMQ;
309*4d7e907cSAndroid Build Coastguard Worker 
310*4d7e907cSAndroid Build Coastguard Worker     class EffectParam {
311*4d7e907cSAndroid Build Coastguard Worker       public:
312*4d7e907cSAndroid Build Coastguard Worker         std::unique_ptr<StatusMQ> statusMQ;
313*4d7e907cSAndroid Build Coastguard Worker         std::unique_ptr<DataMQ> inputMQ;
314*4d7e907cSAndroid Build Coastguard Worker         std::unique_ptr<DataMQ> outputMQ;
315*4d7e907cSAndroid Build Coastguard Worker     };
316*4d7e907cSAndroid Build Coastguard Worker 
317*4d7e907cSAndroid Build Coastguard Worker     template <typename T, Range::Tag tag>
isParameterValid(const T & target,const Descriptor & desc)318*4d7e907cSAndroid Build Coastguard Worker     static bool isParameterValid(const T& target, const Descriptor& desc) {
319*4d7e907cSAndroid Build Coastguard Worker         if (desc.capability.range.getTag() != tag) {
320*4d7e907cSAndroid Build Coastguard Worker             return true;
321*4d7e907cSAndroid Build Coastguard Worker         }
322*4d7e907cSAndroid Build Coastguard Worker         const auto& ranges = desc.capability.range.get<tag>();
323*4d7e907cSAndroid Build Coastguard Worker         return inRange(target, ranges);
324*4d7e907cSAndroid Build Coastguard Worker     }
325*4d7e907cSAndroid Build Coastguard Worker 
326*4d7e907cSAndroid Build Coastguard Worker     /**
327*4d7e907cSAndroid Build Coastguard Worker      * Add to test value set: (min+max)/2, minimum/maximum numeric limits, and min-1/max+1 if
328*4d7e907cSAndroid Build Coastguard Worker      * result still in numeric limits after -1/+1.
329*4d7e907cSAndroid Build Coastguard Worker      * Only use this when the type of test value is basic type (std::is_arithmetic return true).
330*4d7e907cSAndroid Build Coastguard Worker      */
331*4d7e907cSAndroid Build Coastguard Worker     template <typename S, typename = std::enable_if_t<std::is_arithmetic_v<S>>>
expandTestValueBasic(std::set<S> & s)332*4d7e907cSAndroid Build Coastguard Worker     static std::set<S> expandTestValueBasic(std::set<S>& s) {
333*4d7e907cSAndroid Build Coastguard Worker         const auto minLimit = std::numeric_limits<S>::min(),
334*4d7e907cSAndroid Build Coastguard Worker                    maxLimit = std::numeric_limits<S>::max();
335*4d7e907cSAndroid Build Coastguard Worker         if (s.size()) {
336*4d7e907cSAndroid Build Coastguard Worker             const auto min = *s.begin(), max = *s.rbegin();
337*4d7e907cSAndroid Build Coastguard Worker             s.insert((min & max) + ((min ^ max) >> 1));
338*4d7e907cSAndroid Build Coastguard Worker             if (min > minLimit + 1) {
339*4d7e907cSAndroid Build Coastguard Worker                 s.insert(min - 1);
340*4d7e907cSAndroid Build Coastguard Worker             }
341*4d7e907cSAndroid Build Coastguard Worker             if (max < maxLimit - 1) {
342*4d7e907cSAndroid Build Coastguard Worker                 s.insert(max + 1);
343*4d7e907cSAndroid Build Coastguard Worker             }
344*4d7e907cSAndroid Build Coastguard Worker         }
345*4d7e907cSAndroid Build Coastguard Worker         s.insert(minLimit);
346*4d7e907cSAndroid Build Coastguard Worker         s.insert(maxLimit);
347*4d7e907cSAndroid Build Coastguard Worker         return s;
348*4d7e907cSAndroid Build Coastguard Worker     }
349*4d7e907cSAndroid Build Coastguard Worker 
350*4d7e907cSAndroid Build Coastguard Worker     template <typename T, typename S, Range::Tag R, typename T::Tag tag>
getTestValueSet(std::vector<std::pair<std::shared_ptr<IFactory>,Descriptor>> descList)351*4d7e907cSAndroid Build Coastguard Worker     static std::set<S> getTestValueSet(
352*4d7e907cSAndroid Build Coastguard Worker             std::vector<std::pair<std::shared_ptr<IFactory>, Descriptor>> descList) {
353*4d7e907cSAndroid Build Coastguard Worker         std::set<S> result;
354*4d7e907cSAndroid Build Coastguard Worker         for (const auto& [_, desc] : descList) {
355*4d7e907cSAndroid Build Coastguard Worker             if (desc.capability.range.getTag() == R) {
356*4d7e907cSAndroid Build Coastguard Worker                 const auto& ranges = desc.capability.range.get<R>();
357*4d7e907cSAndroid Build Coastguard Worker                 for (const auto& range : ranges) {
358*4d7e907cSAndroid Build Coastguard Worker                     if (range.min.getTag() == tag) {
359*4d7e907cSAndroid Build Coastguard Worker                         result.insert(range.min.template get<tag>());
360*4d7e907cSAndroid Build Coastguard Worker                     }
361*4d7e907cSAndroid Build Coastguard Worker                     if (range.max.getTag() == tag) {
362*4d7e907cSAndroid Build Coastguard Worker                         result.insert(range.max.template get<tag>());
363*4d7e907cSAndroid Build Coastguard Worker                     }
364*4d7e907cSAndroid Build Coastguard Worker                 }
365*4d7e907cSAndroid Build Coastguard Worker             }
366*4d7e907cSAndroid Build Coastguard Worker         }
367*4d7e907cSAndroid Build Coastguard Worker         return result;
368*4d7e907cSAndroid Build Coastguard Worker     }
369*4d7e907cSAndroid Build Coastguard Worker 
370*4d7e907cSAndroid Build Coastguard Worker     template <typename T, typename S, Range::Tag R, typename T::Tag tag, typename Functor>
getTestValueSet(std::vector<std::pair<std::shared_ptr<IFactory>,Descriptor>> descList,Functor functor)371*4d7e907cSAndroid Build Coastguard Worker     static std::set<S> getTestValueSet(
372*4d7e907cSAndroid Build Coastguard Worker             std::vector<std::pair<std::shared_ptr<IFactory>, Descriptor>> descList,
373*4d7e907cSAndroid Build Coastguard Worker             Functor functor) {
374*4d7e907cSAndroid Build Coastguard Worker         auto result = getTestValueSet<T, S, R, tag>(descList);
375*4d7e907cSAndroid Build Coastguard Worker         return functor(result);
376*4d7e907cSAndroid Build Coastguard Worker     }
377*4d7e907cSAndroid Build Coastguard Worker 
378*4d7e907cSAndroid Build Coastguard Worker     // keep writing data to the FMQ until effect transit from DRAINING to IDLE
waitForDrain(std::vector<float> & inputBuffer,std::vector<float> & outputBuffer,const std::shared_ptr<IEffect> & effect,std::unique_ptr<EffectHelper::StatusMQ> & statusMQ,std::unique_ptr<EffectHelper::DataMQ> & inputMQ,std::unique_ptr<EffectHelper::DataMQ> & outputMQ,int version)379*4d7e907cSAndroid Build Coastguard Worker     static void waitForDrain(std::vector<float>& inputBuffer, std::vector<float>& outputBuffer,
380*4d7e907cSAndroid Build Coastguard Worker                              const std::shared_ptr<IEffect>& effect,
381*4d7e907cSAndroid Build Coastguard Worker                              std::unique_ptr<EffectHelper::StatusMQ>& statusMQ,
382*4d7e907cSAndroid Build Coastguard Worker                              std::unique_ptr<EffectHelper::DataMQ>& inputMQ,
383*4d7e907cSAndroid Build Coastguard Worker                              std::unique_ptr<EffectHelper::DataMQ>& outputMQ, int version) {
384*4d7e907cSAndroid Build Coastguard Worker         State state;
385*4d7e907cSAndroid Build Coastguard Worker         while (effect->getState(&state).getStatus() == EX_NONE && state == State::DRAINING) {
386*4d7e907cSAndroid Build Coastguard Worker             EXPECT_NO_FATAL_FAILURE(
387*4d7e907cSAndroid Build Coastguard Worker                     EffectHelper::writeToFmq(statusMQ, inputMQ, inputBuffer, version));
388*4d7e907cSAndroid Build Coastguard Worker             EXPECT_NO_FATAL_FAILURE(EffectHelper::readFromFmq(
389*4d7e907cSAndroid Build Coastguard Worker                     statusMQ, 1, outputMQ, outputBuffer.size(), outputBuffer, std::nullopt));
390*4d7e907cSAndroid Build Coastguard Worker         }
391*4d7e907cSAndroid Build Coastguard Worker         ASSERT_TRUE(State::IDLE == state);
392*4d7e907cSAndroid Build Coastguard Worker         EXPECT_NO_FATAL_FAILURE(EffectHelper::readFromFmq(statusMQ, 0, outputMQ, 0, outputBuffer));
393*4d7e907cSAndroid Build Coastguard Worker         return;
394*4d7e907cSAndroid Build Coastguard Worker     }
395*4d7e907cSAndroid Build Coastguard Worker 
396*4d7e907cSAndroid Build Coastguard Worker     static void processAndWriteToOutput(std::vector<float>& inputBuffer,
397*4d7e907cSAndroid Build Coastguard Worker                                         std::vector<float>& outputBuffer,
398*4d7e907cSAndroid Build Coastguard Worker                                         const std::shared_ptr<IEffect>& effect,
399*4d7e907cSAndroid Build Coastguard Worker                                         IEffect::OpenEffectReturn* openEffectReturn,
400*4d7e907cSAndroid Build Coastguard Worker                                         int version = -1, int times = 1,
401*4d7e907cSAndroid Build Coastguard Worker                                         bool callStopReset = true) {
402*4d7e907cSAndroid Build Coastguard Worker         // Initialize AidlMessagequeues
403*4d7e907cSAndroid Build Coastguard Worker         auto statusMQ = std::make_unique<EffectHelper::StatusMQ>(openEffectReturn->statusMQ);
404*4d7e907cSAndroid Build Coastguard Worker         ASSERT_TRUE(statusMQ->isValid());
405*4d7e907cSAndroid Build Coastguard Worker         auto inputMQ = std::make_unique<EffectHelper::DataMQ>(openEffectReturn->inputDataMQ);
406*4d7e907cSAndroid Build Coastguard Worker         ASSERT_TRUE(inputMQ->isValid());
407*4d7e907cSAndroid Build Coastguard Worker         auto outputMQ = std::make_unique<EffectHelper::DataMQ>(openEffectReturn->outputDataMQ);
408*4d7e907cSAndroid Build Coastguard Worker         ASSERT_TRUE(outputMQ->isValid());
409*4d7e907cSAndroid Build Coastguard Worker 
410*4d7e907cSAndroid Build Coastguard Worker         // Enabling the process
411*4d7e907cSAndroid Build Coastguard Worker         ASSERT_NO_FATAL_FAILURE(command(effect, CommandId::START));
412*4d7e907cSAndroid Build Coastguard Worker 
413*4d7e907cSAndroid Build Coastguard Worker         // Write from buffer to message queues and calling process
414*4d7e907cSAndroid Build Coastguard Worker         if (version == -1) {
415*4d7e907cSAndroid Build Coastguard Worker             ASSERT_IS_OK(effect->getInterfaceVersion(&version));
416*4d7e907cSAndroid Build Coastguard Worker         }
417*4d7e907cSAndroid Build Coastguard Worker 
418*4d7e907cSAndroid Build Coastguard Worker         for (int i = 0; i < times; i++) {
419*4d7e907cSAndroid Build Coastguard Worker             EXPECT_NO_FATAL_FAILURE(
420*4d7e907cSAndroid Build Coastguard Worker                     EffectHelper::writeToFmq(statusMQ, inputMQ, inputBuffer, version));
421*4d7e907cSAndroid Build Coastguard Worker             // Read the updated message queues into buffer
422*4d7e907cSAndroid Build Coastguard Worker             EXPECT_NO_FATAL_FAILURE(EffectHelper::readFromFmq(statusMQ, 1, outputMQ,
423*4d7e907cSAndroid Build Coastguard Worker                                                               outputBuffer.size(), outputBuffer));
424*4d7e907cSAndroid Build Coastguard Worker         }
425*4d7e907cSAndroid Build Coastguard Worker 
426*4d7e907cSAndroid Build Coastguard Worker         // Disable the process
427*4d7e907cSAndroid Build Coastguard Worker         if (callStopReset) {
428*4d7e907cSAndroid Build Coastguard Worker             ASSERT_NO_FATAL_FAILURE(command(effect, CommandId::STOP));
429*4d7e907cSAndroid Build Coastguard Worker             EXPECT_NO_FATAL_FAILURE(waitForDrain(inputBuffer, outputBuffer, effect, statusMQ,
430*4d7e907cSAndroid Build Coastguard Worker                                                  inputMQ, outputMQ, version));
431*4d7e907cSAndroid Build Coastguard Worker         }
432*4d7e907cSAndroid Build Coastguard Worker 
433*4d7e907cSAndroid Build Coastguard Worker         if (callStopReset) {
434*4d7e907cSAndroid Build Coastguard Worker             ASSERT_NO_FATAL_FAILURE(command(effect, CommandId::RESET));
435*4d7e907cSAndroid Build Coastguard Worker         }
436*4d7e907cSAndroid Build Coastguard Worker     }
437*4d7e907cSAndroid Build Coastguard Worker 
438*4d7e907cSAndroid Build Coastguard Worker     // Find FFT bin indices for testFrequencies and get bin center frequencies
roundToFreqCenteredToFftBin(std::vector<int> & testFrequencies,std::vector<int> & binOffsets,const float kBinWidth)439*4d7e907cSAndroid Build Coastguard Worker     void roundToFreqCenteredToFftBin(std::vector<int>& testFrequencies,
440*4d7e907cSAndroid Build Coastguard Worker                                      std::vector<int>& binOffsets, const float kBinWidth) {
441*4d7e907cSAndroid Build Coastguard Worker         for (size_t i = 0; i < testFrequencies.size(); i++) {
442*4d7e907cSAndroid Build Coastguard Worker             binOffsets[i] = std::round(testFrequencies[i] / kBinWidth);
443*4d7e907cSAndroid Build Coastguard Worker             testFrequencies[i] = std::round(binOffsets[i] * kBinWidth);
444*4d7e907cSAndroid Build Coastguard Worker         }
445*4d7e907cSAndroid Build Coastguard Worker     }
446*4d7e907cSAndroid Build Coastguard Worker 
447*4d7e907cSAndroid Build Coastguard Worker     // Fill inputBuffer with random values between -maxAudioSampleValue to maxAudioSampleValue
448*4d7e907cSAndroid Build Coastguard Worker     void generateInputBuffer(std::vector<float>& inputBuffer, size_t startPosition, bool isStrip,
449*4d7e907cSAndroid Build Coastguard Worker                              size_t channelCount,
450*4d7e907cSAndroid Build Coastguard Worker                              float maxAudioSampleValue = kMaxAudioSampleValue) {
451*4d7e907cSAndroid Build Coastguard Worker         size_t increment = isStrip ? 1 /*Fill input at all the channels*/
452*4d7e907cSAndroid Build Coastguard Worker                                    : channelCount /*Fill input at only one channel*/;
453*4d7e907cSAndroid Build Coastguard Worker 
454*4d7e907cSAndroid Build Coastguard Worker         for (size_t i = startPosition; i < inputBuffer.size(); i += increment) {
455*4d7e907cSAndroid Build Coastguard Worker             inputBuffer[i] =
456*4d7e907cSAndroid Build Coastguard Worker                     ((static_cast<float>(std::rand()) / RAND_MAX) * 2 - 1) * maxAudioSampleValue;
457*4d7e907cSAndroid Build Coastguard Worker         }
458*4d7e907cSAndroid Build Coastguard Worker     }
459*4d7e907cSAndroid Build Coastguard Worker 
460*4d7e907cSAndroid Build Coastguard Worker     // Generate multitone input between -amplitude to +amplitude using testFrequencies
461*4d7e907cSAndroid Build Coastguard Worker     // All test frequencies are considered having the same amplitude
462*4d7e907cSAndroid Build Coastguard Worker     void generateSineWave(const std::vector<int>& testFrequencies, std::vector<float>& input,
463*4d7e907cSAndroid Build Coastguard Worker                           const float amplitude = 1.0,
464*4d7e907cSAndroid Build Coastguard Worker                           const int samplingFrequency = kSamplingFrequency) {
465*4d7e907cSAndroid Build Coastguard Worker         for (size_t i = 0; i < input.size(); i++) {
466*4d7e907cSAndroid Build Coastguard Worker             input[i] = 0;
467*4d7e907cSAndroid Build Coastguard Worker 
468*4d7e907cSAndroid Build Coastguard Worker             for (size_t j = 0; j < testFrequencies.size(); j++) {
469*4d7e907cSAndroid Build Coastguard Worker                 input[i] += sin(2 * M_PI * testFrequencies[j] * i / samplingFrequency);
470*4d7e907cSAndroid Build Coastguard Worker             }
471*4d7e907cSAndroid Build Coastguard Worker             input[i] *= amplitude / testFrequencies.size();
472*4d7e907cSAndroid Build Coastguard Worker         }
473*4d7e907cSAndroid Build Coastguard Worker     }
474*4d7e907cSAndroid Build Coastguard Worker 
475*4d7e907cSAndroid Build Coastguard Worker     // Generate single tone input between -amplitude to +amplitude using testFrequency
476*4d7e907cSAndroid Build Coastguard Worker     void generateSineWave(const int testFrequency, std::vector<float>& input,
477*4d7e907cSAndroid Build Coastguard Worker                           const float amplitude = 1.0,
478*4d7e907cSAndroid Build Coastguard Worker                           const int samplingFrequency = kSamplingFrequency) {
479*4d7e907cSAndroid Build Coastguard Worker         generateSineWave(std::vector<int>{testFrequency}, input, amplitude, samplingFrequency);
480*4d7e907cSAndroid Build Coastguard Worker     }
481*4d7e907cSAndroid Build Coastguard Worker 
482*4d7e907cSAndroid Build Coastguard Worker     // Use FFT transform to convert the buffer to frequency domain
483*4d7e907cSAndroid Build Coastguard Worker     // Compute its magnitude at binOffsets
calculateMagnitude(const std::vector<float> & buffer,const std::vector<int> & binOffsets,const int nPointFFT)484*4d7e907cSAndroid Build Coastguard Worker     std::vector<float> calculateMagnitude(const std::vector<float>& buffer,
485*4d7e907cSAndroid Build Coastguard Worker                                           const std::vector<int>& binOffsets, const int nPointFFT) {
486*4d7e907cSAndroid Build Coastguard Worker         std::vector<float> fftInput(nPointFFT);
487*4d7e907cSAndroid Build Coastguard Worker         PFFFT_Setup* inputHandle = pffft_new_setup(nPointFFT, PFFFT_REAL);
488*4d7e907cSAndroid Build Coastguard Worker         pffft_transform_ordered(inputHandle, buffer.data(), fftInput.data(), nullptr,
489*4d7e907cSAndroid Build Coastguard Worker                                 PFFFT_FORWARD);
490*4d7e907cSAndroid Build Coastguard Worker         pffft_destroy_setup(inputHandle);
491*4d7e907cSAndroid Build Coastguard Worker         std::vector<float> bufferMag(binOffsets.size());
492*4d7e907cSAndroid Build Coastguard Worker         for (size_t i = 0; i < binOffsets.size(); i++) {
493*4d7e907cSAndroid Build Coastguard Worker             size_t k = binOffsets[i];
494*4d7e907cSAndroid Build Coastguard Worker             bufferMag[i] = sqrt((fftInput[k * 2] * fftInput[k * 2]) +
495*4d7e907cSAndroid Build Coastguard Worker                                 (fftInput[k * 2 + 1] * fftInput[k * 2 + 1]));
496*4d7e907cSAndroid Build Coastguard Worker         }
497*4d7e907cSAndroid Build Coastguard Worker 
498*4d7e907cSAndroid Build Coastguard Worker         return bufferMag;
499*4d7e907cSAndroid Build Coastguard Worker     }
500*4d7e907cSAndroid Build Coastguard Worker 
updateFrameSize(const Parameter::Common & common)501*4d7e907cSAndroid Build Coastguard Worker     void updateFrameSize(const Parameter::Common& common) {
502*4d7e907cSAndroid Build Coastguard Worker         mInputFrameSize = ::aidl::android::hardware::audio::common::getFrameSizeInBytes(
503*4d7e907cSAndroid Build Coastguard Worker                 common.input.base.format, common.input.base.channelMask);
504*4d7e907cSAndroid Build Coastguard Worker         mInputSamples = common.input.frameCount * mInputFrameSize / sizeof(float);
505*4d7e907cSAndroid Build Coastguard Worker         mOutputFrameSize = ::aidl::android::hardware::audio::common::getFrameSizeInBytes(
506*4d7e907cSAndroid Build Coastguard Worker                 common.output.base.format, common.output.base.channelMask);
507*4d7e907cSAndroid Build Coastguard Worker         mOutputSamples = common.output.frameCount * mOutputFrameSize / sizeof(float);
508*4d7e907cSAndroid Build Coastguard Worker     }
509*4d7e907cSAndroid Build Coastguard Worker 
510*4d7e907cSAndroid Build Coastguard Worker     void generateInput(std::vector<float>& input, float inputFrequency, float samplingFrequency,
511*4d7e907cSAndroid Build Coastguard Worker                        size_t inputSize = 0) {
512*4d7e907cSAndroid Build Coastguard Worker         if (inputSize == 0 || inputSize > input.size()) {
513*4d7e907cSAndroid Build Coastguard Worker             inputSize = input.size();
514*4d7e907cSAndroid Build Coastguard Worker         }
515*4d7e907cSAndroid Build Coastguard Worker 
516*4d7e907cSAndroid Build Coastguard Worker         for (size_t i = 0; i < inputSize; i++) {
517*4d7e907cSAndroid Build Coastguard Worker             input[i] = sin(2 * M_PI * inputFrequency * i / samplingFrequency);
518*4d7e907cSAndroid Build Coastguard Worker         }
519*4d7e907cSAndroid Build Coastguard Worker     }
520*4d7e907cSAndroid Build Coastguard Worker 
521*4d7e907cSAndroid Build Coastguard Worker     bool mIsSpatializer;
522*4d7e907cSAndroid Build Coastguard Worker     Descriptor mDescriptor;
523*4d7e907cSAndroid Build Coastguard Worker     size_t mInputFrameSize, mOutputFrameSize;
524*4d7e907cSAndroid Build Coastguard Worker     size_t mInputSamples, mOutputSamples;
525*4d7e907cSAndroid Build Coastguard Worker };
526