zynaddsubfx

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

commit 8918867e6e15afa19cf5ac62399b3d3579c165d8
parent 865f3a8888af378ef0487898d44dac9c92ee218c
Author: fundamental <[email protected]>
Date:   Mon, 19 Aug 2013 00:03:27 -0400

Adsynth: Speedup Linear Interpolator

Replace unneeded floating point ops with fixed point ones
resulting in a ~15% speedup observed in ADnoteTest

Diffstat:
Msrc/Synth/ADnote.cpp | 39+++++++++++++++++++++++++--------------
1 file changed, 25 insertions(+), 14 deletions(-)

diff --git a/src/Synth/ADnote.cpp b/src/Synth/ADnote.cpp @@ -23,6 +23,8 @@ #include <cstdlib> #include <cstdio> #include <cstring> +#include <cassert> +#include <stdint.h> #include "../globals.h" #include "../Misc/Util.h" @@ -1102,30 +1104,39 @@ inline void ADnote::fadein(float *smps) const /* * Computes the Oscillator (Without Modulation) - LinearInterpolation */ + +/* As the code here is a bit odd due to optimization, here is what happens + * First the current possition and frequency are retrieved from the running + * state. These are broken up into high and low portions to indicate how many + * samples are skipped in one step and how many fractional samples are skipped. + * Outside of this method the fractional samples are just handled with floating + * point code, but that's a bit slower than it needs to be. In this code the low + * portions are known to exist between 0.0 and 1.0 and it is known that they are + * stored in single precision floating point IEEE numbers. This implies that + * a maximum of 24 bits are significant. The below code does your standard + * linear interpolation that you'll see throughout this codebase, but by + * sticking to integers for tracking the overflow of the low portion, around 15% + * of the execution time was shaved off in the ADnote test. + */ inline void ADnote::ComputeVoiceOscillator_LinearInterpolation(int nvoice) { - int i, poshi; - float poslo; - for(int k = 0; k < unison_size[nvoice]; ++k) { - poshi = oscposhi[nvoice][k]; - poslo = oscposlo[nvoice][k]; + int poshi = oscposhi[nvoice][k]; + int poslo = oscposlo[nvoice][k] * (1<<24); int freqhi = oscfreqhi[nvoice][k]; - float freqlo = oscfreqlo[nvoice][k]; + int freqlo = oscfreqlo[nvoice][k] * (1<<24); float *smps = NoteVoicePar[nvoice].OscilSmp; float *tw = tmpwave_unison[k]; - for(i = 0; i < synth->buffersize; ++i) { - tw[i] = smps[poshi] * (1.0f - poslo) + smps[poshi + 1] * poslo; + assert(oscfreqlo[nvoice][k] < 1.0f); + for(int i = 0; i < synth->buffersize; ++i) { + tw[i] = (smps[poshi] * ((1<<24) - poslo) + smps[poshi + 1] * poslo)/(1.0f*(1<<24)); poslo += freqlo; - if(poslo >= 1.0f) { - poslo -= 1.0f; - poshi++; - } - poshi += freqhi; + poshi += freqhi + (poslo>>24); + poslo &= 0xffffff; poshi &= synth->oscilsize - 1; } oscposhi[nvoice][k] = poshi; - oscposlo[nvoice][k] = poslo; + oscposlo[nvoice][k] = poslo/(1.0f*(1<<24)); } }