moog filter

This commit is contained in:
Connor Olding 2015-06-03 09:33:27 -07:00
parent 834e424087
commit ec0fcf908c
3 changed files with 254 additions and 1 deletions

View file

@ -5,7 +5,7 @@ FULLNAME = $(DISTNAME)-$(VERSION)
BIN ?= ./bin
VST_SDK_DIR ?= .
BOTH = eq eq_const eq_const_T420 eq_const_T420_svf noise tube
BOTH = eq eq_const eq_const_T420 eq_const_T420_svf mugi4 noise tube
LADSPA = $(BOTH)
VST = $(BOTH) delay_test

208
crap/mugi4.h Normal file
View file

@ -0,0 +1,208 @@
#define ID (0xD8D0D8D0)
#define LABEL "crap_mugi4"
#define NAME "crap mugi4 (moog-like)"
#define AUTHOR "Connor Olding"
#define COPYRIGHT "MIT"
#define PARAMETERS 3
/*
an implementation of:
S. D Angelo and V. Välimäki. Generalized Moog Ladder Filter: Part II
Explicit Nonlinear Model through a Novel Delay-Free Loop Implementation Method.
IEEE Trans. Audio, Speech, and Lang. Process.,
vol. 22, no. 12, pp. 1873 1883, December 2014.
https://aaltodoc.aalto.fi/bitstream/handle/123456789/14420/article6.pdf
*/
#define OVERSAMPLING 2
#define BLOCK_SIZE 256
#define FULL_SIZE (BLOCK_SIZE*OVERSAMPLING)
#include <stdio.h>
#include <string.h>
#include "util.h"
#include "param.h"
#include "os2piir_stereo.h"
#define VT 0.026
#define N 4
#define VT2 V(2.*VT)
typedef struct {
v2df sum, sumback, dout;
} stage;
typedef struct {
v2df g;
v2df p0;
v2df q0, q1, q2, q3;
v2df r1, r2, r3, r4;
v2df L_p0;
v2df L_q0;
v2df L_r1;
} freqdata;
typedef struct {
ulong fs;
halfband_t hb_up, hb_down;
freqdata fd;
stage s1, s2, s3, s4;
v2df sumback1, sumback2, sumback3, sumback4;
v2df drive, feedback;
} personal;
INNER PURE v2df
tanh2(v2df x)
{
//return (v2df){tanh(x[0]), tanh(x[1])};
v2df xx = x*x;
v2df a = ((xx + V(378.))*xx + V(17325.))*xx + V(135135.);
v2df b = ((V(28.)*xx + V(3150.))*xx + V(62370.))*xx + V(135135.);
return x*a/b;
}
INNER v2df
process_stage(stage *s, freqdata fd, v2df in)
{
v2df temp = (in + s->sumback)*VT2*fd.L_p0*fd.g;
v2df out = temp + s->sum;
s->sum += V(2.)*temp;
s->dout = tanh2(out/VT2);
s->sumback = in*fd.L_r1 - s->dout*fd.L_q0;
return out;
}
INNER v2df
process_one(v2df in, personal *data)
{
const freqdata fd = data->fd;
in *= data->drive;
v2df sum = in + data->sumback1;
v2df pre = -fd.p0*sum;
process_stage(&data->s1, fd, tanh2(pre/VT2));
process_stage(&data->s2, fd, data->s1.dout);
process_stage(&data->s3, fd, data->s2.dout);
v2df out = process_stage(&data->s4, fd, data->s3.dout);
v2df back = data->feedback*out;
data->sumback1 = fd.r1*in + fd.q0*back + data->sumback2;
data->sumback2 = fd.r2*in + fd.q1*back + data->sumback3;
data->sumback3 = fd.r3*in + fd.q2*back + data->sumback4;
data->sumback4 = fd.r4*in + fd.q3*back;
v2df compensate = -(data->feedback + V(1.));
return out/data->drive*compensate;
}
static void
process_double(personal *data,
double *in_L, double *in_R,
double *out_L, double *out_R,
unsigned long count)
#include "process_mugi4.h"
static void
process(personal *data,
float *in_L, float *in_R,
float *out_L, float *out_R,
ulong count)
#include "process_mugi4.h"
INNER void
construct(personal *data)
{
memset(data, 0, sizeof(personal));
}
INNER void
construct_params(param *params)
{
sprintf(params[0].name, "Frequency");
params[0].min = 20;
params[0].max = 20000;
params[0].scale = SCALE_HZ;
params[0].def = DEFAULT_MAX;
sprintf(params[1].name, "Drive");
params[1].min = -40;
params[1].max = 0;
params[1].scale = SCALE_DB;
params[1].def = DEFAULT_MIN;
sprintf(params[2].name, "Feedback");
params[2].min = 0;
params[2].max = 1;
params[2].scale = SCALE_FLOAT;
params[2].def = DEFAULT_MIN;
param_reset(&params[0]);
param_reset(&params[1]);
param_reset(&params[2]);
}
INNER void
destruct(personal *data)
{}
INNER void
resume(personal *data)
{
memset(&data->hb_up, 0, sizeof(halfband_t));
memset(&data->hb_down, 0, sizeof(halfband_t));
}
INNER void
pause(personal *data)
{}
INNER void
adjust(personal *data, param *params, ulong fs_long)
{
double fs = fs_long;
data->fs = fs_long;
double f = params[0].value;
if (f < 20) f = 20;
if (f > fs/6*OVERSAMPLING) f = fs/6*OVERSAMPLING;
double drive = DB2LIN(params[1].value);
double k = params[2].value*N;
data->drive = (v2df){drive, drive};
data->feedback = (v2df){k, k};
double bc1 = -4; //-binomial(N, 1);
double bc2 = -6; //-binomial(N, 2);
double bc3 = -4; //-binomial(N, 3);
double bc4 = -1; //-binomial(N, 4);
freqdata fd;
#define fd_set(L, R) double L = R; fd.L = (v2df){L, L}
fd_set(g, tan(M_PI*f/fs/OVERSAMPLING));
double gg1 = g/(g + 1);
double gg1Nk = k*gg1*gg1*gg1*gg1;
double g1g1 = (g - 1)/(g + 1);
fd_set(p0, 1/(1 + gg1Nk));
fd_set(r1, bc1*gg1Nk);
fd_set(r2, bc2*gg1Nk);
fd_set(r3, bc3*gg1Nk);
fd_set(r4, bc4*gg1Nk);
fd_set(q0, r1 + bc1*g1g1);
fd_set(q1, r2 + bc2*g1g1*g1g1);
fd_set(q2, r3 + bc3*g1g1*g1g1*g1g1);
fd_set(q3, r4 + bc4*g1g1*g1g1*g1g1*g1g1);
fd_set(L_p0, 1/(1 + g));
fd_set(L_q0, 1 - g);
fd_set(L_r1, -g);
#undef fd_set
data->fd = fd;
}
INNER void
adjust_one(personal *data, param *params, unsigned int index)
{
adjust(data, params, data->fs);
}

