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.
 
 
 
 
MiniDexed/src/fx_engine.hpp

325 lines
8.0 KiB

#pragma once
#include "fx_components.h"
#include <algorithm>
#include <climits>
#include <cassert>
#define MAKE_INTEGRAL_FRACTIONAL(x) \
int32_t x ## _integral = static_cast<int32_t>(x); \
float x ## _fractional = x - static_cast<float>(x ## _integral);
enum Format
{
FORMAT_12_BIT,
FORMAT_16_BIT,
FORMAT_32_BIT
};
enum LFOIndex
{
LFO_1,
LFO_2
};
template <Format format>
struct DataType
{
};
inline int16_t clip16(int32_t x)
{
if(x > INT16_MAX)
{
return INT16_MAX;
}
if(x < INT16_MIN)
{
return INT16_MIN;
}
return static_cast<int16_t>(x);
}
template <>
struct DataType<FORMAT_16_BIT>
{
typedef uint16_t T;
static inline float32_t decompress(T value)
{
return static_cast<float>(static_cast<int16_t>(value)) / 4096.0f;
}
static inline T compress(float32_t value)
{
return clip16(static_cast<int32_t>(value * 4096.0f));
}
};
template <
size_t size,
Format format = FORMAT_16_BIT,
bool enable_lfo = true>
class FxEngine : public FXBase
{
DISALLOW_COPY_AND_ASSIGN(FxEngine);
public:
typedef typename DataType<format>::T T;
FxEngine(float32_t sampling_rate, float32_t max_lfo1_frequency = 1.0f, float32_t max_lfo2_frequency = 1.0f) :
FXBase(sampling_rate)
{
this->buffer_ = new uint16_t[size];
this->lfo_[LFO_1] = enable_lfo ? new LFO(sampling_rate, LFO::Waveform::Sine, 0.0f, max_lfo1_frequency) : nullptr;
this->lfo_[LFO_2] = enable_lfo ? new LFO(sampling_rate, LFO::Waveform::Sine, 0.0f, max_lfo2_frequency) : nullptr;
this->clear();
}
~FxEngine()
{
delete[] this->buffer_;
if(enable_lfo)
{
delete this->lfo_[LFO_1];
delete this->lfo_[LFO_2];
}
}
void clear()
{
memset(this->buffer_, 0, size * sizeof(uint16_t));
this->write_ptr_ = 0;
}
struct Empty
{
};
template <int32_t l, typename T = Empty>
struct Reserve
{
typedef T Tail;
enum
{
length = l
};
};
template <typename Memory, int32_t index>
struct DelayLine
{
enum
{
length = DelayLine<typename Memory::Tail, index - 1>::length,
base = DelayLine<Memory, index - 1>::base + DelayLine<Memory, index - 1>::length + 1
};
};
template <typename Memory>
struct DelayLine<Memory, 0>
{
enum
{
length = Memory::length,
base = 0
};
};
class Context
{
DISALLOW_COPY_AND_ASSIGN(Context);
friend class FxEngine;
public:
Context() :
accumulator_(0.0f),
previous_read_(0.0f),
buffer_(nullptr),
write_ptr_(0)
{
memset(this->lfo_value_, 0, 2 * sizeof(float32_t));
}
~Context()
{
}
inline void load(float32_t value)
{
this->accumulator_ = value;
}
inline void read(float32_t value, float32_t scale)
{
this->accumulator_ += value * scale;
}
inline void read(float32_t value)
{
this->accumulator_ += value;
}
inline void write(float32_t& value)
{
value = this->accumulator_;
}
inline void write(float32_t& value, float32_t scale)
{
value = this->accumulator_;
this->accumulator_ *= scale;
}
template <typename D>
inline void write(D& d, int32_t offset, float32_t scale)
{
assert(D::base + D::length <= size);
T w = DataType<format>::compress(this->accumulator_);
if(offset == -1)
{
this->buffer_[(this->write_ptr_ + D::base + D::length - 1) & MASK] = w;
}
else
{
this->buffer_[(this->write_ptr_ + D::base + offset) & MASK] = w;
}
this->accumulator_ *= scale;
}
template <typename D>
inline void write(D& d, float32_t scale)
{
this->write(d, 0, scale);
}
template <typename D>
inline void writeAllPass(D& d, int32_t offset, float32_t scale)
{
this->write(d, offset, scale);
this->accumulator_ += this->previous_read_;
}
template <typename D>
inline void writeAllPass(D& d, float32_t scale)
{
this->writeAllPass(d, 0, scale);
}
template <typename D>
inline void read(D& d, int32_t offset, float32_t scale)
{
assert(D::base + D::length <= size);
T r;
if(offset == -1)
{
r = this->buffer_[(this->write_ptr_ + D::base + D::length - 1) & MASK];
}
else
{
r = this->buffer_[(this->write_ptr_ + D::base + offset) & MASK];
}
float32_t r_f = DataType<format>::decompress(r);
this->previous_read_ = r_f;
this->accumulator_ += r_f * scale;
}
template <typename D>
inline void read(D& d, float32_t scale)
{
this->read(d, 0, scale);
}
inline void lp(float32_t& state, float32_t coefficient)
{
state += coefficient * (this->accumulator_ - state);
this->accumulator_ = state;
}
inline void hp(float32_t& state, float32_t coefficient)
{
state += coefficient * (this->accumulator_ - state);
this->accumulator_ -= state;
}
template <typename D>
inline void interpolate(D& d, float32_t offset, float32_t scale)
{
assert(D::base + D::length <= size);
MAKE_INTEGRAL_FRACTIONAL(offset);
float32_t a = DataType<format>::decompress(this->buffer_[(this->write_ptr_ + offset_integral + D::base) & MASK]);
float32_t b = DataType<format>::decompress(this->buffer_[(this->write_ptr_ + offset_integral + D::base + 1) & MASK]);
float32_t x = a + (b - a) * offset_fractional;
this->previous_read_ = x;
this->accumulator_ += x * scale;
}
template <typename D>
inline void interpolate(D& d, float32_t offset, LFOIndex index, float32_t amplitude, float32_t scale)
{
assert(D::base + D::length <= size);
offset += amplitude * this->lfo_value_[index];
MAKE_INTEGRAL_FRACTIONAL(offset);
float32_t a = DataType<format>::decompress(this->buffer_[(this->write_ptr_ + offset_integral + D::base) & MASK]);
float32_t b = DataType<format>::decompress(this->buffer_[(this->write_ptr_ + offset_integral + D::base + 1) & MASK]);
float32_t x = a + (b - a) * offset_fractional;
this->previous_read_ = x;
this->accumulator_ += x * scale;
}
private:
float32_t accumulator_;
float32_t previous_read_;
float32_t lfo_value_[2];
T* buffer_;
int32_t write_ptr_;
};
inline void setLFOFrequency(LFOIndex index, float32_t frequency)
{
if(enable_lfo)
{
this->lfo_[index]->setFrequency(frequency);
}
}
inline void setLFONormalizedFrequency(LFOIndex index, float32_t normalized_frequency)
{
if(enable_lfo)
{
this->lfo_[index]->setNormalizedFrequency(normalized_frequency);
}
}
inline void start(Context *c)
{
--this->write_ptr_;
if(this->write_ptr_ < 0)
{
this->write_ptr_ += size;
}
c->accumulator_ = 0.0f;
c->previous_read_ = 0.0f;
c->buffer_ = buffer_;
c->write_ptr_ = write_ptr_;
if(enable_lfo)
{
c->lfo_value_[LFO_1] = this->lfo_[LFO_1]->process();
c->lfo_value_[LFO_2] = this->lfo_[LFO_2]->process();
}
}
private:
enum
{
MASK = size - 1
};
uint16_t* buffer_;
unsigned write_ptr_;
LFO* lfo_[2];
};