mirror of https://github.com/probonopd/MiniDexed
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
278 lines
6.2 KiB
278 lines
6.2 KiB
#include "fx_components.h"
|
|
|
|
#include <cmath>
|
|
|
|
///////////////////////////////
|
|
// Constants implemlentation //
|
|
///////////////////////////////
|
|
const float32_t Constants::M2PI = 2.0f * PI;
|
|
const float32_t Constants::MPI_2 = PI / 2.0f;
|
|
const float32_t Constants::M1_PI = 1.0f / PI;
|
|
|
|
/////////////////////////
|
|
// LFO implemlentation //
|
|
/////////////////////////
|
|
LFO::LFO(float32_t sampling_rate, Waveform waveform, float32_t min_frequency, float32_t max_frequency, float32_t initial_phase) :
|
|
FXBase(sampling_rate),
|
|
InitialPhase(initial_phase),
|
|
min_frequency_(min_frequency),
|
|
max_frequency_(max_frequency),
|
|
waveform_(waveform),
|
|
normalized_frequency_(-1.0f),
|
|
frequency_(0.0f),
|
|
phase_(initial_phase),
|
|
phase_increment_(0.0f),
|
|
current_sample_(0.0f),
|
|
new_phase_(true),
|
|
rnd_generator_(rnd_device_()),
|
|
rnd_distribution_(-1.0f, 1.0f)
|
|
{
|
|
this->setWaveform(waveform);
|
|
this->setFrequency(this->min_frequency_);
|
|
}
|
|
|
|
LFO::~LFO()
|
|
{
|
|
}
|
|
|
|
void LFO::setWaveform(Waveform waveform)
|
|
{
|
|
this->waveform_ = waveform;
|
|
}
|
|
|
|
LFO::Waveform LFO::getWaveform() const
|
|
{
|
|
return this->waveform_;
|
|
}
|
|
|
|
void LFO::setNormalizedFrequency(float32_t normalized_frequency)
|
|
{
|
|
normalized_frequency = constrain(normalized_frequency, 0.0f, 1.0f);
|
|
if(this->normalized_frequency_ != normalized_frequency)
|
|
{
|
|
float32_t frequency = mapfloat(normalized_frequency, 0.0f, 1.0f, this->min_frequency_, this->max_frequency_);
|
|
this->normalized_frequency_ = normalized_frequency;
|
|
this->frequency_ = frequency;
|
|
this->phase_increment_ = Constants::M2PI * this->frequency_ / this->getSamplingRate();
|
|
}
|
|
}
|
|
|
|
float32_t LFO::getNormalizedFrequency() const
|
|
{
|
|
return this->normalized_frequency_;
|
|
}
|
|
|
|
void LFO::setFrequency(float32_t frequency)
|
|
{
|
|
frequency = constrain(frequency, this->min_frequency_, this->max_frequency_);
|
|
if(this->frequency_ != frequency)
|
|
{
|
|
float32_t normalized_frequency = mapfloat(frequency, this->min_frequency_, this->max_frequency_, 0.0f, 1.0f);
|
|
this->normalized_frequency_ = normalized_frequency;
|
|
this->frequency_ = frequency;
|
|
this->phase_increment_ = Constants::M2PI * this->frequency_ / this->getSamplingRate();
|
|
}
|
|
}
|
|
|
|
float32_t LFO::getFrequency() const
|
|
{
|
|
return this->frequency_;
|
|
}
|
|
|
|
void LFO::reset()
|
|
{
|
|
this->phase_ = this->InitialPhase;
|
|
this->current_sample_ = 0.0f;
|
|
}
|
|
|
|
float32_t LFO::process()
|
|
{
|
|
float32_t out = 0.0f;
|
|
switch(this->waveform_)
|
|
{
|
|
case Waveform::Sine:
|
|
out = std::sin(this->phase_);
|
|
break;
|
|
case Waveform::Saw:
|
|
out = Constants::M1_PI * this->phase_ - 1.0f;
|
|
break;
|
|
case Waveform::Square:
|
|
out = this->phase_ < PI ? 1.0 : -1.0;
|
|
break;
|
|
case Waveform::SH:
|
|
if(this->new_phase_)
|
|
{
|
|
out = this->rnd_distribution_(this->rnd_generator_);
|
|
}
|
|
else
|
|
{
|
|
out = this->current_sample_;
|
|
}
|
|
break;
|
|
case Waveform::Noise:
|
|
out = this->rnd_distribution_(this->rnd_generator_);
|
|
break;
|
|
}
|
|
|
|
this->current_sample_ = out;
|
|
|
|
this->phase_ += this->phase_increment_;
|
|
if(this->phase_ >= Constants::M2PI)
|
|
{
|
|
this->phase_ -= Constants::M2PI;
|
|
this->new_phase_ = true;
|
|
}
|
|
else
|
|
{
|
|
this->new_phase_ = false;
|
|
}
|
|
|
|
return out;
|
|
}
|
|
|
|
float32_t LFO::current() const
|
|
{
|
|
return this->current_sample_;
|
|
}
|
|
|
|
|
|
////////////////////////////////////
|
|
// JitterGenerator implementation //
|
|
////////////////////////////////////
|
|
JitterGenerator::JitterGenerator(float32_t sampling_rate) :
|
|
FXBase(sampling_rate),
|
|
rnd_generator_(rnd_device_()),
|
|
rnd_distribution_(-1.0f, 1.0f),
|
|
speed_(-1.0f),
|
|
magnitude_(-1.0f),
|
|
phase_(0.0f),
|
|
phase_increment_(0.0f)
|
|
{
|
|
this->setSpeed(1.0f);
|
|
this->setMagnitude(0.1f);
|
|
}
|
|
|
|
JitterGenerator::~JitterGenerator()
|
|
{
|
|
}
|
|
|
|
void JitterGenerator::setSpeed(float32_t speed)
|
|
{
|
|
if(this->speed_ != speed)
|
|
{
|
|
this->speed_ = speed;
|
|
this->phase_increment_ = Constants::M2PI * this->speed_ / this->getSamplingRate();
|
|
}
|
|
}
|
|
|
|
float32_t JitterGenerator::getSpeed() const
|
|
{
|
|
return this->speed_;
|
|
}
|
|
|
|
void JitterGenerator::setMagnitude(float32_t magnitude)
|
|
{
|
|
this->magnitude_ = magnitude;
|
|
}
|
|
|
|
float32_t JitterGenerator::getMagnitude() const
|
|
{
|
|
return this->magnitude_;
|
|
}
|
|
|
|
void JitterGenerator::reset()
|
|
{
|
|
this->phase_ = 0.0f;
|
|
}
|
|
|
|
float32_t JitterGenerator::process()
|
|
{
|
|
float32_t out = std::sin(this->phase_);
|
|
|
|
this->phase_ += this->phase_increment_ * (1.0f + this->magnitude_ * this->rnd_distribution_(this->rnd_generator_));
|
|
if(this->phase_ > Constants::M2PI)
|
|
{
|
|
this->phase_ -= Constants::M2PI;
|
|
}
|
|
|
|
return out;
|
|
}
|
|
|
|
//////////////////////////////////
|
|
// softSaturate implemlentation //
|
|
//////////////////////////////////
|
|
float32_t softSaturator1(float32_t in, float32_t threshold)
|
|
{
|
|
float32_t x = std::abs(in);
|
|
float32_t y = 0.0f;
|
|
if(x < threshold)
|
|
{
|
|
y = x;
|
|
}
|
|
else if(x > threshold)
|
|
{
|
|
y = threshold + (x - threshold) / (1.0f + std::pow((x - threshold) / (1.0f - threshold), 2.0f));
|
|
}
|
|
else if(x > 1.0f)
|
|
{
|
|
y = (threshold + 1.0f) / 2.0f;
|
|
}
|
|
|
|
float32_t g = 2.0f / (1.0f + threshold);
|
|
y *= g;
|
|
|
|
return (in < 0.0f) ? -y : y;
|
|
}
|
|
|
|
float32_t softSaturator2(float32_t input, float32_t saturation)
|
|
{
|
|
constexpr static float kTubeCurve = 4.0f;
|
|
constexpr static float kTubeBias = 0.5f;
|
|
|
|
float absInput = std::abs(input);
|
|
float output = 0.0f;
|
|
if(absInput > kTubeBias)
|
|
{
|
|
output = (kTubeCurve + saturation) * (absInput - kTubeBias) / (1.0f - kTubeBias);
|
|
}
|
|
else
|
|
{
|
|
output = (kTubeCurve + saturation) * absInput / (1.0f + kTubeCurve * absInput);
|
|
}
|
|
|
|
// Clip the output if overdrive is set to 1
|
|
// output = std::min(1.0f, output);
|
|
if(output > 1.0f)
|
|
{
|
|
output = 1.0f;
|
|
}
|
|
else
|
|
{
|
|
output -= output * output * output / 3.0f;
|
|
}
|
|
|
|
if(input < 0.0f)
|
|
{
|
|
output = -output;
|
|
}
|
|
|
|
return output;
|
|
}
|
|
|
|
float32_t waveFolder(float32_t input, float32_t bias)
|
|
{
|
|
bias = 0.5 + (2.0f - bias) / 4.0f;
|
|
float32_t output = std::abs(input) / bias;
|
|
|
|
if(output > 1.0f)
|
|
{
|
|
output = 2.0f - output;
|
|
}
|
|
|
|
if(input < 0.0f)
|
|
{
|
|
output = -output;
|
|
}
|
|
|
|
return output;
|
|
} |