45
include/process_mugi4.h Normal file
View file

@ -0,0 +1,45 @@
{
disable_denormals();
v2df buf[BLOCK_SIZE];
v2df over[FULL_SIZE];
halfband_t *hb_up = &data->hb_up;
halfband_t *hb_down = &data->hb_down;
for (ulong pos = 0; pos < count; pos += BLOCK_SIZE) {
ulong rem = BLOCK_SIZE;
if (pos + BLOCK_SIZE > count)
rem = count - pos;
ulong rem2 = rem*OVERSAMPLING;
for (ulong i = 0; i < rem; i++) {
buf[i][0] = in_L[i];
buf[i][1] = in_R[i];
}
for (ulong i = 0; i < rem; i++) {
over[i*2+0] = interpolate_a(hb_up, buf[i]);
over[i*2+1] = interpolate_b(hb_up, buf[i]);
}
for (ulong i = 0; i < rem2; i++) {
over[i] = process_one(over[i], data);
}
for (ulong i = 0; i < rem; i++) {
decimate_a(hb_down, over[i*2+0]);
buf[i] = decimate_b(hb_down, over[i*2+1]);
}
for (ulong i = 0; i < rem; i++) {
out_L[i] = buf[i][0];
out_R[i] = buf[i][1];
}
in_L += BLOCK_SIZE;
in_R += BLOCK_SIZE;
out_L += BLOCK_SIZE;
out_R += BLOCK_SIZE;
}
}