zynaddsubfx

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

commit aae63c815150bb7cd4535991b42d669404879d24
parent 613658446bf6744a64b4436a744bc4c25af56a3c
Author: friedolino78 <[email protected]>
Date:   Sun, 16 Feb 2025 04:11:34 +0100

New chorus mode: "Roland Ensemble" (#274)


Diffstat:
Mdoc/effects.txt | 5+++++
Msrc/Effects/Chorus.cpp | 160+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++--------------------
Msrc/Effects/Chorus.h | 37+++++++++++++++++++++++++++++++++----
Msrc/Effects/EffectLFO.cpp | 42++++++++++++++++++++++++++----------------
Msrc/Effects/EffectLFO.h | 9++++++---
Msrc/UI/EffUI.fl | 29+++++++++++++++++++++++------
6 files changed, 212 insertions(+), 70 deletions(-)

diff --git a/doc/effects.txt b/doc/effects.txt @@ -137,6 +137,10 @@ with very short LFO delay and little LFO depth. You can imagine a flanger as two copies of a sound playing at almost the same time. This leads to interference, which can be clearly heard. It is popular to apply flangers to guitars, giving them more "character". +While in standard CHORUS and FLANGER mode there is only one additional voice that +can be blended with the original one using the amount knob, the DUAL and TRIPLE +mode adds two/three voices with their LFOs each being phase shifted by 180 / 120 degrees. +Those have been used by famous roland string synthesizers. Usage ^^^^^ @@ -153,6 +157,7 @@ the *D/W* knob, but we have not applied panning and subtraction yet. * Next, the signal can be negated. If the *Subtract* checkbox is activated, the amplitude is multiplied by -1. * Finally, *Pan* lets you apply panning. +* *Mode* selector lets you choose between Chorus, Flanger, Dual- and Triple Chorus. Distortion ~~~~~~~~~~ diff --git a/src/Effects/Chorus.cpp b/src/Effects/Chorus.cpp @@ -21,6 +21,10 @@ using namespace std; namespace zyn { +constexpr float PHASE_120 = 0.33333333f; +constexpr float PHASE_180 = 0.5f; +constexpr float PHASE_240 = 0.66666666f; + #define rObject Chorus #define rBegin [](const char *msg, rtosc::RtData &d) { #define rEnd } @@ -28,7 +32,8 @@ namespace zyn { rtosc::Ports Chorus::ports = { {"preset::i", rProp(parameter) rOptions(Chorus1, Chorus2, Chorus3, Celeste1, Celeste2, - Flange1, Flange2, Flange3, Flange4, Flange5) + Flange1, Flange2, Flange3, Flange4, Flange5, + Triple, Dual) rProp(alias) rDefault(0) rDoc("Instrument Presets"), 0, @@ -63,8 +68,8 @@ rtosc::Ports Chorus::ports = { rPresets(64, 64, 90, 90, 31, 62, 109, 54, 97, 17), "Feedback"), rEffPar(Plrcross, 9, rShort("l/r"), rPresets(119, 19, 127, 127, 127), rDefault(0), "Left/Right Crossover"), - rEffParTF(Pflangemode, 10, rShort("flange"), rDefault(false), - "Flange Mode"), + rEffParOpt(Pflangemode, 10, rShort("mode"), rDefault(CHORUS),rOptions(CHORUS_MODES), + "Chorus Mode"), rEffParTF(Poutsub, 11, rShort("sub"), rPreset(4, true), rPreset(7, true), rPreset(9, true), rDefault(false), "Output Subtraction"), @@ -84,8 +89,8 @@ Chorus::Chorus(EffectParams pars) setpreset(Ppreset); changepar(1, 64); lfo.effectlfoout(&lfol, &lfor); - dl2 = getdelay(lfol); - dr2 = getdelay(lfor); + dlNew = getdelay(lfol); + drNew = getdelay(lfor); cleanup(); } @@ -99,7 +104,7 @@ Chorus::~Chorus() float Chorus::getdelay(float xlfo) { float result = - (Pflangemode) ? 0 : (delay + xlfo * depth) * samplerate_f; + (Pflangemode==FLANGE) ? 0 : (delay + xlfo * depth) * samplerate_f; //check if delay is too big (caused by bad setdelay() and setdepth() if((result + 0.5f) >= maxdelay) { @@ -112,15 +117,57 @@ float Chorus::getdelay(float xlfo) return result; } +// sample + +inline float Chorus::getSample(float* delayline, float mdel, int dk) +{ + float samplePos = dk - mdel + float(maxdelay * 2); //where should I get the sample from + return cinterpolate(delayline, maxdelay, samplePos); + +} + //Apply the effect void Chorus::out(const Stereo<float *> &input) { - dl1 = dl2; - dr1 = dr2; + // store old delay value for linear interpolation + dlHist = dlNew; + drHist = drNew; + // calculate new lfo values lfo.effectlfoout(&lfol, &lfor); + // calculate new delay values + dlNew = getdelay(lfol); + drNew = getdelay(lfor); + float fbComp = fb; + if (Pflangemode == DUAL) // ensemble mode + { + // same for second member for ensemble mode with 180° phase offset + dlHist2 = dlNew2; + drHist2 = drNew2; + lfo.effectlfoout(&lfol, &lfor, PHASE_180); + dlNew2 = getdelay(lfol); + drNew2 = getdelay(lfor); + fbComp /= 2.0f; + } - dl2 = getdelay(lfol); - dr2 = getdelay(lfor); + if (Pflangemode == TRIPLE) // ensemble mode + { + // same for second member for ensemble mode with 120° phase offset + dlHist2 = dlNew2; + drHist2 = drNew2; + lfo.effectlfoout(&lfol, &lfor, PHASE_120); + dlNew2 = getdelay(lfol); + drNew2 = getdelay(lfor); + + // same for third member for ensemble mode with 240° phase offset + dlHist3 = dlNew3; + drHist3 = drNew3; + lfo.effectlfoout(&lfol, &lfor, PHASE_240); + dlNew3 = getdelay(lfol); + drNew3 = getdelay(lfor); + // reduce amplitude to match single phase modes + // 0.85 * fbComp / 3 + fbComp /= 3.53f; + } for(int i = 0; i < buffersize; ++i) { float inL = input.l[i]; @@ -131,41 +178,67 @@ void Chorus::out(const Stereo<float *> &input) inR = tmpc.r * (1.0f - lrcross) + tmpc.l * lrcross; //Left channel - - //compute the delay in samples using linear interpolation between the lfo delays - float mdel = - (dl1 * (buffersize - i) + dl2 * i) / buffersize_f; + // reset output accumulator + output = 0.0f; + // increase delay line writing position and handle turnaround if(++dlk >= maxdelay) dlk = 0; - float tmp = dlk - mdel + maxdelay * 2.0f; //where should I get the sample from - - dlhi = (int) tmp; - dlhi %= maxdelay; - - float dlhi2 = (dlhi - 1 + maxdelay) % maxdelay; - float dllo = 1.0f + floorf(tmp) - tmp; - efxoutl[i] = cinterpolate(delaySample.l, maxdelay, dlhi2) * dllo - + cinterpolate(delaySample.l, maxdelay, - dlhi) * (1.0f - dllo); - delaySample.l[dlk] = inL + efxoutl[i] * fb; + // linear interpolate from old to new value over length of the buffer + float dl = (dlHist * (buffersize - i) + dlNew * i) / buffersize_f; + // get sample with that delay from delay line and add to output accumulator + output += getSample(delaySample.l, dl, dlk); + switch (Pflangemode) { + case DUAL: + // calculate and apply delay for second ensemble member + dl = (dlHist2 * (buffersize - i) + dlNew2 * i) / buffersize_f; + output += getSample(delaySample.l, dl, dlk); + break; + case TRIPLE: + // calculate and apply delay for second ensemble member + dl = (dlHist2 * (buffersize - i) + dlNew2 * i) / buffersize_f; + output += getSample(delaySample.l, dl, dlk); + // same for third ensemble member + dl = (dlHist3 * (buffersize - i) + dlNew3 * i) / buffersize_f; + output += getSample(delaySample.l, dl, dlk); + break; + default: + // nothing to do for standard chorus + break; + } + // store current input + feedback to delay line at writing position + delaySample.l[dlk] = inL + output * fbComp; + // write output to output interface + efxoutl[i] = output; //Right channel - - //compute the delay in samples using linear interpolation between the lfo delays - mdel = (dr1 * (buffersize - i) + dr2 * i) / buffersize_f; + output = 0.0f; if(++drk >= maxdelay) drk = 0; - tmp = drk * 1.0f - mdel + maxdelay * 2.0f; //where should I get the sample from - - dlhi = (int) tmp; - dlhi %= maxdelay; + float dr = (drHist * (buffersize - i) + drNew * i) / buffersize_f; + output += getSample(delaySample.r, dr, drk); + switch (Pflangemode) { + case DUAL: + // calculate and apply delay for second ensemble member + dr = (drHist2 * (buffersize - i) + drNew2 * i) / buffersize_f; + output += getSample(delaySample.r, dr, drk); + break; + case TRIPLE: + // calculate and apply delay for second ensemble member + dr = (drHist2 * (buffersize - i) + drNew2 * i) / buffersize_f; + output += getSample(delaySample.r, dr, drk); + // same for third ensemble member + dr = (drHist3 * (buffersize - i) + drNew3 * i) / buffersize_f; + output += getSample(delaySample.r, dr, drk); + // reduce amplitude to match single phase modes + output *= 0.85f; + break; + default: + // nothing to do for standard chorus + break; + } - dlhi2 = (dlhi - 1 + maxdelay) % maxdelay; - dllo = 1.0f + floorf(tmp) - tmp; - efxoutr[i] = cinterpolate(delaySample.r, maxdelay, dlhi2) * dllo - + cinterpolate(delaySample.r, maxdelay, - dlhi) * (1.0f - dllo); - delaySample.r[dlk] = inR + efxoutr[i] * fb; + delaySample.r[drk] = inR + output * fbComp; + efxoutr[i] = output; } if(Poutsub) @@ -216,7 +289,7 @@ void Chorus::setvolume(unsigned char _Pvolume) unsigned char Chorus::getpresetpar(unsigned char npreset, unsigned int npar) { #define PRESET_SIZE 12 -#define NUM_PRESETS 10 +#define NUM_PRESETS 12 static const unsigned char presets[NUM_PRESETS][PRESET_SIZE] = { //Chorus1 {64, 64, 50, 0, 0, 90, 40, 85, 64, 119, 0, 0}, @@ -237,7 +310,11 @@ unsigned char Chorus::getpresetpar(unsigned char npreset, unsigned int npar) //Flange4 {64, 64, 40, 0, 1, 62, 12, 19, 97, 0, 0, 0}, //Flange5 - {64, 64, 55, 105, 0, 24, 39, 19, 17, 0, 0, 1} + {64, 64, 55, 105, 0, 24, 39, 19, 17, 0, 0, 1}, + //Ensemble + {127, 64, 68, 25, 1, 24, 35, 55, 64, 0, TRIPLE, 0}, + //Dual + {127, 64, 55, 25, 1, 24, 32, 55, 80, 0, DUAL, 0} }; if(npreset < NUM_PRESETS && npar < PRESET_SIZE) { return presets[npreset][npar]; @@ -292,7 +369,8 @@ void Chorus::changepar(int npar, unsigned char value) setlrcross(value); break; case 10: - Pflangemode = (value > 1) ? 1 : value; + lfo.updateparams(); + Pflangemode = (value > 3) ? 3 : value; break; case 11: Poutsub = (value > 1) ? 1 : value; diff --git a/src/Effects/Chorus.h b/src/Effects/Chorus.h @@ -17,10 +17,30 @@ #include "EffectLFO.h" #include "../Misc/Stereo.h" -#define MAX_CHORUS_DELAY 250.0f //ms + + + namespace zyn { +#define MAX_CHORUS_DELAY 250.0f //ms + +// Chorus modes +// CHORUS default chorus mode +// FLANGE flanger mode (very short delays) +// TRIPLE 120° triple phase chorus +// DUAL 180° dual phase chorus + +#define CHORUS_MODES \ + CHORUS,\ + FLANGE,\ + TRIPLE,\ + DUAL + +enum { + CHORUS_MODES +}; + /**Chorus and Flange effects*/ class Chorus final:public Effect { @@ -73,12 +93,14 @@ class Chorus final:public Effect static rtosc::Ports ports; private: + inline float getSample(float* delayline, float mdel, int dk); + //Chorus Parameters unsigned char Pvolume; unsigned char Pdepth; //the depth of the Chorus(ms) unsigned char Pdelay; //the delay (ms) unsigned char Pfb; //feedback - unsigned char Pflangemode; //how the LFO is scaled, to result chorus or flange + unsigned char Pflangemode; //mode as described above in CHORUS_MODES unsigned char Poutsub; //if I wish to subtract the output instead of the adding it EffectLFO lfo; //lfo-ul chorus @@ -91,11 +113,18 @@ class Chorus final:public Effect //Internal Values float depth, delay, fb; - float dl1, dl2, dr1, dr2, lfol, lfor; + float dlHist, dlNew, lfol; + float drHist, drNew, lfor; + float dlHist2, dlNew2; + float drHist2, drNew2; + float dlHist3, dlNew3; + float drHist3, drNew3; int maxdelay; Stereo<float *> delaySample; - int dlk, drk, dlhi; + int dlk, drk; float getdelay(float xlfo); + + float output; }; } diff --git a/src/Effects/EffectLFO.cpp b/src/Effects/EffectLFO.cpp @@ -62,6 +62,7 @@ void EffectLFO::updateparams(void) //Compute the shape of the LFO float EffectLFO::getlfoshape(float x) { + x = fmodf(x, 1.0f); float out; switch(lfotype) { case 1: //EffectLFO_TRIANGLE @@ -81,31 +82,40 @@ float EffectLFO::getlfoshape(float x) } //LFO output -void EffectLFO::effectlfoout(float *outl, float *outr) +void EffectLFO::effectlfoout(float *outl, float *outr, float phaseOffset) { float out; - - out = getlfoshape(xl); + // left stereo signal + out = getlfoshape(xl+phaseOffset); if((lfotype == 0) || (lfotype == 1)) out *= (ampl1 + xl * (ampl2 - ampl1)); - xl += incx; - if(xl > 1.0f) { - xl -= 1.0f; - ampl1 = ampl2; - ampl2 = (1.0f - lfornd) + lfornd * RND; - } *outl = (out + 1.0f) * 0.5f; + // update left phase for master lfo + if(phaseOffset==0.0f) { + xl += incx; + if(xl > 1.0f) { + xl -= 1.0f; + ampl1 = ampl2; + ampl2 = (1.0f - lfornd) + lfornd * RND; + } + } - out = getlfoshape(xr); + // right stereo signal + out = getlfoshape(xr+phaseOffset); if((lfotype == 0) || (lfotype == 1)) out *= (ampr1 + xr * (ampr2 - ampr1)); - xr += incx; - if(xr > 1.0f) { - xr -= 1.0f; - ampr1 = ampr2; - ampr2 = (1.0f - lfornd) + lfornd * RND; - } *outr = (out + 1.0f) * 0.5f; + + // update right phase for master lfo + if(phaseOffset==0.0f) { + xr += incx; + if(xr > 1.0f) { + xr -= 1.0f; + ampr1 = ampr2; + ampr2 = (1.0f - lfornd) + lfornd * RND; + } + } + } } diff --git a/src/Effects/EffectLFO.h b/src/Effects/EffectLFO.h @@ -37,13 +37,16 @@ class EffectLFO */ ~EffectLFO(); + // if phaseOffset!=0 we don't want to update the phase but get a phased shifted output of the last one. /** - * Processes the LFO and outputs the result to the provided pointers. - * + * calculate the output of the effect LFO for two signals (stereo) * @param outl Pointer to the left output channel. * @param outr Pointer to the right output channel. + * @param phaseOffset phaseoffset + * if phaseOffset is not 0 we don't want to update the phase + * but get the phased shifted output of the last one. */ - void effectlfoout(float *outl, float *outr); + void effectlfoout(float *outl, float *outr, float phaseOffset = 0.0f); /** * Updates the LFO parameters based on the current settings. diff --git a/src/UI/EffUI.fl b/src/UI/EffUI.fl @@ -442,12 +442,6 @@ if (filterwindow!=NULL){ code0 {o->init("parameter9");} class Fl_Osc_Dial } - Fl_Check_Button {} { - label Flange - xywh {120 10 55 20} box THIN_UP_BOX down_box DOWN_BOX color 230 labelfont 1 labelsize 10 hide deactivate - code0 {o->init("parameter10");} - class Fl_Osc_Check - } Fl_Check_Button chorusp11 { label Subtract tooltip {inverts the output} xywh {185 10 70 20} box THIN_UP_BOX down_box DOWN_BOX color 51 labelsize 10 @@ -469,6 +463,29 @@ if (filterwindow!=NULL){ xywh {25 25 100 20} labelfont 1 labelsize 10 } } + Fl_Choice chorusp10 { + label {Mode} + tooltip {LFO function} xywh {270 15 80 15} box UP_BOX down_box BORDER_BOX labelfont 1 labelsize 10 align 5 textsize 8 + code0 {o->init("parameter10");} + class Fl_Osc_Choice + } { + MenuItem {} { + label CHORUS + xywh {15 15 100 20} labelfont 1 labelsize 10 + } + MenuItem {} { + label FLANGE + xywh {25 25 100 20} labelfont 1 labelsize 10 + } + MenuItem {} { + label TRIPLE + xywh {35 35 100 20} labelfont 1 labelsize 10 + } + MenuItem {} { + label DUAL + xywh {45 45 100 20} labelfont 1 labelsize 10 + } + } } } Function {make_phaser_window()} {} {