zynaddsubfx

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

commit b9038dfef78f01cb7dfefbd560e3b43426b4888e
parent c4e8d94c7c6a1675d26ed6220c74e84888ef9e6b
Author: Friedolino <[email protected]>
Date:   Mon, 11 Mar 2019 00:16:41 +0100

windowed sinc 2x AA filter sinc

Diffstat:
Msrc/Synth/ADnote.cpp | 65+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++--
1 file changed, 63 insertions(+), 2 deletions(-)

diff --git a/src/Synth/ADnote.cpp b/src/Synth/ADnote.cpp @@ -29,7 +29,6 @@ #include "ADnote.h" namespace zyn { - ADnote::ADnote(ADnoteParameters *pars_, SynthParams &spars, WatchManager *wm, const char *prefix) :SynthNote(spars), watch_be4_add(wm, prefix, "noteout/be4_mix"), watch_after_add(wm,prefix,"noteout/after_mix"), @@ -139,6 +138,7 @@ void ADnote::setupVoice(int nvoice) oscfreqhi[nvoice] = memory.valloc<int>(unison); oscfreqlo[nvoice] = memory.valloc<float>(unison); + oscfreqhiFM[nvoice] = memory.valloc<unsigned int>(unison); oscfreqloFM[nvoice] = memory.valloc<float>(unison); oscposhi[nvoice] = memory.valloc<int>(unison); @@ -1033,6 +1033,7 @@ void ADnote::setfreq(int nvoice, float in_freq) F2I(speed, oscfreqhi[nvoice][k]); oscfreqlo[nvoice][k] = speed - floor(speed); + } } @@ -1241,20 +1242,80 @@ inline void ADnote::fadein(float *smps) const */ inline void ADnote::ComputeVoiceOscillator_LinearInterpolation(int nvoice) { + + // windowed sinc kernel factor Fs/4, rejection 80dB + const float_t kernel[21] = { + 0.00017017607982446043, + 0.0009262403856891687, + 0, + -0.008116326243706371, + -0.02476577025825532, + -0.03386903752666308, + 0, + 0.10709721649843373, + 0.27452737291792295, + 0.4339834705604548, + 0.5000933151725993, + 0.4339834705604548, + 0.27452737291792295, + 0.10709721649843373, + 0, + -0.03386903752666308, + -0.02476577025825532, + -0.008116326243706371, + 0, + 0.0009262403856891687, + 0.00017017607982446043 + }; + + for(int k = 0; k < unison_size[nvoice]; ++k) { int poshi = oscposhi[nvoice][k]; int poslo = oscposlo[nvoice][k] * (1<<24); int freqhi = oscfreqhi[nvoice][k]; int freqlo = oscfreqlo[nvoice][k] * (1<<24); + int ovsmpfreqhi = oscfreqhi[nvoice][k] / 2; + int ovsmpfreqlo = (oscfreqlo[nvoice][k] / 2) * (1<<24); + + int ovsmpposlo; + int ovsmpposhi; + int uflow; float *smps = NoteVoicePar[nvoice].OscilSmp; float *tw = tmpwave_unison[k]; assert(oscfreqlo[nvoice][k] < 1.0f); + float out = 0; + for(int i = 0; i < synth.buffersize; ++i) { - tw[i] = (smps[poshi] * ((1<<24) - poslo) + smps[poshi + 1] * poslo)/(1.0f*(1<<24)); + + // linear interpolated sampling of waveform for low frequencies + //tw[i] = (smps[poshi] * ((1<<24) - poslo) + smps[poshi + 1] * poslo)/(1.0f*(1<<24)); + + ovsmpposlo = poslo - 10 * ovsmpfreqlo; + uflow = ovsmpposlo>>24; + ovsmpposhi = poshi - 10 * ovsmpfreqhi - ((0x00 - uflow) & 0xff); + ovsmpposlo &= 0xffffff; + ovsmpposhi &= synth.oscilsize - 1; + out = 0; + for (int l = 0; l<21; l++) { + out += kernel[l] * ( + smps[ovsmpposhi] * ((1<<24) - ovsmpposlo) + + smps[ovsmpposhi + 1] * ovsmpposlo)/(1.0f*(1<<24)); + // advance to next sample + ovsmpposlo += ovsmpfreqlo; + ovsmpposhi += ovsmpfreqhi + (ovsmpposlo>>24); // add the 24-bit overflow + ovsmpposlo &= 0xffffff; + ovsmpposhi &= synth.oscilsize - 1; + + } + + // advance to next sample poslo += freqlo; poshi += freqhi + (poslo>>24); poslo &= 0xffffff; poshi &= synth.oscilsize - 1; + + tw[i] = out; + } oscposhi[nvoice][k] = poshi; oscposlo[nvoice][k] = poslo/(1.0f*(1<<24));