1 /* 2 * Copyright 2017 The Android Open Source Project 3 * 4 * Licensed under the Apache License, Version 2.0 (the "License"); 5 * you may not use this file except in compliance with the License. 6 * You may obtain a copy of the License at 7 * 8 * http://www.apache.org/licenses/LICENSE-2.0 9 * 10 * Unless required by applicable law or agreed to in writing, software 11 * distributed under the License is distributed on an "AS IS" BASIS, 12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 13 * See the License for the specific language governing permissions and 14 * limitations under the License. 15 */ 16 17 #ifndef AAUDIO_LINEAR_RAMP_H 18 #define AAUDIO_LINEAR_RAMP_H 19 20 #include <atomic> 21 #include <stdint.h> 22 23 /** 24 * Generate segments along a linear ramp. 25 * The ramp target can be updated from another thread. 26 * When the target is updated, a new ramp is started from the current position. 27 * 28 * The first ramp starts at 0.0. 29 * 30 */ 31 class LinearRamp { 32 public: LinearRamp()33 LinearRamp() { 34 mTarget.store(1.0f); 35 } 36 setLengthInFrames(int32_t frames)37 void setLengthInFrames(int32_t frames) { 38 mLengthInFrames = frames; 39 } 40 getLengthInFrames()41 int32_t getLengthInFrames() { 42 return mLengthInFrames; 43 } 44 45 /** 46 * This may be called by another thread. 47 * @param target 48 */ setTarget(float target)49 void setTarget(float target) { 50 mTarget.store(target); 51 } 52 getTarget()53 float getTarget() { 54 return mTarget.load(); 55 } 56 57 /** 58 * Force the nextSegment to start from this level. 59 * 60 * WARNING: this can cause a discontinuity if called while the ramp is being used. 61 * Only call this when setting the initial ramp. 62 * 63 * @param level 64 */ forceCurrent(float level)65 void forceCurrent(float level) { 66 mLevelFrom = level; 67 mLevelTo = level; // forces a ramp if it does not match target 68 } 69 getCurrent()70 float getCurrent() { 71 return mLevelFrom; 72 } 73 74 /** 75 * Get levels for next ramp segment. 76 * 77 * @param frames number of frames in the segment 78 * @param levelFrom pointer to starting amplitude 79 * @param levelTo pointer to ending amplitude 80 * @return true if ramp is still moving towards the target 81 */ 82 bool nextSegment(int32_t frames, float *levelFrom, float *levelTo); 83 84 private: 85 86 bool isRamping(); 87 88 std::atomic<float> mTarget; 89 90 int32_t mLengthInFrames = 48000 / 50; // 20 msec at 48000 Hz 91 int32_t mRemaining = 0; 92 float mLevelFrom = 0.0f; 93 float mLevelTo = 0.0f; 94 }; 95 96 97 #endif //AAUDIO_LINEAR_RAMP_H 98