|
|
|
@ -83,7 +83,8 @@ const int16_t AudioWaveformSine[257] = { |
|
|
|
|
-4808, -4011, -3212, -2410, -1608, -804, 0 |
|
|
|
|
}; |
|
|
|
|
|
|
|
|
|
AudioEffectPlateReverb::AudioEffectPlateReverb(float32_t samplerate) |
|
|
|
|
AudioEffectPlateReverb::AudioEffectPlateReverb(float32_t samplerate) : |
|
|
|
|
FXElement(samplerate) |
|
|
|
|
{ |
|
|
|
|
input_attn = 0.5f; |
|
|
|
|
in_allp_k = INP_ALLP_COEFF; |
|
|
|
@ -156,9 +157,33 @@ AudioEffectPlateReverb::AudioEffectPlateReverb(float32_t samplerate) |
|
|
|
|
reverb_level = 0.0f; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
AudioEffectPlateReverb::~AudioEffectPlateReverb() |
|
|
|
|
{ |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
// #define sat16(n, rshift) signed_saturate_rshift((n), 16, (rshift))
|
|
|
|
|
|
|
|
|
|
void AudioEffectPlateReverb::doReverb(const float32_t* inblockL, const float32_t* inblockR, float32_t* rvbblockL, float32_t* rvbblockR, uint16_t len) |
|
|
|
|
void AudioEffectPlateReverb::reset() |
|
|
|
|
{ |
|
|
|
|
memset(in_allp1_bufL, 0, sizeof(in_allp1_bufL)); |
|
|
|
|
memset(in_allp2_bufL, 0, sizeof(in_allp2_bufL)); |
|
|
|
|
memset(in_allp3_bufL, 0, sizeof(in_allp3_bufL)); |
|
|
|
|
memset(in_allp4_bufL, 0, sizeof(in_allp4_bufL)); |
|
|
|
|
memset(in_allp1_bufR, 0, sizeof(in_allp1_bufR)); |
|
|
|
|
memset(in_allp2_bufR, 0, sizeof(in_allp2_bufR)); |
|
|
|
|
memset(in_allp3_bufR, 0, sizeof(in_allp3_bufR)); |
|
|
|
|
memset(in_allp4_bufR, 0, sizeof(in_allp4_bufR)); |
|
|
|
|
memset(lp_allp1_buf, 0, sizeof(lp_allp1_buf)); |
|
|
|
|
memset(lp_allp2_buf, 0, sizeof(lp_allp2_buf)); |
|
|
|
|
memset(lp_allp3_buf, 0, sizeof(lp_allp3_buf)); |
|
|
|
|
memset(lp_allp4_buf, 0, sizeof(lp_allp4_buf)); |
|
|
|
|
memset(lp_dly1_buf, 0, sizeof(lp_dly1_buf)); |
|
|
|
|
memset(lp_dly2_buf, 0, sizeof(lp_dly2_buf)); |
|
|
|
|
memset(lp_dly3_buf, 0, sizeof(lp_dly3_buf)); |
|
|
|
|
memset(lp_dly4_buf, 0, sizeof(lp_dly4_buf)); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void AudioEffectPlateReverb::processSample(float32_t inL, float32_t inR, float32_t& outL, float32_t& outR) |
|
|
|
|
{ |
|
|
|
|
float32_t input, acc, temp1, temp2; |
|
|
|
|
uint16_t temp16; |
|
|
|
@ -169,6 +194,259 @@ void AudioEffectPlateReverb::doReverb(const float32_t* inblockL, const float32_t |
|
|
|
|
int32_t y0, y1; |
|
|
|
|
int64_t y; |
|
|
|
|
uint32_t idx; |
|
|
|
|
|
|
|
|
|
rv_time = rv_time_k; |
|
|
|
|
|
|
|
|
|
// do the LFOs
|
|
|
|
|
lfo1_phase_acc += lfo1_adder; |
|
|
|
|
idx = lfo1_phase_acc >> 24; // 8bit lookup table address
|
|
|
|
|
y0 = AudioWaveformSine[idx]; |
|
|
|
|
y1 = AudioWaveformSine[idx+1]; |
|
|
|
|
idx = lfo1_phase_acc & 0x00FFFFFF; // lower 24 bit = fractional part
|
|
|
|
|
y = (int64_t)y0 * (0x00FFFFFF - idx); |
|
|
|
|
y += (int64_t)y1 * idx; |
|
|
|
|
lfo1_out_sin = (int32_t) (y >> (32-8)); // 16bit output
|
|
|
|
|
idx = ((lfo1_phase_acc >> 24)+64) & 0xFF; |
|
|
|
|
y0 = AudioWaveformSine[idx]; |
|
|
|
|
y1 = AudioWaveformSine[idx + 1]; |
|
|
|
|
y = (int64_t)y0 * (0x00FFFFFF - idx); |
|
|
|
|
y += (int64_t)y1 * idx; |
|
|
|
|
lfo1_out_cos = (int32_t) (y >> (32-8)); // 16bit output
|
|
|
|
|
|
|
|
|
|
lfo2_phase_acc += lfo2_adder; |
|
|
|
|
idx = lfo2_phase_acc >> 24; // 8bit lookup table address
|
|
|
|
|
y0 = AudioWaveformSine[idx]; |
|
|
|
|
y1 = AudioWaveformSine[idx+1]; |
|
|
|
|
idx = lfo2_phase_acc & 0x00FFFFFF; // lower 24 bit = fractional part
|
|
|
|
|
y = (int64_t)y0 * (0x00FFFFFF - idx); |
|
|
|
|
y += (int64_t)y1 * idx; |
|
|
|
|
lfo2_out_sin = (int32_t) (y >> (32-8)); //32-8->output 16bit,
|
|
|
|
|
idx = ((lfo2_phase_acc >> 24)+64) & 0xFF; |
|
|
|
|
y0 = AudioWaveformSine[idx]; |
|
|
|
|
y1 = AudioWaveformSine[idx + 1]; |
|
|
|
|
y = (int64_t)y0 * (0x00FFFFFF - idx); |
|
|
|
|
y += (int64_t)y1 * idx; |
|
|
|
|
lfo2_out_cos = (int32_t) (y >> (32-8)); // 16bit output
|
|
|
|
|
|
|
|
|
|
input = inL * input_attn; |
|
|
|
|
|
|
|
|
|
// chained input allpasses, channel L
|
|
|
|
|
acc = in_allp1_bufL[in_allp1_idxL] + input * in_allp_k;
|
|
|
|
|
in_allp1_bufL[in_allp1_idxL] = input - in_allp_k * acc; |
|
|
|
|
input = acc; |
|
|
|
|
if (++in_allp1_idxL >= sizeof(in_allp1_bufL)/sizeof(float32_t)) in_allp1_idxL = 0; |
|
|
|
|
|
|
|
|
|
acc = in_allp2_bufL[in_allp2_idxL] + input * in_allp_k;
|
|
|
|
|
in_allp2_bufL[in_allp2_idxL] = input - in_allp_k * acc; |
|
|
|
|
input = acc; |
|
|
|
|
if (++in_allp2_idxL >= sizeof(in_allp2_bufL)/sizeof(float32_t)) in_allp2_idxL = 0; |
|
|
|
|
|
|
|
|
|
acc = in_allp3_bufL[in_allp3_idxL] + input * in_allp_k;
|
|
|
|
|
in_allp3_bufL[in_allp3_idxL] = input - in_allp_k * acc; |
|
|
|
|
input = acc; |
|
|
|
|
if (++in_allp3_idxL >= sizeof(in_allp3_bufL)/sizeof(float32_t)) in_allp3_idxL = 0; |
|
|
|
|
|
|
|
|
|
acc = in_allp4_bufL[in_allp4_idxL] + input * in_allp_k;
|
|
|
|
|
in_allp4_bufL[in_allp4_idxL] = input - in_allp_k * acc; |
|
|
|
|
in_allp_out_L = acc; |
|
|
|
|
if (++in_allp4_idxL >= sizeof(in_allp4_bufL)/sizeof(float32_t)) in_allp4_idxL = 0; |
|
|
|
|
|
|
|
|
|
input = inR * input_attn; |
|
|
|
|
|
|
|
|
|
// chained input allpasses, channel R
|
|
|
|
|
acc = in_allp1_bufR[in_allp1_idxR] + input * in_allp_k;
|
|
|
|
|
in_allp1_bufR[in_allp1_idxR] = input - in_allp_k * acc; |
|
|
|
|
input = acc; |
|
|
|
|
if (++in_allp1_idxR >= sizeof(in_allp1_bufR)/sizeof(float32_t)) in_allp1_idxR = 0; |
|
|
|
|
|
|
|
|
|
acc = in_allp2_bufR[in_allp2_idxR] + input * in_allp_k;
|
|
|
|
|
in_allp2_bufR[in_allp2_idxR] = input - in_allp_k * acc; |
|
|
|
|
input = acc; |
|
|
|
|
if (++in_allp2_idxR >= sizeof(in_allp2_bufR)/sizeof(float32_t)) in_allp2_idxR = 0; |
|
|
|
|
|
|
|
|
|
acc = in_allp3_bufR[in_allp3_idxR] + input * in_allp_k;
|
|
|
|
|
in_allp3_bufR[in_allp3_idxR] = input - in_allp_k * acc; |
|
|
|
|
input = acc; |
|
|
|
|
if (++in_allp3_idxR >= sizeof(in_allp3_bufR)/sizeof(float32_t)) in_allp3_idxR = 0; |
|
|
|
|
|
|
|
|
|
acc = in_allp4_bufR[in_allp4_idxR] + input * in_allp_k;
|
|
|
|
|
in_allp4_bufR[in_allp4_idxR] = input - in_allp_k * acc; |
|
|
|
|
in_allp_out_R = acc; |
|
|
|
|
if (++in_allp4_idxR >= sizeof(in_allp4_bufR)/sizeof(float32_t)) in_allp4_idxR = 0; |
|
|
|
|
|
|
|
|
|
// input allpases done, start loop allpases
|
|
|
|
|
input = lp_allp_out + in_allp_out_R;
|
|
|
|
|
acc = lp_allp1_buf[lp_allp1_idx] + input * loop_allp_k; // input is the lp allpass chain output
|
|
|
|
|
lp_allp1_buf[lp_allp1_idx] = input - loop_allp_k * acc; |
|
|
|
|
input = acc; |
|
|
|
|
if (++lp_allp1_idx >= sizeof(lp_allp1_buf)/sizeof(float32_t))lp_allp1_idx = 0; |
|
|
|
|
|
|
|
|
|
acc = lp_dly1_buf[lp_dly1_idx]; // read the end of the delay
|
|
|
|
|
lp_dly1_buf[lp_dly1_idx] = input; // write new sample
|
|
|
|
|
input = acc; |
|
|
|
|
if (++lp_dly1_idx >= sizeof(lp_dly1_buf)/sizeof(float32_t)) lp_dly1_idx = 0; // update index
|
|
|
|
|
|
|
|
|
|
// hi/lo shelving filter
|
|
|
|
|
temp1 = input - lpf1; |
|
|
|
|
lpf1 += temp1 * lp_lowpass_f; |
|
|
|
|
temp2 = input - lpf1; |
|
|
|
|
temp1 = lpf1 - hpf1; |
|
|
|
|
hpf1 += temp1 * lp_hipass_f; |
|
|
|
|
acc = lpf1 + temp2*lp_hidamp_k + hpf1*lp_lodamp_k; |
|
|
|
|
acc = acc * rv_time * rv_time_scaler; // scale by the reveb time
|
|
|
|
|
|
|
|
|
|
input = acc + in_allp_out_L; |
|
|
|
|
|
|
|
|
|
acc = lp_allp2_buf[lp_allp2_idx] + input * loop_allp_k;
|
|
|
|
|
lp_allp2_buf[lp_allp2_idx] = input - loop_allp_k * acc; |
|
|
|
|
input = acc; |
|
|
|
|
if (++lp_allp2_idx >= sizeof(lp_allp2_buf)/sizeof(float32_t)) lp_allp2_idx = 0; |
|
|
|
|
acc = lp_dly2_buf[lp_dly2_idx]; // read the end of the delay
|
|
|
|
|
lp_dly2_buf[lp_dly2_idx] = input; // write new sample
|
|
|
|
|
input = acc; |
|
|
|
|
if (++lp_dly2_idx >= sizeof(lp_dly2_buf)/sizeof(float32_t)) lp_dly2_idx = 0; // update index
|
|
|
|
|
// hi/lo shelving filter
|
|
|
|
|
temp1 = input - lpf2; |
|
|
|
|
lpf2 += temp1 * lp_lowpass_f; |
|
|
|
|
temp2 = input - lpf2; |
|
|
|
|
temp1 = lpf2 - hpf2; |
|
|
|
|
hpf2 += temp1 * lp_hipass_f; |
|
|
|
|
acc = lpf2 + temp2*lp_hidamp_k + hpf2*lp_lodamp_k; |
|
|
|
|
acc = acc * rv_time * rv_time_scaler;
|
|
|
|
|
|
|
|
|
|
input = acc + in_allp_out_R; |
|
|
|
|
|
|
|
|
|
acc = lp_allp3_buf[lp_allp3_idx] + input * loop_allp_k;
|
|
|
|
|
lp_allp3_buf[lp_allp3_idx] = input - loop_allp_k * acc; |
|
|
|
|
input = acc; |
|
|
|
|
if (++lp_allp3_idx >= sizeof(lp_allp3_buf)/sizeof(float32_t)) lp_allp3_idx = 0; |
|
|
|
|
acc = lp_dly3_buf[lp_dly3_idx]; // read the end of the delay
|
|
|
|
|
lp_dly3_buf[lp_dly3_idx] = input; // write new sample
|
|
|
|
|
input = acc; |
|
|
|
|
if (++lp_dly3_idx >= sizeof(lp_dly3_buf)/sizeof(float32_t)) lp_dly3_idx = 0; // update index
|
|
|
|
|
// hi/lo shelving filter
|
|
|
|
|
temp1 = input - lpf3; |
|
|
|
|
lpf3 += temp1 * lp_lowpass_f; |
|
|
|
|
temp2 = input - lpf3; |
|
|
|
|
temp1 = lpf3 - hpf3; |
|
|
|
|
hpf3 += temp1 * lp_hipass_f; |
|
|
|
|
acc = lpf3 + temp2*lp_hidamp_k + hpf3*lp_lodamp_k; |
|
|
|
|
acc = acc * rv_time * rv_time_scaler;
|
|
|
|
|
|
|
|
|
|
input = acc + in_allp_out_L;
|
|
|
|
|
|
|
|
|
|
acc = lp_allp4_buf[lp_allp4_idx] + input * loop_allp_k;
|
|
|
|
|
lp_allp4_buf[lp_allp4_idx] = input - loop_allp_k * acc; |
|
|
|
|
input = acc; |
|
|
|
|
if (++lp_allp4_idx >= sizeof(lp_allp4_buf)/sizeof(float32_t)) lp_allp4_idx = 0; |
|
|
|
|
acc = lp_dly4_buf[lp_dly4_idx]; // read the end of the delay
|
|
|
|
|
lp_dly4_buf[lp_dly4_idx] = input; // write new sample
|
|
|
|
|
input = acc; |
|
|
|
|
if (++lp_dly4_idx >= sizeof(lp_dly4_buf)/sizeof(float32_t)) lp_dly4_idx= 0; // update index
|
|
|
|
|
// hi/lo shelving filter
|
|
|
|
|
temp1 = input - lpf4; |
|
|
|
|
lpf4 += temp1 * lp_lowpass_f; |
|
|
|
|
temp2 = input - lpf4; |
|
|
|
|
temp1 = lpf4 - hpf4; |
|
|
|
|
hpf4 += temp1 * lp_hipass_f; |
|
|
|
|
acc = lpf4 + temp2*lp_hidamp_k + hpf4*lp_lodamp_k; |
|
|
|
|
acc = acc * rv_time * rv_time_scaler;
|
|
|
|
|
|
|
|
|
|
lp_allp_out = acc; |
|
|
|
|
|
|
|
|
|
// channel L:
|
|
|
|
|
#ifdef TAP1_MODULATED |
|
|
|
|
temp16 = (lp_dly1_idx + lp_dly1_offset_L + (lfo1_out_cos>>LFO_FRAC_BITS)) % (sizeof(lp_dly1_buf)/sizeof(float32_t)); |
|
|
|
|
temp1 = lp_dly1_buf[temp16++]; // sample now
|
|
|
|
|
if (temp16 >= sizeof(lp_dly1_buf)/sizeof(float32_t)) temp16 = 0; |
|
|
|
|
temp2 = lp_dly1_buf[temp16]; // sample next
|
|
|
|
|
input = (float32_t)(lfo1_out_cos & LFO_FRAC_MASK) / ((float32_t)LFO_FRAC_MASK); // interp. k
|
|
|
|
|
acc = (temp1*(1.0f-input) + temp2*input)* 0.8f; |
|
|
|
|
#else |
|
|
|
|
temp16 = (lp_dly1_idx + lp_dly1_offset_L) % (sizeof(lp_dly1_buf)/sizeof(float32_t)); |
|
|
|
|
acc = lp_dly1_buf[temp16]* 0.8f; |
|
|
|
|
#endif |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef TAP2_MODULATED |
|
|
|
|
temp16 = (lp_dly2_idx + lp_dly2_offset_L + (lfo1_out_sin>>LFO_FRAC_BITS)) % (sizeof(lp_dly2_buf)/sizeof(float32_t)); |
|
|
|
|
temp1 = lp_dly2_buf[temp16++]; |
|
|
|
|
if (temp16 >= sizeof(lp_dly2_buf)/sizeof(float32_t)) temp16 = 0; |
|
|
|
|
temp2 = lp_dly2_buf[temp16];
|
|
|
|
|
input = (float32_t)(lfo1_out_sin & LFO_FRAC_MASK) / ((float32_t)LFO_FRAC_MASK); // interp. k
|
|
|
|
|
acc += (temp1*(1.0f-input) + temp2*input)* 0.7f; |
|
|
|
|
#else |
|
|
|
|
temp16 = (lp_dly2_idx + lp_dly2_offset_L) % (sizeof(lp_dly2_buf)/sizeof(float32_t)); |
|
|
|
|
acc += (temp1*(1.0f-input) + temp2*input)* 0.6f; |
|
|
|
|
#endif |
|
|
|
|
|
|
|
|
|
temp16 = (lp_dly3_idx + lp_dly3_offset_L + (lfo2_out_cos>>LFO_FRAC_BITS)) % (sizeof(lp_dly3_buf)/sizeof(float32_t)); |
|
|
|
|
temp1 = lp_dly3_buf[temp16++]; |
|
|
|
|
if (temp16 >= sizeof(lp_dly3_buf)/sizeof(float32_t)) temp16 = 0; |
|
|
|
|
temp2 = lp_dly3_buf[temp16];
|
|
|
|
|
input = (float32_t)(lfo2_out_cos & LFO_FRAC_MASK) / ((float32_t)LFO_FRAC_MASK); // interp. k
|
|
|
|
|
acc += (temp1*(1.0f-input) + temp2*input)* 0.6f; |
|
|
|
|
|
|
|
|
|
temp16 = (lp_dly4_idx + lp_dly4_offset_L + (lfo2_out_sin>>LFO_FRAC_BITS)) % (sizeof(lp_dly4_buf)/sizeof(float32_t)); |
|
|
|
|
temp1 = lp_dly4_buf[temp16++]; |
|
|
|
|
if (temp16 >= sizeof(lp_dly4_buf)/sizeof(float32_t)) temp16 = 0; |
|
|
|
|
temp2 = lp_dly4_buf[temp16];
|
|
|
|
|
input = (float32_t)(lfo2_out_sin & LFO_FRAC_MASK) / ((float32_t)LFO_FRAC_MASK); // interp. k
|
|
|
|
|
acc += (temp1*(1.0f-input) + temp2*input)* 0.5f; |
|
|
|
|
|
|
|
|
|
// Master lowpass filter
|
|
|
|
|
temp1 = acc - master_lowpass_l; |
|
|
|
|
master_lowpass_l += temp1 * master_lowpass_f; |
|
|
|
|
|
|
|
|
|
outL = master_lowpass_l; |
|
|
|
|
|
|
|
|
|
// Channel R
|
|
|
|
|
#ifdef TAP1_MODULATED |
|
|
|
|
temp16 = (lp_dly1_idx + lp_dly1_offset_R + (lfo2_out_cos>>LFO_FRAC_BITS)) % (sizeof(lp_dly1_buf)/sizeof(float32_t)); |
|
|
|
|
temp1 = lp_dly1_buf[temp16++]; // sample now
|
|
|
|
|
if (temp16 >= sizeof(lp_dly1_buf)/sizeof(float32_t)) temp16 = 0; |
|
|
|
|
temp2 = lp_dly1_buf[temp16]; // sample next
|
|
|
|
|
input = (float32_t)(lfo2_out_cos & LFO_FRAC_MASK) / ((float32_t)LFO_FRAC_MASK); // interp. k
|
|
|
|
|
|
|
|
|
|
acc = (temp1*(1.0f-input) + temp2*input)* 0.8f; |
|
|
|
|
#else |
|
|
|
|
temp16 = (lp_dly1_idx + lp_dly1_offset_R) % (sizeof(lp_dly1_buf)/sizeof(float32_t)); |
|
|
|
|
acc = lp_dly1_buf[temp16] * 0.8f; |
|
|
|
|
#endif |
|
|
|
|
#ifdef TAP2_MODULATED |
|
|
|
|
temp16 = (lp_dly2_idx + lp_dly2_offset_R + (lfo1_out_cos>>LFO_FRAC_BITS)) % (sizeof(lp_dly2_buf)/sizeof(float32_t)); |
|
|
|
|
temp1 = lp_dly2_buf[temp16++]; |
|
|
|
|
if (temp16 >= sizeof(lp_dly2_buf)/sizeof(float32_t)) temp16 = 0; |
|
|
|
|
temp2 = lp_dly2_buf[temp16];
|
|
|
|
|
input = (float32_t)(lfo1_out_cos & LFO_FRAC_MASK) / ((float32_t)LFO_FRAC_MASK); // interp. k
|
|
|
|
|
acc += (temp1*(1.0f-input) + temp2*input)* 0.7f; |
|
|
|
|
#else |
|
|
|
|
temp16 = (lp_dly2_idx + lp_dly2_offset_R) % (sizeof(lp_dly2_buf)/sizeof(float32_t)); |
|
|
|
|
acc += (temp1*(1.0f-input) + temp2*input)* 0.7f; |
|
|
|
|
#endif |
|
|
|
|
temp16 = (lp_dly3_idx + lp_dly3_offset_R + (lfo2_out_sin>>LFO_FRAC_BITS)) % (sizeof(lp_dly3_buf)/sizeof(float32_t)); |
|
|
|
|
temp1 = lp_dly3_buf[temp16++]; |
|
|
|
|
if (temp16 >= sizeof(lp_dly3_buf)/sizeof(float32_t)) temp16 = 0; |
|
|
|
|
temp2 = lp_dly3_buf[temp16];
|
|
|
|
|
input = (float32_t)(lfo2_out_sin & LFO_FRAC_MASK) / ((float32_t)LFO_FRAC_MASK); // interp. k
|
|
|
|
|
acc += (temp1*(1.0f-input) + temp2*input)* 0.6f; |
|
|
|
|
|
|
|
|
|
temp16 = (lp_dly4_idx + lp_dly4_offset_R + (lfo1_out_sin>>LFO_FRAC_BITS)) % (sizeof(lp_dly4_buf)/sizeof(float32_t)); |
|
|
|
|
temp1 = lp_dly4_buf[temp16++]; |
|
|
|
|
if (temp16 >= sizeof(lp_dly4_buf)/sizeof(float32_t)) temp16 = 0; |
|
|
|
|
temp2 = lp_dly4_buf[temp16];
|
|
|
|
|
input = (float32_t)(lfo2_out_cos & LFO_FRAC_MASK) / ((float32_t)LFO_FRAC_MASK); // interp. k
|
|
|
|
|
acc += (temp1*(1.0f-input) + temp2*input)* 0.5f; |
|
|
|
|
|
|
|
|
|
// Master lowpass filter
|
|
|
|
|
temp1 = acc - master_lowpass_r; |
|
|
|
|
master_lowpass_r += temp1 * master_lowpass_f; |
|
|
|
|
|
|
|
|
|
outR = master_lowpass_r; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void AudioEffectPlateReverb::doReverb(const float32_t* inblockL, const float32_t* inblockR, float32_t* rvbblockL, float32_t* rvbblockR, uint16_t len) |
|
|
|
|
{ |
|
|
|
|
static bool cleanup_done = false; |
|
|
|
|
|
|
|
|
|
// handle bypass, 1st call will clean the buffers to avoid continuing the previous reverb tail
|
|
|
|
@ -176,23 +454,7 @@ void AudioEffectPlateReverb::doReverb(const float32_t* inblockL, const float32_t |
|
|
|
|
{ |
|
|
|
|
if (!cleanup_done) |
|
|
|
|
{ |
|
|
|
|
memset(in_allp1_bufL, 0, sizeof(in_allp1_bufL)); |
|
|
|
|
memset(in_allp2_bufL, 0, sizeof(in_allp2_bufL)); |
|
|
|
|
memset(in_allp3_bufL, 0, sizeof(in_allp3_bufL)); |
|
|
|
|
memset(in_allp4_bufL, 0, sizeof(in_allp4_bufL)); |
|
|
|
|
memset(in_allp1_bufR, 0, sizeof(in_allp1_bufR)); |
|
|
|
|
memset(in_allp2_bufR, 0, sizeof(in_allp2_bufR)); |
|
|
|
|
memset(in_allp3_bufR, 0, sizeof(in_allp3_bufR)); |
|
|
|
|
memset(in_allp4_bufR, 0, sizeof(in_allp4_bufR)); |
|
|
|
|
memset(lp_allp1_buf, 0, sizeof(lp_allp1_buf)); |
|
|
|
|
memset(lp_allp2_buf, 0, sizeof(lp_allp2_buf)); |
|
|
|
|
memset(lp_allp3_buf, 0, sizeof(lp_allp3_buf)); |
|
|
|
|
memset(lp_allp4_buf, 0, sizeof(lp_allp4_buf)); |
|
|
|
|
memset(lp_dly1_buf, 0, sizeof(lp_dly1_buf)); |
|
|
|
|
memset(lp_dly2_buf, 0, sizeof(lp_dly2_buf)); |
|
|
|
|
memset(lp_dly3_buf, 0, sizeof(lp_dly3_buf)); |
|
|
|
|
memset(lp_dly4_buf, 0, sizeof(lp_dly4_buf)); |
|
|
|
|
|
|
|
|
|
this->reset(); |
|
|
|
|
cleanup_done = true; |
|
|
|
|
} |
|
|
|
|
|
|
|
|
@ -200,255 +462,8 @@ void AudioEffectPlateReverb::doReverb(const float32_t* inblockL, const float32_t |
|
|
|
|
} |
|
|
|
|
cleanup_done = false; |
|
|
|
|
|
|
|
|
|
rv_time = rv_time_k; |
|
|
|
|
|
|
|
|
|
for (uint16_t i=0; i < len; i++)
|
|
|
|
|
{ |
|
|
|
|
// do the LFOs
|
|
|
|
|
lfo1_phase_acc += lfo1_adder; |
|
|
|
|
idx = lfo1_phase_acc >> 24; // 8bit lookup table address
|
|
|
|
|
y0 = AudioWaveformSine[idx]; |
|
|
|
|
y1 = AudioWaveformSine[idx+1]; |
|
|
|
|
idx = lfo1_phase_acc & 0x00FFFFFF; // lower 24 bit = fractional part
|
|
|
|
|
y = (int64_t)y0 * (0x00FFFFFF - idx); |
|
|
|
|
y += (int64_t)y1 * idx; |
|
|
|
|
lfo1_out_sin = (int32_t) (y >> (32-8)); // 16bit output
|
|
|
|
|
idx = ((lfo1_phase_acc >> 24)+64) & 0xFF; |
|
|
|
|
y0 = AudioWaveformSine[idx]; |
|
|
|
|
y1 = AudioWaveformSine[idx + 1]; |
|
|
|
|
y = (int64_t)y0 * (0x00FFFFFF - idx); |
|
|
|
|
y += (int64_t)y1 * idx; |
|
|
|
|
lfo1_out_cos = (int32_t) (y >> (32-8)); // 16bit output
|
|
|
|
|
|
|
|
|
|
lfo2_phase_acc += lfo2_adder; |
|
|
|
|
idx = lfo2_phase_acc >> 24; // 8bit lookup table address
|
|
|
|
|
y0 = AudioWaveformSine[idx]; |
|
|
|
|
y1 = AudioWaveformSine[idx+1]; |
|
|
|
|
idx = lfo2_phase_acc & 0x00FFFFFF; // lower 24 bit = fractional part
|
|
|
|
|
y = (int64_t)y0 * (0x00FFFFFF - idx); |
|
|
|
|
y += (int64_t)y1 * idx; |
|
|
|
|
lfo2_out_sin = (int32_t) (y >> (32-8)); //32-8->output 16bit,
|
|
|
|
|
idx = ((lfo2_phase_acc >> 24)+64) & 0xFF; |
|
|
|
|
y0 = AudioWaveformSine[idx]; |
|
|
|
|
y1 = AudioWaveformSine[idx + 1]; |
|
|
|
|
y = (int64_t)y0 * (0x00FFFFFF - idx); |
|
|
|
|
y += (int64_t)y1 * idx; |
|
|
|
|
lfo2_out_cos = (int32_t) (y >> (32-8)); // 16bit output
|
|
|
|
|
|
|
|
|
|
input = inblockL[i] * input_attn; |
|
|
|
|
|
|
|
|
|
// chained input allpasses, channel L
|
|
|
|
|
acc = in_allp1_bufL[in_allp1_idxL] + input * in_allp_k;
|
|
|
|
|
in_allp1_bufL[in_allp1_idxL] = input - in_allp_k * acc; |
|
|
|
|
input = acc; |
|
|
|
|
if (++in_allp1_idxL >= sizeof(in_allp1_bufL)/sizeof(float32_t)) in_allp1_idxL = 0; |
|
|
|
|
|
|
|
|
|
acc = in_allp2_bufL[in_allp2_idxL] + input * in_allp_k;
|
|
|
|
|
in_allp2_bufL[in_allp2_idxL] = input - in_allp_k * acc; |
|
|
|
|
input = acc; |
|
|
|
|
if (++in_allp2_idxL >= sizeof(in_allp2_bufL)/sizeof(float32_t)) in_allp2_idxL = 0; |
|
|
|
|
|
|
|
|
|
acc = in_allp3_bufL[in_allp3_idxL] + input * in_allp_k;
|
|
|
|
|
in_allp3_bufL[in_allp3_idxL] = input - in_allp_k * acc; |
|
|
|
|
input = acc; |
|
|
|
|
if (++in_allp3_idxL >= sizeof(in_allp3_bufL)/sizeof(float32_t)) in_allp3_idxL = 0; |
|
|
|
|
|
|
|
|
|
acc = in_allp4_bufL[in_allp4_idxL] + input * in_allp_k;
|
|
|
|
|
in_allp4_bufL[in_allp4_idxL] = input - in_allp_k * acc; |
|
|
|
|
in_allp_out_L = acc; |
|
|
|
|
if (++in_allp4_idxL >= sizeof(in_allp4_bufL)/sizeof(float32_t)) in_allp4_idxL = 0; |
|
|
|
|
|
|
|
|
|
input = inblockR[i] * input_attn; |
|
|
|
|
|
|
|
|
|
// chained input allpasses, channel R
|
|
|
|
|
acc = in_allp1_bufR[in_allp1_idxR] + input * in_allp_k;
|
|
|
|
|
in_allp1_bufR[in_allp1_idxR] = input - in_allp_k * acc; |
|
|
|
|
input = acc; |
|
|
|
|
if (++in_allp1_idxR >= sizeof(in_allp1_bufR)/sizeof(float32_t)) in_allp1_idxR = 0; |
|
|
|
|
|
|
|
|
|
acc = in_allp2_bufR[in_allp2_idxR] + input * in_allp_k;
|
|
|
|
|
in_allp2_bufR[in_allp2_idxR] = input - in_allp_k * acc; |
|
|
|
|
input = acc; |
|
|
|
|
if (++in_allp2_idxR >= sizeof(in_allp2_bufR)/sizeof(float32_t)) in_allp2_idxR = 0; |
|
|
|
|
|
|
|
|
|
acc = in_allp3_bufR[in_allp3_idxR] + input * in_allp_k;
|
|
|
|
|
in_allp3_bufR[in_allp3_idxR] = input - in_allp_k * acc; |
|
|
|
|
input = acc; |
|
|
|
|
if (++in_allp3_idxR >= sizeof(in_allp3_bufR)/sizeof(float32_t)) in_allp3_idxR = 0; |
|
|
|
|
|
|
|
|
|
acc = in_allp4_bufR[in_allp4_idxR] + input * in_allp_k;
|
|
|
|
|
in_allp4_bufR[in_allp4_idxR] = input - in_allp_k * acc; |
|
|
|
|
in_allp_out_R = acc; |
|
|
|
|
if (++in_allp4_idxR >= sizeof(in_allp4_bufR)/sizeof(float32_t)) in_allp4_idxR = 0; |
|
|
|
|
|
|
|
|
|
// input allpases done, start loop allpases
|
|
|
|
|
input = lp_allp_out + in_allp_out_R;
|
|
|
|
|
acc = lp_allp1_buf[lp_allp1_idx] + input * loop_allp_k; // input is the lp allpass chain output
|
|
|
|
|
lp_allp1_buf[lp_allp1_idx] = input - loop_allp_k * acc; |
|
|
|
|
input = acc; |
|
|
|
|
if (++lp_allp1_idx >= sizeof(lp_allp1_buf)/sizeof(float32_t)) lp_allp1_idx = 0; |
|
|
|
|
|
|
|
|
|
acc = lp_dly1_buf[lp_dly1_idx]; // read the end of the delay
|
|
|
|
|
lp_dly1_buf[lp_dly1_idx] = input; // write new sample
|
|
|
|
|
input = acc; |
|
|
|
|
if (++lp_dly1_idx >= sizeof(lp_dly1_buf)/sizeof(float32_t)) lp_dly1_idx = 0; // update index
|
|
|
|
|
|
|
|
|
|
// hi/lo shelving filter
|
|
|
|
|
temp1 = input - lpf1; |
|
|
|
|
lpf1 += temp1 * lp_lowpass_f; |
|
|
|
|
temp2 = input - lpf1; |
|
|
|
|
temp1 = lpf1 - hpf1; |
|
|
|
|
hpf1 += temp1 * lp_hipass_f; |
|
|
|
|
acc = lpf1 + temp2*lp_hidamp_k + hpf1*lp_lodamp_k; |
|
|
|
|
acc = acc * rv_time * rv_time_scaler; // scale by the reveb time
|
|
|
|
|
|
|
|
|
|
input = acc + in_allp_out_L; |
|
|
|
|
|
|
|
|
|
acc = lp_allp2_buf[lp_allp2_idx] + input * loop_allp_k;
|
|
|
|
|
lp_allp2_buf[lp_allp2_idx] = input - loop_allp_k * acc; |
|
|
|
|
input = acc; |
|
|
|
|
if (++lp_allp2_idx >= sizeof(lp_allp2_buf)/sizeof(float32_t)) lp_allp2_idx = 0; |
|
|
|
|
acc = lp_dly2_buf[lp_dly2_idx]; // read the end of the delay
|
|
|
|
|
lp_dly2_buf[lp_dly2_idx] = input; // write new sample
|
|
|
|
|
input = acc; |
|
|
|
|
if (++lp_dly2_idx >= sizeof(lp_dly2_buf)/sizeof(float32_t)) lp_dly2_idx = 0; // update index
|
|
|
|
|
// hi/lo shelving filter
|
|
|
|
|
temp1 = input - lpf2; |
|
|
|
|
lpf2 += temp1 * lp_lowpass_f; |
|
|
|
|
temp2 = input - lpf2; |
|
|
|
|
temp1 = lpf2 - hpf2; |
|
|
|
|
hpf2 += temp1 * lp_hipass_f; |
|
|
|
|
acc = lpf2 + temp2*lp_hidamp_k + hpf2*lp_lodamp_k; |
|
|
|
|
acc = acc * rv_time * rv_time_scaler;
|
|
|
|
|
|
|
|
|
|
input = acc + in_allp_out_R; |
|
|
|
|
|
|
|
|
|
acc = lp_allp3_buf[lp_allp3_idx] + input * loop_allp_k;
|
|
|
|
|
lp_allp3_buf[lp_allp3_idx] = input - loop_allp_k * acc; |
|
|
|
|
input = acc; |
|
|
|
|
if (++lp_allp3_idx >= sizeof(lp_allp3_buf)/sizeof(float32_t)) lp_allp3_idx = 0; |
|
|
|
|
acc = lp_dly3_buf[lp_dly3_idx]; // read the end of the delay
|
|
|
|
|
lp_dly3_buf[lp_dly3_idx] = input; // write new sample
|
|
|
|
|
input = acc; |
|
|
|
|
if (++lp_dly3_idx >= sizeof(lp_dly3_buf)/sizeof(float32_t)) lp_dly3_idx = 0; // update index
|
|
|
|
|
// hi/lo shelving filter
|
|
|
|
|
temp1 = input - lpf3; |
|
|
|
|
lpf3 += temp1 * lp_lowpass_f; |
|
|
|
|
temp2 = input - lpf3; |
|
|
|
|
temp1 = lpf3 - hpf3; |
|
|
|
|
hpf3 += temp1 * lp_hipass_f; |
|
|
|
|
acc = lpf3 + temp2*lp_hidamp_k + hpf3*lp_lodamp_k; |
|
|
|
|
acc = acc * rv_time * rv_time_scaler;
|
|
|
|
|
|
|
|
|
|
input = acc + in_allp_out_L;
|
|
|
|
|
|
|
|
|
|
acc = lp_allp4_buf[lp_allp4_idx] + input * loop_allp_k;
|
|
|
|
|
lp_allp4_buf[lp_allp4_idx] = input - loop_allp_k * acc; |
|
|
|
|
input = acc; |
|
|
|
|
if (++lp_allp4_idx >= sizeof(lp_allp4_buf)/sizeof(float32_t)) lp_allp4_idx = 0; |
|
|
|
|
acc = lp_dly4_buf[lp_dly4_idx]; // read the end of the delay
|
|
|
|
|
lp_dly4_buf[lp_dly4_idx] = input; // write new sample
|
|
|
|
|
input = acc; |
|
|
|
|
if (++lp_dly4_idx >= sizeof(lp_dly4_buf)/sizeof(float32_t)) lp_dly4_idx= 0; // update index
|
|
|
|
|
// hi/lo shelving filter
|
|
|
|
|
temp1 = input - lpf4; |
|
|
|
|
lpf4 += temp1 * lp_lowpass_f; |
|
|
|
|
temp2 = input - lpf4; |
|
|
|
|
temp1 = lpf4 - hpf4; |
|
|
|
|
hpf4 += temp1 * lp_hipass_f; |
|
|
|
|
acc = lpf4 + temp2*lp_hidamp_k + hpf4*lp_lodamp_k; |
|
|
|
|
acc = acc * rv_time * rv_time_scaler;
|
|
|
|
|
|
|
|
|
|
lp_allp_out = acc; |
|
|
|
|
|
|
|
|
|
// channel L:
|
|
|
|
|
#ifdef TAP1_MODULATED |
|
|
|
|
temp16 = (lp_dly1_idx + lp_dly1_offset_L + (lfo1_out_cos>>LFO_FRAC_BITS)) % (sizeof(lp_dly1_buf)/sizeof(float32_t)); |
|
|
|
|
temp1 = lp_dly1_buf[temp16++]; // sample now
|
|
|
|
|
if (temp16 >= sizeof(lp_dly1_buf)/sizeof(float32_t)) temp16 = 0; |
|
|
|
|
temp2 = lp_dly1_buf[temp16]; // sample next
|
|
|
|
|
input = (float32_t)(lfo1_out_cos & LFO_FRAC_MASK) / ((float32_t)LFO_FRAC_MASK); // interp. k
|
|
|
|
|
acc = (temp1*(1.0f-input) + temp2*input)* 0.8f; |
|
|
|
|
#else |
|
|
|
|
temp16 = (lp_dly1_idx + lp_dly1_offset_L) % (sizeof(lp_dly1_buf)/sizeof(float32_t)); |
|
|
|
|
acc = lp_dly1_buf[temp16]* 0.8f; |
|
|
|
|
#endif |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#ifdef TAP2_MODULATED |
|
|
|
|
temp16 = (lp_dly2_idx + lp_dly2_offset_L + (lfo1_out_sin>>LFO_FRAC_BITS)) % (sizeof(lp_dly2_buf)/sizeof(float32_t)); |
|
|
|
|
temp1 = lp_dly2_buf[temp16++]; |
|
|
|
|
if (temp16 >= sizeof(lp_dly2_buf)/sizeof(float32_t)) temp16 = 0; |
|
|
|
|
temp2 = lp_dly2_buf[temp16];
|
|
|
|
|
input = (float32_t)(lfo1_out_sin & LFO_FRAC_MASK) / ((float32_t)LFO_FRAC_MASK); // interp. k
|
|
|
|
|
acc += (temp1*(1.0f-input) + temp2*input)* 0.7f; |
|
|
|
|
#else |
|
|
|
|
temp16 = (lp_dly2_idx + lp_dly2_offset_L) % (sizeof(lp_dly2_buf)/sizeof(float32_t)); |
|
|
|
|
acc += (temp1*(1.0f-input) + temp2*input)* 0.6f; |
|
|
|
|
#endif |
|
|
|
|
|
|
|
|
|
temp16 = (lp_dly3_idx + lp_dly3_offset_L + (lfo2_out_cos>>LFO_FRAC_BITS)) % (sizeof(lp_dly3_buf)/sizeof(float32_t)); |
|
|
|
|
temp1 = lp_dly3_buf[temp16++]; |
|
|
|
|
if (temp16 >= sizeof(lp_dly3_buf)/sizeof(float32_t)) temp16 = 0; |
|
|
|
|
temp2 = lp_dly3_buf[temp16];
|
|
|
|
|
input = (float32_t)(lfo2_out_cos & LFO_FRAC_MASK) / ((float32_t)LFO_FRAC_MASK); // interp. k
|
|
|
|
|
acc += (temp1*(1.0f-input) + temp2*input)* 0.6f; |
|
|
|
|
|
|
|
|
|
temp16 = (lp_dly4_idx + lp_dly4_offset_L + (lfo2_out_sin>>LFO_FRAC_BITS)) % (sizeof(lp_dly4_buf)/sizeof(float32_t)); |
|
|
|
|
temp1 = lp_dly4_buf[temp16++]; |
|
|
|
|
if (temp16 >= sizeof(lp_dly4_buf)/sizeof(float32_t)) temp16 = 0; |
|
|
|
|
temp2 = lp_dly4_buf[temp16];
|
|
|
|
|
input = (float32_t)(lfo2_out_sin & LFO_FRAC_MASK) / ((float32_t)LFO_FRAC_MASK); // interp. k
|
|
|
|
|
acc += (temp1*(1.0f-input) + temp2*input)* 0.5f; |
|
|
|
|
|
|
|
|
|
// Master lowpass filter
|
|
|
|
|
temp1 = acc - master_lowpass_l; |
|
|
|
|
master_lowpass_l += temp1 * master_lowpass_f; |
|
|
|
|
|
|
|
|
|
rvbblockL[i] = master_lowpass_l; |
|
|
|
|
|
|
|
|
|
// Channel R
|
|
|
|
|
#ifdef TAP1_MODULATED |
|
|
|
|
temp16 = (lp_dly1_idx + lp_dly1_offset_R + (lfo2_out_cos>>LFO_FRAC_BITS)) % (sizeof(lp_dly1_buf)/sizeof(float32_t)); |
|
|
|
|
temp1 = lp_dly1_buf[temp16++]; // sample now
|
|
|
|
|
if (temp16 >= sizeof(lp_dly1_buf)/sizeof(float32_t)) temp16 = 0; |
|
|
|
|
temp2 = lp_dly1_buf[temp16]; // sample next
|
|
|
|
|
input = (float32_t)(lfo2_out_cos & LFO_FRAC_MASK) / ((float32_t)LFO_FRAC_MASK); // interp. k
|
|
|
|
|
|
|
|
|
|
acc = (temp1*(1.0f-input) + temp2*input)* 0.8f; |
|
|
|
|
#else |
|
|
|
|
temp16 = (lp_dly1_idx + lp_dly1_offset_R) % (sizeof(lp_dly1_buf)/sizeof(float32_t)); |
|
|
|
|
acc = lp_dly1_buf[temp16] * 0.8f; |
|
|
|
|
#endif |
|
|
|
|
#ifdef TAP2_MODULATED |
|
|
|
|
temp16 = (lp_dly2_idx + lp_dly2_offset_R + (lfo1_out_cos>>LFO_FRAC_BITS)) % (sizeof(lp_dly2_buf)/sizeof(float32_t)); |
|
|
|
|
temp1 = lp_dly2_buf[temp16++]; |
|
|
|
|
if (temp16 >= sizeof(lp_dly2_buf)/sizeof(float32_t)) temp16 = 0; |
|
|
|
|
temp2 = lp_dly2_buf[temp16];
|
|
|
|
|
input = (float32_t)(lfo1_out_cos & LFO_FRAC_MASK) / ((float32_t)LFO_FRAC_MASK); // interp. k
|
|
|
|
|
acc += (temp1*(1.0f-input) + temp2*input)* 0.7f; |
|
|
|
|
#else |
|
|
|
|
temp16 = (lp_dly2_idx + lp_dly2_offset_R) % (sizeof(lp_dly2_buf)/sizeof(float32_t)); |
|
|
|
|
acc += (temp1*(1.0f-input) + temp2*input)* 0.7f; |
|
|
|
|
#endif |
|
|
|
|
temp16 = (lp_dly3_idx + lp_dly3_offset_R + (lfo2_out_sin>>LFO_FRAC_BITS)) % (sizeof(lp_dly3_buf)/sizeof(float32_t)); |
|
|
|
|
temp1 = lp_dly3_buf[temp16++]; |
|
|
|
|
if (temp16 >= sizeof(lp_dly3_buf)/sizeof(float32_t)) temp16 = 0; |
|
|
|
|
temp2 = lp_dly3_buf[temp16];
|
|
|
|
|
input = (float32_t)(lfo2_out_sin & LFO_FRAC_MASK) / ((float32_t)LFO_FRAC_MASK); // interp. k
|
|
|
|
|
acc += (temp1*(1.0f-input) + temp2*input)* 0.6f; |
|
|
|
|
|
|
|
|
|
temp16 = (lp_dly4_idx + lp_dly4_offset_R + (lfo1_out_sin>>LFO_FRAC_BITS)) % (sizeof(lp_dly4_buf)/sizeof(float32_t)); |
|
|
|
|
temp1 = lp_dly4_buf[temp16++]; |
|
|
|
|
if (temp16 >= sizeof(lp_dly4_buf)/sizeof(float32_t)) temp16 = 0; |
|
|
|
|
temp2 = lp_dly4_buf[temp16];
|
|
|
|
|
input = (float32_t)(lfo2_out_cos & LFO_FRAC_MASK) / ((float32_t)LFO_FRAC_MASK); // interp. k
|
|
|
|
|
acc += (temp1*(1.0f-input) + temp2*input)* 0.5f; |
|
|
|
|
|
|
|
|
|
// Master lowpass filter
|
|
|
|
|
temp1 = acc - master_lowpass_r; |
|
|
|
|
master_lowpass_r += temp1 * master_lowpass_f; |
|
|
|
|
|
|
|
|
|
rvbblockR[i] = master_lowpass_r; |
|
|
|
|
this->processSample(inblockL[i], inblockR[i], rvbblockL[i], rvbblockR[i]); |
|
|
|
|
} |
|
|
|
|
} |
|
|
|
|