1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74
|
/*
This file is part of the SimpleCompressor project.
https://github.com/DanielRudrich/SimpleCompressor
Copyright (c) 2019 Daniel Rudrich
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, version 3.
This program is distributed in the hope that it will be useful, but
WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
#pragma once
#include <vector>
namespace DanielRudrich {
/** This class acts as a delay line for gain-reduction samples, which additionally fades in high gain-reduction values in order to avoid distortion when limiting an audio signal.
*/
class LookAheadGainReduction
{
public:
LookAheadGainReduction() : sampleRate (0.0) {}
~LookAheadGainReduction() {}
void setDelayTime (float delayTimeInSeconds);
const int getDelayInSamples() { return delayInSamples; }
/** Prepares the processor so it can resize the buffers depending on samplerate and the expected buffersize.
*/
void prepare (const double sampleRate, const int blockSize);
/** Writes gain-reduction samples into the delay-line. Make sure you call process() afterwards, and read the same amount of samples with the readSamples method. Make also sure the pushed samples are decibel values.
*/
void pushSamples (const float* src, const int numSamples);
/** Processes the data within the delay line, i.e. fades-in high gain-reduction, in order to reduce distortions.
*/
void process();
/** Reads smoothed gain-reduction samples back to the destination. Make sure you read as many samples as you've pushed before!
*/
void readSamples (float* dest, const int numSamples);
private:
/** A little helper-function which calulcates how many samples we should process in a first step before we have to wrap around, as our buffer is a ring-buffer.
*/
inline void getProcessPositions (int startIndex, int numSamples, int& blockSize1, int& blockSize2);
inline void getWritePositions (int numSamples, int& startIndex, int& blockSize1, int& blockSize2);
inline void getReadPositions (int numSamples, int& startIndex, int& blockSize1, int& blockSize2);
private:
//==============================================================================
double sampleRate;
int blockSize;
float delay;
int delayInSamples = 0;
int writePosition = 0;
int lastPushedSamples = 0;
std::vector<float> buffer;
};
} // namespace DanielRudrich
|