mirror of
https://github.com/airwindows/airwindows.git
synced 2026-05-15 14:16:00 -06:00
628 lines
27 KiB
C++
Executable file
628 lines
27 KiB
C++
Executable file
/* ========================================
|
|
* ToTape8 - ToTape8.h
|
|
* Copyright (c) airwindows, Airwindows uses the MIT license
|
|
* ======================================== */
|
|
|
|
#ifndef __ToTape8_H
|
|
#include "ToTape8.h"
|
|
#endif
|
|
|
|
void ToTape8::processReplacing(float **inputs, float **outputs, VstInt32 sampleFrames)
|
|
{
|
|
float* in1 = inputs[0];
|
|
float* in2 = inputs[1];
|
|
float* out1 = outputs[0];
|
|
float* out2 = outputs[1];
|
|
|
|
double overallscale = 1.0;
|
|
overallscale /= 44100.0;
|
|
overallscale *= getSampleRate();
|
|
|
|
int spacing = floor(overallscale); //should give us working basic scaling, usually 2 or 4
|
|
if (spacing < 1) spacing = 1; if (spacing > 16) spacing = 16;
|
|
|
|
double inputGain = pow(A*2.0,2.0);
|
|
|
|
double dublyAmount = B*2.0;
|
|
double outlyAmount = (1.0-B)*-2.0;
|
|
if (outlyAmount < -1.0) outlyAmount = -1.0;
|
|
double iirEncFreq = (1.0-C)/overallscale;
|
|
double iirDecFreq = C/overallscale;
|
|
double iirMidFreq = ((C * 0.618) + 0.382)/overallscale;
|
|
|
|
double flutDepth = pow(D,6)*overallscale*50;
|
|
if (flutDepth > 498.0) flutDepth = 498.0;
|
|
double flutFrequency = (0.02*pow(E,3))/overallscale;
|
|
double bias = (F*2.0)-1.0;
|
|
double underBias = (pow(bias,4)*0.25)/overallscale;
|
|
double overBias = pow(1.0-bias,3)/overallscale;
|
|
if (bias > 0.0) underBias = 0.0;
|
|
if (bias < 0.0) overBias = 1.0/overallscale;
|
|
|
|
gslew[threshold9] = overBias;
|
|
overBias *= 1.618033988749894848204586;
|
|
gslew[threshold8] = overBias;
|
|
overBias *= 1.618033988749894848204586;
|
|
gslew[threshold7] = overBias;
|
|
overBias *= 1.618033988749894848204586;
|
|
gslew[threshold6] = overBias;
|
|
overBias *= 1.618033988749894848204586;
|
|
gslew[threshold5] = overBias;
|
|
overBias *= 1.618033988749894848204586;
|
|
gslew[threshold4] = overBias;
|
|
overBias *= 1.618033988749894848204586;
|
|
gslew[threshold3] = overBias;
|
|
overBias *= 1.618033988749894848204586;
|
|
gslew[threshold2] = overBias;
|
|
overBias *= 1.618033988749894848204586;
|
|
gslew[threshold1] = overBias;
|
|
overBias *= 1.618033988749894848204586;
|
|
|
|
double headBumpDrive = (G*0.1)/overallscale;
|
|
double headBumpMix = G*0.5;
|
|
double subCurve = sin(G*M_PI);
|
|
double iirSubFreq = (subCurve*0.008)/overallscale;
|
|
|
|
hdbA[hdb_freq] = (((H*H)*175.0)+25.0)/getSampleRate();
|
|
hdbB[hdb_freq] = hdbA[hdb_freq]*0.9375;
|
|
hdbB[hdb_reso] = hdbA[hdb_reso] = 0.618033988749894848204586;
|
|
hdbB[hdb_a1] = hdbA[hdb_a1] = 0.0;
|
|
|
|
double K = tan(M_PI * hdbA[hdb_freq]);
|
|
double norm = 1.0 / (1.0 + K / hdbA[hdb_reso] + K * K);
|
|
hdbA[hdb_a0] = K / hdbA[hdb_reso] * norm;
|
|
hdbA[hdb_a2] = -hdbA[hdb_a0];
|
|
hdbA[hdb_b1] = 2.0 * (K * K - 1.0) * norm;
|
|
hdbA[hdb_b2] = (1.0 - K / hdbA[hdb_reso] + K * K) * norm;
|
|
K = tan(M_PI * hdbB[hdb_freq]);
|
|
norm = 1.0 / (1.0 + K / hdbB[hdb_reso] + K * K);
|
|
hdbB[hdb_a0] = K / hdbB[hdb_reso] * norm;
|
|
hdbB[hdb_a2] = -hdbB[hdb_a0];
|
|
hdbB[hdb_b1] = 2.0 * (K * K - 1.0) * norm;
|
|
hdbB[hdb_b2] = (1.0 - K / hdbB[hdb_reso] + K * K) * norm;
|
|
|
|
double outputGain = I*2.0;
|
|
|
|
while (--sampleFrames >= 0)
|
|
{
|
|
double inputSampleL = *in1;
|
|
double inputSampleR = *in2;
|
|
if (fabs(inputSampleL)<1.18e-23) inputSampleL = fpdL * 1.18e-17;
|
|
if (fabs(inputSampleR)<1.18e-23) inputSampleR = fpdR * 1.18e-17;
|
|
|
|
if (inputGain != 1.0) {
|
|
inputSampleL *= inputGain;
|
|
inputSampleR *= inputGain;
|
|
}
|
|
|
|
//Dubly encode
|
|
iirEncL = (iirEncL * (1.0 - iirEncFreq)) + (inputSampleL * iirEncFreq);
|
|
double highPart = ((inputSampleL-iirEncL)*2.848);
|
|
highPart += avgEncL; avgEncL = (inputSampleL-iirEncL)*1.152;
|
|
if (highPart > 1.0) highPart = 1.0; if (highPart < -1.0) highPart = -1.0;
|
|
double dubly = fabs(highPart);
|
|
if (dubly > 0.0) {
|
|
double adjust = log(1.0+(255.0*dubly))/2.40823996531;
|
|
if (adjust > 0.0) dubly /= adjust;
|
|
compEncL = (compEncL*(1.0-iirEncFreq))+(dubly*iirEncFreq);
|
|
inputSampleL += ((highPart*compEncL)*dublyAmount);
|
|
} //end Dubly encode L
|
|
iirEncR = (iirEncR * (1.0 - iirEncFreq)) + (inputSampleR * iirEncFreq);
|
|
highPart = ((inputSampleR-iirEncR)*2.848);
|
|
highPart += avgEncR; avgEncR = (inputSampleR-iirEncR)*1.152;
|
|
if (highPart > 1.0) highPart = 1.0; if (highPart < -1.0) highPart = -1.0;
|
|
dubly = fabs(highPart);
|
|
if (dubly > 0.0) {
|
|
double adjust = log(1.0+(255.0*dubly))/2.40823996531;
|
|
if (adjust > 0.0) dubly /= adjust;
|
|
compEncR = (compEncR*(1.0-iirEncFreq))+(dubly*iirEncFreq);
|
|
inputSampleR += ((highPart*compEncR)*dublyAmount);
|
|
} //end Dubly encode R
|
|
|
|
|
|
//begin Flutter
|
|
if (flutDepth > 0.0) {
|
|
if (gcount < 0 || gcount > 999) gcount = 999;
|
|
dL[gcount] = inputSampleL;
|
|
int count = gcount;
|
|
double offset = flutDepth + (flutDepth * sin(sweepL));
|
|
sweepL += nextmaxL * flutFrequency;
|
|
if (sweepL > (M_PI*2.0)) {
|
|
sweepL -= M_PI*2.0;
|
|
double flutA = 0.24 + (fpdL / (double)UINT32_MAX * 0.74);
|
|
fpdL ^= fpdL << 13; fpdL ^= fpdL >> 17; fpdL ^= fpdL << 5;
|
|
double flutB = 0.24 + (fpdL / (double)UINT32_MAX * 0.74);
|
|
if (fabs(flutA-sin(sweepR+nextmaxR))<fabs(flutB-sin(sweepR+nextmaxR))) nextmaxL = flutA; else nextmaxL = flutB;
|
|
}
|
|
count += (int)floor(offset);
|
|
inputSampleL = (dL[count-((count > 999)?1000:0)] * (1-(offset-floor(offset))));
|
|
inputSampleL += (dL[count+1-((count+1 > 999)?1000:0)] * (offset-floor(offset)));
|
|
dR[gcount] = inputSampleR;
|
|
count = gcount;
|
|
offset = flutDepth + (flutDepth * sin(sweepR));
|
|
sweepR += nextmaxR * flutFrequency;
|
|
if (sweepR > (M_PI*2.0)) {
|
|
sweepR -= M_PI*2.0;
|
|
double flutA = 0.24 + (fpdR / (double)UINT32_MAX * 0.74);
|
|
fpdR ^= fpdR << 13; fpdR ^= fpdR >> 17; fpdR ^= fpdR << 5;
|
|
double flutB = 0.24 + (fpdR / (double)UINT32_MAX * 0.74);
|
|
if (fabs(flutA-sin(sweepL+nextmaxL))<fabs(flutB-sin(sweepL+nextmaxL))) nextmaxR = flutA; else nextmaxR = flutB;
|
|
}
|
|
count += (int)floor(offset);
|
|
inputSampleR = (dR[count-((count > 999)?1000:0)] * (1-(offset-floor(offset))));
|
|
inputSampleR += (dR[count+1-((count+1 > 999)?1000:0)] * (offset-floor(offset)));
|
|
gcount--;
|
|
}
|
|
//end Flutter
|
|
|
|
//start bias routine
|
|
if (fabs(bias) > 0.001) {
|
|
for (int x = 0; x < gslew_total; x += 3) {
|
|
if (underBias > 0.0) {
|
|
double stuck = fabs(inputSampleL - (gslew[x]/0.975)) / underBias;
|
|
if (stuck < 1.0) inputSampleL = (inputSampleL * stuck) + ((gslew[x]/0.975)*(1.0-stuck));
|
|
stuck = fabs(inputSampleR - (gslew[x+1]/0.975)) / underBias;
|
|
if (stuck < 1.0) inputSampleR = (inputSampleR * stuck) + ((gslew[x+1]/0.975)*(1.0-stuck));
|
|
}
|
|
if ((inputSampleL - gslew[x]) > gslew[x+2]) inputSampleL = gslew[x] + gslew[x+2];
|
|
if (-(inputSampleL - gslew[x]) > gslew[x+2]) inputSampleL = gslew[x] - gslew[x+2];
|
|
gslew[x] = inputSampleL * 0.975;
|
|
if ((inputSampleR - gslew[x+1]) > gslew[x+2]) inputSampleR = gslew[x+1] + gslew[x+2];
|
|
if (-(inputSampleR - gslew[x+1]) > gslew[x+2]) inputSampleR = gslew[x+1] - gslew[x+2];
|
|
gslew[x+1] = inputSampleR * 0.975;
|
|
}
|
|
}
|
|
//end bias routine
|
|
|
|
//toTape basic algorithm L
|
|
iirMidRollerL = (iirMidRollerL * (1.0-iirMidFreq)) + (inputSampleL*iirMidFreq);
|
|
double HighsSampleL = inputSampleL - iirMidRollerL;
|
|
double LowsSampleL = iirMidRollerL;
|
|
if (iirSubFreq > 0.0) {
|
|
iirLowCutoffL = (iirLowCutoffL * (1.0-iirSubFreq)) + (LowsSampleL*iirSubFreq);
|
|
LowsSampleL -= iirLowCutoffL;
|
|
}
|
|
if (LowsSampleL > 1.57079633) LowsSampleL = 1.57079633;
|
|
if (LowsSampleL < -1.57079633) LowsSampleL = -1.57079633;
|
|
LowsSampleL = sin(LowsSampleL);
|
|
double thinnedHighSample = fabs(HighsSampleL)*1.57079633;
|
|
if (thinnedHighSample > 1.57079633) thinnedHighSample = 1.57079633;
|
|
thinnedHighSample = 1.0-cos(thinnedHighSample);
|
|
if (HighsSampleL < 0) thinnedHighSample = -thinnedHighSample;
|
|
HighsSampleL -= thinnedHighSample;
|
|
|
|
//toTape basic algorithm R
|
|
iirMidRollerR = (iirMidRollerR * (1.0-iirMidFreq)) + (inputSampleR*iirMidFreq);
|
|
double HighsSampleR = inputSampleR - iirMidRollerR;
|
|
double LowsSampleR = iirMidRollerR;
|
|
if (iirSubFreq > 0.0) {
|
|
iirLowCutoffR = (iirLowCutoffR * (1.0-iirSubFreq)) + (LowsSampleR*iirSubFreq);
|
|
LowsSampleR -= iirLowCutoffR;
|
|
}
|
|
if (LowsSampleR > 1.57079633) LowsSampleR = 1.57079633;
|
|
if (LowsSampleR < -1.57079633) LowsSampleR = -1.57079633;
|
|
LowsSampleR = sin(LowsSampleR);
|
|
thinnedHighSample = fabs(HighsSampleR)*1.57079633;
|
|
if (thinnedHighSample > 1.57079633) thinnedHighSample = 1.57079633;
|
|
thinnedHighSample = 1.0-cos(thinnedHighSample);
|
|
if (HighsSampleR < 0) thinnedHighSample = -thinnedHighSample;
|
|
HighsSampleR -= thinnedHighSample;
|
|
|
|
//begin HeadBump
|
|
double headBumpSampleL = 0.0;
|
|
double headBumpSampleR = 0.0;
|
|
if (headBumpMix > 0.0) {
|
|
headBumpL += (LowsSampleL * headBumpDrive);
|
|
headBumpL -= (headBumpL * headBumpL * headBumpL * (0.0618/sqrt(overallscale)));
|
|
headBumpR += (LowsSampleR * headBumpDrive);
|
|
headBumpR -= (headBumpR * headBumpR * headBumpR * (0.0618/sqrt(overallscale)));
|
|
double headBiqSampleL = (headBumpL * hdbA[hdb_a0]) + hdbA[hdb_sL1];
|
|
hdbA[hdb_sL1] = (headBumpL * hdbA[hdb_a1]) - (headBiqSampleL * hdbA[hdb_b1]) + hdbA[hdb_sL2];
|
|
hdbA[hdb_sL2] = (headBumpL * hdbA[hdb_a2]) - (headBiqSampleL * hdbA[hdb_b2]);
|
|
headBumpSampleL = (headBiqSampleL * hdbB[hdb_a0]) + hdbB[hdb_sL1];
|
|
hdbB[hdb_sL1] = (headBiqSampleL * hdbB[hdb_a1]) - (headBumpSampleL * hdbB[hdb_b1]) + hdbB[hdb_sL2];
|
|
hdbB[hdb_sL2] = (headBiqSampleL * hdbB[hdb_a2]) - (headBumpSampleL * hdbB[hdb_b2]);
|
|
double headBiqSampleR = (headBumpR * hdbA[hdb_a0]) + hdbA[hdb_sR1];
|
|
hdbA[hdb_sR1] = (headBumpR * hdbA[hdb_a1]) - (headBiqSampleR * hdbA[hdb_b1]) + hdbA[hdb_sR2];
|
|
hdbA[hdb_sR2] = (headBumpR * hdbA[hdb_a2]) - (headBiqSampleR * hdbA[hdb_b2]);
|
|
headBumpSampleR = (headBiqSampleR * hdbB[hdb_a0]) + hdbB[hdb_sR1];
|
|
hdbB[hdb_sR1] = (headBiqSampleR * hdbB[hdb_a1]) - (headBumpSampleR * hdbB[hdb_b1]) + hdbB[hdb_sR2];
|
|
hdbB[hdb_sR2] = (headBiqSampleR * hdbB[hdb_a2]) - (headBumpSampleR * hdbB[hdb_b2]);
|
|
}
|
|
//end HeadBump
|
|
|
|
inputSampleL = LowsSampleL + HighsSampleL + (headBumpSampleL * headBumpMix);
|
|
inputSampleR = LowsSampleR + HighsSampleR + (headBumpSampleR * headBumpMix);
|
|
|
|
//Dubly decode
|
|
iirDecL = (iirDecL * (1.0 - iirDecFreq)) + (inputSampleL * iirDecFreq);
|
|
highPart = ((inputSampleL-iirDecL)*2.628);
|
|
highPart += avgDecL; avgDecL = (inputSampleL-iirDecL)*1.372;
|
|
if (highPart > 1.0) highPart = 1.0; if (highPart < -1.0) highPart = -1.0;
|
|
dubly = fabs(highPart);
|
|
if (dubly > 0.0) {
|
|
double adjust = log(1.0+(255.0*dubly))/2.40823996531;
|
|
if (adjust > 0.0) dubly /= adjust;
|
|
compDecL = (compDecL*(1.0-iirDecFreq))+(dubly*iirDecFreq);
|
|
inputSampleL += ((highPart*compDecL)*outlyAmount);
|
|
} //end Dubly decode L
|
|
iirDecR = (iirDecR * (1.0 - iirDecFreq)) + (inputSampleR * iirDecFreq);
|
|
highPart = ((inputSampleR-iirDecR)*2.628);
|
|
highPart += avgDecR; avgDecR = (inputSampleR-iirDecR)*1.372;
|
|
if (highPart > 1.0) highPart = 1.0; if (highPart < -1.0) highPart = -1.0;
|
|
dubly = fabs(highPart);
|
|
if (dubly > 0.0) {
|
|
double adjust = log(1.0+(255.0*dubly))/2.40823996531;
|
|
if (adjust > 0.0) dubly /= adjust;
|
|
compDecR = (compDecR*(1.0-iirDecFreq))+(dubly*iirDecFreq);
|
|
inputSampleR += ((highPart*compDecR)*outlyAmount);
|
|
} //end Dubly decode R
|
|
|
|
|
|
if (outputGain != 1.0) {
|
|
inputSampleL *= outputGain;
|
|
inputSampleR *= outputGain;
|
|
}
|
|
|
|
//begin ClipOnly2 stereo as a little, compressed chunk that can be dropped into code
|
|
if (inputSampleL > 4.0) inputSampleL = 4.0; if (inputSampleL < -4.0) inputSampleL = -4.0;
|
|
if (wasPosClipL == true) { //current will be over
|
|
if (inputSampleL<lastSampleL) lastSampleL=0.7058208+(inputSampleL*0.2609148);
|
|
else lastSampleL = 0.2491717+(lastSampleL*0.7390851);
|
|
} wasPosClipL = false;
|
|
if (inputSampleL>0.9549925859) {wasPosClipL=true;inputSampleL=0.7058208+(lastSampleL*0.2609148);}
|
|
if (wasNegClipL == true) { //current will be -over
|
|
if (inputSampleL > lastSampleL) lastSampleL=-0.7058208+(inputSampleL*0.2609148);
|
|
else lastSampleL=-0.2491717+(lastSampleL*0.7390851);
|
|
} wasNegClipL = false;
|
|
if (inputSampleL<-0.9549925859) {wasNegClipL=true;inputSampleL=-0.7058208+(lastSampleL*0.2609148);}
|
|
intermediateL[spacing] = inputSampleL;
|
|
inputSampleL = lastSampleL; //Latency is however many samples equals one 44.1k sample
|
|
for (int x = spacing; x > 0; x--) intermediateL[x-1] = intermediateL[x];
|
|
lastSampleL = intermediateL[0]; //run a little buffer to handle this
|
|
|
|
if (inputSampleR > 4.0) inputSampleR = 4.0; if (inputSampleR < -4.0) inputSampleR = -4.0;
|
|
if (wasPosClipR == true) { //current will be over
|
|
if (inputSampleR<lastSampleR) lastSampleR=0.7058208+(inputSampleR*0.2609148);
|
|
else lastSampleR = 0.2491717+(lastSampleR*0.7390851);
|
|
} wasPosClipR = false;
|
|
if (inputSampleR>0.9549925859) {wasPosClipR=true;inputSampleR=0.7058208+(lastSampleR*0.2609148);}
|
|
if (wasNegClipR == true) { //current will be -over
|
|
if (inputSampleR > lastSampleR) lastSampleR=-0.7058208+(inputSampleR*0.2609148);
|
|
else lastSampleR=-0.2491717+(lastSampleR*0.7390851);
|
|
} wasNegClipR = false;
|
|
if (inputSampleR<-0.9549925859) {wasNegClipR=true;inputSampleR=-0.7058208+(lastSampleR*0.2609148);}
|
|
intermediateR[spacing] = inputSampleR;
|
|
inputSampleR = lastSampleR; //Latency is however many samples equals one 44.1k sample
|
|
for (int x = spacing; x > 0; x--) intermediateR[x-1] = intermediateR[x];
|
|
lastSampleR = intermediateR[0]; //run a little buffer to handle this
|
|
//end ClipOnly2 stereo as a little, compressed chunk that can be dropped into code
|
|
|
|
//begin 32 bit stereo floating point dither
|
|
int expon; frexpf((float)inputSampleL, &expon);
|
|
fpdL ^= fpdL << 13; fpdL ^= fpdL >> 17; fpdL ^= fpdL << 5;
|
|
inputSampleL += ((double(fpdL)-uint32_t(0x7fffffff)) * 5.5e-36l * pow(2,expon+62));
|
|
frexpf((float)inputSampleR, &expon);
|
|
fpdR ^= fpdR << 13; fpdR ^= fpdR >> 17; fpdR ^= fpdR << 5;
|
|
inputSampleR += ((double(fpdR)-uint32_t(0x7fffffff)) * 5.5e-36l * pow(2,expon+62));
|
|
//end 32 bit stereo floating point dither
|
|
|
|
*out1 = inputSampleL;
|
|
*out2 = inputSampleR;
|
|
|
|
in1++;
|
|
in2++;
|
|
out1++;
|
|
out2++;
|
|
}
|
|
}
|
|
|
|
void ToTape8::processDoubleReplacing(double **inputs, double **outputs, VstInt32 sampleFrames)
|
|
{
|
|
double* in1 = inputs[0];
|
|
double* in2 = inputs[1];
|
|
double* out1 = outputs[0];
|
|
double* out2 = outputs[1];
|
|
|
|
double overallscale = 1.0;
|
|
overallscale /= 44100.0;
|
|
overallscale *= getSampleRate();
|
|
|
|
int spacing = floor(overallscale); //should give us working basic scaling, usually 2 or 4
|
|
if (spacing < 1) spacing = 1; if (spacing > 16) spacing = 16;
|
|
|
|
double inputGain = pow(A*2.0,2.0);
|
|
|
|
double dublyAmount = B*2.0;
|
|
double outlyAmount = (1.0-B)*-2.0;
|
|
if (outlyAmount < -1.0) outlyAmount = -1.0;
|
|
double iirEncFreq = (1.0-C)/overallscale;
|
|
double iirDecFreq = C/overallscale;
|
|
double iirMidFreq = ((C * 0.618) + 0.382)/overallscale;
|
|
|
|
double flutDepth = pow(D,6)*overallscale*50;
|
|
if (flutDepth > 498.0) flutDepth = 498.0;
|
|
double flutFrequency = (0.02*pow(E,3))/overallscale;
|
|
double bias = (F*2.0)-1.0;
|
|
double underBias = (pow(bias,4)*0.25)/overallscale;
|
|
double overBias = pow(1.0-bias,3)/overallscale;
|
|
if (bias > 0.0) underBias = 0.0;
|
|
if (bias < 0.0) overBias = 1.0/overallscale;
|
|
|
|
gslew[threshold9] = overBias;
|
|
overBias *= 1.618033988749894848204586;
|
|
gslew[threshold8] = overBias;
|
|
overBias *= 1.618033988749894848204586;
|
|
gslew[threshold7] = overBias;
|
|
overBias *= 1.618033988749894848204586;
|
|
gslew[threshold6] = overBias;
|
|
overBias *= 1.618033988749894848204586;
|
|
gslew[threshold5] = overBias;
|
|
overBias *= 1.618033988749894848204586;
|
|
gslew[threshold4] = overBias;
|
|
overBias *= 1.618033988749894848204586;
|
|
gslew[threshold3] = overBias;
|
|
overBias *= 1.618033988749894848204586;
|
|
gslew[threshold2] = overBias;
|
|
overBias *= 1.618033988749894848204586;
|
|
gslew[threshold1] = overBias;
|
|
overBias *= 1.618033988749894848204586;
|
|
|
|
double headBumpDrive = (G*0.1)/overallscale;
|
|
double headBumpMix = G*0.5;
|
|
double subCurve = sin(G*M_PI);
|
|
double iirSubFreq = (subCurve*0.008)/overallscale;
|
|
|
|
hdbA[hdb_freq] = (((H*H)*175.0)+25.0)/getSampleRate();
|
|
hdbB[hdb_freq] = hdbA[hdb_freq]*0.9375;
|
|
hdbB[hdb_reso] = hdbA[hdb_reso] = 0.618033988749894848204586;
|
|
hdbB[hdb_a1] = hdbA[hdb_a1] = 0.0;
|
|
|
|
double K = tan(M_PI * hdbA[hdb_freq]);
|
|
double norm = 1.0 / (1.0 + K / hdbA[hdb_reso] + K * K);
|
|
hdbA[hdb_a0] = K / hdbA[hdb_reso] * norm;
|
|
hdbA[hdb_a2] = -hdbA[hdb_a0];
|
|
hdbA[hdb_b1] = 2.0 * (K * K - 1.0) * norm;
|
|
hdbA[hdb_b2] = (1.0 - K / hdbA[hdb_reso] + K * K) * norm;
|
|
K = tan(M_PI * hdbB[hdb_freq]);
|
|
norm = 1.0 / (1.0 + K / hdbB[hdb_reso] + K * K);
|
|
hdbB[hdb_a0] = K / hdbB[hdb_reso] * norm;
|
|
hdbB[hdb_a2] = -hdbB[hdb_a0];
|
|
hdbB[hdb_b1] = 2.0 * (K * K - 1.0) * norm;
|
|
hdbB[hdb_b2] = (1.0 - K / hdbB[hdb_reso] + K * K) * norm;
|
|
|
|
double outputGain = I*2.0;
|
|
|
|
while (--sampleFrames >= 0)
|
|
{
|
|
double inputSampleL = *in1;
|
|
double inputSampleR = *in2;
|
|
if (fabs(inputSampleL)<1.18e-23) inputSampleL = fpdL * 1.18e-17;
|
|
if (fabs(inputSampleR)<1.18e-23) inputSampleR = fpdR * 1.18e-17;
|
|
|
|
if (inputGain != 1.0) {
|
|
inputSampleL *= inputGain;
|
|
inputSampleR *= inputGain;
|
|
}
|
|
|
|
//Dubly encode
|
|
iirEncL = (iirEncL * (1.0 - iirEncFreq)) + (inputSampleL * iirEncFreq);
|
|
double highPart = ((inputSampleL-iirEncL)*2.848);
|
|
highPart += avgEncL; avgEncL = (inputSampleL-iirEncL)*1.152;
|
|
if (highPart > 1.0) highPart = 1.0; if (highPart < -1.0) highPart = -1.0;
|
|
double dubly = fabs(highPart);
|
|
if (dubly > 0.0) {
|
|
double adjust = log(1.0+(255.0*dubly))/2.40823996531;
|
|
if (adjust > 0.0) dubly /= adjust;
|
|
compEncL = (compEncL*(1.0-iirEncFreq))+(dubly*iirEncFreq);
|
|
inputSampleL += ((highPart*compEncL)*dublyAmount);
|
|
} //end Dubly encode L
|
|
iirEncR = (iirEncR * (1.0 - iirEncFreq)) + (inputSampleR * iirEncFreq);
|
|
highPart = ((inputSampleR-iirEncR)*2.848);
|
|
highPart += avgEncR; avgEncR = (inputSampleR-iirEncR)*1.152;
|
|
if (highPart > 1.0) highPart = 1.0; if (highPart < -1.0) highPart = -1.0;
|
|
dubly = fabs(highPart);
|
|
if (dubly > 0.0) {
|
|
double adjust = log(1.0+(255.0*dubly))/2.40823996531;
|
|
if (adjust > 0.0) dubly /= adjust;
|
|
compEncR = (compEncR*(1.0-iirEncFreq))+(dubly*iirEncFreq);
|
|
inputSampleR += ((highPart*compEncR)*dublyAmount);
|
|
} //end Dubly encode R
|
|
|
|
|
|
//begin Flutter
|
|
if (flutDepth > 0.0) {
|
|
if (gcount < 0 || gcount > 999) gcount = 999;
|
|
dL[gcount] = inputSampleL;
|
|
int count = gcount;
|
|
double offset = flutDepth + (flutDepth * sin(sweepL));
|
|
sweepL += nextmaxL * flutFrequency;
|
|
if (sweepL > (M_PI*2.0)) {
|
|
sweepL -= M_PI*2.0;
|
|
double flutA = 0.24 + (fpdL / (double)UINT32_MAX * 0.74);
|
|
fpdL ^= fpdL << 13; fpdL ^= fpdL >> 17; fpdL ^= fpdL << 5;
|
|
double flutB = 0.24 + (fpdL / (double)UINT32_MAX * 0.74);
|
|
if (fabs(flutA-sin(sweepR+nextmaxR))<fabs(flutB-sin(sweepR+nextmaxR))) nextmaxL = flutA; else nextmaxL = flutB;
|
|
}
|
|
count += (int)floor(offset);
|
|
inputSampleL = (dL[count-((count > 999)?1000:0)] * (1-(offset-floor(offset))));
|
|
inputSampleL += (dL[count+1-((count+1 > 999)?1000:0)] * (offset-floor(offset)));
|
|
dR[gcount] = inputSampleR;
|
|
count = gcount;
|
|
offset = flutDepth + (flutDepth * sin(sweepR));
|
|
sweepR += nextmaxR * flutFrequency;
|
|
if (sweepR > (M_PI*2.0)) {
|
|
sweepR -= M_PI*2.0;
|
|
double flutA = 0.24 + (fpdR / (double)UINT32_MAX * 0.74);
|
|
fpdR ^= fpdR << 13; fpdR ^= fpdR >> 17; fpdR ^= fpdR << 5;
|
|
double flutB = 0.24 + (fpdR / (double)UINT32_MAX * 0.74);
|
|
if (fabs(flutA-sin(sweepL+nextmaxL))<fabs(flutB-sin(sweepL+nextmaxL))) nextmaxR = flutA; else nextmaxR = flutB;
|
|
}
|
|
count += (int)floor(offset);
|
|
inputSampleR = (dR[count-((count > 999)?1000:0)] * (1-(offset-floor(offset))));
|
|
inputSampleR += (dR[count+1-((count+1 > 999)?1000:0)] * (offset-floor(offset)));
|
|
gcount--;
|
|
}
|
|
//end Flutter
|
|
|
|
//start bias routine
|
|
if (fabs(bias) > 0.001) {
|
|
for (int x = 0; x < gslew_total; x += 3) {
|
|
if (underBias > 0.0) {
|
|
double stuck = fabs(inputSampleL - (gslew[x]/0.975)) / underBias;
|
|
if (stuck < 1.0) inputSampleL = (inputSampleL * stuck) + ((gslew[x]/0.975)*(1.0-stuck));
|
|
stuck = fabs(inputSampleR - (gslew[x+1]/0.975)) / underBias;
|
|
if (stuck < 1.0) inputSampleR = (inputSampleR * stuck) + ((gslew[x+1]/0.975)*(1.0-stuck));
|
|
}
|
|
if ((inputSampleL - gslew[x]) > gslew[x+2]) inputSampleL = gslew[x] + gslew[x+2];
|
|
if (-(inputSampleL - gslew[x]) > gslew[x+2]) inputSampleL = gslew[x] - gslew[x+2];
|
|
gslew[x] = inputSampleL * 0.975;
|
|
if ((inputSampleR - gslew[x+1]) > gslew[x+2]) inputSampleR = gslew[x+1] + gslew[x+2];
|
|
if (-(inputSampleR - gslew[x+1]) > gslew[x+2]) inputSampleR = gslew[x+1] - gslew[x+2];
|
|
gslew[x+1] = inputSampleR * 0.975;
|
|
}
|
|
}
|
|
//end bias routine
|
|
|
|
//toTape basic algorithm L
|
|
iirMidRollerL = (iirMidRollerL * (1.0-iirMidFreq)) + (inputSampleL*iirMidFreq);
|
|
double HighsSampleL = inputSampleL - iirMidRollerL;
|
|
double LowsSampleL = iirMidRollerL;
|
|
if (iirSubFreq > 0.0) {
|
|
iirLowCutoffL = (iirLowCutoffL * (1.0-iirSubFreq)) + (LowsSampleL*iirSubFreq);
|
|
LowsSampleL -= iirLowCutoffL;
|
|
}
|
|
if (LowsSampleL > 1.57079633) LowsSampleL = 1.57079633;
|
|
if (LowsSampleL < -1.57079633) LowsSampleL = -1.57079633;
|
|
LowsSampleL = sin(LowsSampleL);
|
|
double thinnedHighSample = fabs(HighsSampleL)*1.57079633;
|
|
if (thinnedHighSample > 1.57079633) thinnedHighSample = 1.57079633;
|
|
thinnedHighSample = 1.0-cos(thinnedHighSample);
|
|
if (HighsSampleL < 0) thinnedHighSample = -thinnedHighSample;
|
|
HighsSampleL -= thinnedHighSample;
|
|
|
|
//toTape basic algorithm R
|
|
iirMidRollerR = (iirMidRollerR * (1.0-iirMidFreq)) + (inputSampleR*iirMidFreq);
|
|
double HighsSampleR = inputSampleR - iirMidRollerR;
|
|
double LowsSampleR = iirMidRollerR;
|
|
if (iirSubFreq > 0.0) {
|
|
iirLowCutoffR = (iirLowCutoffR * (1.0-iirSubFreq)) + (LowsSampleR*iirSubFreq);
|
|
LowsSampleR -= iirLowCutoffR;
|
|
}
|
|
if (LowsSampleR > 1.57079633) LowsSampleR = 1.57079633;
|
|
if (LowsSampleR < -1.57079633) LowsSampleR = -1.57079633;
|
|
LowsSampleR = sin(LowsSampleR);
|
|
thinnedHighSample = fabs(HighsSampleR)*1.57079633;
|
|
if (thinnedHighSample > 1.57079633) thinnedHighSample = 1.57079633;
|
|
thinnedHighSample = 1.0-cos(thinnedHighSample);
|
|
if (HighsSampleR < 0) thinnedHighSample = -thinnedHighSample;
|
|
HighsSampleR -= thinnedHighSample;
|
|
|
|
//begin HeadBump
|
|
double headBumpSampleL = 0.0;
|
|
double headBumpSampleR = 0.0;
|
|
if (headBumpMix > 0.0) {
|
|
headBumpL += (LowsSampleL * headBumpDrive);
|
|
headBumpL -= (headBumpL * headBumpL * headBumpL * (0.0618/sqrt(overallscale)));
|
|
headBumpR += (LowsSampleR * headBumpDrive);
|
|
headBumpR -= (headBumpR * headBumpR * headBumpR * (0.0618/sqrt(overallscale)));
|
|
double headBiqSampleL = (headBumpL * hdbA[hdb_a0]) + hdbA[hdb_sL1];
|
|
hdbA[hdb_sL1] = (headBumpL * hdbA[hdb_a1]) - (headBiqSampleL * hdbA[hdb_b1]) + hdbA[hdb_sL2];
|
|
hdbA[hdb_sL2] = (headBumpL * hdbA[hdb_a2]) - (headBiqSampleL * hdbA[hdb_b2]);
|
|
headBumpSampleL = (headBiqSampleL * hdbB[hdb_a0]) + hdbB[hdb_sL1];
|
|
hdbB[hdb_sL1] = (headBiqSampleL * hdbB[hdb_a1]) - (headBumpSampleL * hdbB[hdb_b1]) + hdbB[hdb_sL2];
|
|
hdbB[hdb_sL2] = (headBiqSampleL * hdbB[hdb_a2]) - (headBumpSampleL * hdbB[hdb_b2]);
|
|
double headBiqSampleR = (headBumpR * hdbA[hdb_a0]) + hdbA[hdb_sR1];
|
|
hdbA[hdb_sR1] = (headBumpR * hdbA[hdb_a1]) - (headBiqSampleR * hdbA[hdb_b1]) + hdbA[hdb_sR2];
|
|
hdbA[hdb_sR2] = (headBumpR * hdbA[hdb_a2]) - (headBiqSampleR * hdbA[hdb_b2]);
|
|
headBumpSampleR = (headBiqSampleR * hdbB[hdb_a0]) + hdbB[hdb_sR1];
|
|
hdbB[hdb_sR1] = (headBiqSampleR * hdbB[hdb_a1]) - (headBumpSampleR * hdbB[hdb_b1]) + hdbB[hdb_sR2];
|
|
hdbB[hdb_sR2] = (headBiqSampleR * hdbB[hdb_a2]) - (headBumpSampleR * hdbB[hdb_b2]);
|
|
}
|
|
//end HeadBump
|
|
|
|
inputSampleL = LowsSampleL + HighsSampleL + (headBumpSampleL * headBumpMix);
|
|
inputSampleR = LowsSampleR + HighsSampleR + (headBumpSampleR * headBumpMix);
|
|
|
|
//Dubly decode
|
|
iirDecL = (iirDecL * (1.0 - iirDecFreq)) + (inputSampleL * iirDecFreq);
|
|
highPart = ((inputSampleL-iirDecL)*2.628);
|
|
highPart += avgDecL; avgDecL = (inputSampleL-iirDecL)*1.372;
|
|
if (highPart > 1.0) highPart = 1.0; if (highPart < -1.0) highPart = -1.0;
|
|
dubly = fabs(highPart);
|
|
if (dubly > 0.0) {
|
|
double adjust = log(1.0+(255.0*dubly))/2.40823996531;
|
|
if (adjust > 0.0) dubly /= adjust;
|
|
compDecL = (compDecL*(1.0-iirDecFreq))+(dubly*iirDecFreq);
|
|
inputSampleL += ((highPart*compDecL)*outlyAmount);
|
|
} //end Dubly decode L
|
|
iirDecR = (iirDecR * (1.0 - iirDecFreq)) + (inputSampleR * iirDecFreq);
|
|
highPart = ((inputSampleR-iirDecR)*2.628);
|
|
highPart += avgDecR; avgDecR = (inputSampleR-iirDecR)*1.372;
|
|
if (highPart > 1.0) highPart = 1.0; if (highPart < -1.0) highPart = -1.0;
|
|
dubly = fabs(highPart);
|
|
if (dubly > 0.0) {
|
|
double adjust = log(1.0+(255.0*dubly))/2.40823996531;
|
|
if (adjust > 0.0) dubly /= adjust;
|
|
compDecR = (compDecR*(1.0-iirDecFreq))+(dubly*iirDecFreq);
|
|
inputSampleR += ((highPart*compDecR)*outlyAmount);
|
|
} //end Dubly decode R
|
|
|
|
|
|
if (outputGain != 1.0) {
|
|
inputSampleL *= outputGain;
|
|
inputSampleR *= outputGain;
|
|
}
|
|
|
|
//begin ClipOnly2 stereo as a little, compressed chunk that can be dropped into code
|
|
if (inputSampleL > 4.0) inputSampleL = 4.0; if (inputSampleL < -4.0) inputSampleL = -4.0;
|
|
if (wasPosClipL == true) { //current will be over
|
|
if (inputSampleL<lastSampleL) lastSampleL=0.7058208+(inputSampleL*0.2609148);
|
|
else lastSampleL = 0.2491717+(lastSampleL*0.7390851);
|
|
} wasPosClipL = false;
|
|
if (inputSampleL>0.9549925859) {wasPosClipL=true;inputSampleL=0.7058208+(lastSampleL*0.2609148);}
|
|
if (wasNegClipL == true) { //current will be -over
|
|
if (inputSampleL > lastSampleL) lastSampleL=-0.7058208+(inputSampleL*0.2609148);
|
|
else lastSampleL=-0.2491717+(lastSampleL*0.7390851);
|
|
} wasNegClipL = false;
|
|
if (inputSampleL<-0.9549925859) {wasNegClipL=true;inputSampleL=-0.7058208+(lastSampleL*0.2609148);}
|
|
intermediateL[spacing] = inputSampleL;
|
|
inputSampleL = lastSampleL; //Latency is however many samples equals one 44.1k sample
|
|
for (int x = spacing; x > 0; x--) intermediateL[x-1] = intermediateL[x];
|
|
lastSampleL = intermediateL[0]; //run a little buffer to handle this
|
|
|
|
if (inputSampleR > 4.0) inputSampleR = 4.0; if (inputSampleR < -4.0) inputSampleR = -4.0;
|
|
if (wasPosClipR == true) { //current will be over
|
|
if (inputSampleR<lastSampleR) lastSampleR=0.7058208+(inputSampleR*0.2609148);
|
|
else lastSampleR = 0.2491717+(lastSampleR*0.7390851);
|
|
} wasPosClipR = false;
|
|
if (inputSampleR>0.9549925859) {wasPosClipR=true;inputSampleR=0.7058208+(lastSampleR*0.2609148);}
|
|
if (wasNegClipR == true) { //current will be -over
|
|
if (inputSampleR > lastSampleR) lastSampleR=-0.7058208+(inputSampleR*0.2609148);
|
|
else lastSampleR=-0.2491717+(lastSampleR*0.7390851);
|
|
} wasNegClipR = false;
|
|
if (inputSampleR<-0.9549925859) {wasNegClipR=true;inputSampleR=-0.7058208+(lastSampleR*0.2609148);}
|
|
intermediateR[spacing] = inputSampleR;
|
|
inputSampleR = lastSampleR; //Latency is however many samples equals one 44.1k sample
|
|
for (int x = spacing; x > 0; x--) intermediateR[x-1] = intermediateR[x];
|
|
lastSampleR = intermediateR[0]; //run a little buffer to handle this
|
|
//end ClipOnly2 stereo as a little, compressed chunk that can be dropped into code
|
|
|
|
//begin 64 bit stereo floating point dither
|
|
//int expon; frexp((double)inputSampleL, &expon);
|
|
fpdL ^= fpdL << 13; fpdL ^= fpdL >> 17; fpdL ^= fpdL << 5;
|
|
//inputSampleL += ((double(fpdL)-uint32_t(0x7fffffff)) * 1.1e-44l * pow(2,expon+62));
|
|
//frexp((double)inputSampleR, &expon);
|
|
fpdR ^= fpdR << 13; fpdR ^= fpdR >> 17; fpdR ^= fpdR << 5;
|
|
//inputSampleR += ((double(fpdR)-uint32_t(0x7fffffff)) * 1.1e-44l * pow(2,expon+62));
|
|
//end 64 bit stereo floating point dither
|
|
|
|
*out1 = inputSampleL;
|
|
*out2 = inputSampleR;
|
|
|
|
in1++;
|
|
in2++;
|
|
out1++;
|
|
out2++;
|
|
}
|
|
}
|