2016-05-13 01:46:41 +00:00
|
|
|
#include "amuse/EffectReverb.hpp"
|
2019-08-25 10:28:43 +00:00
|
|
|
|
2020-03-27 20:55:41 +00:00
|
|
|
#include <algorithm>
|
2016-05-13 01:46:41 +00:00
|
|
|
#include <cmath>
|
|
|
|
|
2019-08-25 10:28:43 +00:00
|
|
|
#include "amuse/IBackendVoice.hpp"
|
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
namespace amuse {
|
2016-05-13 01:46:41 +00:00
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
/* clang-format off */
|
|
|
|
|
2016-05-13 01:46:41 +00:00
|
|
|
/* Comb-filter delays */
|
2020-03-27 20:55:41 +00:00
|
|
|
constexpr std::array<size_t, 3> CTapDelays{
|
2016-05-13 01:46:41 +00:00
|
|
|
1789,
|
|
|
|
1999,
|
|
|
|
2333
|
|
|
|
};
|
|
|
|
|
|
|
|
/* All-pass-filter delays */
|
2020-03-27 20:55:41 +00:00
|
|
|
constexpr std::array<size_t, 2> APTapDelays{
|
2016-05-13 01:46:41 +00:00
|
|
|
433,
|
2020-03-27 20:55:41 +00:00
|
|
|
149,
|
2016-05-13 01:46:41 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/* Per-channel low-pass delays (Hi-quality reverb only) */
|
2020-03-27 20:55:41 +00:00
|
|
|
constexpr std::array<size_t, 8> LPTapDelays{
|
2016-05-13 01:46:41 +00:00
|
|
|
47,
|
|
|
|
73,
|
|
|
|
67,
|
|
|
|
57,
|
|
|
|
43,
|
|
|
|
57,
|
|
|
|
83,
|
|
|
|
73
|
|
|
|
};
|
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
/* clang-format on */
|
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
void ReverbDelayLine::allocate(int32_t delay) {
|
|
|
|
delay += 2;
|
|
|
|
x8_length = delay;
|
2020-03-27 20:55:41 +00:00
|
|
|
xc_inputs = std::make_unique<float[]>(delay);
|
2018-12-08 05:20:09 +00:00
|
|
|
x10_lastInput = 0.f;
|
|
|
|
setdelay(delay / 2);
|
|
|
|
x0_inPoint = 0;
|
|
|
|
x4_outPoint = 0;
|
2016-05-13 01:46:41 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
void ReverbDelayLine::setdelay(int32_t delay) {
|
|
|
|
x4_outPoint = x0_inPoint - delay;
|
|
|
|
while (x4_outPoint < 0)
|
|
|
|
x4_outPoint += x8_length;
|
2016-05-13 01:46:41 +00:00
|
|
|
}
|
|
|
|
|
2016-07-14 04:54:46 +00:00
|
|
|
EffectReverbStd::EffectReverbStd(float coloration, float mix, float time, float damping, float preDelay)
|
2019-09-07 10:21:07 +00:00
|
|
|
: x140_x1c8_coloration(std::clamp(coloration, 0.f, 1.f))
|
|
|
|
, x144_x1cc_mix(std::clamp(mix, 0.f, 1.f))
|
|
|
|
, x148_x1d0_time(std::clamp(time, 0.01f, 10.f))
|
|
|
|
, x14c_x1d4_damping(std::clamp(damping, 0.f, 1.f))
|
|
|
|
, x150_x1d8_preDelay(std::clamp(preDelay, 0.f, 0.1f)) {}
|
2016-07-14 04:54:46 +00:00
|
|
|
|
|
|
|
EffectReverbHi::EffectReverbHi(float coloration, float mix, float time, float damping, float preDelay, float crosstalk)
|
2019-09-07 10:21:07 +00:00
|
|
|
: EffectReverbStd(coloration, mix, time, damping, preDelay), x1dc_crosstalk(std::clamp(crosstalk, 0.f, 1.0f)) {}
|
2016-05-14 04:46:39 +00:00
|
|
|
|
2020-09-28 20:42:51 +00:00
|
|
|
EffectReverbStdImp::EffectReverbStdImp(float coloration, float mix, float time, float damping, float preDelay,
|
2016-07-14 04:54:46 +00:00
|
|
|
double sampleRate)
|
2018-12-08 05:20:09 +00:00
|
|
|
: EffectReverbStd(coloration, mix, time, damping, preDelay) {
|
|
|
|
_setup(sampleRate);
|
2016-06-01 04:49:35 +00:00
|
|
|
}
|
|
|
|
|
2020-09-28 20:42:51 +00:00
|
|
|
void EffectReverbStdImp::_setup(double sampleRate) {
|
2018-12-08 05:20:09 +00:00
|
|
|
m_sampleRate = sampleRate;
|
|
|
|
_update();
|
2016-06-01 04:49:35 +00:00
|
|
|
}
|
2016-05-13 01:46:41 +00:00
|
|
|
|
2020-09-28 20:42:51 +00:00
|
|
|
void EffectReverbStdImp::_update() {
|
2018-12-08 05:20:09 +00:00
|
|
|
float timeSamples = x148_x1d0_time * m_sampleRate;
|
|
|
|
double rateRatio = m_sampleRate / NativeSampleRate;
|
2020-03-27 20:55:41 +00:00
|
|
|
for (size_t c = 0; c < NumChannels; ++c) {
|
|
|
|
for (size_t t = 0; t < x78_C[c].size(); ++t) {
|
2018-12-08 05:20:09 +00:00
|
|
|
ReverbDelayLine& combLine = x78_C[c][t];
|
|
|
|
size_t tapDelay = CTapDelays[t] * rateRatio;
|
|
|
|
combLine.allocate(tapDelay);
|
|
|
|
combLine.setdelay(tapDelay);
|
|
|
|
xf4_combCoef[c][t] = std::pow(10.f, tapDelay * -3.f / timeSamples);
|
|
|
|
}
|
2016-05-13 01:46:41 +00:00
|
|
|
|
2020-03-27 20:55:41 +00:00
|
|
|
for (size_t t = 0; t < x0_AP[c].size(); ++t) {
|
2018-12-08 05:20:09 +00:00
|
|
|
ReverbDelayLine& allPassLine = x0_AP[c][t];
|
|
|
|
size_t tapDelay = APTapDelays[t] * rateRatio;
|
|
|
|
allPassLine.allocate(tapDelay);
|
|
|
|
allPassLine.setdelay(tapDelay);
|
2016-05-13 01:46:41 +00:00
|
|
|
}
|
2018-12-08 05:20:09 +00:00
|
|
|
}
|
2016-05-13 01:46:41 +00:00
|
|
|
|
2018-12-08 05:20:09 +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
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
if (x11c_damping < 0.05f)
|
|
|
|
x11c_damping = 0.05f;
|
2016-05-13 01:46:41 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
x11c_damping = 1.f - (x11c_damping * 0.8f + 0.05);
|
2016-05-13 01:46:41 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
if (x150_x1d8_preDelay != 0.f) {
|
|
|
|
x120_preDelayTime = m_sampleRate * x150_x1d8_preDelay;
|
2020-03-27 20:55:41 +00:00
|
|
|
for (size_t i = 0; i < NumChannels; ++i) {
|
|
|
|
x124_preDelayLine[i] = std::make_unique<float[]>(x120_preDelayTime);
|
2018-12-08 05:20:09 +00:00
|
|
|
x130_preDelayPtr[i] = x124_preDelayLine[i].get();
|
2016-05-13 01:46:41 +00:00
|
|
|
}
|
2018-12-08 05:20:09 +00:00
|
|
|
} else {
|
|
|
|
x120_preDelayTime = 0;
|
2020-03-27 20:55:41 +00:00
|
|
|
for (auto& delayLine : x124_preDelayLine) {
|
|
|
|
delayLine.reset();
|
2016-05-13 01:46:41 +00:00
|
|
|
}
|
2020-03-27 20:55:41 +00:00
|
|
|
x130_preDelayPtr.fill(nullptr);
|
2018-12-08 05:20:09 +00:00
|
|
|
}
|
2016-05-13 01:46:41 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
m_dirty = false;
|
2016-05-13 01:46:41 +00:00
|
|
|
}
|
|
|
|
|
2020-09-28 20:42:51 +00:00
|
|
|
void EffectReverbStdImp::applyEffect(float* audio, size_t frameCount, const ChannelMap& chanMap) {
|
2018-12-08 05:20:09 +00:00
|
|
|
if (m_dirty)
|
|
|
|
_update();
|
|
|
|
|
2020-03-27 20:55:41 +00:00
|
|
|
const float dampWet = x118_level * 0.6f;
|
|
|
|
const float dampDry = 0.6f - dampWet;
|
2018-12-08 05:20:09 +00:00
|
|
|
|
|
|
|
for (size_t f = 0; f < frameCount; f += 160) {
|
|
|
|
for (unsigned c = 0; c < chanMap.m_channelCount; ++c) {
|
2020-03-27 20:55:41 +00:00
|
|
|
const auto& combCoefs = xf4_combCoef[c];
|
2018-12-08 05:20:09 +00:00
|
|
|
float& lpLastOut = x10c_lpLastout[c];
|
|
|
|
float* preDelayLine = x124_preDelayLine[c].get();
|
|
|
|
float* preDelayPtr = x130_preDelayPtr[c];
|
|
|
|
float* lastPreDelaySamp = &preDelayLine[x120_preDelayTime - 1];
|
|
|
|
|
2020-03-27 20:55:41 +00:00
|
|
|
auto& linesC = x78_C[c];
|
|
|
|
auto& linesAP = x0_AP[c];
|
2018-12-08 05:20:09 +00:00
|
|
|
|
2020-03-27 20:55:41 +00:00
|
|
|
const int procSamples = std::min(size_t(160), frameCount - f);
|
2018-12-08 05:20:09 +00:00
|
|
|
for (int s = 0; s < procSamples; ++s) {
|
2020-03-27 20:55:41 +00:00
|
|
|
const float sample = audio[s * chanMap.m_channelCount + c];
|
2018-12-08 05:20:09 +00:00
|
|
|
|
|
|
|
/* Pre-delay stage */
|
|
|
|
float sample2 = sample;
|
|
|
|
if (x120_preDelayTime != 0) {
|
|
|
|
sample2 = *preDelayPtr;
|
|
|
|
*preDelayPtr = sample;
|
|
|
|
preDelayPtr += 1;
|
|
|
|
if (preDelayPtr == lastPreDelaySamp)
|
|
|
|
preDelayPtr = preDelayLine;
|
2016-05-13 01:46:41 +00:00
|
|
|
}
|
2018-12-08 05:20:09 +00:00
|
|
|
|
|
|
|
/* Comb filter stage */
|
|
|
|
linesC[0].xc_inputs[linesC[0].x0_inPoint] = combCoefs[0] * linesC[0].x10_lastInput + sample2;
|
|
|
|
linesC[0].x0_inPoint += 1;
|
|
|
|
|
|
|
|
linesC[1].xc_inputs[linesC[1].x0_inPoint] = combCoefs[1] * linesC[1].x10_lastInput + sample2;
|
|
|
|
linesC[1].x0_inPoint += 1;
|
|
|
|
|
|
|
|
linesC[0].x10_lastInput = linesC[0].xc_inputs[linesC[0].x4_outPoint];
|
|
|
|
linesC[0].x4_outPoint += 1;
|
|
|
|
|
|
|
|
linesC[1].x10_lastInput = linesC[1].xc_inputs[linesC[1].x4_outPoint];
|
|
|
|
linesC[1].x4_outPoint += 1;
|
|
|
|
|
|
|
|
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] =
|
|
|
|
xf0_allPassCoef * linesAP[0].x10_lastInput + linesC[0].x10_lastInput + linesC[1].x10_lastInput;
|
2020-03-27 20:55:41 +00:00
|
|
|
const float lowPass =
|
|
|
|
-(xf0_allPassCoef * linesAP[0].xc_inputs[linesAP[0].x0_inPoint] - linesAP[0].x10_lastInput);
|
2018-12-08 05:20:09 +00:00
|
|
|
linesAP[0].x0_inPoint += 1;
|
|
|
|
|
|
|
|
linesAP[0].x10_lastInput = linesAP[0].xc_inputs[linesAP[0].x4_outPoint];
|
|
|
|
linesAP[0].x4_outPoint += 1;
|
|
|
|
|
|
|
|
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;
|
|
|
|
|
|
|
|
lpLastOut = x11c_damping * lpLastOut + lowPass * 0.3f;
|
|
|
|
linesAP[1].xc_inputs[linesAP[1].x0_inPoint] = xf0_allPassCoef * linesAP[1].x10_lastInput + lpLastOut;
|
2020-03-27 20:55:41 +00:00
|
|
|
const float allPass =
|
|
|
|
-(xf0_allPassCoef * linesAP[1].xc_inputs[linesAP[1].x0_inPoint] - linesAP[1].x10_lastInput);
|
2018-12-08 05:20:09 +00:00
|
|
|
linesAP[1].x0_inPoint += 1;
|
|
|
|
|
|
|
|
linesAP[1].x10_lastInput = linesAP[1].xc_inputs[linesAP[1].x4_outPoint];
|
|
|
|
linesAP[1].x4_outPoint += 1;
|
|
|
|
|
|
|
|
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 */
|
2020-09-28 20:42:51 +00:00
|
|
|
audio[s * chanMap.m_channelCount + c] = dampWet * allPass + dampDry * sample;
|
2018-12-08 05:20:09 +00:00
|
|
|
}
|
|
|
|
x130_preDelayPtr[c] = preDelayPtr;
|
2016-05-13 01:46:41 +00:00
|
|
|
}
|
2018-12-08 05:20:09 +00:00
|
|
|
audio += chanMap.m_channelCount * 160;
|
|
|
|
}
|
2016-05-13 01:46:41 +00:00
|
|
|
}
|
|
|
|
|
2020-09-28 20:42:51 +00:00
|
|
|
EffectReverbHiImp::EffectReverbHiImp(float coloration, float mix, float time, float damping, float preDelay,
|
2016-07-14 04:54:46 +00:00
|
|
|
float crosstalk, double sampleRate)
|
2018-12-08 05:20:09 +00:00
|
|
|
: EffectReverbHi(coloration, mix, time, damping, preDelay, crosstalk) {
|
|
|
|
_setup(sampleRate);
|
2016-06-01 04:49:35 +00:00
|
|
|
}
|
|
|
|
|
2020-09-28 20:42:51 +00:00
|
|
|
void EffectReverbHiImp::_setup(double sampleRate) {
|
2018-12-08 05:20:09 +00:00
|
|
|
m_sampleRate = sampleRate;
|
|
|
|
_update();
|
2016-05-13 01:46:41 +00:00
|
|
|
}
|
|
|
|
|
2020-09-28 20:42:51 +00:00
|
|
|
void EffectReverbHiImp::_update() {
|
2020-03-27 20:55:41 +00:00
|
|
|
const float timeSamples = x148_x1d0_time * m_sampleRate;
|
|
|
|
const double rateRatio = m_sampleRate / NativeSampleRate;
|
|
|
|
|
|
|
|
for (size_t c = 0; c < NumChannels; ++c) {
|
|
|
|
for (size_t t = 0; t < xb4_C[c].size(); ++t) {
|
2018-12-08 05:20:09 +00:00
|
|
|
ReverbDelayLine& combLine = xb4_C[c][t];
|
2020-03-27 20:55:41 +00:00
|
|
|
const size_t tapDelay = CTapDelays[t] * rateRatio;
|
2018-12-08 05:20:09 +00:00
|
|
|
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
|
|
|
|
2020-03-27 20:55:41 +00:00
|
|
|
for (size_t t = 0; t < x0_AP[c].size(); ++t) {
|
2018-12-08 05:20:09 +00:00
|
|
|
ReverbDelayLine& allPassLine = x0_AP[c][t];
|
2020-03-27 20:55:41 +00:00
|
|
|
const size_t tapDelay = APTapDelays[t] * rateRatio;
|
2018-12-08 05:20:09 +00:00
|
|
|
allPassLine.allocate(tapDelay);
|
|
|
|
allPassLine.setdelay(tapDelay);
|
2016-05-13 01:46:41 +00:00
|
|
|
}
|
2016-05-14 04:46:39 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
ReverbDelayLine& lpLine = x78_LP[c];
|
2020-03-27 20:55:41 +00:00
|
|
|
const size_t tapDelay = LPTapDelays[c] * rateRatio;
|
2018-12-08 05:20:09 +00:00
|
|
|
lpLine.allocate(tapDelay);
|
|
|
|
lpLine.setdelay(tapDelay);
|
|
|
|
}
|
2016-05-14 04:46:39 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
x168_allPassCoef = x140_x1c8_coloration;
|
|
|
|
x19c_level = x144_x1cc_mix;
|
|
|
|
x1a0_damping = x14c_x1d4_damping;
|
2016-05-14 04:46:39 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
if (x1a0_damping < 0.05f)
|
|
|
|
x1a0_damping = 0.05f;
|
2016-05-14 04:46:39 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
x1a0_damping = 1.f - (x1a0_damping * 0.8f + 0.05);
|
|
|
|
|
|
|
|
if (x150_x1d8_preDelay != 0.f) {
|
|
|
|
x1a4_preDelayTime = m_sampleRate * x150_x1d8_preDelay;
|
2020-03-27 20:55:41 +00:00
|
|
|
for (size_t i = 0; i < NumChannels; ++i) {
|
|
|
|
x1ac_preDelayLine[i] = std::make_unique<float[]>(x1a4_preDelayTime);
|
2018-12-08 05:20:09 +00:00
|
|
|
x1b8_preDelayPtr[i] = x1ac_preDelayLine[i].get();
|
2016-05-14 04:46:39 +00:00
|
|
|
}
|
2018-12-08 05:20:09 +00:00
|
|
|
} else {
|
|
|
|
x1a4_preDelayTime = 0;
|
2020-03-27 20:55:41 +00:00
|
|
|
for (auto& delayLine : x1ac_preDelayLine) {
|
|
|
|
delayLine.reset();
|
2016-05-14 04:46:39 +00:00
|
|
|
}
|
2020-03-27 20:55:41 +00:00
|
|
|
x1b8_preDelayPtr.fill(nullptr);
|
2018-12-08 05:20:09 +00:00
|
|
|
}
|
2016-05-14 04:46:39 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
x1a8_internalCrosstalk = x1dc_crosstalk;
|
|
|
|
m_dirty = false;
|
2016-05-13 01:46:41 +00:00
|
|
|
}
|
|
|
|
|
2020-09-28 20:42:51 +00:00
|
|
|
void EffectReverbHiImp::_handleReverb(float* audio, int c, int chanCount, int sampleCount) {
|
2020-03-27 20:55:41 +00:00
|
|
|
const float dampWet = x19c_level * 0.6f;
|
|
|
|
const float dampDry = 0.6f - dampWet;
|
2018-12-08 05:20:09 +00:00
|
|
|
|
2020-03-27 20:55:41 +00:00
|
|
|
const auto& combCoefs = x16c_combCoef[c];
|
2018-12-08 05:20:09 +00:00
|
|
|
float& lpLastOut = x190_lpLastout[c];
|
|
|
|
float* preDelayLine = x1ac_preDelayLine[c].get();
|
|
|
|
float* preDelayPtr = x1b8_preDelayPtr[c];
|
|
|
|
float* lastPreDelaySamp = &preDelayLine[x1a4_preDelayTime - 1];
|
|
|
|
|
2020-03-27 20:55:41 +00:00
|
|
|
auto& linesC = xb4_C[c];
|
|
|
|
auto& linesAP = x0_AP[c];
|
2018-12-08 05:20:09 +00:00
|
|
|
ReverbDelayLine& lineLP = x78_LP[c];
|
|
|
|
|
2020-03-27 20:55:41 +00:00
|
|
|
const float allPassCoef = x168_allPassCoef;
|
|
|
|
const float damping = x1a0_damping;
|
|
|
|
const int32_t preDelayTime = x1a4_preDelayTime;
|
2018-12-08 05:20:09 +00:00
|
|
|
|
|
|
|
for (int s = 0; s < sampleCount; ++s) {
|
2020-03-27 20:55:41 +00:00
|
|
|
const float sample = audio[s * chanCount + c];
|
2018-12-08 05:20:09 +00:00
|
|
|
|
|
|
|
/* Pre-delay stage */
|
|
|
|
float sample2 = sample;
|
|
|
|
if (preDelayTime != 0) {
|
|
|
|
sample2 = *preDelayPtr;
|
|
|
|
*preDelayPtr = sample;
|
|
|
|
preDelayPtr += 1;
|
|
|
|
if (preDelayPtr == lastPreDelaySamp)
|
|
|
|
preDelayPtr = preDelayLine;
|
|
|
|
}
|
2016-05-13 01:46:41 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
/* Comb filter stage */
|
|
|
|
linesC[0].xc_inputs[linesC[0].x0_inPoint] = combCoefs[0] * linesC[0].x10_lastInput + sample2;
|
|
|
|
linesC[0].x0_inPoint += 1;
|
2016-05-13 01:46:41 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
linesC[1].xc_inputs[linesC[1].x0_inPoint] = combCoefs[1] * linesC[1].x10_lastInput + sample2;
|
|
|
|
linesC[1].x0_inPoint += 1;
|
2016-05-14 22:38:37 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
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
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
linesC[0].x10_lastInput = linesC[0].xc_inputs[linesC[0].x4_outPoint];
|
|
|
|
linesC[0].x4_outPoint += 1;
|
2016-05-13 01:46:41 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
linesC[1].x10_lastInput = linesC[1].xc_inputs[linesC[1].x4_outPoint];
|
|
|
|
linesC[1].x4_outPoint += 1;
|
2016-05-14 22:38:37 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
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
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
if (linesC[0].x0_inPoint == linesC[0].x8_length)
|
|
|
|
linesC[0].x0_inPoint = 0;
|
2016-05-13 01:46:41 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
if (linesC[1].x0_inPoint == linesC[1].x8_length)
|
|
|
|
linesC[1].x0_inPoint = 0;
|
2016-05-13 01:46:41 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
if (linesC[2].x0_inPoint == linesC[2].x8_length)
|
|
|
|
linesC[2].x0_inPoint = 0;
|
2016-05-14 22:38:37 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
if (linesC[0].x4_outPoint == linesC[0].x8_length)
|
|
|
|
linesC[0].x4_outPoint = 0;
|
2016-05-13 01:46:41 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
if (linesC[1].x4_outPoint == linesC[1].x8_length)
|
|
|
|
linesC[1].x4_outPoint = 0;
|
2016-05-13 01:46:41 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
if (linesC[2].x4_outPoint == linesC[2].x8_length)
|
|
|
|
linesC[2].x4_outPoint = 0;
|
2016-05-14 22:38:37 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
/* All-pass filter stage */
|
|
|
|
linesAP[0].xc_inputs[linesAP[0].x0_inPoint] = 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
|
|
|
|
2018-12-08 05:20:09 +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);
|
2016-05-13 01:46:41 +00:00
|
|
|
|
2020-03-27 20:55:41 +00:00
|
|
|
const float lowPass = -(allPassCoef * linesAP[1].xc_inputs[linesAP[1].x0_inPoint] - linesAP[1].x10_lastInput);
|
2018-12-08 05:20:09 +00:00
|
|
|
linesAP[0].x0_inPoint += 1;
|
|
|
|
linesAP[1].x0_inPoint += 1;
|
2016-05-13 01:46:41 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
if (linesAP[0].x0_inPoint == linesAP[0].x8_length)
|
|
|
|
linesAP[0].x0_inPoint = 0;
|
2016-05-13 01:46:41 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
if (linesAP[1].x0_inPoint == linesAP[1].x8_length)
|
|
|
|
linesAP[1].x0_inPoint = 0;
|
2016-05-13 01:46:41 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
linesAP[0].x10_lastInput = linesAP[0].xc_inputs[linesAP[0].x4_outPoint];
|
|
|
|
linesAP[0].x4_outPoint += 1;
|
2016-05-13 01:46:41 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
linesAP[1].x10_lastInput = linesAP[1].xc_inputs[linesAP[1].x4_outPoint];
|
|
|
|
linesAP[1].x4_outPoint += 1;
|
2016-05-13 01:46:41 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
if (linesAP[0].x4_outPoint == linesAP[0].x8_length)
|
|
|
|
linesAP[0].x4_outPoint = 0;
|
2016-05-13 01:46:41 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
if (linesAP[1].x4_outPoint == linesAP[1].x8_length)
|
|
|
|
linesAP[1].x4_outPoint = 0;
|
2016-05-13 01:46:41 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
lpLastOut = damping * lpLastOut + lowPass * 0.3f;
|
|
|
|
lineLP.xc_inputs[lineLP.x0_inPoint] = allPassCoef * lineLP.x10_lastInput + lpLastOut;
|
2020-03-27 20:55:41 +00:00
|
|
|
const float allPass = -(allPassCoef * lineLP.xc_inputs[lineLP.x0_inPoint] - lineLP.x10_lastInput);
|
2018-12-08 05:20:09 +00:00
|
|
|
lineLP.x0_inPoint += 1;
|
2016-05-13 01:46:41 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
lineLP.x10_lastInput = lineLP.xc_inputs[lineLP.x4_outPoint];
|
|
|
|
lineLP.x4_outPoint += 1;
|
2016-05-13 01:46:41 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
if (lineLP.x0_inPoint == lineLP.x8_length)
|
|
|
|
lineLP.x0_inPoint = 0;
|
2016-05-13 01:46:41 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
if (lineLP.x4_outPoint == lineLP.x8_length)
|
|
|
|
lineLP.x4_outPoint = 0;
|
2016-05-13 01:46:41 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
/* Mix out */
|
2020-09-28 20:42:51 +00:00
|
|
|
audio[s * chanCount + c] = dampWet * allPass + dampDry * sample;
|
2018-12-08 05:20:09 +00:00
|
|
|
}
|
2016-05-14 04:46:39 +00:00
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
x1b8_preDelayPtr[c] = preDelayPtr;
|
2016-05-13 01:46:41 +00:00
|
|
|
}
|
|
|
|
|
2020-09-28 20:42:51 +00:00
|
|
|
void EffectReverbHiImp::_doCrosstalk(float* audio, float wet, float dry, int chanCount, int sampleCount) {
|
2018-12-08 05:20:09 +00:00
|
|
|
for (int i = 0; i < sampleCount; ++i) {
|
2020-09-28 20:42:51 +00:00
|
|
|
auto* base = &audio[chanCount * i];
|
2018-12-08 05:20:09 +00:00
|
|
|
float allWet = 0;
|
|
|
|
for (int c = 0; c < chanCount; ++c) {
|
|
|
|
allWet += base[c] * wet;
|
|
|
|
base[c] *= dry;
|
2016-05-13 01:46:41 +00:00
|
|
|
}
|
2018-12-08 05:20:09 +00:00
|
|
|
for (int c = 0; c < chanCount; ++c)
|
2020-09-28 20:42:51 +00:00
|
|
|
base[c] = base[c] + allWet;
|
2018-12-08 05:20:09 +00:00
|
|
|
}
|
2016-05-13 01:46:41 +00:00
|
|
|
}
|
|
|
|
|
2020-09-28 20:42:51 +00:00
|
|
|
void EffectReverbHiImp::applyEffect(float* audio, size_t frameCount, const ChannelMap& chanMap) {
|
2018-12-08 05:20:09 +00:00
|
|
|
if (m_dirty)
|
|
|
|
_update();
|
|
|
|
|
|
|
|
for (size_t f = 0; f < frameCount; f += 160) {
|
|
|
|
size_t blockSamples = std::min(size_t(160), frameCount - f);
|
|
|
|
for (unsigned i = 0; i < chanMap.m_channelCount; ++i) {
|
|
|
|
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);
|
2016-05-13 01:46:41 +00:00
|
|
|
}
|
2018-12-08 05:20:09 +00:00
|
|
|
audio += chanMap.m_channelCount * 160;
|
|
|
|
}
|
2016-05-13 01:46:41 +00:00
|
|
|
}
|
|
|
|
|
2018-12-08 05:20:09 +00:00
|
|
|
} // namespace amuse
|