amuse/lib/EffectReverb.cpp

494 lines
16 KiB
C++
Raw Normal View History

2016-05-13 01:46:41 +00:00
#include "amuse/EffectReverb.hpp"
#include "amuse/IBackendVoice.hpp"
#include <cmath>
#include <string.h>
namespace amuse
{
/* Comb-filter delays */
static const size_t CTapDelays[] =
{
1789,
1999,
2333
};
/* All-pass-filter delays */
static const size_t APTapDelays[] =
{
433,
149
};
/* Per-channel low-pass delays (Hi-quality reverb only) */
static const size_t LPTapDelays[] =
{
47,
73,
67,
57,
43,
57,
83,
73
};
void ReverbDelayLine::allocate(int32_t delay)
{
delay += 2;
2016-05-14 22:38:37 +00:00
x8_length = delay;
2016-05-13 01:46:41 +00:00
xc_inputs.reset(new float[delay]);
2016-05-14 22:38:37 +00:00
memset(xc_inputs.get(), 0, x8_length * sizeof(float));
2016-05-13 01:46:41 +00:00
x10_lastInput = 0.f;
setdelay(delay / 2);
x0_inPoint = 0;
x4_outPoint = 0;
}
void ReverbDelayLine::setdelay(int32_t delay)
{
2016-05-14 22:38:37 +00:00
x4_outPoint = x0_inPoint - delay;
2016-05-13 01:46:41 +00:00
while (x4_outPoint < 0)
x4_outPoint += x8_length;
}
2016-05-16 21:00:37 +00:00
EffectReverbStd::EffectReverbStd(float coloration, float mix, float time,
2016-05-14 04:46:39 +00:00
float damping, float preDelay)
2016-05-13 01:46:41 +00:00
: x140_x1c8_coloration(clamp(0.f, coloration, 1.f)),
x144_x1cc_mix(clamp(0.f, mix, 1.f)),
x148_x1d0_time(clamp(0.01f, time, 10.f)),
x14c_x1d4_damping(clamp(0.f, damping, 1.f)),
2016-05-14 04:46:39 +00:00
x150_x1d8_preDelay(clamp(0.f, preDelay, 0.1f))
{}
EffectReverbHi::EffectReverbHi(float coloration, float mix, float time,
float damping, float preDelay, float crosstalk)
2016-05-16 21:00:37 +00:00
: EffectReverbStd(coloration, mix, time, damping, preDelay),
2016-05-14 04:46:39 +00:00
x1dc_crosstalk(clamp(0.f, crosstalk, 1.0f))
{}
template <typename T>
EffectReverbStdImp<T>::EffectReverbStdImp(float coloration, float mix, float time,
float damping, float preDelay, double sampleRate)
2016-06-01 04:49:35 +00:00
: EffectReverbStd(coloration, mix, time, damping, preDelay)
{
_setup(sampleRate);
}
template <typename T>
void EffectReverbStdImp<T>::_setup(double sampleRate)
{
m_sampleRate = sampleRate;
_update();
}
2016-05-13 01:46:41 +00:00
2016-05-14 04:46:39 +00:00
template <typename T>
void EffectReverbStdImp<T>::_update()
2016-05-13 01:46:41 +00:00
{
float timeSamples = x148_x1d0_time * m_sampleRate;
for (int c=0 ; c<8 ; ++c)
{
2016-05-14 04:46:39 +00:00
for (int t=0 ; t<2 ; ++t)
2016-05-13 01:46:41 +00:00
{
2016-05-14 04:46:39 +00:00
ReverbDelayLine& combLine = x78_C[c][t];
2016-05-13 01:46:41 +00:00
size_t tapDelay = CTapDelays[t] * m_sampleRate / 32000.0;
combLine.allocate(tapDelay);
combLine.setdelay(tapDelay);
2016-05-16 21:00:37 +00:00
xf4_combCoef[c][t] = std::pow(10.f, tapDelay * -3.f / timeSamples);
2016-05-13 01:46:41 +00:00
}
2016-05-14 04:46:39 +00:00
for (int t=0 ; t<2 ; ++t)
2016-05-13 01:46:41 +00:00
{
2016-05-14 04:46:39 +00:00
ReverbDelayLine& allPassLine = x0_AP[c][t];
2016-05-13 01:46:41 +00:00
size_t tapDelay = APTapDelays[t] * m_sampleRate / 32000.0;
allPassLine.allocate(tapDelay);
allPassLine.setdelay(tapDelay);
}
}
2016-05-14 04:46:39 +00:00
xf0_allPassCoef = x140_x1c8_coloration;
x118_level = x144_x1cc_mix;
x11c_damping = x14c_x1d4_damping;
2016-05-13 01:46:41 +00:00
2016-05-14 04:46:39 +00:00
if (x11c_damping < 0.05f)
x11c_damping = 0.05f;
2016-05-13 01:46:41 +00:00
2016-05-14 04:46:39 +00:00
x11c_damping = 1.f - (x11c_damping * 0.8f + 0.05);
2016-05-13 01:46:41 +00:00
if (x150_x1d8_preDelay != 0.f)
{
2016-05-14 04:46:39 +00:00
x120_preDelayTime = m_sampleRate * x150_x1d8_preDelay;
2016-05-13 01:46:41 +00:00
for (int i=0 ; i<8 ; ++i)
{
2016-05-14 04:46:39 +00:00
x124_preDelayLine[i].reset(new float[x120_preDelayTime]);
memset(x124_preDelayLine[i].get(), 0, x120_preDelayTime * sizeof(float));
x130_preDelayPtr[i] = x124_preDelayLine[i].get();
2016-05-13 01:46:41 +00:00
}
}
else
{
2016-05-14 04:46:39 +00:00
x120_preDelayTime = 0;
2016-05-13 01:46:41 +00:00
for (int i=0 ; i<8 ; ++i)
{
2016-05-14 04:46:39 +00:00
x124_preDelayLine[i] = nullptr;
x130_preDelayPtr[i] = nullptr;
2016-05-13 01:46:41 +00:00
}
}
m_dirty = false;
}
2016-05-14 04:46:39 +00:00
template <typename T>
void EffectReverbStdImp<T>::applyEffect(T* audio, size_t frameCount, const ChannelMap& chanMap)
2016-05-13 01:46:41 +00:00
{
if (m_dirty)
_update();
2016-05-14 04:46:39 +00:00
float dampWet = x118_level * 0.6f;
2016-05-13 01:46:41 +00:00
float dampDry = 0.6f - dampWet;
for (size_t f=0 ; f<frameCount ; f+=160)
2016-05-13 01:46:41 +00:00
{
2016-05-31 22:25:05 +00:00
for (unsigned c=0 ; c<chanMap.m_channelCount ; ++c)
2016-05-13 01:46:41 +00:00
{
2016-05-14 04:46:39 +00:00
float* combCoefs = xf4_combCoef[c];
float& lpLastOut = x10c_lpLastout[c];
float* preDelayLine = x124_preDelayLine[c].get();
float* preDelayPtr = x130_preDelayPtr[c];
float* lastPreDelaySamp = &preDelayLine[x120_preDelayTime - 1];
2016-05-13 01:46:41 +00:00
2016-05-14 04:46:39 +00:00
ReverbDelayLine* linesC = x78_C[c];
ReverbDelayLine* linesAP = x0_AP[c];
2016-05-13 01:46:41 +00:00
2016-05-31 22:25:05 +00:00
int procSamples = std::min(size_t(160), frameCount - f);
for (int s=0 ; s<procSamples ; ++s)
2016-05-13 01:46:41 +00:00
{
2016-05-16 21:00:37 +00:00
float sample = audio[s * chanMap.m_channelCount + c];
2016-05-13 01:46:41 +00:00
/* Pre-delay stage */
float sample2 = sample;
2016-05-14 04:46:39 +00:00
if (x120_preDelayTime != 0)
2016-05-13 01:46:41 +00:00
{
sample2 = *preDelayPtr;
*preDelayPtr = sample;
2016-05-14 22:38:37 +00:00
preDelayPtr += 1;
2016-05-13 01:46:41 +00:00
if (preDelayPtr == lastPreDelaySamp)
preDelayPtr = preDelayLine;
}
/* Comb filter stage */
linesC[0].xc_inputs[linesC[0].x0_inPoint] = combCoefs[0] * linesC[0].x10_lastInput + sample2;
2016-05-14 22:38:37 +00:00
linesC[0].x0_inPoint += 1;
2016-05-13 01:46:41 +00:00
linesC[1].xc_inputs[linesC[1].x0_inPoint] = combCoefs[1] * linesC[1].x10_lastInput + sample2;
2016-05-14 22:38:37 +00:00
linesC[1].x0_inPoint += 1;
2016-05-13 01:46:41 +00:00
linesC[0].x10_lastInput = linesC[0].xc_inputs[linesC[0].x4_outPoint];
2016-05-14 22:38:37 +00:00
linesC[0].x4_outPoint += 1;
2016-05-13 01:46:41 +00:00
linesC[1].x10_lastInput = linesC[1].xc_inputs[linesC[1].x4_outPoint];
2016-05-14 22:38:37 +00:00
linesC[1].x4_outPoint += 1;
2016-05-13 01:46:41 +00:00
if (linesC[0].x0_inPoint == linesC[0].x8_length)
linesC[0].x0_inPoint = 0;
if (linesC[1].x0_inPoint == linesC[1].x8_length)
linesC[1].x0_inPoint = 0;
if (linesC[0].x4_outPoint == linesC[0].x8_length)
linesC[0].x4_outPoint = 0;
if (linesC[1].x4_outPoint == linesC[1].x8_length)
linesC[1].x4_outPoint = 0;
/* All-pass filter stage */
linesAP[0].xc_inputs[linesAP[0].x0_inPoint] =
2016-05-14 04:46:39 +00:00
xf0_allPassCoef * linesAP[0].x10_lastInput + linesC[0].x10_lastInput + linesC[1].x10_lastInput;
float lowPass = -(xf0_allPassCoef * linesAP[0].xc_inputs[linesAP[0].x0_inPoint] -
2016-05-13 01:46:41 +00:00
linesAP[0].x10_lastInput);
2016-05-14 22:38:37 +00:00
linesAP[0].x0_inPoint += 1;
2016-05-13 01:46:41 +00:00
linesAP[0].x10_lastInput = linesAP[0].xc_inputs[linesAP[0].x4_outPoint];
2016-05-14 22:38:37 +00:00
linesAP[0].x4_outPoint += 1;
2016-05-13 01:46:41 +00:00
if (linesAP[0].x0_inPoint == linesAP[0].x8_length)
linesAP[0].x0_inPoint = 0;
if (linesAP[0].x4_outPoint == linesAP[0].x8_length)
linesAP[0].x4_outPoint = 0;
2016-05-14 04:46:39 +00:00
lpLastOut = x11c_damping * lpLastOut + lowPass * 0.3f;
linesAP[1].xc_inputs[linesAP[1].x0_inPoint] = xf0_allPassCoef * linesAP[1].x10_lastInput + lpLastOut;
float allPass = -(xf0_allPassCoef * linesAP[1].xc_inputs[linesAP[1].x0_inPoint] -
2016-05-13 01:46:41 +00:00
linesAP[1].x10_lastInput);
2016-05-14 22:38:37 +00:00
linesAP[1].x0_inPoint += 1;
2016-05-13 01:46:41 +00:00
linesAP[1].x10_lastInput = linesAP[1].xc_inputs[linesAP[1].x4_outPoint];
2016-05-14 22:38:37 +00:00
linesAP[1].x4_outPoint += 1;
2016-05-13 01:46:41 +00:00
if (linesAP[1].x0_inPoint == linesAP[1].x8_length)
linesAP[1].x0_inPoint = 0;
if (linesAP[1].x4_outPoint == linesAP[1].x8_length)
linesAP[1].x4_outPoint = 0;
/* Mix out */
2016-05-16 21:00:37 +00:00
audio[s * chanMap.m_channelCount + c] = ClampFull<T>(dampWet * allPass + dampDry * sample);
2016-05-13 01:46:41 +00:00
}
2016-05-14 04:46:39 +00:00
x130_preDelayPtr[c] = preDelayPtr;
2016-05-13 01:46:41 +00:00
}
audio += chanMap.m_channelCount * 160;
}
}
template <typename T>
2016-05-14 04:46:39 +00:00
EffectReverbHiImp<T>::EffectReverbHiImp(float coloration, float mix, float time,
float damping, float preDelay, float crosstalk,
double sampleRate)
2016-06-01 04:49:35 +00:00
: EffectReverbHi(coloration, mix, time, damping, preDelay, crosstalk)
{
_setup(sampleRate);
}
template <typename T>
void EffectReverbHiImp<T>::_setup(double sampleRate)
2016-05-13 01:46:41 +00:00
{
2016-06-01 04:49:35 +00:00
m_sampleRate = sampleRate;
2016-05-14 04:46:39 +00:00
_update();
2016-05-13 01:46:41 +00:00
}
template <typename T>
2016-05-14 04:46:39 +00:00
void EffectReverbHiImp<T>::_update()
2016-05-13 01:46:41 +00:00
{
2016-05-14 04:46:39 +00:00
float timeSamples = x148_x1d0_time * m_sampleRate;
2016-05-13 01:46:41 +00:00
for (int c=0 ; c<8 ; ++c)
{
2016-05-14 22:38:37 +00:00
for (int t=0 ; t<3 ; ++t)
2016-05-14 04:46:39 +00:00
{
ReverbDelayLine& combLine = xb4_C[c][t];
size_t tapDelay = CTapDelays[t] * m_sampleRate / 32000.0;
combLine.allocate(tapDelay);
combLine.setdelay(tapDelay);
x16c_combCoef[c][t] = std::pow(10.f, tapDelay * -3.f / timeSamples);
2016-05-14 04:46:39 +00:00
}
for (int t=0 ; t<2 ; ++t)
{
ReverbDelayLine& allPassLine = x0_AP[c][t];
size_t tapDelay = APTapDelays[t] * m_sampleRate / 32000.0;
allPassLine.allocate(tapDelay);
allPassLine.setdelay(tapDelay);
}
ReverbDelayLine& lpLine = x78_LP[c];
2016-05-14 04:46:39 +00:00
size_t tapDelay = LPTapDelays[c] * m_sampleRate / 32000.0;
lpLine.allocate(tapDelay);
lpLine.setdelay(tapDelay);
2016-05-13 01:46:41 +00:00
}
2016-05-14 04:46:39 +00:00
x168_allPassCoef = x140_x1c8_coloration;
x19c_level = x144_x1cc_mix;
x1a0_damping = x14c_x1d4_damping;
if (x1a0_damping < 0.05f)
x1a0_damping = 0.05f;
x1a0_damping = 1.f - (x1a0_damping * 0.8f + 0.05);
if (x150_x1d8_preDelay != 0.f)
{
x1a4_preDelayTime = m_sampleRate * x150_x1d8_preDelay;
for (int i=0 ; i<8 ; ++i)
{
x1ac_preDelayLine[i].reset(new float[x1a4_preDelayTime]);
memset(x1ac_preDelayLine[i].get(), 0, x1a4_preDelayTime * sizeof(float));
x1b8_preDelayPtr[i] = x1ac_preDelayLine[i].get();
}
}
else
{
x1a4_preDelayTime = 0;
for (int i=0 ; i<8 ; ++i)
{
x1ac_preDelayLine[i] = nullptr;
x1b8_preDelayPtr[i] = nullptr;
}
}
x1a8_internalCrosstalk = x1dc_crosstalk;
2016-05-14 04:46:39 +00:00
m_dirty = false;
2016-05-13 01:46:41 +00:00
}
template <typename T>
2016-05-14 04:46:39 +00:00
void EffectReverbHiImp<T>::_handleReverb(T* audio, int c, int chanCount, int sampleCount)
2016-05-13 01:46:41 +00:00
{
2016-05-14 04:46:39 +00:00
float dampWet = x19c_level * 0.6f;
2016-05-13 01:46:41 +00:00
float dampDry = 0.6f - dampWet;
2016-05-14 04:46:39 +00:00
float* combCoefs = x16c_combCoef[c];
float& lpLastOut = x190_lpLastout[c];
float* preDelayLine = x1ac_preDelayLine[c].get();
float* preDelayPtr = x1b8_preDelayPtr[c];
float* lastPreDelaySamp = &preDelayLine[x1a4_preDelayTime - 1];
2016-05-13 01:46:41 +00:00
2016-05-14 04:46:39 +00:00
ReverbDelayLine* linesC = xb4_C[c];
ReverbDelayLine* linesAP = x0_AP[c];
ReverbDelayLine& lineLP = x78_LP[c];
2016-05-13 01:46:41 +00:00
2016-05-14 04:46:39 +00:00
float allPassCoef = x168_allPassCoef;
float damping = x1a0_damping;
int32_t preDelayTime = x1a4_preDelayTime;
2016-05-13 01:46:41 +00:00
for (int s=0 ; s<sampleCount ; ++s)
2016-05-13 01:46:41 +00:00
{
float sample = audio[s * chanCount + c];
2016-05-13 01:46:41 +00:00
/* Pre-delay stage */
float sample2 = sample;
if (preDelayTime != 0)
{
sample2 = *preDelayPtr;
*preDelayPtr = sample;
2016-05-14 22:38:37 +00:00
preDelayPtr += 1;
2016-05-13 01:46:41 +00:00
if (preDelayPtr == lastPreDelaySamp)
preDelayPtr = preDelayLine;
}
/* Comb filter stage */
linesC[0].xc_inputs[linesC[0].x0_inPoint] = combCoefs[0] * linesC[0].x10_lastInput + sample2;
2016-05-14 22:38:37 +00:00
linesC[0].x0_inPoint += 1;
2016-05-13 01:46:41 +00:00
linesC[1].xc_inputs[linesC[1].x0_inPoint] = combCoefs[1] * linesC[1].x10_lastInput + sample2;
2016-05-14 22:38:37 +00:00
linesC[1].x0_inPoint += 1;
linesC[2].xc_inputs[linesC[2].x0_inPoint] = combCoefs[2] * linesC[2].x10_lastInput + sample2;
linesC[2].x0_inPoint += 1;
2016-05-13 01:46:41 +00:00
linesC[0].x10_lastInput = linesC[0].xc_inputs[linesC[0].x4_outPoint];
2016-05-14 22:38:37 +00:00
linesC[0].x4_outPoint += 1;
2016-05-13 01:46:41 +00:00
linesC[1].x10_lastInput = linesC[1].xc_inputs[linesC[1].x4_outPoint];
2016-05-14 22:38:37 +00:00
linesC[1].x4_outPoint += 1;
linesC[2].x10_lastInput = linesC[2].xc_inputs[linesC[2].x4_outPoint];
linesC[2].x4_outPoint += 1;
2016-05-13 01:46:41 +00:00
if (linesC[0].x0_inPoint == linesC[0].x8_length)
linesC[0].x0_inPoint = 0;
if (linesC[1].x0_inPoint == linesC[1].x8_length)
linesC[1].x0_inPoint = 0;
2016-05-14 22:38:37 +00:00
if (linesC[2].x0_inPoint == linesC[2].x8_length)
linesC[2].x0_inPoint = 0;
2016-05-13 01:46:41 +00:00
if (linesC[0].x4_outPoint == linesC[0].x8_length)
linesC[0].x4_outPoint = 0;
if (linesC[1].x4_outPoint == linesC[1].x8_length)
linesC[1].x4_outPoint = 0;
2016-05-14 22:38:37 +00:00
if (linesC[2].x4_outPoint == linesC[2].x8_length)
linesC[2].x4_outPoint = 0;
2016-05-13 01:46:41 +00:00
/* All-pass filter stage */
linesAP[0].xc_inputs[linesAP[0].x0_inPoint] =
2016-05-14 22:38:37 +00:00
allPassCoef * linesAP[0].x10_lastInput + linesC[0].x10_lastInput + linesC[1].x10_lastInput + linesC[2].x10_lastInput;
2016-05-13 01:46:41 +00:00
linesAP[1].xc_inputs[linesAP[1].x0_inPoint] =
allPassCoef * linesAP[1].x10_lastInput -
(allPassCoef * linesAP[0].xc_inputs[linesAP[0].x0_inPoint] - linesAP[0].x10_lastInput);
float lowPass = -(allPassCoef * linesAP[1].xc_inputs[linesAP[1].x0_inPoint] - linesAP[1].x10_lastInput);
2016-05-14 22:38:37 +00:00
linesAP[0].x0_inPoint += 1;
linesAP[1].x0_inPoint += 1;
2016-05-13 01:46:41 +00:00
if (linesAP[0].x0_inPoint == linesAP[0].x8_length)
linesAP[0].x0_inPoint = 0;
if (linesAP[1].x0_inPoint == linesAP[1].x8_length)
linesAP[1].x0_inPoint = 0;
linesAP[0].x10_lastInput = linesAP[0].xc_inputs[linesAP[0].x4_outPoint];
2016-05-14 22:38:37 +00:00
linesAP[0].x4_outPoint += 1;
2016-05-13 01:46:41 +00:00
linesAP[1].x10_lastInput = linesAP[1].xc_inputs[linesAP[1].x4_outPoint];
2016-05-14 22:38:37 +00:00
linesAP[1].x4_outPoint += 1;
2016-05-13 01:46:41 +00:00
if (linesAP[0].x4_outPoint == linesAP[0].x8_length)
linesAP[0].x4_outPoint = 0;
if (linesAP[1].x4_outPoint == linesAP[1].x8_length)
linesAP[1].x4_outPoint = 0;
lpLastOut = damping * lpLastOut + lowPass * 0.3f;
lineLP.xc_inputs[lineLP.x0_inPoint] = allPassCoef * lineLP.x10_lastInput + lpLastOut;
float allPass = -(allPassCoef * lineLP.xc_inputs[lineLP.x0_inPoint] - lineLP.x10_lastInput);
2016-05-14 22:38:37 +00:00
lineLP.x0_inPoint += 1;
2016-05-13 01:46:41 +00:00
lineLP.x10_lastInput = lineLP.xc_inputs[lineLP.x4_outPoint];
2016-05-14 22:38:37 +00:00
lineLP.x4_outPoint += 1;
2016-05-13 01:46:41 +00:00
if (lineLP.x0_inPoint == lineLP.x8_length)
lineLP.x0_inPoint = 0;
2016-05-13 01:46:41 +00:00
if (lineLP.x4_outPoint == lineLP.x8_length)
lineLP.x4_outPoint = 0;
2016-05-13 01:46:41 +00:00
/* Mix out */
audio[s * chanCount + c] = ClampFull<T>(dampWet * allPass + dampDry * sample);
2016-05-13 01:46:41 +00:00
}
2016-05-14 04:46:39 +00:00
x1b8_preDelayPtr[c] = preDelayPtr;
2016-05-13 01:46:41 +00:00
}
template <typename T>
2016-05-14 04:46:39 +00:00
void EffectReverbHiImp<T>::_doCrosstalk(T* audio, float wet, float dry, int chanCount, int sampleCount)
2016-05-13 01:46:41 +00:00
{
for (int i=0 ; i<sampleCount ; ++i)
{
T* base = &audio[chanCount*i];
float allWet = 0;
for (int c=0 ; c<chanCount ; ++c)
{
allWet += base[c] * wet;
base[c] *= dry;
}
for (int c=0 ; c<chanCount ; ++c)
base[c] = ClampFull<T>(base[c] + allWet);
}
}
template <typename T>
2016-05-14 04:46:39 +00:00
void EffectReverbHiImp<T>::applyEffect(T* audio, size_t frameCount, const ChannelMap& chanMap)
2016-05-13 01:46:41 +00:00
{
2016-05-14 04:46:39 +00:00
if (m_dirty)
2016-05-13 01:46:41 +00:00
_update();
for (size_t f=0 ; f<frameCount ; f+=160)
{
size_t blockSamples = std::min(size_t(160), frameCount - f);
2016-05-31 22:25:05 +00:00
for (unsigned i=0 ; i<chanMap.m_channelCount ; ++i)
2016-05-13 01:46:41 +00:00
{
if (i == 0 && x1a8_internalCrosstalk != 0.f)
{
float crossWet = x1a8_internalCrosstalk * 0.5;
_doCrosstalk(audio, crossWet, 1.f - crossWet, chanMap.m_channelCount, blockSamples);
}
_handleReverb(audio, i, chanMap.m_channelCount, blockSamples);
}
audio += chanMap.m_channelCount * 160;
}
}
2016-05-14 04:46:39 +00:00
template class EffectReverbStdImp<int16_t>;
template class EffectReverbStdImp<int32_t>;
template class EffectReverbStdImp<float>;
2016-05-13 01:46:41 +00:00
2016-05-14 04:46:39 +00:00
template class EffectReverbHiImp<int16_t>;
template class EffectReverbHiImp<int32_t>;
template class EffectReverbHiImp<float>;
2016-05-13 01:46:41 +00:00
}