/* * Copyright (C) 2016 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. */ #ifndef SINE_GENERATOR_H #define SINE_GENERATOR_H #include <math.h> class SineGenerator { public: SineGenerator() {} virtual ~SineGenerator() = default; void setup(double frequency, double frameRate) { mFrameRate = frameRate; mPhaseIncrement = frequency * M_PI * 2 / frameRate; } void setSweep(double frequencyLow, double frequencyHigh, double seconds) { mSweeping = seconds > 0.0; if (mSweeping) { mPhaseIncrementLow = frequencyLow * M_PI * 2 / mFrameRate; mPhaseIncrementHigh = frequencyHigh * M_PI * 2 / mFrameRate; double numFrames = seconds * mFrameRate; mUpScaler = pow((frequencyHigh / frequencyLow), (1.0 / numFrames)); mDownScaler = 1.0 / mUpScaler; } } void render(int16_t *buffer, int32_t channelStride, int32_t numFrames) { int sampleIndex = 0; for (int i = 0; i < numFrames; i++) { buffer[sampleIndex] = (int16_t) (INT16_MAX * sin(mPhase) * mAmplitude); sampleIndex += channelStride; advancePhase(); } } void render(float *buffer, int32_t channelStride, int32_t numFrames) { int sampleIndex = 0; for (int i = 0; i < numFrames; i++) { buffer[sampleIndex] = sin(mPhase) * mAmplitude; sampleIndex += channelStride; advancePhase(); } } void setAmplitude(double amplitude) { mAmplitude = amplitude; } double getAmplitude() const { return mAmplitude; } private: void advancePhase() { mPhase += mPhaseIncrement; if (mPhase > M_PI * 2) { mPhase -= M_PI * 2; } if (mSweeping) { if (mGoingUp) { mPhaseIncrement *= mUpScaler; if (mPhaseIncrement > mPhaseIncrementHigh) { mGoingUp = false; } } else { mPhaseIncrement *= mDownScaler; if (mPhaseIncrement < mPhaseIncrementLow) { mGoingUp = true; } } } } double mAmplitude = 0.05; // unitless scaler double mPhase = 0.0; double mPhaseIncrement = 440 * M_PI * 2 / 48000; double mFrameRate = 48000; double mPhaseIncrementLow; double mPhaseIncrementHigh; double mUpScaler = 1.0; double mDownScaler = 1.0; bool mGoingUp = false; bool mSweeping = false; }; #endif /* SINE_GENERATOR_H */