/*
* Copyright (C) 2018 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 DPFREQUENCY_H_
#define DPFREQUENCY_H_
#include <Eigen/Dense>
#include <unsupported/Eigen/FFT>
#include "RDsp.h"
#include "SHCircularBuffer.h"
#include "DPBase.h"
namespace dp_fx {
using FXBuffer = SHCircularBuffer<float>;
class ChannelBuffer {
public:
FXBuffer cBInput; // Circular Buffer input
FXBuffer cBOutput; // Circular Buffer output
FloatVec input; // time domain temp vector for input
FloatVec output; // time domain temp vector for output
FloatVec outTail; // time domain temp vector for output tail (for overlap-add method)
Eigen::VectorXcf complexTemp; // complex temp vector for frequency domain operations
//Current parameters
float inputGainDb;
float outputGainDb;
struct BandParams {
bool enabled;
float freqCutoffHz;
size_t binStart;
size_t binStop;
};
struct EqBandParams : public BandParams {
float gainDb;
};
struct MbcBandParams : public BandParams {
float gainPreDb;
float gainPostDb;
float attackTimeMs;
float releaseTimeMs;
float ratio;
float thresholdDb;
float kneeWidthDb;
float noiseGateThresholdDb;
float expanderRatio;
//Historic values
float previousEnvelope;
};
struct LimiterParams {
int32_t linkGroup;
float attackTimeMs;
float releaseTimeMs;
float ratio;
float thresholdDb;
float postGainDb;
//Historic values
float previousEnvelope;
float newFactor;
float linkFactor;
};
bool mPreEqInUse;
bool mPreEqEnabled;
std::vector<EqBandParams> mPreEqBands;
bool mMbcInUse;
bool mMbcEnabled;
std::vector<MbcBandParams> mMbcBands;
bool mPostEqInUse;
bool mPostEqEnabled;
std::vector<EqBandParams> mPostEqBands;
bool mLimiterInUse;
bool mLimiterEnabled;
LimiterParams mLimiterParams;
FloatVec mPreEqFactorVector; // temp pre-computed vector to shape spectrum at preEQ stage
FloatVec mPostEqFactorVector; // temp pre-computed vector to shape spectrum at postEQ stage
void initBuffers(unsigned int blockSize, unsigned int overlapSize, unsigned int halfFftSize,
unsigned int samplingRate, DPBase &dpBase);
void computeBinStartStop(BandParams &bp, size_t binStart);
private:
unsigned int mSamplingRate;
unsigned int mBlockSize;
};
using CBufferVector = std::vector<ChannelBuffer>;
using GroupsMap = std::map<int32_t, IntVec>;
class LinkedLimiters {
public:
void reset();
void update(int32_t group, int index);
void remove(int index);
GroupsMap mGroupsMap;
};
class DPFrequency : public DPBase {
public:
virtual size_t processSamples(const float *in, float *out, size_t samples);
virtual void reset();
void configure(size_t blockSize, size_t overlapSize, size_t samplingRate);
static size_t getMinBockSize();
static size_t getMaxBockSize();
private:
void updateParameters(ChannelBuffer &cb, int channelIndex);
size_t processMono(ChannelBuffer &cb);
size_t processOneVector(FloatVec &output, FloatVec &input, ChannelBuffer &cb);
size_t processChannelBuffers(CBufferVector &channelBuffers);
size_t processFirstStages(ChannelBuffer &cb);
size_t processLastStages(ChannelBuffer &cb);
void processLinkedLimiters(CBufferVector &channelBuffers);
size_t mBlockSize;
size_t mHalfFFTSize;
size_t mOverlapSize;
size_t mSamplingRate;
float mBlocksPerSecond;
CBufferVector mChannelBuffers;
LinkedLimiters mLinkedLimiters;
//dsp
FloatVec mVWindow; //window class.
float mWindowRms;
Eigen::FFT<float> mFftServer;
};
} //namespace dp_fx
#endif // DPFREQUENCY_H_