zynaddsubfx

ZynAddSubFX open source synthesizer
Log | Files | Refs | Submodules | LICENSE

commit 5c3b6e6e097b18c033f36331bb85f3188a2baffa
parent eb427f941f0f6d60845a66ec678d3db7b2b52802
Author: Jonathan Moore Liles <[email protected]>
Date:   Fri, 11 Dec 2020 17:37:41 -0800

Revert "SVFilter: Use Value_Smoothing_Filter to smooth cutoff frequency changes."

Still has zipper artifact, needs more work.

Diffstat:
Msrc/DSP/SVFilter.cpp | 87++++++++++++++++++++++++++++++++++++++++++++++++++++++++-----------------------
Msrc/DSP/SVFilter.h | 12+++++++-----
2 files changed, 69 insertions(+), 30 deletions(-)

diff --git a/src/DSP/SVFilter.cpp b/src/DSP/SVFilter.cpp @@ -26,6 +26,12 @@ namespace zyn { +enum FilterInterpolationType { + INTERPOLATE_EXTREME = 0x01, + INTERPOLATE_NON_ZERO, + INTERPOLATE_NONE +}; + SVFilter::SVFilter(unsigned char Ftype, float Ffreq, float Fq, unsigned char Fstages, unsigned int srate, int bufsize) :Filter(srate, bufsize), @@ -33,15 +39,15 @@ SVFilter::SVFilter(unsigned char Ftype, float Ffreq, float Fq, stages(Fstages), freq(Ffreq), q(Fq), - gain(1.0f) + gain(1.0f), + needsinterpolation(INTERPOLATE_NONE), + firsttime(true) { if(stages >= MAX_FILTER_STAGES) stages = MAX_FILTER_STAGES; outgain = 1.0f; cleanup(); setfreq_and_q(Ffreq, Fq); - freq_smoothing.reset(Ffreq); - freq_smoothing.sample_rate(bufsize); } SVFilter::~SVFilter() @@ -51,6 +57,8 @@ void SVFilter::cleanup() { for(int i = 0; i < MAX_FILTER_STAGES + 1; ++i) st[i].low = st[i].high = st[i].band = st[i].notch = 0.0f; + oldabovenq = false; + abovenq = false; } SVFilter::response::response(float b0, float b1, float b2, @@ -119,8 +127,26 @@ void SVFilter::setfreq(float frequency) if(rap < 1.0f) rap = 1.0f / rap; + oldabovenq = abovenq; + abovenq = frequency > (samplerate_f / 2 - 500.0f); + + bool nyquistthresh = (abovenq ^ oldabovenq); + + //if the frequency is changed fast, it needs interpolation + if((rap > 3.0f) || nyquistthresh) { //(now, filter and coefficients backup) + if(!firsttime) + needsinterpolation = INTERPOLATE_EXTREME; + ipar = par; + } else if(rap != 1.0) { + if (!firsttime) + needsinterpolation = INTERPOLATE_NON_ZERO; + ipar = par; + } else { + needsinterpolation = INTERPOLATE_NONE; + } freq = frequency; computefiltercoefs(); + firsttime = false; } void SVFilter::setfreq_and_q(float frequency, float q_) @@ -178,6 +204,21 @@ float *SVFilter::getfilteroutfortype(SVFilter::fstage &x) { return out; } +void SVFilter::singlefilterout_with_par_interpolation(float *smp, fstage &x, parameters &par1, parameters &par2) +{ + float *out = getfilteroutfortype(x); + for(int i = 0; i < buffersize; ++i) { + float p = i / buffersize_f; + float f = par1.f + (par2.f - par1.f) * p; + float q = par1.q + (par2.q - par1.q) * p; + float q_sqrt = sqrtf(q); + x.low = x.low + f * x.band; + x.high = q_sqrt * smp[i] - x.low - q * x.band; + x.band = f * x.high + x.band; + x.notch = x.high + x.low; + smp[i] = *out; + } +} // simplifying the responses // xl = xl*z(-1) + pf*xb*z(-1) @@ -191,7 +232,7 @@ float *SVFilter::getfilteroutfortype(SVFilter::fstage &x) { -void SVFilter::singlefilterout(float *smp, SVFilter::fstage &x, SVFilter::parameters &par, int buffersize ) +void SVFilter::singlefilterout(float *smp, SVFilter::fstage &x, SVFilter::parameters &par) { float *out = getfilteroutfortype(x); for(int i = 0; i < buffersize; ++i) { @@ -205,28 +246,24 @@ void SVFilter::singlefilterout(float *smp, SVFilter::fstage &x, SVFilter::parame void SVFilter::filterout(float *smp) { - assert((buffersize % 8) == 0); - - float freqbuf[buffersize]; - - if ( freq_smoothing.apply( freqbuf, buffersize, freq ) ) - { - /* 8 sample chunks seems to work OK for AnalogFilter, so do that here too. */ - for ( int i = 0; i < buffersize; i += 8 ) - { - freq = freqbuf[i]; - computefiltercoefs(); - - for(int j = 0; j < stages + 1; ++j) - singlefilterout(smp + i, st[j], par, 8 ); - } - - freq = freqbuf[buffersize - 1]; - computefiltercoefs(); + if (needsinterpolation == INTERPOLATE_EXTREME) { + float ismp[buffersize]; + for(int i = 0; i < stages + 1; ++i) + singlefilterout(smp, st[i], par); + memcpy(ismp, smp, bufferbytes); + for(int i = 0; i < stages + 1; ++i) + singlefilterout(ismp, st[i], ipar); + for(int i = 0; i < buffersize; ++i) { + float x = i / buffersize_f; + smp[i] = ismp[i] * (1.0f - x) + smp[i] * x; + } + } else if (needsinterpolation == INTERPOLATE_NON_ZERO) { + for(int i = 0; i < stages + 1; ++i) + singlefilterout_with_par_interpolation(smp, st[i], ipar, par); + } else { + for(int i = 0; i < stages + 1; ++i) + singlefilterout(smp, st[i], par); } - else - for(int i = 0; i < stages + 1; ++i) - singlefilterout(smp, st[i], par, buffersize ); for(int i = 0; i < buffersize; ++i) smp[i] *= outgain; diff --git a/src/DSP/SVFilter.h b/src/DSP/SVFilter.h @@ -16,7 +16,6 @@ #include "../globals.h" #include "Filter.h" -#include "Value_Smoothing_Filter.h" namespace zyn { @@ -58,16 +57,19 @@ class SVFilter:public Filter } par, ipar; float *getfilteroutfortype(SVFilter::fstage &x); - void singlefilterout(float *smp, fstage &x, parameters &par, int buffersize ); - - void computefiltercoefs(void); + void singlefilterout(float *smp, fstage &x, parameters &par); + void singlefilterout_with_par_interpolation(float *smp, fstage &x, parameters &par1, parameters &par2); + void computefiltercoefs(void); int type; // The type of the filter (LPF1,HPF1,LPF2,HPF2...) int stages; // how many times the filter is applied (0->1,1->2,etc.) float freq; // Frequency given in Hz float q; // Q factor (resonance or Q factor) float gain; // the gain of the filter (if are shelf/peak) filters - Value_Smoothing_Filter freq_smoothing; + bool abovenq, //if the frequency is above the nyquist + oldabovenq; + int needsinterpolation; + bool firsttime; }; }