zynaddsubfx

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

commit 107267b9a4761bc13faa37a8dfb3cc50c2c5d20e
parent dea5fe9666952d3596ef242df0231fbc7a1280b2
Author: fundamental <[email protected]>
Date:   Wed, 23 Dec 2009 21:29:16 -0500

Echo: reworking delay resize

- Still some issues (crackles from improperly distribution workload)

Diffstat:
Msrc/Effects/Echo.cpp | 120+++++++++++++++++++++++++++++++++++++++++--------------------------------------
Msrc/Effects/Echo.h | 13++++++-------
Msrc/Samples/Sample.cpp | 88+++++++++++++++++++++++++++++++++++++++----------------------------------------
Msrc/Samples/Sample.h | 3---
4 files changed, 111 insertions(+), 113 deletions(-)

diff --git a/src/Effects/Echo.cpp b/src/Effects/Echo.cpp @@ -32,13 +32,11 @@ Echo::Echo(const int &insertion_, :Effect(insertion_, efxoutl_, efxoutr_, NULL, 0), Pvolume(50), Ppanning(64), //Pdelay(60), Plrdelay(100), Plrcross(100), Pfb(40), Phidamp(60), - dl(0), dr(0), lrdelay(0), delaySample(1), old(0.0), - itr(0), - maxDelay(SAMPLE_RATE * (1.5 + (pow(2.0, 9) - 1.0) /1000.0) + 1) + delayTime(1), dl(1), dr(1), lrdelay(0), delay(128000), old(0.0), + pos(0) + { pthread_mutex_init(&mutex, NULL); - delaySample.l() = Sample(maxDelay, 0.0); - delaySample.r() = Sample(maxDelay, 0.0); initdelays(); setpreset(Ppreset); } @@ -50,8 +48,8 @@ Echo::~Echo() {} */ void Echo::cleanup() { - delaySample.l().clear(); - delaySample.r().clear(); + delay.l().clear(); + delay.r().clear(); old = Stereo<REALTYPE>(0.0); } @@ -61,47 +59,35 @@ void Echo::cleanup() */ void Echo::initdelays() { + int dl = (int)(delayCtl.getiVal() * SAMPLE_RATE - lrdelay); + if(dl < 1) + dl = 1; + + int dr = (int)(delayCtl.getiVal() * SAMPLE_RATE + lrdelay); + if(dr < 1) + dr = 1; + + if(dl == delay.l().size() && dr == delay.r().size()) + return; //no need to do anything here + + //resetting the loop to a known state pthread_mutex_lock(&mutex); - int ndl = (int)(delay.getiVal() * SAMPLE_RATE - lrdelay); - if(ndl < 0) - ndl = dl = 0; - - int ndr = (int)(delay.getiVal() * SAMPLE_RATE + lrdelay); - if(ndr < 0) - ndr = dr = 0; - - if(ndl != dl) { //lets rewite history - Sample rewrite(ndl); - int j = 0; - //this is where real iterators would be useful - for(int i=(itr>dl?itr-dl:itr+maxDelay-dl); i%maxDelay!=itr; ++i) - rewrite[j++] = delaySample.l()[i]; - rewrite.resize(ndl); - j = 0; - for(int i=(itr>ndl?itr-ndl:itr+maxDelay-ndl); i%maxDelay!=itr; ++i) - delaySample.l()[i] = rewrite[j++]; - } - - if(ndr != dr) { //lets rewite history - Sample rewrite(ndr); - int j = 0; - //this is where real iterators would be useful - for(int i=(itr>dr?itr-dr:itr+maxDelay-dr); i%maxDelay!=itr; ++i) - rewrite[j++] = delaySample.r()[i]; - rewrite.resize(ndr); - j = 0; - for(int i=(itr>ndr?itr-ndr:itr+maxDelay-ndr); i%maxDelay!=itr; ++i) - delaySample.r()[i] = rewrite[j++]; - } - dl = ndl; - dr = ndr; - - - //cout << "dl: " << dl << endl; - //cout << "dr: " << dr << endl; - //cout << "max: " << maxDelay << endl; + Sample tmpl(delay.l().size()); + for(int i = 0; i < delay.l().size(); ++i) + tmpl[i] = delay.l()[pos.l()+i]; + + Sample tmpr(delay.r().size()); + for(int i = 0; i < delay.r().size(); ++i) + tmpr[i] = delay.r()[pos.r()+i]; + + tmpl.resize(dl); + tmpr.resize(dr); + + delay.l() = tmpl; + delay.r() = tmpr; + pos.l() = 0; + pos.r() = 0; pthread_mutex_unlock(&mutex); - } void Echo::out(const Stereo<Sample> &input) @@ -110,28 +96,46 @@ void Echo::out(const Stereo<Sample> &input) pthread_mutex_lock(&mutex); for(int i = 0; i < input.l().size(); ++i) { - //get past samples (maxDelay is used due to implementaiton of Sample) - dl; - "foo"; - itr; - ldl = delaySample.l()[itr-dl+maxDelay]; - rdl = delaySample.r()[itr-dr+maxDelay]; + //get past samples (delay .size() is used due to implementaiton of Sample) + //should get fixed so negative indexes are properly referenced + //or iterators should work + ldl = delay.l()[pos.l()]; + rdl = delay.r()[pos.r()]; ldl = ldl * (1.0 - lrcross) + rdl * lrcross; rdl = rdl * (1.0 - lrcross) + ldl * lrcross; efxoutl[i] = ldl * 2.0; efxoutr[i] = rdl * 2.0; + if(rdl != rdl) { + cout << "hi" << hidamp << endl; + cout << "l" << ldl << endl; + cout << "r" << rdl << endl; + cout << "ol " << efxoutl[i] << endl; + cout << "or " << efxoutr[i] << endl; + cout << "cross " << lrcross << endl; + cout << pos.l() << endl; + cout << pos.r() << endl; + cout << input.l()[0] << endl; + cout << input.r()[0] << endl; + for(int i=pos.l()-SOUND_BUFFER_SIZE; i<pos.l(); ++i) + cout << i << ": " << delay.l()[i] << endl; + exit(1); + } ldl = input.l()[i] * panning - ldl * fb; rdl = input.r()[i] * (1.0 - panning) - rdl * fb; //LowPass Filter - old.l() = delaySample.l()[itr] = ldl * hidamp + old.l() * (1.0 - hidamp); - old.r() = delaySample.r()[itr] = rdl * hidamp + old.r() * (1.0 - hidamp); - ++itr; + old.l() = delay.l()[pos.l()] + = ldl * hidamp + old.l() * (1.0 - hidamp); + old.r() = delay.r()[pos.r()] + = rdl * hidamp + old.r() * (1.0 - hidamp); + ++pos.l(); + ++pos.r(); } - itr %= maxDelay; + pos.l() %= delay.l().size(); + pos.r() %= delay.r().size(); pthread_mutex_unlock(&mutex); } @@ -162,7 +166,7 @@ void Echo::setpanning(unsigned char Ppanning) void Echo::setdelay(unsigned char Pdelay) { - delay.setmVal(Pdelay); + delayCtl.setmVal(Pdelay); //this->Pdelay=Pdelay; //delay=1+(int)(Pdelay/127.0*SAMPLE_RATE*1.5);//0 .. 1.5 sec initdelays(); @@ -272,7 +276,7 @@ unsigned char Echo::getpar(int npar) const return Ppanning; break; case 2: - return delay.getmVal(); + return delayCtl.getmVal(); break; case 3: return Plrdelay; diff --git a/src/Effects/Echo.h b/src/Effects/Echo.h @@ -103,7 +103,7 @@ class Echo:public Effect //Parameters char Pvolume; /**<#1 Volume or Dry/Wetness*/ char Ppanning; /**<#2 Panning*/ - DelayCtl delay; /**<#3 Delay of the Echo*/ + DelayCtl delayCtl; /**<#3 Delay of the Echo*/ char Plrdelay; /**<#4 L/R delay difference*/ char Plrcross; /**<#5 L/R Mixing*/ char Pfb; /**<#6Feedback*/ @@ -120,17 +120,16 @@ class Echo:public Effect //Real Parameters REALTYPE panning, lrcross, fb, hidamp; //needs better names //Left/Right delay lengths + Stereo<int> delayTime; int dl, dr, lrdelay; -// int ndl, ndr; //used for shifting delay amounts - void initdelays(); - Stereo<Sample> delaySample; + //2 channel ring buffer + Stereo<Sample> delay; Stereo<REALTYPE> old; - int itr; - //int kl, kr; - int maxDelay; + //position of reading/writing from delaysample + Stereo<int> pos; mutable pthread_mutex_t mutex; }; diff --git a/src/Samples/Sample.cpp b/src/Samples/Sample.cpp @@ -20,8 +20,11 @@ */ #include <cmath> #include <cstring>//for memcpy/memset +#include <iostream> #include "Sample.h" +using namespace std; + #warning TODO Think about renaming Sample to Frame /**\TODO start using pointer math here as these will be Frequency called * functions throughout the code*/ @@ -103,52 +106,21 @@ bool Sample::operator==(const Sample &smp) const * @param xb X of point b * @return estimated Y of test point */ -float linearEstimate(float ya, float yb, float xt, float xa = 0.0, float xb =1.0) +float linearEstimate(float ya, float yb, float xt, int xa = 0, int xb = 1) { -#warning TODO this could be done with a good bit less computation - //Lets make this simple by normalizing the x axis + if(xa == xb) + return ya; //Normalize point a - xb -= xa; - xt -= xa; - xa -= xa; + //xb -= xa; + //xt -= xa; //Normalize point b - xt /= xb; - xb /= xb; + //xt /= xb; //Now xa=0 xb=1 0<=xt<=1 //simpily use y=mx+b - return (yb-ya) * xt + ya; -} - - -void Sample::resample(const unsigned int rate, const unsigned int nrate) -{ - //does not call resize, as I have a feeling that that could lose precision - //(have not tested, so feel free to prove me wrong) - if(rate == nrate) - return; //no resampling here - else {//resampling occurs here - int itr = 0; - float ratio = (nrate * 1.0) / (rate * 1.0); - - int nBufferSize = (int)bufferSize * ratio; - float *nBuffer = new float[nBufferSize]; - - //addition is done to avoid 0 edge case - for(int i = 0; i < nBufferSize; ++i) - nBuffer[i] = linearEstimate(buffer[(int)floor(i/ratio)], - buffer[(int)ceil((i+1)/ratio)], - i, - floor(i/ratio), - ceil((i+1)/ratio)); - - //put the new data in - delete buffer; - buffer = nBuffer; - bufferSize = nBufferSize; - } + return (yb-ya) * (xt-xa)/(xb-xa) + ya; } void Sample::resize(unsigned int nsize) @@ -162,16 +134,42 @@ void Sample::resize(unsigned int nsize) int nBufferSize = nsize; float *nBuffer = new float[nBufferSize]; + //take care of edge cases + *nBuffer = *buffer; + *(nBuffer+nBufferSize-1) = *(buffer+bufferSize-1); + //addition is done to avoid 0 edge case - for(int i = 0; i < nBufferSize; ++i) - nBuffer[i] = linearEstimate(buffer[(int)floor(i/ratio)], - buffer[(int)ceil((i+1)/ratio)], - i, - floor(i/ratio), - ceil((i+1)/ratio)); + for(int i = 1; i < nBufferSize - 1; ++i) + { + float left = floor(i/ratio); + float right = ceil((i+1)/ratio); + float test = i/ratio; + if(left > bufferSize - 1) + left = bufferSize - 1; + if(right > bufferSize - 1) + right = bufferSize - 1; + if(left > test) + test = left; + nBuffer[i] = linearEstimate(buffer[(int)left], + buffer[(int)right], + test, (int)left, (int)right); + if(nBuffer[i] != nBuffer[i]) + { + cout << "ERROR: " << nBuffer[i] << endl; + cout << "ERROR2: " << buffer[(int)left] << endl; + cout << "ERROR33: " << buffer[(int)right] << endl; + cout << "ERROR444: " << right << endl; + cout << "ERROR5555: " << left << endl; + cout << "ERROR66666: " << i << endl; + cout << "ERROR7777: " << i/ratio << endl; + cout << "ERROR888: " << test << endl; + exit(1); + } + + } //put the new data in - delete buffer; + delete[] buffer; buffer = nBuffer; bufferSize = nBufferSize; } diff --git a/src/Samples/Sample.h b/src/Samples/Sample.h @@ -57,9 +57,6 @@ class Sample * sparingly*/ const REALTYPE *c_buf() const {return buffer;} - /**Change the sampling rate of the Sample*/ - void resample(const unsigned int rate, const unsigned int nrate); - /**Change the size of the sample*/ void resize(unsigned int nsize);