#include "fx_delay.h" #include #define MAX_DELAY_TIME 2.0f #define MAX_FLUTTER_DELAY_TIME 0.001f #define LPF_CUTOFF_REF 12000.0f #define HPF_CUTOFF_REF 80.0f 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), ratio_(1.0f) { this->setCutoffChangeRatio(0.0f); this->lpf_.setGainDB(0.82f); this->hpf_.setGainDB(0.82f); } Delay::LowHighPassFilter::~LowHighPassFilter() { } void Delay::LowHighPassFilter::setCutoffChangeRatio(float32_t ratio) { static const float32_t weight = 4.0f; ratio = constrain(ratio, -1.0f, 1.0f); if(ratio != this->ratio_) { this->ratio_ = ratio; ratio /= 10.0f; this->lpf_.setCutoff(LPF_CUTOFF_REF * (1.0f - ratio / weight)); this->hpf_.setCutoff(HPF_CUTOFF_REF * (1.0f + ratio * weight)); } } void Delay::LowHighPassFilter::reset() { this->lpf_.reset(); this->hpf_.reset(); } void Delay::LowHighPassFilter::processSample(float32_t inL, float32_t inR, float32_t& outL, float32_t& outR) { this->lpf_.processSample(inL, inR, outL, outR); this->hpf_.processSample(outL, outR, outL, outR); } Delay::Delay(const float32_t sampling_rate, float32_t default_delay_time, float32_t default_flutter_level, float32_t default_feedback_level) : FXElement(sampling_rate), MaxSampleDelayTime((MAX_DELAY_TIME + MAX_FLUTTER_DELAY_TIME) * sampling_rate * MAX_DELAY_TIME), read_pos_L_(0), read_pos_R_(0), filter_(sampling_rate), jitter_generator_(sampling_rate) { this->buffer_L_ = new float32_t[this->MaxSampleDelayTime]; this->buffer_R_ = new float32_t[this->MaxSampleDelayTime]; this->setLeftDelayTime(default_delay_time); this->setRightDelayTime(default_delay_time); this->setFeedback(default_feedback_level); this->setFlutterRate(0.2f); this->setFlutterAmount(0.05f); this->reset(); } Delay::~Delay() { delete[] this->buffer_L_; delete[] this->buffer_R_; } void Delay::reset() { memset(this->buffer_L_, 0, this->MaxSampleDelayTime * sizeof(float32_t)); memset(this->buffer_R_, 0, this->MaxSampleDelayTime * sizeof(float32_t)); this->read_pos_L_ = 0; this->read_pos_R_ = 0; this->filter_.reset(); this->jitter_generator_.reset(); } void Delay::processSample(float32_t inL, float32_t inR, float32_t& outL, float32_t& outR) { static const float32_t max_delay_time = MAX_DELAY_TIME * this->getSamplingRate(); float32_t jitter_delay_time = 0.0f; if(this->jitter_amount_ != 0.0f) { float32_t jitter_ratio = this->jitter_generator_.process(); if(jitter_ratio != 0.0f) { jitter_ratio *= this->jitter_amount_; jitter_delay_time = MAX_FLUTTER_DELAY_TIME * jitter_ratio * this->getSamplingRate(); } } // this->filter_.setCutoffChangeRatio(jitter_ratio); float32_t delay_time_L = jitter_delay_time + max_delay_time * this->getLeftDelayTime(); float32_t delay_time_R = jitter_delay_time + max_delay_time * this->getRightDelayTime(); // Calculate write positions unsigned write_pos_L = static_cast(this->MaxSampleDelayTime + this->read_pos_L_ + delay_time_L) % this->MaxSampleDelayTime; unsigned write_pos_R = static_cast(this->MaxSampleDelayTime + this->read_pos_R_ + delay_time_R) % this->MaxSampleDelayTime; // Write input to delay buffers this->buffer_L_[write_pos_L] = inL; this->buffer_R_[write_pos_R] = inR; // Read from delay buffers and apply feedback this->filter_.processSample( this->buffer_L_[this->read_pos_L_], this->buffer_R_[this->read_pos_R_], outL, outR ); this->buffer_L_[write_pos_L] += outL * this->getFeedback(); this->buffer_R_[write_pos_R] += outR * this->getFeedback(); // Increment read positions ++this->read_pos_L_; if(this->read_pos_L_ >= this->MaxSampleDelayTime) { this->read_pos_L_ -= this->MaxSampleDelayTime; } ++this->read_pos_R_; if(this->read_pos_R_ >= this->MaxSampleDelayTime) { this->read_pos_R_ -= this->MaxSampleDelayTime; } } void Delay::setLeftDelayTime(float32_t delay_time) { this->delay_time_L_ = constrain(delay_time, 0.0f, 1.0f); } float32_t Delay::getLeftDelayTime() const { return this->delay_time_L_; } void Delay::setRightDelayTime(float32_t delay_time) { this->delay_time_R_ = constrain(delay_time, 0.0f, 1.0f); } float32_t Delay::getRightDelayTime() const { return this->delay_time_R_; } void Delay::setFeedback(float32_t feedback) { this->feedback_ = constrain(feedback, 0.0, 1.0); } float32_t Delay::getFeedback() const { return this->feedback_; } void Delay::setFlutterRate(float32_t rate) { this->jitter_generator_.setRate(rate); } float32_t Delay::getFlutterRate() const { return this->jitter_generator_.getRate(); } void Delay::setFlutterAmount(float32_t amount) { this->jitter_amount_ = constrain(amount, 0.0f, 1.0f); } float32_t Delay::getFlutterAmount() const { return this->jitter_amount_; }