diff --git a/src/fx_chorus.cpp b/src/fx_chorus.cpp index b22f71c..b941236 100644 --- a/src/fx_chorus.cpp +++ b/src/fx_chorus.cpp @@ -13,10 +13,10 @@ Chorus::Chorus(float32_t sampling_rate) : fullscale_depth_(0.0f), feedback_(0.0f) { - this->lfo_[LFOIndex::Sin1] = new LFO(sampling_rate, 0.0f, LFO1_MAX_FREQ); - this->lfo_[LFOIndex::Cos1] = new LFO(sampling_rate, 0.0f, LFO1_MAX_FREQ, Constants::MPI_2); - this->lfo_[LFOIndex::Sin2] = new LFO(sampling_rate, 0.0f, LFO2_MAX_FREQ); - this->lfo_[LFOIndex::Cos2] = new LFO(sampling_rate, 0.0f, LFO2_MAX_FREQ, Constants::MPI_2); + this->lfo_[LFOIndex::Sin1] = new LFO(sampling_rate, 0.0f, LFO1_MAX_FREQ, 0.0f, false); + this->lfo_[LFOIndex::Cos1] = new LFO(sampling_rate, 0.0f, LFO1_MAX_FREQ, Constants::MPI_2, false); + this->lfo_[LFOIndex::Sin2] = new LFO(sampling_rate, 0.0f, LFO2_MAX_FREQ, 0.0f, false); + this->lfo_[LFOIndex::Cos2] = new LFO(sampling_rate, 0.0f, LFO2_MAX_FREQ, Constants::MPI_2, false); this->setRate(0.1f); this->setDepth(0.15f); diff --git a/src/fx_components.cpp b/src/fx_components.cpp index c3bdc27..75d4b39 100644 --- a/src/fx_components.cpp +++ b/src/fx_components.cpp @@ -5,6 +5,13 @@ /////////////////////////////// // Constants implemlentation // /////////////////////////////// +const float32_t Constants::M_PI_POW_2 = PI * PI; +const float32_t Constants::M_PI_POW_3 = Constants::M_PI_POW_2 * PI; +const float32_t Constants::M_PI_POW_5 = Constants::M_PI_POW_2 * Constants::M_PI_POW_3; +const float32_t Constants::M_PI_POW_7 = Constants::M_PI_POW_2 * Constants::M_PI_POW_5; +const float32_t Constants::M_PI_POW_9 = Constants::M_PI_POW_2 * Constants::M_PI_POW_7; +const float32_t Constants::M_PI_POW_11 = Constants::M_PI_POW_2 * Constants::M_PI_POW_9; + const float32_t Constants::M2PI = 2.0f * PI; const float32_t Constants::MPI_2 = PI / 2.0f; const float32_t Constants::MPI_3 = PI / 3.0f; @@ -15,11 +22,12 @@ const float32_t Constants::M1_PI = 1.0f / PI; ///////////////////////////// // FastLFO implemlentation // ///////////////////////////// -FastLFO::FastLFO(float32_t sampling_rate, float32_t min_frequency, float32_t max_frequency, float32_t initial_phase) : +FastLFO::FastLFO(float32_t sampling_rate, float32_t min_frequency, float32_t max_frequency, float32_t initial_phase, bool centered) : FXBase(sampling_rate), InitialPhase(initial_phase), min_frequency_(min_frequency), max_frequency_(max_frequency), + centered_(centered), frequency_(0.0f), y0_(0.0f), y1_(0.0f), @@ -132,7 +140,14 @@ float32_t FastLFO::process() float32_t temp = this->y0_; this->y0_ = this->iir_coefficient_ * this->y0_ - this->y1_; this->y1_ = temp; - this->current_ = (temp + 0.5f) * 2.0f - 1.0f; + if(this->centered_) + { + this->current_ = (temp + 0.5f) * 2.0f - 1.0f; + } + else + { + this->current_ = temp + 0.5f; + } return this->current_; } @@ -152,22 +167,25 @@ bool InterpolatedSineOscillator::ClassInitializer() float32_t phase = 0.0; for(size_t i = 0; i <= InterpolatedSineOscillator::DataPointSize; ++i) { - InterpolatedSineOscillator::DataPoints[i] = std::sin(phase); + InterpolatedSineOscillator::CenteredDataPoints[i] = std::sin(phase); + InterpolatedSineOscillator::UpliftDataPoints[i] = InterpolatedSineOscillator::CenteredDataPoints[i] * 0.5f + 0.5f; phase += phase_increment; } return true; } -float32_t InterpolatedSineOscillator::DataPoints[InterpolatedSineOscillator::DataPointSize + 1]; +float32_t InterpolatedSineOscillator::CenteredDataPoints[InterpolatedSineOscillator::DataPointSize + 1]; +float32_t InterpolatedSineOscillator::UpliftDataPoints[InterpolatedSineOscillator::DataPointSize + 1]; const float32_t InterpolatedSineOscillator::DeltaTime = Constants::M2PI / static_cast(InterpolatedSineOscillator::DataPointSize); -InterpolatedSineOscillator::InterpolatedSineOscillator(float32_t sampling_rate, float32_t min_frequency, float32_t max_frequency, float32_t initial_phase) : +InterpolatedSineOscillator::InterpolatedSineOscillator(float32_t sampling_rate, float32_t min_frequency, float32_t max_frequency, float32_t initial_phase, bool centered) : FXBase(sampling_rate), InitialPhase(initial_phase), min_frequency_(min_frequency), max_frequency_(max_frequency), + centered_(centered), frequency_(0.0f), normalized_frequency_(-1.0f), phase_index_(initial_phase / InterpolatedSineOscillator::DeltaTime), @@ -232,14 +250,16 @@ void InterpolatedSineOscillator::reset() float32_t InterpolatedSineOscillator::process() { + float32_t* dataPoints = this->centered_ ? InterpolatedSineOscillator::CenteredDataPoints : InterpolatedSineOscillator::UpliftDataPoints; + float32_t out = 0.0f; float32_t findex = this->phase_index_; size_t index1 = static_cast(findex); size_t index2 = index1 + 1; - float32_t f1 = InterpolatedSineOscillator::DataPoints[index1]; - float32_t f2 = InterpolatedSineOscillator::DataPoints[index2]; + float32_t f1 = dataPoints[index1]; + float32_t f2 = dataPoints[index2]; float32_t r = findex - index1; out = f1 + (f2 - f1) * r * InterpolatedSineOscillator::DeltaTime; @@ -262,11 +282,12 @@ float32_t InterpolatedSineOscillator::current() const //////////////////////////////// // ComplexLFO implemlentation // //////////////////////////////// -ComplexLFO::ComplexLFO(float32_t sampling_rate, float32_t min_frequency, float32_t max_frequency, float32_t initial_phase) : +ComplexLFO::ComplexLFO(float32_t sampling_rate, float32_t min_frequency, float32_t max_frequency, float32_t initial_phase, bool centered) : FXBase(sampling_rate), InitialPhase(initial_phase), min_frequency_(min_frequency), max_frequency_(max_frequency), + centered_(centered), normalized_frequency_(-1.0f), frequency_(0.0f), phase_(initial_phase), @@ -366,6 +387,11 @@ float32_t ComplexLFO::process() break; } + if(!this->centered_) + { + out = out * 0.5f + 0.5f; + } + this->current_sample_ = out; this->phase_ += this->phase_increment_; diff --git a/src/fx_components.h b/src/fx_components.h index 044d4ae..5718c5f 100644 --- a/src/fx_components.h +++ b/src/fx_components.h @@ -26,6 +26,13 @@ struct Constants { + const static float32_t M_PI_POW_2; // PI^2 + const static float32_t M_PI_POW_3; // PI^3 + const static float32_t M_PI_POW_5; // PI^5 + const static float32_t M_PI_POW_7; // PI^7 + const static float32_t M_PI_POW_9; // PI^9 + const static float32_t M_PI_POW_11; // PI^11 + const static float32_t M2PI; // 2 * PI const static float32_t MPI_2; // PI / 2 const static float32_t MPI_3; // PI / 3 @@ -39,7 +46,7 @@ class FastLFO : public FXBase DISALLOW_COPY_AND_ASSIGN(FastLFO); public: - FastLFO(float32_t sampling_rate, float32_t min_frequency = 0.01f, float32_t max_frequency = 10.0f, float32_t initial_phase = 0.0f); + FastLFO(float32_t sampling_rate, float32_t min_frequency = 0.01f, float32_t max_frequency = 10.0f, float32_t initial_phase = 0.0f, bool centered = true); virtual ~FastLFO(); void setNormalizedFrequency(float32_t normalized_frequency); @@ -58,6 +65,7 @@ private: const float32_t InitialPhase; const float32_t min_frequency_; const float32_t max_frequency_; + const bool centered_; float32_t frequency_; float32_t normalized_frequency_; float32_t unitary_frequency_; @@ -135,7 +143,7 @@ class InterpolatedSineOscillator : public FXBase DISALLOW_COPY_AND_ASSIGN(InterpolatedSineOscillator); public: - InterpolatedSineOscillator(float32_t sampling_rate, float32_t min_frequency = 0.01f, float32_t max_frequency = 10.0f, float32_t initial_phase = 0.0f); + InterpolatedSineOscillator(float32_t sampling_rate, float32_t min_frequency = 0.01f, float32_t max_frequency = 10.0f, float32_t initial_phase = 0.0f, bool centered = true); virtual ~InterpolatedSineOscillator(); void setNormalizedFrequency(float32_t normalized_frequency); @@ -152,16 +160,18 @@ private: static bool ClassInitializer(); static const size_t DataPointSize = 176400; static const float32_t DeltaTime; - static float32_t DataPoints[]; + static float32_t CenteredDataPoints[]; + static float32_t UpliftDataPoints[]; - const float32_t InitialPhase; - const float32_t min_frequency_; - const float32_t max_frequency_; - float32_t frequency_; - float32_t normalized_frequency_; - float32_t phase_index_; - float32_t phase_index_increment_; - float32_t current_sample_; + const float32_t InitialPhase; + const float32_t min_frequency_; + const float32_t max_frequency_; + const bool centered_; + float32_t frequency_; + float32_t normalized_frequency_; + float32_t phase_index_; + float32_t phase_index_increment_; + float32_t current_sample_; IMPLEMENT_DUMP( const size_t space = 22; @@ -227,7 +237,7 @@ public: Noise } Waveform; - ComplexLFO(float32_t sampling_rate, float32_t min_frequency = 0.01f, float32_t max_frequency = 10.0f, float32_t initial_phase = 0.0f); + ComplexLFO(float32_t sampling_rate, float32_t min_frequency = 0.01f, float32_t max_frequency = 10.0f, float32_t initial_phase = 0.0f, bool centered = true); virtual ~ComplexLFO(); void setWaveform(Waveform waveform); @@ -247,6 +257,7 @@ private: const float32_t InitialPhase; const float32_t min_frequency_; const float32_t max_frequency_; + const bool centered_; Waveform waveform_; float32_t normalized_frequency_; float32_t frequency_; @@ -462,4 +473,4 @@ float32_t softSaturator2(float32_t in, float32_t saturation); float32_t softSaturator3(float32_t in, float32_t saturation); float32_t softSaturator4(float32_t in, float32_t saturation); -float32_t waveFolder(float32_t input, float32_t bias); \ No newline at end of file +float32_t waveFolder(float32_t input, float32_t bias); diff --git a/src/fx_delay.cpp b/src/fx_delay.cpp index 06335c5..b9cf5a6 100644 --- a/src/fx_delay.cpp +++ b/src/fx_delay.cpp @@ -10,8 +10,8 @@ Delay::LowHighPassFilter::LowHighPassFilter(float32_t sampling_rate) : FXElement(sampling_rate), - lpf_(sampling_rate, StateVariableFilter::Type::LPF, LPF_CUTOFF_REF), - hpf_(sampling_rate, StateVariableFilter::Type::HPF, HPF_CUTOFF_REF), + lpf_(sampling_rate, StateVariableFilter::FilterMode::LPF, LPF_CUTOFF_REF), + hpf_(sampling_rate, StateVariableFilter::FilterMode::HPF, HPF_CUTOFF_REF), ratio_(1.0f) { this->setCutoffChangeRatio(0.0f); diff --git a/src/fx_diffuser.cpp b/src/fx_diffuser.cpp index d657169..4976098 100644 --- a/src/fx_diffuser.cpp +++ b/src/fx_diffuser.cpp @@ -5,9 +5,9 @@ #define TAIL , -1 -Diffuser::Diffuser(float32_t sampling_rate) : - FXElement(sampling_rate), - engine_(sampling_rate) +Diffuser::Diffuser(float32_t sampling_frequency) : + FXElement(sampling_frequency), + engine_(sampling_frequency) { } diff --git a/src/fx_diffuser.h b/src/fx_diffuser.h index 561b03e..9f2ef44 100644 --- a/src/fx_diffuser.h +++ b/src/fx_diffuser.h @@ -30,7 +30,7 @@ class Diffuser : public FXElement DISALLOW_COPY_AND_ASSIGN(Diffuser); public: - Diffuser(float32_t sampling_rate); + Diffuser(float32_t sampling_frequency); virtual ~Diffuser(); virtual void reset() override; @@ -40,6 +40,25 @@ private: typedef FxEngine Engine; Engine engine_; - IMPLEMENT_DUMP() - IMPLEMENT_INSPECT(return 0u;) + IMPLEMENT_DUMP( + out << "START " << tag << "(" << typeid(*this).name() << ") dump" << std::endl << std::endl; + + if(deepInspection) + { + this->engine_.dump(out, deepInspection, tag + ".engine_"); + } + + out << "END " << tag << "(" << typeid(*this).name() << ") dump" << std::endl << std::endl; + ) + + IMPLEMENT_INSPECT( + size_t nb_errors = 0u; + + if(deepInspection) + { + nb_errors += this->engine_.inspect(inspector, deepInspection, tag + ".engine_"); + } + + return nb_errors; + ) }; diff --git a/src/fx_engine.hpp b/src/fx_engine.hpp index 2fd6d5b..02f1166 100644 --- a/src/fx_engine.hpp +++ b/src/fx_engine.hpp @@ -121,7 +121,7 @@ public: write_ptr_(0) { this->buffer_ = new T[size]; - for(unsigned i = 0; i < LFOIndex::kLFOCount; ++i) this->lfo_[i] = enable_lfo ? new LFO(sampling_rate, 0.0f, max_lfo_frequency) : nullptr; + for(unsigned i = 0; i < LFOIndex::kLFOCount; ++i) this->lfo_[i] = enable_lfo ? new LFO(sampling_rate, 0.0f, max_lfo_frequency, 0.0f, false) : nullptr; this->clear(); } @@ -354,7 +354,7 @@ public: { assert(index < LFOIndex::kLFOCount); - this->interpolate(d, offset + amplitude * this->lfo_value_[index], scale); + this->interpolate(d, offset + amplitude * (this->lfo_value_[index] * 0.5f + 0.5f), scale); } private: diff --git a/src/fx_flanger.cpp b/src/fx_flanger.cpp index 8362f9b..fca4b2f 100644 --- a/src/fx_flanger.cpp +++ b/src/fx_flanger.cpp @@ -8,8 +8,8 @@ Flanger::Flanger(float32_t sampling_rate, float32_t rate, float32_t depth, float this->delay_lineL_ = new float32_t[this->MaxDelayLineSize]; this->delay_lineR_ = new float32_t[this->MaxDelayLineSize]; - this->lfo_[LFOIndex::LFO_L] = new LFO(sampling_rate, 0.1f, 5.0f); - this->lfo_[LFOIndex::LFO_R] = new LFO(sampling_rate, 0.1f, 5.0f, Constants::MPI_2); + this->lfo_[LFOIndex::LFO_L] = new LFO(sampling_rate, 0.1f, 5.0f, 0.0f, false); + this->lfo_[LFOIndex::LFO_R] = new LFO(sampling_rate, 0.1f, 5.0f, Constants::MPI_2, false); this->setRate(rate); this->setDepth(depth); diff --git a/src/fx_orbitone.cpp b/src/fx_orbitone.cpp index 686c6f6..ca18f35 100644 --- a/src/fx_orbitone.cpp +++ b/src/fx_orbitone.cpp @@ -9,13 +9,13 @@ Orbitone::Orbitone(float32_t sampling_rate, float32_t rate, float32_t depth) : depth_(0.0f), fullscale_depth_(0.0f) { - this->lfo_[LFOIndex::Slow0 ] = new LFO(sampling_rate, 0.0f, LFO_SLOW_MAX_FREQUENCY, 0.0f); - this->lfo_[LFOIndex::Slow120] = new LFO(sampling_rate, 0.0f, LFO_SLOW_MAX_FREQUENCY, 2.0f * PI / 3.0); - this->lfo_[LFOIndex::Slow240] = new LFO(sampling_rate, 0.0f, LFO_SLOW_MAX_FREQUENCY, 4.0f * PI / 3.0); + this->lfo_[LFOIndex::Slow0 ] = new LFO(sampling_rate, 0.0f, LFO_SLOW_MAX_FREQUENCY, 0.0f, false); + this->lfo_[LFOIndex::Slow120] = new LFO(sampling_rate, 0.0f, LFO_SLOW_MAX_FREQUENCY, 2.0f * PI / 3.0, false); + this->lfo_[LFOIndex::Slow240] = new LFO(sampling_rate, 0.0f, LFO_SLOW_MAX_FREQUENCY, 4.0f * PI / 3.0, false); - this->lfo_[LFOIndex::Fast0 ] = new LFO(sampling_rate, 0.0f, LFO_FAST_MAX_FREQUENCY, 0.0f); - this->lfo_[LFOIndex::Fast120] = new LFO(sampling_rate, 0.0f, LFO_FAST_MAX_FREQUENCY, 2.0f * PI / 3.0); - this->lfo_[LFOIndex::Fast240] = new LFO(sampling_rate, 0.0f, LFO_FAST_MAX_FREQUENCY, 4.0f * PI / 3.0); + this->lfo_[LFOIndex::Fast0 ] = new LFO(sampling_rate, 0.0f, LFO_FAST_MAX_FREQUENCY, 0.0f, false); + this->lfo_[LFOIndex::Fast120] = new LFO(sampling_rate, 0.0f, LFO_FAST_MAX_FREQUENCY, 2.0f * PI / 3.0, false); + this->lfo_[LFOIndex::Fast240] = new LFO(sampling_rate, 0.0f, LFO_FAST_MAX_FREQUENCY, 4.0f * PI / 3.0, false); for(unsigned i = 0; i < LFOIndex::kLFOCount; ++i) { diff --git a/src/fx_phaser.cpp b/src/fx_phaser.cpp index 241d3cc..12e930e 100644 --- a/src/fx_phaser.cpp +++ b/src/fx_phaser.cpp @@ -40,8 +40,8 @@ Phaser::Phaser(float32_t sampling_rate, float32_t rate, float32_t depth, float32 dmin_(0.0f), dmax_(0.0f) { - this->lfo_[StereoChannels::Left ] = new LFO(sampling_rate, 0.0f, 2.5f); - this->lfo_[StereoChannels::Right] = new LFO(sampling_rate, 0.0f, 2.5f, Constants::MPI_2); + this->lfo_[StereoChannels::Left ] = new LFO(sampling_rate, 0.0f, 2.5f, 0.0f, false); + this->lfo_[StereoChannels::Right] = new LFO(sampling_rate, 0.0f, 2.5f, Constants::MPI_2, false); this->setRate(rate); this->setDepth(depth); diff --git a/src/fx_pitch_shifter.cpp b/src/fx_pitch_shifter.cpp index 550af94..5bb053a 100644 --- a/src/fx_pitch_shifter.cpp +++ b/src/fx_pitch_shifter.cpp @@ -4,15 +4,14 @@ #include #include -#define PITCH_SHIFT_BOUND 36.0f - #define ONE_POLE(out, in, coefficient) out += (coefficient) * ((in) - out); #define TAIL , -1 -PitchShifter::PitchShifter(float32_t sampling_rate) : +PitchShifter::PitchShifter(float32_t sampling_rate, float32_t transpose_boundary) : FXElement(sampling_rate), engine_(sampling_rate), + TransposeBoundary(transpose_boundary), phase_(0.0f), transpose_(0.0f), ratio_(0.0f), @@ -74,7 +73,7 @@ void PitchShifter::processSample(float32_t inL, float32_t inR, float32_t& outL, void PitchShifter::setTranspose(float32_t transpose) { - transpose = constrain(transpose, -PITCH_SHIFT_BOUND, PITCH_SHIFT_BOUND); + transpose = constrain(transpose, -this->TransposeBoundary, this->TransposeBoundary); if(this->transpose_ != transpose) { this->transpose_ = transpose; diff --git a/src/fx_pitch_shifter.h b/src/fx_pitch_shifter.h index 5d31d17..4b45030 100644 --- a/src/fx_pitch_shifter.h +++ b/src/fx_pitch_shifter.h @@ -24,13 +24,14 @@ #include "fx_engine.hpp" #define PITCH_SHIFTER_BUFFER_SIZE 4096 +#define PITCH_SHIFTER_TRANSPOSE_BOUNDARY 36.0f class PitchShifter : public FXElement { DISALLOW_COPY_AND_ASSIGN(PitchShifter); public: - PitchShifter(float32_t sampling_rate); + PitchShifter(float32_t sampling_rate, float32_t transpose_boundary = PITCH_SHIFTER_TRANSPOSE_BOUNDARY); virtual ~PitchShifter(); virtual void reset() override; @@ -46,12 +47,68 @@ private: typedef FxEngine Engine; Engine engine_; + const float32_t TransposeBoundary; float32_t phase_; float32_t transpose_; float32_t ratio_; float32_t size_; float32_t sample_size_; - IMPLEMENT_DUMP() - IMPLEMENT_INSPECT(return 0u;) + 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, '|', "phase_"); + SS__TEXT(ss, ' ', space, std::left, '|', "transpose_"); + SS__TEXT(ss, ' ', space, std::left, '|', "ratio_"); + SS__TEXT(ss, ' ', space, std::left, '|', "size_"); + SS__TEXT(ss, ' ', space, std::left, '|', "sample_size_"); + 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, '+'); + out << "\t" << ss.str() << std::endl; + + SS_RESET(ss, precision, std::left); + SS__TEXT(ss, ' ', space - 1, std::right, " |", this->phase_); + SS__TEXT(ss, ' ', space - 1, std::right, " |", this->transpose_); + SS__TEXT(ss, ' ', space - 1, std::right, " |", this->ratio_); + SS__TEXT(ss, ' ', space - 1, std::right, " |", this->size_); + SS__TEXT(ss, ' ', space - 1, std::right, " |", this->sample_size_); + out << "\t" << ss.str() << std::endl; + + if(deepInspection) + { + this->engine_.dump(out, deepInspection, tag + ".engine_"); + } + + out << "END " << tag << "(" << typeid(*this).name() << ") dump" << std::endl << std::endl; + ) + + IMPLEMENT_INSPECT( + size_t nb_errors = 0u; + + nb_errors += inspector(tag + ".phase_", this->phase_, 0.0f, 1.0f, deepInspection); + nb_errors += inspector(tag + ".transpose_", this->transpose_, -PITCH_SHIFTER_TRANSPOSE_BOUNDARY, PITCH_SHIFTER_TRANSPOSE_BOUNDARY, deepInspection); + nb_errors += inspector(tag + ".ratio_", this->ratio_, 0.0f, 1.0f, deepInspection); + nb_errors += inspector(tag + ".size_", this->size_, 0.0f, 1.0f, deepInspection); + nb_errors += inspector(tag + ".sample_size_", this->sample_size_, 0.0f, 1.0f, deepInspection); + + if(deepInspection) + { + nb_errors += this->engine_.inspect(inspector, deepInspection, tag + ".engine_"); + } + + return nb_errors; + ) }; diff --git a/src/fx_reverberator.h b/src/fx_reverberator.h index b8a8204..c01cbff 100644 --- a/src/fx_reverberator.h +++ b/src/fx_reverberator.h @@ -99,6 +99,8 @@ private: { this->engine_.dump(out, deepInspection, tag + ".engine_"); } + + out << "END " << tag << "(" << typeid(*this).name() << ") dump" << std::endl << std::endl; ) IMPLEMENT_INSPECT( @@ -108,8 +110,8 @@ private: nb_errors += inspector(tag + ".reverb_time_", this->reverb_time_, 0.0f, 1.0f, deepInspection); nb_errors += inspector(tag + ".diffusion_", this->diffusion_, 0.0f, 1.0f, deepInspection); nb_errors += inspector(tag + ".lp_", this->lp_, 0.0f, 1.0f, deepInspection); - nb_errors += inspector(tag + ".lp_decay_1_", this->lp_decay_1_, 0.0f, 1.0f, deepInspection); - nb_errors += inspector(tag + ".lp_decay_2_", this->lp_decay_2_, 0.0f, 1.0f, deepInspection); + nb_errors += inspector(tag + ".lp_decay_1_", this->lp_decay_1_, -1.0f, 1.0f, deepInspection); + nb_errors += inspector(tag + ".lp_decay_2_", this->lp_decay_2_, -1.0f, 1.0f, deepInspection); if(deepInspection) { diff --git a/src/fx_shimmer_reverb.cpp b/src/fx_shimmer_reverb.cpp index b8819e4..758857f 100644 --- a/src/fx_shimmer_reverb.cpp +++ b/src/fx_shimmer_reverb.cpp @@ -2,23 +2,23 @@ #define TAIL , -1 -ShimmerReverb::ShimmerReverb(float32_t sampling_rate) : - FXElement(sampling_rate), - engine_(sampling_rate), - input_gain_(-1.0f), - reverb_time_(0.0f), - diffusion_(-1.0f), - lp_(-1.0f), - lp_decay_1_(0.0f), - lp_decay_2_(0.0f) -{ - this->engine_.setLFOFrequency(Engine::LFOIndex::LFO_1, 0.5f); - this->engine_.setLFOFrequency(Engine::LFOIndex::LFO_2, 0.3f); - - this->setInputGain(1.0f); - this->setTime(0.7f); - this->setDiffusion(0.625f); - this->setLP(0.7f); +ShimmerReverb::ShimmerReverb(float32_t sampling_frequency) : + FXElement(sampling_frequency, 1.2f), + pitch_shifter_(sampling_frequency, PITCH_SHIFTER_TRANSPOSE_BOUNDARY), + lp_filter_(sampling_frequency, SVF::FilterMode::SVF_LP), + hp_filter_(sampling_frequency, SVF::FilterMode::SVF_HP), + reverberator_(sampling_frequency), + texture_(0.0f), + lp_cutoff_(0.0f), + hp_cutoff_(0.0f), + lpq_(0.0f), + amount_(0.0f), + feedback_(0.0f), + cutoff_(0.0f) +{ + this->setInputGain(0.2f); + this->setDiffusion(0.7f); + this->setCutoff(1.0f); this->reset(); } @@ -29,132 +29,141 @@ ShimmerReverb::~ShimmerReverb() void ShimmerReverb::reset() { - this->engine_.reset(); - this->lp_decay_1_ = 0.0f; - this->lp_decay_2_ = 0.0f; + this->pitch_shifter_.reset(); + this->lp_filter_.reset(); + this->hp_filter_.reset(); + this->reverberator_.reset(); } void ShimmerReverb::processSample(float32_t inL, float32_t inR, float32_t& outL, float32_t& outR) { - // This is the Griesinger topology described in the Dattorro paper - // (4 AP diffusers on the input, then a loop of 2x 2AP+1Delay). - // Modulation is applied in the loop of the first diffuser AP for additional - // smearing; and to the two long delays for a slow shimmer/chorus effect. - typedef Engine::Reserve< 113, - Engine::Reserve< 162, - Engine::Reserve< 241, - Engine::Reserve< 399, - Engine::Reserve<1653, - Engine::Reserve<2038, - Engine::Reserve<3411, - Engine::Reserve<1913, - Engine::Reserve<1663, - Engine::Reserve<4782> > > > > > > > > > Memory; - Engine::DelayLine ap1; - Engine::DelayLine ap2; - Engine::DelayLine ap3; - Engine::DelayLine ap4; - Engine::DelayLine dap1a; - Engine::DelayLine dap1b; - Engine::DelayLine del1; - Engine::DelayLine dap2a; - Engine::DelayLine dap2b; - Engine::DelayLine del2; - Engine::Context c; - - const float32_t kap = this->diffusion_; - const float32_t klp = this->lp_; - const float32_t krt = this->reverb_time_; - const float32_t gain = this->input_gain_; - - float32_t lp_1 = this->lp_decay_1_; - float32_t lp_2 = this->lp_decay_2_; - - float32_t wet = 0.0f; - float32_t apout = 0.0f; - engine_.start(&c); - - // Smear AP1 inside the loop. - c.interpolate(ap1, 10.0f, Engine::LFOIndex::LFO_1, 60.0f, 1.0f); - c.writeAndLoad(ap1, 100, 0.0f); - c.read(inL + inR, gain); - - // Diffuse through 4 allpasses. - c.read(ap1 TAIL, kap); - c.writeAllPass(ap1, -kap); - c.read(ap2 TAIL, kap); - c.writeAllPass(ap2, -kap); - c.read(ap3 TAIL, kap); - c.writeAllPass(ap3, -kap); - c.read(ap4 TAIL, kap); - c.writeAllPass(ap4, -kap); - c.write(apout); - - // Main reverb loop. - c.load(apout); - c.interpolate(del2, 4680.0f, Engine::LFOIndex::LFO_2, 100.0f, krt); - c.lp(lp_1, klp); - c.read(dap1a TAIL, -kap); - c.writeAllPass(dap1a, kap); - c.read(dap1b TAIL, kap); - c.writeAllPass(dap1b, -kap); - c.write(del1, 2.0f); - c.writeAndLoad(wet, 0.0f); - - outL = wet; - - c.load(apout); - c.read(del1 TAIL, krt); - c.lp(lp_2, klp); - c.read(dap2a TAIL, kap); - c.writeAllPass(dap2a, -kap); - c.read(dap2b TAIL, -kap); - c.writeAllPass(dap2b, kap); - c.write(del2, 2.0f); - c.writeAndLoad(wet, 0.0f); - - outR = wet; - - this->lp_decay_1_ = lp_1; - this->lp_decay_2_ = lp_2; -} - -void ShimmerReverb::setInputGain(float32_t gain) -{ - this->input_gain_ = constrain(gain, 0.0f, 1.0f); + this->pitch_shifter_.processSample(inL, inR, outL, outR); + this->lp_filter_.processSample(outL, outR, outL, outR); + this->hp_filter_.processSample(outL, outR, outL, outR); + this->reverberator_.processSample(outL, outR, outL, outR); + + outL *= this->OutputLevelCorrector; + outR *= this->OutputLevelCorrector; +} + +void ShimmerReverb::setInputGain(float32_t input_gain) +{ + this->reverberator_.setInputGain(input_gain); } float32_t ShimmerReverb::getInputGain() const { - return this->input_gain_; + return this->reverberator_.getInputGain(); +} + +void ShimmerReverb::setDiffusion(float32_t diffusion) +{ + this->reverberator_.setDiffusion(diffusion); +} + +float32_t ShimmerReverb::getDiffusion() const +{ + return this->reverberator_.getDiffusion(); } void ShimmerReverb::setTime(float32_t time) { - this->reverb_time_ = constrain(time, 0.0f, 1.0f); + this->reverberator_.setTime(time); } float32_t ShimmerReverb::getTime() const { - return this->reverb_time_; + return this->reverberator_.getTime(); } -void ShimmerReverb::setDiffusion(float32_t diffusion) +void ShimmerReverb::setReverbAmount(float32_t amount) { - this->diffusion_ = constrain(diffusion, 0.0f, 1.0f); + amount = constrain(amount, 0.0f, 1.0f); + if(this->amount_ != amount) + { + this->amount_ = amount; + this->updateReverberatorCoefficients(); + } } -float32_t ShimmerReverb::getDiffusion() const +void ShimmerReverb::setTexture(float32_t texture) +{ + texture = constrain(texture, 0.0f, 1.0f); + if(this->texture_ != texture) + { + this->texture_ = texture; + this->updateFilterCoefficients(); + } +} + +float32_t ShimmerReverb::getTexture() const +{ + return this->texture_; +} + +void ShimmerReverb::setFeedback(float32_t feedback) +{ + feedback = constrain(feedback, 0.0f, 1.0f); + if(this->feedback_ != feedback) + { + this->feedback_ = feedback; + this->updateFilterCoefficients(); + this->updateReverberatorCoefficients(); + } +} + +float32_t ShimmerReverb::getFeedback() const +{ + return this->feedback_; +} + +void ShimmerReverb::setCutoff(float32_t cutoff) +{ + cutoff = constrain(cutoff, 0.0f, 1.0f); + if(this->cutoff_ != cutoff) + { + this->cutoff_ = cutoff; + this->updateFilterCoefficients(); + } +} + +void ShimmerReverb::updateFilterCoefficients() +{ + this->lp_cutoff_ = constrain(0.50f * semitoneToRatio((this->cutoff_ < 0.5f ? this->cutoff_ - 0.5f : 0.0f ) * 216.0f), 0.0f, 0.499f); + this->hp_cutoff_ = constrain(0.25f * semitoneToRatio((this->cutoff_ < 0.5f ? -0.5f : this->cutoff_ - 1.0f) * 216.0f), 0.0f, 0.499f); + this->lpq_ = 1.0f + 3.0f * (1.0f - this->feedback_) * (0.5f - this->lp_cutoff_); + + this->lp_filter_.setFQ(this->lp_cutoff_, this->lpq_); + this->hp_filter_.setFQ(this->hp_cutoff_, 1.0f); + + this->reverberator_.setLP(0.6f + 0.37f * this->feedback_); +} + +void ShimmerReverb::updateReverberatorCoefficients() +{ + float32_t reverb_amount = this->amount_ * 0.95f; + reverb_amount += this->feedback_ * (2.0f - this->feedback_); + reverb_amount = constrain(reverb_amount, 0.0f, 1.0f); + + this->setTime(0.35f + 0.63f * reverb_amount); +} + +void ShimmerReverb::setPitch(float32_t pitch) +{ + this->pitch_shifter_.setTranspose(pitch); +} + +float32_t ShimmerReverb::getPitch() const { - return this->diffusion_; + return this->pitch_shifter_.getTranspose(); } -void ShimmerReverb::setLP(float32_t lp) +void ShimmerReverb::setSize(float32_t size) { - this->lp_ = constrain(lp, 0.0f, 1.0f); + this->pitch_shifter_.setSize(size); } -float32_t ShimmerReverb::getLP() const +float32_t ShimmerReverb::getSize() const { - return this->lp_; + return this->pitch_shifter_.getSize(); } diff --git a/src/fx_shimmer_reverb.h b/src/fx_shimmer_reverb.h index 5848ef1..e0535e5 100644 --- a/src/fx_shimmer_reverb.h +++ b/src/fx_shimmer_reverb.h @@ -21,9 +21,10 @@ #pragma once #include "fx_components.h" -#include "fx_engine.hpp" - -#define SHIMMER_REVERB_BUFFER_SIZE 16384 +#include "fx_svf.h" +#include "fx_shimmer_helper.h" +#include "fx_pitch_shifter.h" +#include "fx_reverberator.h" class ShimmerReverb : public FXElement { @@ -36,30 +37,74 @@ public: virtual void reset() override; virtual void processSample(float32_t inL, float32_t inR, float32_t& outL, float32_t& outR) override; - void setInputGain(float32_t gain); + void setInputGain(float32_t input_gain); float32_t getInputGain() const; + void setDiffusion(float32_t diffusion); + float32_t getDiffusion() const; + void setTime(float32_t time); float32_t getTime() const; - void setDiffusion(float32_t diffusion); - float32_t getDiffusion() const; + void setReverbAmount(float32_t amount); + + void setTexture(float32_t texture); + float32_t getTexture() const; - void setLP(float32_t lp); - float32_t getLP() const; + void setFeedback(float32_t feedback); + float32_t getFeedback() const; + + void setPitch(float32_t pitch); + float32_t getPitch() const; + + void setSize(float32_t size); + float32_t getSize() const; + + void setCutoff(float32_t cutoff); + float32_t getCutoff() const; private: - typedef FxEngine Engine; - Engine engine_; + void updateFilterCoefficients(); + void updateReverberatorCoefficients(); + + PitchShifter pitch_shifter_; + SVF lp_filter_; + SVF hp_filter_; + Reverberator reverberator_; + + float32_t texture_; + float32_t lp_cutoff_; + float32_t hp_cutoff_; + float32_t lpq_; + float32_t amount_; + float32_t feedback_; + float32_t cutoff_; + + IMPLEMENT_DUMP( + out << "START " << tag << "(" << typeid(*this).name() << ") dump" << std::endl << std::endl; + + out << "END " << tag << "(" << typeid(*this).name() << ") dump" << std::endl << std::endl; + ) + + IMPLEMENT_INSPECT( + size_t nb_errors = 0u; - float32_t input_gain_; - float32_t reverb_time_; - float32_t diffusion_; - float32_t lp_; + nb_errors += inspector(tag + ".texture_", this->texture_, 0.0f, 1.0f, deepInspection); + nb_errors += inspector(tag + ".lp_cutoff_", this->lp_cutoff_, 0.0f, 1.0f, deepInspection); + nb_errors += inspector(tag + ".hp_cutoff_", this->hp_cutoff_, 0.0f, 1.0f, deepInspection); + nb_errors += inspector(tag + ".lpq_", this->lpq_, 0.0f, 1.0f, deepInspection); + nb_errors += inspector(tag + ".amount_", this->amount_, 0.0f, 1.0f, deepInspection); + nb_errors += inspector(tag + ".feedback_", this->feedback_, 0.0f, 1.0f, deepInspection); + nb_errors += inspector(tag + ".cutoff_", this->cutoff_, 0.0f, 1.0f, deepInspection); - float32_t lp_decay_1_; - float32_t lp_decay_2_; + if(deepInspection) + { + nb_errors += this->pitch_shifter_.inspect(inspector, deepInspection, tag + ".pitch_shifter_"); + nb_errors += this->lp_filter_.inspect(inspector, deepInspection, tag + ".lp_filter_"); + nb_errors += this->hp_filter_.inspect(inspector, deepInspection, tag + ".hp_filter_"); + nb_errors += this->reverberator_.inspect(inspector, deepInspection, tag + ".reverberator_"); + } - IMPLEMENT_DUMP() - IMPLEMENT_INSPECT(return 0u;) + return nb_errors; + ) }; diff --git a/src/fx_svf.cpp b/src/fx_svf.cpp index c4f71ec..aa282c4 100644 --- a/src/fx_svf.cpp +++ b/src/fx_svf.cpp @@ -2,9 +2,9 @@ #include -StateVariableFilter::StateVariableFilter(float32_t sampling_rate, Type type, float32_t cutoff) : +StateVariableFilter::StateVariableFilter(float32_t sampling_rate, FilterMode mode, float32_t cutoff) : FXElement(sampling_rate), - type_(type), + mode_(mode), gain_(-1.0f), cutoff_(cutoff), resonance_(0.0f) @@ -20,11 +20,11 @@ StateVariableFilter::~StateVariableFilter() { } -void StateVariableFilter::setFilterType(Type type) +void StateVariableFilter::setFilterMode(FilterMode mode) { - if(this->type_ != type) + if(this->mode_ != mode) { - this->type_ = type; + this->mode_ = mode; this->updateCoefficients(); } } @@ -72,17 +72,19 @@ void StateVariableFilter::updateCoefficients() this->c1_ = a_b / (1.0f + 0.5f * this->a_ + 0.25f * this->b_); this->c2_ = this->b_ / a_b; - switch(this->type_) + switch(this->mode_) { - case Type::LPF: + case FilterMode::LPF: this->d1_ = 0.0f; this->d0_ = 0.25f * this->c1_ * this->c2_; break; - case Type::HPF: + + case FilterMode::HPF: this->d1_ = 0.0f; this->d0_ = 1.0f - 0.5f * this->c1_ + 0.25f * this->c1_ * this->c2_; break; - case Type::BPF: + + case FilterMode::BPF: this->d1_ = 1.0f - this->c2_; this->d0_ = this->d1_ * this->c1_ * 0.5f; break; @@ -101,9 +103,9 @@ void StateVariableFilter::processSample(float32_t inL, float32_t inR, float32_t& { const float32_t gain = this->g_; - switch(this->type_) + switch(this->mode_) { - case Type::LPF: + case FilterMode::LPF: { const float32_t x = inL - this->z1_[StereoChannels::Left] - this->z2_[StereoChannels::Left] + 1e-20f; this->z2_[StereoChannels::Left] += this->c2_ * this->z1_[StereoChannels::Left]; @@ -118,7 +120,7 @@ void StateVariableFilter::processSample(float32_t inL, float32_t inR, float32_t& } break; - case Type::HPF: + case FilterMode::HPF: { const float32_t x = inL - this->z1_[StereoChannels::Left] - this->z2_[StereoChannels::Left] + 1e-20f; outL = gain * this->d0_ * x; @@ -133,7 +135,7 @@ void StateVariableFilter::processSample(float32_t inL, float32_t inR, float32_t& } break; - case Type::BPF: + case FilterMode::BPF: { const float32_t x = inL - this->z1_[StereoChannels::Left] - this->z2_[StereoChannels::Left] + 1e-20f; outL = gain * (this->d0_ * x) + this->d1_ * this->z1_[StereoChannels::Left]; diff --git a/src/fx_svf.h b/src/fx_svf.h index d1d4ec5..3f8119b 100644 --- a/src/fx_svf.h +++ b/src/fx_svf.h @@ -17,25 +17,27 @@ // // 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: - typedef enum + enum FilterMode { LPF, // Low pass filter HPF, // High pass filter BPF // Band pass filter - } Type; + }; - StateVariableFilter(float32_t sampling_rate, Type type, float32_t cutoff); + StateVariableFilter(float32_t sampling_rate, FilterMode mode, float32_t cutoff); virtual ~StateVariableFilter(); - void setFilterType(Type type); + void setFilterMode(FilterMode mode); void setGainDB(float32_t gainDB); void setCutoff(float32_t cutoff); void setResonance(float32_t resonance); @@ -46,7 +48,7 @@ public: private: void updateCoefficients(); - Type type_; + FilterMode mode_; float32_t gain_; float32_t cutoff_; float32_t resonance_; @@ -70,6 +72,7 @@ private: 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_"); @@ -91,9 +94,11 @@ private: 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_); @@ -123,4 +128,235 @@ private: return nb_errors; ) -}; \ No newline at end of file +}; + +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; + ) +}; diff --git a/src/mixing_console.hpp b/src/mixing_console.hpp index f602c74..d7da09b 100644 --- a/src/mixing_console.hpp +++ b/src/mixing_console.hpp @@ -94,7 +94,7 @@ private: FXUnit2* phaser_; FXUnit2* delay_; FXUnit2* plate_reverb_; - FXUnit2* shimmer_reverb_; + FXUnit2* reverberator_; FXUnit2* dry_; IMPLEMENT_DUMP( @@ -247,7 +247,7 @@ private: this->phaser_->dump(out, deepInspection, tag + ".phaser_"); this->delay_->dump(out, deepInspection, tag + ".delay_"); this->plate_reverb_->dump(out, deepInspection, tag + ".plate_reverb_"); - this->shimmer_reverb_->dump(out, deepInspection, tag + ".shimmer_reverb_"); + this->reverberator_->dump(out, deepInspection, tag + ".reverberator_"); this->dry_->dump(out, deepInspection, tag + ".dry_"); } @@ -303,7 +303,7 @@ private: nb_errors += this->phaser_->inspect(inspector, deepInspection, tag + ".phaser_"); nb_errors += this->delay_->inspect(inspector, deepInspection, tag + ".delay_"); nb_errors += this->plate_reverb_->inspect(inspector, deepInspection, tag + ".plate_reverb_"); - nb_errors += this->shimmer_reverb_->inspect(inspector, deepInspection, tag + ".shimmer_reverb_"); + nb_errors += this->reverberator_->inspect(inspector, deepInspection, tag + ".reverberator_"); nb_errors += this->dry_->inspect(inspector, deepInspection, tag + ".dry_"); } @@ -332,7 +332,7 @@ MixingConsole::MixingConsole(float32_t sampling_rate, size_t buffer_s this->fx_[MixerOutput::FX_Phaser] = this->phaser_ = new FXUnit2(sampling_rate); this->fx_[MixerOutput::FX_Delay] = this->delay_ = new FXUnit2(sampling_rate); this->fx_[MixerOutput::FX_PlateReverb] = this->plate_reverb_ = new FXUnit2(sampling_rate); - this->fx_[MixerOutput::FX_Reverberator] = this->shimmer_reverb_ = new FXUnit2(sampling_rate); + this->fx_[MixerOutput::FX_Reverberator] = this->reverberator_ = new FXUnit2(sampling_rate); this->fx_[MixerOutput::MainOutput] = this->dry_ = new FXUnit2(sampling_rate); this->init(); @@ -513,7 +513,7 @@ FXUnit2* MixingConsole::getPlateReverb() template FXUnit2* MixingConsole::getReverberator() { - return this->shimmer_reverb_; + return this->reverberator_; } template diff --git a/src/test/Makefile b/src/test/Makefile index f37dbf5..25347ae 100644 --- a/src/test/Makefile +++ b/src/test/Makefile @@ -5,15 +5,15 @@ OUTPUT_FOLDER := results EXE := $(BINDIR)/all_tests.bin BETA := $(BINDIR)/beta.bin -CXX = g++ +CXX = g++ CXXFLAGS = -g -Wall -std=c++20 -MMD -MP DEFINES = -DCPU=x86 -DDEBUG -DOUTPUT_FOLDER="\"$(OUTPUT_FOLDER)\"" INCLUDES = -I../../CMSIS_5/CMSIS/DSP/Include/ \ -I../../CMSIS_5/CMSIS/Core/Include/ \ -I../../Synth_Dexed/src/ -CPPCHECK = cppcheck -CHECKFLAGS = -q -j 8 --enable=all --force --language=c++ \ +CPPCHECK = cppcheck +CHECKFLAGS = -q -j 8 --enable=all --force --language=c++ \ $(INCLUDES) --platform=unix64 \ --error-exitcode=0 \ --suppressions-list=cppcheck-suppression-list.txt @@ -21,7 +21,7 @@ CHECKFLAGS = -q -j 8 --enable=all --force --language=c++ \ -include $(TST_OBJS:.o=.d) -include $(FX__OBJS:.o=.d) -LD := g++ +LD := g++ LIBS := -lm -lstdc++ -lgtest -lpthread FX__SRCS := ../fx.cpp diff --git a/src/test/test_fx_components.cpp b/src/test/test_fx_components.cpp index 1e4b726..7084da8 100644 --- a/src/test/test_fx_components.cpp +++ b/src/test/test_fx_components.cpp @@ -59,10 +59,10 @@ TEST(FXComponent, SVF) { float32_t inL, inR; float32_t outL, outR; - StateVariableFilter svf(SAMPLING_FREQUENCY, StateVariableFilter::Type::LPF, 12000.0f); + StateVariableFilter svf(SAMPLING_FREQUENCY, StateVariableFilter::FilterMode::LPF, 12000.0f); { - svf.setFilterType(StateVariableFilter::Type::LPF); + svf.setFilterMode(StateVariableFilter::FilterMode::LPF); svf.setCutoff(12000.0f); svf.setResonance(0.0f); unsigned nbSamples = 0; @@ -83,7 +83,7 @@ TEST(FXComponent, SVF) } { - svf.setFilterType(StateVariableFilter::Type::LPF); + svf.setFilterMode(StateVariableFilter::FilterMode::LPF); svf.setCutoff(60.0f); svf.setResonance(0.0f); unsigned nbSamples = 0; diff --git a/src/test/test_unitFXTuning.cpp b/src/test/test_unitFXTuning.cpp index d9a5293..ee03536 100644 --- a/src/test/test_unitFXTuning.cpp +++ b/src/test/test_unitFXTuning.cpp @@ -130,8 +130,8 @@ TEST(UnitFXTuning, Diffuser) TEST(UnitFXTuning, PitchShifter) { PitchShifter fx(SAMPLING_FREQUENCY); - fx.setSize(0.2f); - fx.setTranspose(24.0f); + fx.setSize(0.5f); + fx.setTranspose(12.0f); PREPARE_AUDIO_TEST(size, inSamples, outSamples, full_test_name); SIMPLE_AUDIO_LOOP(inSamples, outSamples, size, inL, inR, outL, outR, fx); @@ -153,3 +153,19 @@ TEST(UnitFXTuning, Reverberator) CLEANUP_AUDIO_TEST(inSamples, outSamples); } +TEST(UnitFXTuning, ShimmerReverb) +{ + const float32_t amount = 0.6f; + + ShimmerReverb fx(SAMPLING_FREQUENCY); + fx.setInputGain(0.2f); + fx.setReverbAmount(amount); + fx.setDiffusion(0.7f); + fx.setFeedback(0.8f); + + PREPARE_AUDIO_TEST(size, inSamples, outSamples, full_test_name); + SIMPLE_AUDIO_LOOP(inSamples, outSamples, size, inL, inR, outL, outR, fx); + SAVE_AUDIO_RESULTS(full_test_name, outSamples, size); + CLEANUP_AUDIO_TEST(inSamples, outSamples); +} +