/* * Stereo Low Pass Filter * Javier Nonis (https://github.com/jnonis) - 2024 */ #ifndef _EFFECT_LPF_H #define _EFFECT_LPF_H #include "effect_base.h" class AudioEffectLPF : public AudioEffect { public: static constexpr float32_t MIN_CUTOFF = 0.00001f; static constexpr float32_t MAX_CUTOFF = 8000.0f; static constexpr float32_t MIN_RES = 0.0f; static constexpr float32_t MAX_RES = 1.0f; struct LPFState { float32_t y1; float32_t y2; float32_t y3; float32_t y4; float32_t oldx; float32_t oldy1; float32_t oldy2; float32_t oldy3; }; enum Param { BYPASS, CUTOFF, RESONANCE, UNKNOWN }; AudioEffectLPF(float32_t samplerate) : AudioEffect(samplerate) { // Setup init values this->setCutoff(2000.0f); this->setResonance(MIN_RES); this->stateL = new LPFState(); this->stateR = new LPFState(); } virtual ~AudioEffectLPF() { delete this->stateL; delete this->stateR; } virtual unsigned getId() { return EFFECT_LPF; } virtual void setParameter(unsigned param, unsigned value) { switch (param) { case AudioEffectLPF::Param::BYPASS: this->setBypass(value == 1); break; case AudioEffectLPF::Param::CUTOFF: this->setCutoff(((float32_t) value / 100.0f) * MAX_CUTOFF); break; case AudioEffectLPF::Param::RESONANCE: this->setResonance((float32_t) value / 100.0f); break; default: break; } } virtual unsigned getParameter(unsigned param) { switch (param) { case AudioEffectLPF::Param::BYPASS: return this->getBypass() ? 1 : 0; case AudioEffectLPF::Param::CUTOFF: return roundf((this->cutoff / MAX_CUTOFF) * 100); case AudioEffectLPF::Param::RESONANCE: return roundf(this->resonance * 100); default: return 0; } } float32_t processSampleL(float32_t input) { return processSample(input, this->stateL); } float32_t processSampleR(float32_t input) { return processSample(input, this->stateR); } protected: virtual size_t getParametersSize() { return sizeof(AudioEffectLPF::Param); } virtual void doProcess(const float32_t* inblockL, const float32_t* inblockR, float32_t* outblockL, float32_t* outblockR, uint16_t len) { for (int i = 0; i < len; i++) { outblockL[i] = processSampleL(inblockL[i]); outblockR[i] = processSampleR(inblockR[i]); } } private: float32_t cutoff; float32_t resonance; float32_t r, p, k; LPFState* stateL; LPFState* stateR; /** * Set the static cutoff frequency of the filter. * Cutoff frequency must be between MIN_CUTOFF and MAX_CUTOFF. * Envelope signal varies the cutoff frequency from this static value. */ void setCutoff(float32_t cutoff) { // Check limits cutoff = (cutoff < MIN_CUTOFF) ? MIN_CUTOFF : cutoff; cutoff = (cutoff > MAX_CUTOFF) ? MAX_CUTOFF : cutoff; this->cutoff = cutoff; recalculate(); } /** * Set the resonance of the filter. * Valid values are between 0.0 and 1.0 where * 0.0 is no resonance and 1.0 is full resonance or oscillation. */ void setResonance(float32_t resonance) { // Check limits resonance = (resonance < MIN_RES) ? MIN_RES : resonance; resonance = (resonance > MAX_RES) ? MAX_RES : resonance; this->resonance = resonance; recalculate(); } /** * Recalculate filter parameters on changes to cutoff or resonance */ void recalculate() { float32_t f = (cutoff + cutoff) / this->samplerate; p = f * (1.8 - (0.8 * f)); k = p + p - 1.0; float32_t t = (1.0 - p) * 1.386249; float32_t t2 = 12.0 + t * t; r = resonance * (t2 + 6.0 * t) / (t2 - 6.0 * t); } /** * Process a single sample through the filter */ float32_t processSample(float32_t input, LPFState* state) { float32_t y1 = state->y1; float32_t y2 = state->y2; float32_t y3 = state->y3; float32_t y4 = state->y4; float32_t oldx = state->oldx; float32_t oldy1 = state->oldy1; float32_t oldy2 = state->oldy2; float32_t oldy3 = state->oldy3; // Process input float32_t x = input - r * y4; // Four cascaded one pole filters (bilinear transform) y1 = x * p + oldx * p - k * y1; y2 = y1 * p + oldy1 * p - k * y2; y3 = y2 * p + oldy2 * p - k * y3; y4 = y3 * p + oldy3 * p - k * y4; // Clipper band limited sigmoid y4 -= (y4 * y4 * y4) / 6.0; state->y1 = y1; state->y2 = y2; state->y3 = y3; state->y4 = y4; state->oldx = x; state->oldy1 = y1; state->oldy2 = y2; state->oldy3 = y3; return y4; } }; #endif // _EFFECT_LPF_H