// 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, either version 3 of the License, or // (at your option) any later version. // // 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 . // // // fx_svf.h // // State Variable Filter used in Tape Delay // #pragma once #include "fx.h" #include "fx_components.h" class StateVariableFilter : public FXElement { DISALLOW_COPY_AND_ASSIGN(StateVariableFilter); public: enum FilterMode { LPF, // Low pass filter HPF, // High pass filter BPF // Band pass filter }; StateVariableFilter(float32_t sampling_rate, FilterMode mode, float32_t cutoff); virtual ~StateVariableFilter(); void setFilterMode(FilterMode mode); void setGainDB(float32_t gainDB); void setCutoff(float32_t cutoff); void setResonance(float32_t resonance); virtual void reset() override; virtual void processSample(float32_t inL, float32_t inR, float32_t& outL, float32_t& outR) override; private: void updateCoefficients(); FilterMode mode_; float32_t gain_; float32_t cutoff_; float32_t resonance_; float32_t g_; float32_t w_; float32_t a_; float32_t b_; float32_t c1_; float32_t c2_; float32_t d0_; float32_t d1_; float32_t z1_[StereoChannels::kNumChannels]; float32_t z2_[StereoChannels::kNumChannels]; IMPLEMENT_DUMP( const size_t space = 12; const size_t precision = 6; std::stringstream ss; out << "START " << tag << "(" << typeid(*this).name() << ") dump" << std::endl << std::endl; SS_RESET(ss, precision, std::left); SS__TEXT(ss, ' ', space, std::left, '|', "mode_"); SS__TEXT(ss, ' ', space, std::left, '|', "gain_"); SS__TEXT(ss, ' ', space, std::left, '|', "cutoff_"); SS__TEXT(ss, ' ', space, std::left, '|', "resonance_"); SS__TEXT(ss, ' ', space, std::left, '|', "g_"); SS__TEXT(ss, ' ', space, std::left, '|', "w_"); SS__TEXT(ss, ' ', space, std::left, '|', "a_"); SS__TEXT(ss, ' ', space, std::left, '|', "b_"); SS__TEXT(ss, ' ', space, std::left, '|', "c1_"); SS__TEXT(ss, ' ', space, std::left, '|', "c2_"); out << "\t" << ss.str() << std::endl; SS_RESET(ss, precision, std::left); SS_SPACE(ss, '-', space, std::left, '+'); SS_SPACE(ss, '-', space, std::left, '+'); SS_SPACE(ss, '-', space, std::left, '+'); SS_SPACE(ss, '-', space, std::left, '+'); SS_SPACE(ss, '-', space, std::left, '+'); SS_SPACE(ss, '-', space, std::left, '+'); SS_SPACE(ss, '-', space, std::left, '+'); SS_SPACE(ss, '-', space, std::left, '+'); SS_SPACE(ss, '-', space, std::left, '+'); SS_SPACE(ss, '-', space, std::left, '+'); out << "\t" << ss.str() << std::endl; SS_RESET(ss, precision, std::left); SS__TEXT(ss, ' ', space - 1, std::right, " |", this->mode_); SS__TEXT(ss, ' ', space - 1, std::right, " |", this->gain_); SS__TEXT(ss, ' ', space - 1, std::right, " |", this->cutoff_); SS__TEXT(ss, ' ', space - 1, std::right, " |", this->resonance_); SS__TEXT(ss, ' ', space - 1, std::right, " |", this->g_); SS__TEXT(ss, ' ', space - 1, std::right, " |", this->w_); SS__TEXT(ss, ' ', space - 1, std::right, " |", this->a_); SS__TEXT(ss, ' ', space - 1, std::right, " |", this->b_); SS__TEXT(ss, ' ', space - 1, std::right, " |", this->c1_); SS__TEXT(ss, ' ', space - 1, std::right, " |", this->c2_); out << "\t" << ss.str() << std::endl; out << "END " << tag << "(" << typeid(*this).name() << ") dump" << std::endl << std::endl; ) IMPLEMENT_INSPECT( size_t nb_errors = 0u; nb_errors += inspector(tag + "gain_", this->gain_, -1.0f, 1.0f, deepInspection); nb_errors += inspector(tag + "cutoff_", this->cutoff_, 1.0f, this->getSamplingRate() / 2.0f, deepInspection); nb_errors += inspector(tag + "resonance_", this->resonance_, 0.005f, 1.0f, deepInspection); nb_errors += inspector(tag + "g_", this->g_, 0.0f, 16.0f, deepInspection); nb_errors += inspector(tag + "w_", this->w_, 0.0f, 13.0f, deepInspection); nb_errors += inspector(tag + "a_", this->a_, 0.0f, 2526.0f, deepInspection); nb_errors += inspector(tag + "b_", this->b_, 0.0f, 160.0f, deepInspection); nb_errors += inspector(tag + "c1_", this->c1_, 0.0f, 2.06f, deepInspection); nb_errors += inspector(tag + "c2_", this->c2_, 0.0f, 0.06f, deepInspection); return nb_errors; ) }; class SVF : public FXElement { DISALLOW_COPY_AND_ASSIGN(SVF); public: enum FrequencyApproximation { FrequencyExact, FrequencyAccurate, FrequencyFast, FrequencyDirty }; enum FilterMode { SVF_LP, SVF_BP, SVF_BP_NORMALIZED, SVF_HP }; SVF(float32_t sampling_frequency, FilterMode mode = FilterMode::SVF_LP) : FXElement(sampling_frequency), Mode(mode), g_(0.0f), r_(0.0f), h_(0.0f) { this->reset(); } virtual ~SVF() { } inline virtual void reset() override { memset(this->state1_, 0, StereoChannels::kNumChannels * sizeof(float32_t)); memset(this->state2_, 0, StereoChannels::kNumChannels * sizeof(float32_t)); } virtual void processSample(float32_t inL, float32_t inR, float32_t& outL, float32_t& outR) override { float32_t hp, bp, lp; { hp = (inL - this->r_ * this->state1_[StereoChannels::Left ] - this->g_ * this->state1_[StereoChannels::Left ] - this->state2_[StereoChannels::Left ]) * this->h_; bp = this->g_ * hp + this->state1_[StereoChannels::Left ]; this->state1_[StereoChannels::Left ] = this->g_ * hp + bp; lp = this->g_ * bp + this->state2_[StereoChannels::Left ]; this->state2_[StereoChannels::Left ] = this->g_ * bp + lp; switch(this->Mode) { case FilterMode::SVF_LP: outL = lp; break; case FilterMode::SVF_BP: outL = bp; break; case FilterMode::SVF_BP_NORMALIZED: outL = bp * this->r_; break; case FilterMode::SVF_HP: outL = hp; break; } } { hp = (inR - this->r_ * this->state1_[StereoChannels::Right] - this->g_ * this->state1_[StereoChannels::Right] - this->state2_[StereoChannels::Right]) * this->h_; bp = this->g_ * hp + this->state1_[StereoChannels::Right]; this->state1_[StereoChannels::Right] = this->g_ * hp + bp; lp = this->g_ * bp + this->state2_[StereoChannels::Right]; this->state2_[StereoChannels::Right] = this->g_ * bp + lp; switch(this->Mode) { case FilterMode::SVF_LP: outR = lp; break; case FilterMode::SVF_BP: outR = bp; break; case FilterMode::SVF_BP_NORMALIZED: outR = bp * this->r_; break; case FilterMode::SVF_HP: outR = hp; break; } } } inline void setGRH(float32_t g, float32_t r, float32_t h) { this->g_ = g; this->r_ = r; this->h_ = h; } inline void setGR(float32_t g, float32_t r) { this->g_ = g; this->r_ = r; this->h_ = 1.0f / (1.0f + this->r_ * this->g_ * this->g_ * this->g_); } template inline void setFQ(float32_t frequency, float32_t resonance) { this->g_ = SVF::tan(frequency); this->r_ = 1.0f / resonance; this->h_ = 1.0f / (1.0f + this->r_ * this->g_ * this->g_ * this->g_); } private: template static inline float32_t tan(float32_t f) { switch(approximation) { case FrequencyApproximation::FrequencyExact: { // Clip coefficient to about 100. f = constrain(f, 0.0f, 0.497f); return ::tan(PI * f); } case FrequencyApproximation::FrequencyDirty: { // Optimized for frequencies below 8kHz. const float32_t a = 3.736e-01 * Constants::M_PI_POW_3; return f * (PI + a * f * f); } case FrequencyApproximation::FrequencyFast: { // The usual tangent approximation uses 3.1755e-01 and 2.033e-01, but // the coefficients used here are optimized to minimize error for the // 16Hz to 16kHz range, with a sample rate of 48kHz. const float a = 3.260e-01 * Constants::M_PI_POW_3; const float b = 1.823e-01 * Constants::M_PI_POW_5; float f2 = f * f; return f * (PI + f2 * (a + b * f2)); } case FrequencyApproximation::FrequencyAccurate: { // These coefficients don't need to be tweaked for the audio range. const float a = 3.333314036e-01 * Constants::M_PI_POW_3; const float b = 1.333923995e-01 * Constants::M_PI_POW_5; const float c = 5.33740603e-02 * Constants::M_PI_POW_7; const float d = 2.900525e-03 * Constants::M_PI_POW_9; const float e = 9.5168091e-03 * Constants::M_PI_POW_11; float f2 = f * f; return f * (PI + f2 * (a + f2 * (b + f2 * (c + f2 * (d + f2 * e))))); } } } const FilterMode Mode; float32_t g_; float32_t r_; float32_t h_; float32_t state1_[StereoChannels::kNumChannels]; float32_t state2_[StereoChannels::kNumChannels]; IMPLEMENT_DUMP( const size_t space = 12; const size_t precision = 6; std::stringstream ss; out << "START " << tag << "(" << typeid(*this).name() << ") dump" << std::endl << std::endl; SS_RESET(ss, precision, std::left); SS__TEXT(ss, ' ', space, std::left, '|', "g_"); SS__TEXT(ss, ' ', space, std::left, '|', "r_"); SS__TEXT(ss, ' ', space, std::left, '|', "h_"); SS__TEXT(ss, ' ', space, std::left, '|', "state1_[ L ]"); SS__TEXT(ss, ' ', space, std::left, '|', "state1_[ R ]"); SS__TEXT(ss, ' ', space, std::left, '|', "state2_[ L ]"); SS__TEXT(ss, ' ', space, std::left, '|', "state2_[ R ]"); out << "\t" << ss.str() << std::endl; SS_RESET(ss, precision, std::left); SS_SPACE(ss, '-', space, std::left, '+'); SS_SPACE(ss, '-', space, std::left, '+'); SS_SPACE(ss, '-', space, std::left, '+'); SS_SPACE(ss, '-', space, std::left, '+'); SS_SPACE(ss, '-', space, std::left, '+'); SS_SPACE(ss, '-', space, std::left, '+'); SS_SPACE(ss, '-', space, std::left, '+'); out << "\t" << ss.str() << std::endl; SS_RESET(ss, precision, std::left); SS__TEXT(ss, ' ', space - 1, std::right, " |", this->g_); SS__TEXT(ss, ' ', space - 1, std::right, " |", this->r_); SS__TEXT(ss, ' ', space - 1, std::right, " |", this->r_); SS__TEXT(ss, ' ', space - 1, std::right, " |", this->state1_[StereoChannels::Left ]); SS__TEXT(ss, ' ', space - 1, std::right, " |", this->state1_[StereoChannels::Right]); SS__TEXT(ss, ' ', space - 1, std::right, " |", this->state2_[StereoChannels::Left ]); SS__TEXT(ss, ' ', space - 1, std::right, " |", this->state2_[StereoChannels::Right]); out << "\t" << ss.str() << std::endl; out << "END " << tag << "(" << typeid(*this).name() << ") dump" << std::endl << std::endl; ) IMPLEMENT_INSPECT( size_t nb_errors = 0u; nb_errors += inspector(tag + ".g_", this->g_, 0.0f, 106.11f, deepInspection); nb_errors += inspector(tag + ".r_", this->r_, 0.0f, 1.0f, deepInspection); nb_errors += inspector(tag + ".h_", this->h_, 0.0f, 1.0f, deepInspection); nb_errors += inspector(tag + ".state1_[ L ]", this->state1_[StereoChannels::Left ], -1.0f, 1.0f, deepInspection); nb_errors += inspector(tag + ".state1_[ R ]", this->state1_[StereoChannels::Right], -1.0f, 1.0f, deepInspection); nb_errors += inspector(tag + ".state2_[ L ]", this->state2_[StereoChannels::Left ], -1.0f, 1.0f, deepInspection); nb_errors += inspector(tag + ".state2_[ R ]", this->state2_[StereoChannels::Right], -1.0f, 1.0f, deepInspection); return nb_errors; ) };