pull/495/head
abscisys 2 years ago
parent 4de38cbc70
commit cf20795131
  1. 8
      src/fx_chorus.cpp
  2. 42
      src/fx_components.cpp
  3. 37
      src/fx_components.h
  4. 4
      src/fx_delay.cpp
  5. 6
      src/fx_diffuser.cpp
  6. 25
      src/fx_diffuser.h
  7. 4
      src/fx_engine.hpp
  8. 4
      src/fx_flanger.cpp
  9. 12
      src/fx_orbitone.cpp
  10. 4
      src/fx_phaser.cpp
  11. 7
      src/fx_pitch_shifter.cpp
  12. 63
      src/fx_pitch_shifter.h
  13. 6
      src/fx_reverberator.h
  14. 243
      src/fx_shimmer_reverb.cpp
  15. 81
      src/fx_shimmer_reverb.h
  16. 28
      src/fx_svf.cpp
  17. 248
      src/fx_svf.h
  18. 10
      src/mixing_console.hpp
  19. 8
      src/test/Makefile
  20. 6
      src/test/test_fx_components.cpp
  21. 20
      src/test/test_unitFXTuning.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);

@ -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<float32_t>(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<size_t>(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_;

@ -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);
float32_t waveFolder(float32_t input, float32_t bias);

@ -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);

@ -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)
{
}

@ -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<DIFFUSER_BUFFER_SIZE, Format::FORMAT_FLOAT32, false> 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;
)
};

@ -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:

@ -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);

@ -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)
{

@ -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);

@ -4,15 +4,14 @@
#include <cmath>
#include <algorithm>
#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;

@ -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<PITCH_SHIFTER_BUFFER_SIZE, Format::FORMAT_FLOAT32, false> 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;
)
};

@ -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)
{

@ -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<Memory, 0> ap1;
Engine::DelayLine<Memory, 1> ap2;
Engine::DelayLine<Memory, 2> ap3;
Engine::DelayLine<Memory, 3> ap4;
Engine::DelayLine<Memory, 4> dap1a;
Engine::DelayLine<Memory, 5> dap1b;
Engine::DelayLine<Memory, 6> del1;
Engine::DelayLine<Memory, 7> dap2a;
Engine::DelayLine<Memory, 8> dap2b;
Engine::DelayLine<Memory, 9> 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<SVF::FrequencyApproximation::FrequencyFast>(this->lp_cutoff_, this->lpq_);
this->hp_filter_.setFQ<SVF::FrequencyApproximation::FrequencyFast>(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();
}

@ -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<SHIMMER_REVERB_BUFFER_SIZE, Format::FORMAT_FLOAT32, true> 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;
)
};

@ -2,9 +2,9 @@
#include <cmath>
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];

@ -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;
)
};
};
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<FrequencyApproximation approximation>
inline void setFQ(float32_t frequency, float32_t resonance)
{
this->g_ = SVF::tan<approximation>(frequency);
this->r_ = 1.0f / resonance;
this->h_ = 1.0f / (1.0f + this->r_ * this->g_ * this->g_ * this->g_);
}
private:
template<FrequencyApproximation approximation>
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;
)
};

@ -94,7 +94,7 @@ private:
FXUnit2<Phaser>* phaser_;
FXUnit2<Delay>* delay_;
FXUnit2<AudioEffectPlateReverb>* plate_reverb_;
FXUnit2<Reverberator>* shimmer_reverb_;
FXUnit2<Reverberator>* reverberator_;
FXUnit2<Dry>* 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<nb_inputs>::MixingConsole(float32_t sampling_rate, size_t buffer_s
this->fx_[MixerOutput::FX_Phaser] = this->phaser_ = new FXUnit2<Phaser>(sampling_rate);
this->fx_[MixerOutput::FX_Delay] = this->delay_ = new FXUnit2<Delay>(sampling_rate);
this->fx_[MixerOutput::FX_PlateReverb] = this->plate_reverb_ = new FXUnit2<AudioEffectPlateReverb>(sampling_rate);
this->fx_[MixerOutput::FX_Reverberator] = this->shimmer_reverb_ = new FXUnit2<Reverberator>(sampling_rate);
this->fx_[MixerOutput::FX_Reverberator] = this->reverberator_ = new FXUnit2<Reverberator>(sampling_rate);
this->fx_[MixerOutput::MainOutput] = this->dry_ = new FXUnit2<Dry>(sampling_rate);
this->init();
@ -513,7 +513,7 @@ FXUnit2<AudioEffectPlateReverb>* MixingConsole<nb_inputs>::getPlateReverb()
template<size_t nb_inputs>
FXUnit2<Reverberator>* MixingConsole<nb_inputs>::getReverberator()
{
return this->shimmer_reverb_;
return this->reverberator_;
}
template<size_t nb_inputs>

@ -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

@ -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;

@ -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);
}

Loading…
Cancel
Save