From e360047054117d63fb579ec9231e9dc77c99f12a Mon Sep 17 00:00:00 2001 From: David Robillard Date: Fri, 8 Aug 2008 22:45:58 +0000 Subject: Add preliminary (library side only) LV2 port of MDA (open-sourced VST plugins). git-svn-id: http://svn.drobilla.net/lad/mda-lv2@1321 a436a847-0d15-0410-975c-d299462d15a1 --- src/mdaRingMod.cpp | 189 +++++++++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 189 insertions(+) create mode 100644 src/mdaRingMod.cpp (limited to 'src/mdaRingMod.cpp') diff --git a/src/mdaRingMod.cpp b/src/mdaRingMod.cpp new file mode 100644 index 0000000..42b7eda --- /dev/null +++ b/src/mdaRingMod.cpp @@ -0,0 +1,189 @@ +#include "mdaRingMod.h" + +#include + +AudioEffect *createEffectInstance(audioMasterCallback audioMaster) +{ + return new mdaRingMod(audioMaster); +} + +mdaRingMod::mdaRingMod(audioMasterCallback audioMaster) : AudioEffectX(audioMaster, 1, 3) // 1 program, 3 parameter only +{ + fParam1 = (float)0.0625; //1kHz + fParam2 = (float)0.0; + fParam3 = (float)0.0; + fPhi = 0.0; + twoPi = (float)6.2831853; + fdPhi = (float)(twoPi * 100.0 * (fParam2 + (160.0 * fParam1))/ getSampleRate()); + ffb = 0.f; + fprev = 0.f; + + setNumInputs(2); // stereo in + setNumOutputs(2); // stereo out + setUniqueID("mdaR"); // identify + DECLARE_LVZ_DEPRECATED(canMono) (); + canProcessReplacing(); // supports both accumulating and replacing output + strcpy(programName, "Ring Modulator"); // default program name +} + +mdaRingMod::~mdaRingMod() +{ + // nothing to do here +} + +bool mdaRingMod::getProductString(char* text) { strcpy(text, "mda RingMod"); return true; } +bool mdaRingMod::getVendorString(char* text) { strcpy(text, "mda"); return true; } +bool mdaRingMod::getEffectName(char* name) { strcpy(name, "RingMod"); return true; } + +void mdaRingMod::setProgramName(char *name) +{ + strcpy(programName, name); +} + +void mdaRingMod::getProgramName(char *name) +{ + strcpy(name, programName); +} + +void mdaRingMod::setParameter(LvzInt32 index, float value) +{ + switch(index) + { + case 0: fParam1 = value; break; + case 1: fParam2 = value; break; + case 2: fParam3 = value; break; + } + fdPhi = (float) (twoPi * 100.0 * (fParam2 + (160.0 * fParam1))/ getSampleRate()); + ffb = 0.95f * fParam3; +} + +float mdaRingMod::getParameter(LvzInt32 index) +{ + float v=0; + + switch(index) + { + case 0: v = fParam1; break; + case 1: v = fParam2; break; + case 2: v = fParam3; break; + } + return v; +} + +void mdaRingMod::getParameterName(LvzInt32 index, char *label) +{ + switch(index) + { + case 0: strcpy(label, "Freq"); break; + case 1: strcpy(label, "Fine"); break; + case 2: strcpy(label, "Feedback"); break; + } +} + +#include +void long2string(long value, char *string) { sprintf(string, "%ld", value); } + +void mdaRingMod::getParameterDisplay(LvzInt32 index, char *text) +{ + switch(index) + { + case 0: long2string((long)(100. * floor(160. * fParam1)), text); break; + case 1: long2string((long)(100. * fParam2), text); break; + case 2: long2string((long)(100. * fParam3), text); break; + } + +} + +void mdaRingMod::getParameterLabel(LvzInt32 index, char *label) +{ + switch(index) + { + case 0: strcpy(label, "Hz"); break; + case 1: strcpy(label, "Hz"); break; + case 2: strcpy(label, "%"); break; + } +} + +//-------------------------------------------------------------------------------- +// process + +void mdaRingMod::process(float **inputs, float **outputs, LvzInt32 sampleFrames) +{ + float *in1 = inputs[0]; + float *in2 = inputs[1]; + float *out1 = outputs[0]; + float *out2 = outputs[1]; + float a, b, c, d, g; // use registers in sample loops! + float p, dp, tp = twoPi, fb, fp, fp2; + + p = fPhi; + dp = fdPhi; + fb = ffb; + fp = fprev; + + --in1; // pre-decrement so we can use pre-increment in the loop + --in2; // this is because pre-increment is fast on power pc + --out1; + --out2; + while(--sampleFrames >= 0) + { + a = *++in1; // try to do load operations first... + b = *++in2; + + g = (float)sin(p); //instantaneous gain + p = (float)fmod( p + dp, tp ); //oscillator phase + + c = out1[1]; // get output, as we need to accumulate + d = out2[1]; + + fp = (fb * fp + a) * g; + fp2 = (fb * fp + b) * g; + + c += fp; // accumulate to output buss + d += fp2; + + *++out1 = c; // ...and store operations at the end, + *++out2 = d; // as this uses the cache efficiently. + } + fPhi = p; + fprev = fp; +} + +void mdaRingMod::processReplacing(float **inputs, float **outputs, LvzInt32 sampleFrames) +{ + float *in1 = inputs[0]; + float *in2 = inputs[1]; + float *out1 = outputs[0]; + float *out2 = outputs[1]; + float a, b, c, d, g; + float p, dp, tp = twoPi, fb, fp, fp2; + + p = fPhi; + dp = fdPhi; + fb = ffb; + fp = fprev; + + --in1; + --in2; + --out1; + --out2; + while(--sampleFrames >= 0) + { + a = *++in1; + b = *++in2; + + g = (float)sin(p); + p = (float)fmod( p + dp, tp ); + + fp = (fb * fp + a) * g; + fp2 = (fb * fp + b) * g; + + c = fp; + d = fp2; + + *++out1 = c; + *++out2 = d; + } + fPhi = p; + fprev = fp; +} -- cgit v1.2.1