#include "fx_chorus.h" #include #define CHORUS_BUFFER_SIZE 8192 #define LFO1_MAX_FREQ 0.25f #define LFO2_MAX_FREQ 0.35f #define FULLSCALE_DEPTH_RATIO 384.0f Chorus::Chorus(float32_t sampling_rate) : FXElement(sampling_rate), engine_(sampling_rate, LFO1_MAX_FREQ, LFO2_MAX_FREQ), rate_(0.0f), depth_(0.0f), fullscale_depth_(0.0f), feedback_(0.0f) { this->lfo_[LFO_Index::Sin1] = new LFO(sampling_rate, LFO::Waveform::Sine, 0.0f, LFO1_MAX_FREQ); this->lfo_[LFO_Index::Cos1] = new LFO(sampling_rate, LFO::Waveform::Sine, 0.0f, LFO1_MAX_FREQ, Constants::MPI_2); this->lfo_[LFO_Index::Sin2] = new LFO(sampling_rate, LFO::Waveform::Sine, 0.0f, LFO2_MAX_FREQ); this->lfo_[LFO_Index::Cos2] = new LFO(sampling_rate, LFO::Waveform::Sine, 0.0f, LFO2_MAX_FREQ, Constants::MPI_2); this->setRate(0.1f); this->setDepth(0.15f); this->setFeedback(0.15f); } Chorus::~Chorus() { for(unsigned i = 0; i < 4; ++i) { delete this->lfo_[i]; } } void Chorus::processSample(float32_t inL, float32_t inR, float32_t& outL, float32_t& outR) { typedef Engine::Reserve<2047> Memory; Engine::DelayLine line; Engine::Context c; engine_.start(&c); // Update LFO. float32_t sin_1 = this->lfo_[LFO_Index::Sin1]->process() * 4.0f; float32_t cos_1 = this->lfo_[LFO_Index::Cos1]->process() * 4.0f; float32_t sin_2 = this->lfo_[LFO_Index::Sin2]->process() * 4.0f; float32_t cos_2 = this->lfo_[LFO_Index::Cos2]->process() * 4.0f; float32_t wet; // Sum L & R channel to send to chorus line. c.read(inL, 0.5f); c.read(inR, 0.5f); c.write(line, 0.0f); c.interpolate(line, sin_1 * this->fullscale_depth_ + 1200, 0.5f); c.interpolate(line, sin_2 * this->fullscale_depth_ + 800, 0.5f); c.write(wet, 0.0f); outL = wet; c.interpolate(line, cos_1 * this->fullscale_depth_ + 800, 0.5f); c.interpolate(line, cos_2 * this->fullscale_depth_ + 1200, 0.5f); c.write(wet, 0.0f); outR = wet; } void Chorus::setRate(float32_t rate) { this->rate_ = constrain(rate, 0.0f, 1.0f); for(unsigned i = 0; i < 4; ++i) { this->lfo_[i]->setNormalizedFrequency(this->rate_); } } float32_t Chorus::getRate() const { return this->depth_; } void Chorus::setDepth(float32_t depth) { this->depth_ = constrain(depth, 0.0f, 1.0f); this->fullscale_depth_ = this->depth_ * FULLSCALE_DEPTH_RATIO; } float32_t Chorus::getDepth() const { return this->depth_; } void Chorus::setFeedback(float32_t feedback) { this->feedback_ = constrain(feedback, 0.0f, 1.0f); } float32_t Chorus::getFeedback() const { return this->feedback_; }