xref: /aosp_15_r20/external/oboe/src/flowgraph/RampLinear.cpp (revision 05767d913155b055644481607e6fa1e35e2fe72c)
1*05767d91SRobert Wu /*
2*05767d91SRobert Wu  * Copyright 2015 The Android Open Source Project
3*05767d91SRobert Wu  *
4*05767d91SRobert Wu  * Licensed under the Apache License, Version 2.0 (the "License");
5*05767d91SRobert Wu  * you may not use this file except in compliance with the License.
6*05767d91SRobert Wu  * You may obtain a copy of the License at
7*05767d91SRobert Wu  *
8*05767d91SRobert Wu  *      http://www.apache.org/licenses/LICENSE-2.0
9*05767d91SRobert Wu  *
10*05767d91SRobert Wu  * Unless required by applicable law or agreed to in writing, software
11*05767d91SRobert Wu  * distributed under the License is distributed on an "AS IS" BASIS,
12*05767d91SRobert Wu  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*05767d91SRobert Wu  * See the License for the specific language governing permissions and
14*05767d91SRobert Wu  * limitations under the License.
15*05767d91SRobert Wu  */
16*05767d91SRobert Wu 
17*05767d91SRobert Wu #include <algorithm>
18*05767d91SRobert Wu #include <unistd.h>
19*05767d91SRobert Wu #include "FlowGraphNode.h"
20*05767d91SRobert Wu #include "RampLinear.h"
21*05767d91SRobert Wu 
22*05767d91SRobert Wu using namespace FLOWGRAPH_OUTER_NAMESPACE::flowgraph;
23*05767d91SRobert Wu 
RampLinear(int32_t channelCount)24*05767d91SRobert Wu RampLinear::RampLinear(int32_t channelCount)
25*05767d91SRobert Wu         : FlowGraphFilter(channelCount) {
26*05767d91SRobert Wu     mTarget.store(1.0f);
27*05767d91SRobert Wu }
28*05767d91SRobert Wu 
setLengthInFrames(int32_t frames)29*05767d91SRobert Wu void RampLinear::setLengthInFrames(int32_t frames) {
30*05767d91SRobert Wu     mLengthInFrames = frames;
31*05767d91SRobert Wu }
32*05767d91SRobert Wu 
setTarget(float target)33*05767d91SRobert Wu void RampLinear::setTarget(float target) {
34*05767d91SRobert Wu     mTarget.store(target);
35*05767d91SRobert Wu     // If the ramp has not been used then start immediately at this level.
36*05767d91SRobert Wu     if (mLastCallCount == kInitialCallCount) {
37*05767d91SRobert Wu         forceCurrent(target);
38*05767d91SRobert Wu     }
39*05767d91SRobert Wu }
40*05767d91SRobert Wu 
interpolateCurrent()41*05767d91SRobert Wu float RampLinear::interpolateCurrent() {
42*05767d91SRobert Wu     return mLevelTo - (mRemaining * mScaler);
43*05767d91SRobert Wu }
44*05767d91SRobert Wu 
onProcess(int32_t numFrames)45*05767d91SRobert Wu int32_t RampLinear::onProcess(int32_t numFrames) {
46*05767d91SRobert Wu     const float *inputBuffer = input.getBuffer();
47*05767d91SRobert Wu     float *outputBuffer = output.getBuffer();
48*05767d91SRobert Wu     int32_t channelCount = output.getSamplesPerFrame();
49*05767d91SRobert Wu 
50*05767d91SRobert Wu     float target = getTarget();
51*05767d91SRobert Wu     if (target != mLevelTo) {
52*05767d91SRobert Wu         // Start new ramp. Continue from previous level.
53*05767d91SRobert Wu         mLevelFrom = interpolateCurrent();
54*05767d91SRobert Wu         mLevelTo = target;
55*05767d91SRobert Wu         mRemaining = mLengthInFrames;
56*05767d91SRobert Wu         mScaler = (mLevelTo - mLevelFrom) / mLengthInFrames; // for interpolation
57*05767d91SRobert Wu     }
58*05767d91SRobert Wu 
59*05767d91SRobert Wu     int32_t framesLeft = numFrames;
60*05767d91SRobert Wu 
61*05767d91SRobert Wu     if (mRemaining > 0) { // Ramping? This doesn't happen very often.
62*05767d91SRobert Wu         int32_t framesToRamp = std::min(framesLeft, mRemaining);
63*05767d91SRobert Wu         framesLeft -= framesToRamp;
64*05767d91SRobert Wu         while (framesToRamp > 0) {
65*05767d91SRobert Wu             float currentLevel = interpolateCurrent();
66*05767d91SRobert Wu             for (int ch = 0; ch < channelCount; ch++) {
67*05767d91SRobert Wu                 *outputBuffer++ = *inputBuffer++ * currentLevel;
68*05767d91SRobert Wu             }
69*05767d91SRobert Wu             mRemaining--;
70*05767d91SRobert Wu             framesToRamp--;
71*05767d91SRobert Wu         }
72*05767d91SRobert Wu     }
73*05767d91SRobert Wu 
74*05767d91SRobert Wu     // Process any frames after the ramp.
75*05767d91SRobert Wu     int32_t samplesLeft = framesLeft * channelCount;
76*05767d91SRobert Wu     for (int i = 0; i < samplesLeft; i++) {
77*05767d91SRobert Wu         *outputBuffer++ = *inputBuffer++ * mLevelTo;
78*05767d91SRobert Wu     }
79*05767d91SRobert Wu 
80*05767d91SRobert Wu     return numFrames;
81*05767d91SRobert Wu }
82