xref: /aosp_15_r20/hardware/interfaces/audio/aidl/default/eraser/Eraser.cpp (revision 4d7e907c777eeecc4c5bd7cf640a754fac206ff7)
1*4d7e907cSAndroid Build Coastguard Worker /*
2*4d7e907cSAndroid Build Coastguard Worker  * Copyright (C) 2024 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 #define LOG_TAG "AHAL_Eraser"
18*4d7e907cSAndroid Build Coastguard Worker 
19*4d7e907cSAndroid Build Coastguard Worker #include "Eraser.h"
20*4d7e907cSAndroid Build Coastguard Worker 
21*4d7e907cSAndroid Build Coastguard Worker #include <android-base/logging.h>
22*4d7e907cSAndroid Build Coastguard Worker #include <system/audio_effects/effect_uuid.h>
23*4d7e907cSAndroid Build Coastguard Worker 
24*4d7e907cSAndroid Build Coastguard Worker #include <optional>
25*4d7e907cSAndroid Build Coastguard Worker 
26*4d7e907cSAndroid Build Coastguard Worker using aidl::android::hardware::audio::common::getChannelCount;
27*4d7e907cSAndroid Build Coastguard Worker using aidl::android::hardware::audio::effect::Descriptor;
28*4d7e907cSAndroid Build Coastguard Worker using aidl::android::hardware::audio::effect::EraserSw;
29*4d7e907cSAndroid Build Coastguard Worker using aidl::android::hardware::audio::effect::getEffectImplUuidEraserSw;
30*4d7e907cSAndroid Build Coastguard Worker using aidl::android::hardware::audio::effect::getEffectTypeUuidEraser;
31*4d7e907cSAndroid Build Coastguard Worker using aidl::android::hardware::audio::effect::IEffect;
32*4d7e907cSAndroid Build Coastguard Worker using aidl::android::hardware::audio::effect::State;
33*4d7e907cSAndroid Build Coastguard Worker using aidl::android::media::audio::common::AudioChannelLayout;
34*4d7e907cSAndroid Build Coastguard Worker using aidl::android::media::audio::common::AudioUuid;
35*4d7e907cSAndroid Build Coastguard Worker 
createEffect(const AudioUuid * in_impl_uuid,std::shared_ptr<IEffect> * instanceSpp)36*4d7e907cSAndroid Build Coastguard Worker extern "C" binder_exception_t createEffect(const AudioUuid* in_impl_uuid,
37*4d7e907cSAndroid Build Coastguard Worker                                            std::shared_ptr<IEffect>* instanceSpp) {
38*4d7e907cSAndroid Build Coastguard Worker     if (!in_impl_uuid || *in_impl_uuid != getEffectImplUuidEraserSw()) {
39*4d7e907cSAndroid Build Coastguard Worker         LOG(ERROR) << __func__ << "uuid not supported";
40*4d7e907cSAndroid Build Coastguard Worker         return EX_ILLEGAL_ARGUMENT;
41*4d7e907cSAndroid Build Coastguard Worker     }
42*4d7e907cSAndroid Build Coastguard Worker     if (!instanceSpp) {
43*4d7e907cSAndroid Build Coastguard Worker         LOG(ERROR) << __func__ << " invalid input parameter!";
44*4d7e907cSAndroid Build Coastguard Worker         return EX_ILLEGAL_ARGUMENT;
45*4d7e907cSAndroid Build Coastguard Worker     }
46*4d7e907cSAndroid Build Coastguard Worker 
47*4d7e907cSAndroid Build Coastguard Worker     *instanceSpp = ndk::SharedRefBase::make<EraserSw>();
48*4d7e907cSAndroid Build Coastguard Worker     LOG(DEBUG) << __func__ << " instance " << instanceSpp->get() << " created";
49*4d7e907cSAndroid Build Coastguard Worker     return EX_NONE;
50*4d7e907cSAndroid Build Coastguard Worker }
51*4d7e907cSAndroid Build Coastguard Worker 
queryEffect(const AudioUuid * in_impl_uuid,Descriptor * _aidl_return)52*4d7e907cSAndroid Build Coastguard Worker extern "C" binder_exception_t queryEffect(const AudioUuid* in_impl_uuid, Descriptor* _aidl_return) {
53*4d7e907cSAndroid Build Coastguard Worker     if (!in_impl_uuid || *in_impl_uuid != getEffectImplUuidEraserSw()) {
54*4d7e907cSAndroid Build Coastguard Worker         LOG(ERROR) << __func__ << "uuid not supported";
55*4d7e907cSAndroid Build Coastguard Worker         return EX_ILLEGAL_ARGUMENT;
56*4d7e907cSAndroid Build Coastguard Worker     }
57*4d7e907cSAndroid Build Coastguard Worker     *_aidl_return = EraserSw::kDescriptor;
58*4d7e907cSAndroid Build Coastguard Worker     return EX_NONE;
59*4d7e907cSAndroid Build Coastguard Worker }
60*4d7e907cSAndroid Build Coastguard Worker 
61*4d7e907cSAndroid Build Coastguard Worker namespace aidl::android::hardware::audio::effect {
62*4d7e907cSAndroid Build Coastguard Worker 
63*4d7e907cSAndroid Build Coastguard Worker const std::string EraserSw::kEffectName = "EraserSw";
64*4d7e907cSAndroid Build Coastguard Worker const Descriptor EraserSw::kDescriptor = {
65*4d7e907cSAndroid Build Coastguard Worker         .common = {.id = {.type = getEffectTypeUuidEraser(), .uuid = getEffectImplUuidEraserSw()},
66*4d7e907cSAndroid Build Coastguard Worker                    .flags = {.type = Flags::Type::INSERT,
67*4d7e907cSAndroid Build Coastguard Worker                              .insert = Flags::Insert::FIRST,
68*4d7e907cSAndroid Build Coastguard Worker                              .hwAcceleratorMode = Flags::HardwareAccelerator::NONE},
69*4d7e907cSAndroid Build Coastguard Worker                    .name = EraserSw::kEffectName,
70*4d7e907cSAndroid Build Coastguard Worker                    .implementor = "The Android Open Source Project"}};
71*4d7e907cSAndroid Build Coastguard Worker 
getDescriptor(Descriptor * _aidl_return)72*4d7e907cSAndroid Build Coastguard Worker ndk::ScopedAStatus EraserSw::getDescriptor(Descriptor* _aidl_return) {
73*4d7e907cSAndroid Build Coastguard Worker     LOG(DEBUG) << __func__ << kDescriptor.toString();
74*4d7e907cSAndroid Build Coastguard Worker     *_aidl_return = kDescriptor;
75*4d7e907cSAndroid Build Coastguard Worker     return ndk::ScopedAStatus::ok();
76*4d7e907cSAndroid Build Coastguard Worker }
77*4d7e907cSAndroid Build Coastguard Worker 
setParameterSpecific(const Parameter::Specific & specific)78*4d7e907cSAndroid Build Coastguard Worker ndk::ScopedAStatus EraserSw::setParameterSpecific(const Parameter::Specific& specific) {
79*4d7e907cSAndroid Build Coastguard Worker     RETURN_IF(Parameter::Specific::eraser != specific.getTag(), EX_ILLEGAL_ARGUMENT,
80*4d7e907cSAndroid Build Coastguard Worker               "EffectNotSupported");
81*4d7e907cSAndroid Build Coastguard Worker     RETURN_IF(!mContext, EX_NULL_POINTER, "nullContext");
82*4d7e907cSAndroid Build Coastguard Worker 
83*4d7e907cSAndroid Build Coastguard Worker     auto& param = specific.get<Parameter::Specific::eraser>();
84*4d7e907cSAndroid Build Coastguard Worker     return mContext->setParam(param.getTag(), param);
85*4d7e907cSAndroid Build Coastguard Worker }
86*4d7e907cSAndroid Build Coastguard Worker 
getParameterSpecific(const Parameter::Id & id,Parameter::Specific * specific)87*4d7e907cSAndroid Build Coastguard Worker ndk::ScopedAStatus EraserSw::getParameterSpecific(const Parameter::Id& id,
88*4d7e907cSAndroid Build Coastguard Worker                                                   Parameter::Specific* specific) {
89*4d7e907cSAndroid Build Coastguard Worker     RETURN_IF(!mContext, EX_NULL_POINTER, "nullContext");
90*4d7e907cSAndroid Build Coastguard Worker 
91*4d7e907cSAndroid Build Coastguard Worker     auto tag = id.getTag();
92*4d7e907cSAndroid Build Coastguard Worker     RETURN_IF(Parameter::Id::eraserTag != tag, EX_ILLEGAL_ARGUMENT, "wrongIdTag");
93*4d7e907cSAndroid Build Coastguard Worker     auto eraserId = id.get<Parameter::Id::eraserTag>();
94*4d7e907cSAndroid Build Coastguard Worker     auto eraserTag = eraserId.getTag();
95*4d7e907cSAndroid Build Coastguard Worker     switch (eraserTag) {
96*4d7e907cSAndroid Build Coastguard Worker         case Eraser::Id::commonTag: {
97*4d7e907cSAndroid Build Coastguard Worker             auto specificTag = eraserId.get<Eraser::Id::commonTag>();
98*4d7e907cSAndroid Build Coastguard Worker             std::optional<Eraser> param = mContext->getParam(specificTag);
99*4d7e907cSAndroid Build Coastguard Worker             if (!param.has_value()) {
100*4d7e907cSAndroid Build Coastguard Worker                 return ndk::ScopedAStatus::fromExceptionCodeWithMessage(EX_ILLEGAL_ARGUMENT,
101*4d7e907cSAndroid Build Coastguard Worker                                                                         "EraserTagNotSupported");
102*4d7e907cSAndroid Build Coastguard Worker             }
103*4d7e907cSAndroid Build Coastguard Worker             specific->set<Parameter::Specific::eraser>(param.value());
104*4d7e907cSAndroid Build Coastguard Worker             break;
105*4d7e907cSAndroid Build Coastguard Worker         }
106*4d7e907cSAndroid Build Coastguard Worker         default: {
107*4d7e907cSAndroid Build Coastguard Worker             LOG(ERROR) << __func__ << " unsupported tag: " << toString(tag);
108*4d7e907cSAndroid Build Coastguard Worker             return ndk::ScopedAStatus::fromExceptionCodeWithMessage(EX_ILLEGAL_ARGUMENT,
109*4d7e907cSAndroid Build Coastguard Worker                                                                     "EraserTagNotSupported");
110*4d7e907cSAndroid Build Coastguard Worker         }
111*4d7e907cSAndroid Build Coastguard Worker     }
112*4d7e907cSAndroid Build Coastguard Worker     return ndk::ScopedAStatus::ok();
113*4d7e907cSAndroid Build Coastguard Worker }
114*4d7e907cSAndroid Build Coastguard Worker 
createContext(const Parameter::Common & common)115*4d7e907cSAndroid Build Coastguard Worker std::shared_ptr<EffectContext> EraserSw::createContext(const Parameter::Common& common) {
116*4d7e907cSAndroid Build Coastguard Worker     if (mContext) {
117*4d7e907cSAndroid Build Coastguard Worker         LOG(DEBUG) << __func__ << " context already exist";
118*4d7e907cSAndroid Build Coastguard Worker     } else {
119*4d7e907cSAndroid Build Coastguard Worker         mContext = std::make_shared<EraserSwContext>(1 /* statusFmqDepth */, common);
120*4d7e907cSAndroid Build Coastguard Worker     }
121*4d7e907cSAndroid Build Coastguard Worker     return mContext;
122*4d7e907cSAndroid Build Coastguard Worker }
123*4d7e907cSAndroid Build Coastguard Worker 
releaseContext()124*4d7e907cSAndroid Build Coastguard Worker RetCode EraserSw::releaseContext() {
125*4d7e907cSAndroid Build Coastguard Worker     if (mContext) {
126*4d7e907cSAndroid Build Coastguard Worker         mContext.reset();
127*4d7e907cSAndroid Build Coastguard Worker     }
128*4d7e907cSAndroid Build Coastguard Worker     return RetCode::SUCCESS;
129*4d7e907cSAndroid Build Coastguard Worker }
130*4d7e907cSAndroid Build Coastguard Worker 
~EraserSw()131*4d7e907cSAndroid Build Coastguard Worker EraserSw::~EraserSw() {
132*4d7e907cSAndroid Build Coastguard Worker     cleanUp();
133*4d7e907cSAndroid Build Coastguard Worker     LOG(DEBUG) << __func__;
134*4d7e907cSAndroid Build Coastguard Worker }
135*4d7e907cSAndroid Build Coastguard Worker 
command(CommandId command)136*4d7e907cSAndroid Build Coastguard Worker ndk::ScopedAStatus EraserSw::command(CommandId command) {
137*4d7e907cSAndroid Build Coastguard Worker     std::lock_guard lg(mImplMutex);
138*4d7e907cSAndroid Build Coastguard Worker     RETURN_IF(mState == State::INIT, EX_ILLEGAL_STATE, "instanceNotOpen");
139*4d7e907cSAndroid Build Coastguard Worker 
140*4d7e907cSAndroid Build Coastguard Worker     switch (command) {
141*4d7e907cSAndroid Build Coastguard Worker         case CommandId::START:
142*4d7e907cSAndroid Build Coastguard Worker             RETURN_OK_IF(mState == State::PROCESSING);
143*4d7e907cSAndroid Build Coastguard Worker             mState = State::PROCESSING;
144*4d7e907cSAndroid Build Coastguard Worker             mContext->enable();
145*4d7e907cSAndroid Build Coastguard Worker             startThread();
146*4d7e907cSAndroid Build Coastguard Worker             RETURN_IF(notifyEventFlag(mDataMqNotEmptyEf) != RetCode::SUCCESS, EX_ILLEGAL_STATE,
147*4d7e907cSAndroid Build Coastguard Worker                       "notifyEventFlagNotEmptyFailed");
148*4d7e907cSAndroid Build Coastguard Worker             break;
149*4d7e907cSAndroid Build Coastguard Worker         case CommandId::STOP:
150*4d7e907cSAndroid Build Coastguard Worker             RETURN_OK_IF(mState == State::IDLE || mState == State::DRAINING);
151*4d7e907cSAndroid Build Coastguard Worker             if (mVersion < kDrainSupportedVersion) {
152*4d7e907cSAndroid Build Coastguard Worker                 mState = State::IDLE;
153*4d7e907cSAndroid Build Coastguard Worker                 stopThread();
154*4d7e907cSAndroid Build Coastguard Worker                 mContext->disable();
155*4d7e907cSAndroid Build Coastguard Worker             } else {
156*4d7e907cSAndroid Build Coastguard Worker                 mState = State::DRAINING;
157*4d7e907cSAndroid Build Coastguard Worker                 startDraining();
158*4d7e907cSAndroid Build Coastguard Worker                 mContext->startDraining();
159*4d7e907cSAndroid Build Coastguard Worker             }
160*4d7e907cSAndroid Build Coastguard Worker             RETURN_IF(notifyEventFlag(mDataMqNotEmptyEf) != RetCode::SUCCESS, EX_ILLEGAL_STATE,
161*4d7e907cSAndroid Build Coastguard Worker                       "notifyEventFlagNotEmptyFailed");
162*4d7e907cSAndroid Build Coastguard Worker             break;
163*4d7e907cSAndroid Build Coastguard Worker         case CommandId::RESET:
164*4d7e907cSAndroid Build Coastguard Worker             mState = State::IDLE;
165*4d7e907cSAndroid Build Coastguard Worker             RETURN_IF(notifyEventFlag(mDataMqNotEmptyEf) != RetCode::SUCCESS, EX_ILLEGAL_STATE,
166*4d7e907cSAndroid Build Coastguard Worker                       "notifyEventFlagNotEmptyFailed");
167*4d7e907cSAndroid Build Coastguard Worker             stopThread();
168*4d7e907cSAndroid Build Coastguard Worker             mImplContext->disable();
169*4d7e907cSAndroid Build Coastguard Worker             mImplContext->reset();
170*4d7e907cSAndroid Build Coastguard Worker             mImplContext->resetBuffer();
171*4d7e907cSAndroid Build Coastguard Worker             break;
172*4d7e907cSAndroid Build Coastguard Worker         default:
173*4d7e907cSAndroid Build Coastguard Worker             LOG(ERROR) << getEffectNameWithVersion() << __func__ << " instance still processing";
174*4d7e907cSAndroid Build Coastguard Worker             return ndk::ScopedAStatus::fromExceptionCodeWithMessage(EX_ILLEGAL_ARGUMENT,
175*4d7e907cSAndroid Build Coastguard Worker                                                                     "CommandIdNotSupported");
176*4d7e907cSAndroid Build Coastguard Worker     }
177*4d7e907cSAndroid Build Coastguard Worker     LOG(VERBOSE) << getEffectNameWithVersion() << __func__
178*4d7e907cSAndroid Build Coastguard Worker                  << " transfer to state: " << toString(mState);
179*4d7e907cSAndroid Build Coastguard Worker     return ndk::ScopedAStatus::ok();
180*4d7e907cSAndroid Build Coastguard Worker }
181*4d7e907cSAndroid Build Coastguard Worker 
182*4d7e907cSAndroid Build Coastguard Worker // Processing method running in EffectWorker thread.
effectProcessImpl(float * in,float * out,int samples)183*4d7e907cSAndroid Build Coastguard Worker IEffect::Status EraserSw::effectProcessImpl(float* in, float* out, int samples) {
184*4d7e907cSAndroid Build Coastguard Worker     RETURN_VALUE_IF(!mContext, (IEffect::Status{EX_NULL_POINTER, 0, 0}), "nullContext");
185*4d7e907cSAndroid Build Coastguard Worker     IEffect::Status procStatus{STATUS_NOT_ENOUGH_DATA, 0, 0};
186*4d7e907cSAndroid Build Coastguard Worker     procStatus = mContext->process(in, out, samples);
187*4d7e907cSAndroid Build Coastguard Worker     if (mState == State::DRAINING && procStatus.status == STATUS_NOT_ENOUGH_DATA) {
188*4d7e907cSAndroid Build Coastguard Worker         drainingComplete_l();
189*4d7e907cSAndroid Build Coastguard Worker     }
190*4d7e907cSAndroid Build Coastguard Worker 
191*4d7e907cSAndroid Build Coastguard Worker     return procStatus;
192*4d7e907cSAndroid Build Coastguard Worker }
193*4d7e907cSAndroid Build Coastguard Worker 
drainingComplete_l()194*4d7e907cSAndroid Build Coastguard Worker void EraserSw::drainingComplete_l() {
195*4d7e907cSAndroid Build Coastguard Worker     if (mState != State::DRAINING) return;
196*4d7e907cSAndroid Build Coastguard Worker 
197*4d7e907cSAndroid Build Coastguard Worker     LOG(DEBUG) << getEffectNameWithVersion() << __func__;
198*4d7e907cSAndroid Build Coastguard Worker     finishDraining();
199*4d7e907cSAndroid Build Coastguard Worker     mState = State::IDLE;
200*4d7e907cSAndroid Build Coastguard Worker }
201*4d7e907cSAndroid Build Coastguard Worker 
EraserSwContext(int statusDepth,const Parameter::Common & common)202*4d7e907cSAndroid Build Coastguard Worker EraserSwContext::EraserSwContext(int statusDepth, const Parameter::Common& common)
203*4d7e907cSAndroid Build Coastguard Worker     : EffectContext(statusDepth, common) {
204*4d7e907cSAndroid Build Coastguard Worker     LOG(DEBUG) << __func__;
205*4d7e907cSAndroid Build Coastguard Worker }
206*4d7e907cSAndroid Build Coastguard Worker 
~EraserSwContext()207*4d7e907cSAndroid Build Coastguard Worker EraserSwContext::~EraserSwContext() {
208*4d7e907cSAndroid Build Coastguard Worker     LOG(DEBUG) << __func__;
209*4d7e907cSAndroid Build Coastguard Worker }
210*4d7e907cSAndroid Build Coastguard Worker 
211*4d7e907cSAndroid Build Coastguard Worker template <typename TAG>
getParam(TAG tag)212*4d7e907cSAndroid Build Coastguard Worker std::optional<Eraser> EraserSwContext::getParam(TAG tag) {
213*4d7e907cSAndroid Build Coastguard Worker     if (mParamsMap.find(tag) != mParamsMap.end()) {
214*4d7e907cSAndroid Build Coastguard Worker         return mParamsMap.at(tag);
215*4d7e907cSAndroid Build Coastguard Worker     }
216*4d7e907cSAndroid Build Coastguard Worker     return std::nullopt;
217*4d7e907cSAndroid Build Coastguard Worker }
218*4d7e907cSAndroid Build Coastguard Worker 
219*4d7e907cSAndroid Build Coastguard Worker template <typename TAG>
setParam(TAG tag,Eraser eraser)220*4d7e907cSAndroid Build Coastguard Worker ndk::ScopedAStatus EraserSwContext::setParam(TAG tag, Eraser eraser) {
221*4d7e907cSAndroid Build Coastguard Worker     mParamsMap[tag] = eraser;
222*4d7e907cSAndroid Build Coastguard Worker     return ndk::ScopedAStatus::ok();
223*4d7e907cSAndroid Build Coastguard Worker }
224*4d7e907cSAndroid Build Coastguard Worker 
process(float * in,float * out,int samples)225*4d7e907cSAndroid Build Coastguard Worker IEffect::Status EraserSwContext::process(float* in, float* out, int samples) {
226*4d7e907cSAndroid Build Coastguard Worker     LOG(DEBUG) << __func__ << " in " << in << " out " << out << " samples " << samples;
227*4d7e907cSAndroid Build Coastguard Worker     IEffect::Status procStatus = {EX_ILLEGAL_ARGUMENT, 0, 0};
228*4d7e907cSAndroid Build Coastguard Worker     const auto inputChannelCount = getChannelCount(mCommon.input.base.channelMask);
229*4d7e907cSAndroid Build Coastguard Worker     const auto outputChannelCount = getChannelCount(mCommon.output.base.channelMask);
230*4d7e907cSAndroid Build Coastguard Worker     if (inputChannelCount < outputChannelCount) {
231*4d7e907cSAndroid Build Coastguard Worker         LOG(ERROR) << __func__ << " invalid channel count, in: " << inputChannelCount
232*4d7e907cSAndroid Build Coastguard Worker                    << " out: " << outputChannelCount;
233*4d7e907cSAndroid Build Coastguard Worker         return procStatus;
234*4d7e907cSAndroid Build Coastguard Worker     }
235*4d7e907cSAndroid Build Coastguard Worker 
236*4d7e907cSAndroid Build Coastguard Worker     if (samples <= 0 || 0 != samples % inputChannelCount) {
237*4d7e907cSAndroid Build Coastguard Worker         LOG(ERROR) << __func__ << " invalid samples: " << samples;
238*4d7e907cSAndroid Build Coastguard Worker         return procStatus;
239*4d7e907cSAndroid Build Coastguard Worker     }
240*4d7e907cSAndroid Build Coastguard Worker 
241*4d7e907cSAndroid Build Coastguard Worker     const int iFrames = samples / inputChannelCount;
242*4d7e907cSAndroid Build Coastguard Worker     const float gainPerSample = 1.f / iFrames;
243*4d7e907cSAndroid Build Coastguard Worker     for (int i = 0; i < iFrames; i++) {
244*4d7e907cSAndroid Build Coastguard Worker         if (isDraining()) {
245*4d7e907cSAndroid Build Coastguard Worker             const float gain = (iFrames - i - 1) * gainPerSample;
246*4d7e907cSAndroid Build Coastguard Worker             for (size_t c = 0; c < outputChannelCount; c++) {
247*4d7e907cSAndroid Build Coastguard Worker                 out[c] = in[c] * gain;
248*4d7e907cSAndroid Build Coastguard Worker             }
249*4d7e907cSAndroid Build Coastguard Worker         } else {
250*4d7e907cSAndroid Build Coastguard Worker             std::memcpy(out, in, outputChannelCount * sizeof(float));
251*4d7e907cSAndroid Build Coastguard Worker         }
252*4d7e907cSAndroid Build Coastguard Worker 
253*4d7e907cSAndroid Build Coastguard Worker         in += inputChannelCount;
254*4d7e907cSAndroid Build Coastguard Worker         out += outputChannelCount;
255*4d7e907cSAndroid Build Coastguard Worker     }
256*4d7e907cSAndroid Build Coastguard Worker 
257*4d7e907cSAndroid Build Coastguard Worker     // drain for one cycle
258*4d7e907cSAndroid Build Coastguard Worker     if (isDraining()) {
259*4d7e907cSAndroid Build Coastguard Worker         procStatus.status = STATUS_NOT_ENOUGH_DATA;
260*4d7e907cSAndroid Build Coastguard Worker         finishDraining();
261*4d7e907cSAndroid Build Coastguard Worker     } else {
262*4d7e907cSAndroid Build Coastguard Worker         procStatus.status = STATUS_OK;
263*4d7e907cSAndroid Build Coastguard Worker     }
264*4d7e907cSAndroid Build Coastguard Worker     procStatus.fmqConsumed = static_cast<int32_t>(iFrames * inputChannelCount);
265*4d7e907cSAndroid Build Coastguard Worker     procStatus.fmqProduced = static_cast<int32_t>(iFrames * outputChannelCount);
266*4d7e907cSAndroid Build Coastguard Worker 
267*4d7e907cSAndroid Build Coastguard Worker     return procStatus;
268*4d7e907cSAndroid Build Coastguard Worker }
269*4d7e907cSAndroid Build Coastguard Worker 
270*4d7e907cSAndroid Build Coastguard Worker }  // namespace aidl::android::hardware::audio::effect
271