aboutsummaryrefslogtreecommitdiff
path: root/adaptive/src/lms2~.c
diff options
context:
space:
mode:
Diffstat (limited to 'adaptive/src/lms2~.c')
-rwxr-xr-xadaptive/src/lms2~.c332
1 files changed, 332 insertions, 0 deletions
diff --git a/adaptive/src/lms2~.c b/adaptive/src/lms2~.c
new file mode 100755
index 0000000..c3c3d3c
--- /dev/null
+++ b/adaptive/src/lms2~.c
@@ -0,0 +1,332 @@
+/******************************************************
+ *
+ * Adaptive Systems for PD
+ *
+ * copyleft (c) Gerda Strobl, Georg Holzmann
+ * 2005
+ *
+ * for complaints, suggestions: grh@mur.at
+ *
+ ******************************************************
+ *
+ * license: GNU General Public License v.2
+ *
+ ******************************************************/
+
+#include "adaptive.h"
+
+
+/* ------------------------ lms2~ ------------------------- */
+
+static t_class *lms2_tilde_class;
+
+typedef struct _lms2
+{
+ t_object x_obj;
+ t_float f;
+ t_atom *coef;
+ t_sample *buf;
+ t_sample *y_tmp;
+ t_sample *e_tmp;
+ t_int bufsize;
+ t_outlet *c_out;
+ int adapt; // enable/disable adaptation
+
+ t_int N; //number of coefficients of the adaptive system
+ t_float *c; // coefficients of the system
+ t_float mu; // step-size parameter (learning rate)
+
+ t_canvas *x_canvas;
+} t_lms2_tilde;
+
+static void lms2_tilde_a(t_lms2_tilde *x, t_floatarg f)
+{
+ x->adapt = (f==0) ? 0 : 1;
+}
+
+static void lms2_tilde_geta(t_lms2_tilde *x)
+{
+ if(x->adapt==0)
+ post("lms2~: adaptation is currently OFF");
+ else
+ post("lms2~: adaptation is currently ON");
+}
+
+static void lms2_tilde_mu(t_lms2_tilde *x, t_floatarg f)
+{
+ x->mu = f;
+}
+
+static void lms2_tilde_getmu(t_lms2_tilde *x)
+{
+ post("mu (step-size parameter): %f", x->mu);
+}
+
+static void lms2_tilde_getN(t_lms2_tilde *x)
+{
+ post("N (number of coefficients): %d", x->N);
+}
+
+static void lms2_tilde_clear(t_lms2_tilde *x)
+{
+ int i;
+
+ // clear coefficients
+ for(i=0; i<x->N; i++)
+ x->c[i] = 0;
+
+ // clear temp buffer
+ for(i=0; i<x->N-1; i++)
+ x->buf[i] = 0;
+}
+
+static void lms2_tilde_init(t_lms2_tilde *x)
+{
+ int i;
+
+ // set the first coefficient to 1, all others to 0
+ // so this is a delay free transmission
+ x->c[0] = 1;
+ for(i=1; i<x->N; i++)
+ x->c[i] = 0;
+
+ // clear temp buffers
+ for(i=0; i<x->N-1; i++)
+ x->buf[i] = 0;
+}
+
+static void lms2_tilde_print(t_lms2_tilde *x)
+{
+ int i;
+
+ // print coefficients
+ post("\nNr. of coefficients: %d",x->N);
+ post("coefficients:");
+ for(i=0; i<x->N; i++)
+ post("\t%d: %f",i,x->c[i]);
+}
+
+static void lms2_tilde_write(t_lms2_tilde *x, t_symbol *s)
+{
+ // make correct path
+ char filnam[MAXPDSTRING];
+ char filename[MAXPDSTRING];
+ canvas_makefilename(x->x_canvas, s->s_name, filnam, MAXPDSTRING);
+ sys_bashfilename(filnam, filename);
+
+ // save to file
+ adaptation_write(filename, x->N, x->mu, x->c);
+}
+
+static void lms2_tilde_read(t_lms2_tilde *x, t_symbol *s)
+{
+ // make correct path
+ char filnam[MAXPDSTRING];
+ char filename[MAXPDSTRING];
+ int n = x->N;
+ canvas_makefilename(x->x_canvas, s->s_name, filnam, MAXPDSTRING);
+ sys_bashfilename(filnam, filename);
+
+ // read file
+ adaptation_read(filename, &x->N, &x->mu, x->c, x->buf);
+
+ // if length changes:
+ if(x->N != n)
+ {
+ if(x->coef) freebytes(x->coef, sizeof(t_atom) * x->N);
+ x->coef = (t_atom *)getbytes(sizeof(t_atom) * x->N);
+ }
+}
+
+static t_int *lms2_tilde_perform(t_int *w)
+{
+ t_sample *x_ = (t_sample *)(w[1]);
+ t_sample *d_ = (t_sample *)(w[2]);
+ t_sample *y_ = (t_sample *)(w[3]);
+ t_sample *e_ = (t_sample *)(w[4]);
+ int n = (int)(w[5]);
+ t_lms2_tilde *x = (t_lms2_tilde *)(w[6]);
+ int i, j, tmp;
+
+
+ for(i=0; i<n; i++)
+ {
+ // calc output (filter)
+
+ x->y_tmp[i]=0;
+
+ // y_[i] += x->c[j] * x_[i-j];
+ // so lets split in two halfs, so that
+ // negative indezes get samples from the
+ // last audioblock (x->buf) ...
+
+ tmp = (i+1 - x->N)*(-1);
+ tmp = tmp<0 ? 0 : tmp;
+
+ for(j=0; j<x->N-tmp; j++)
+ x->y_tmp[i] += x->c[j] * x_[i-j];
+
+ for(j=x->N-tmp; j<x->N; j++)
+ x->y_tmp[i] += x->c[j] * x->buf[(i-j)*(-1)-1];
+
+ if(x->adapt)
+ {
+ // error computation
+ x->e_tmp[i] = d_[i] - x->y_tmp[i];
+
+ // coefficient adaptation
+ // (split in the same way as above)
+
+ for(j=0; j<x->N-tmp; j++)
+ x->c[j] = x->c[j] + x->mu * x_[i-j] * x->e_tmp[i];
+
+ for(j=x->N-tmp; j<x->N; j++)
+ x->c[j] = x->c[j] + x->mu * x->buf[(i-j)*(-1)-1] * x->e_tmp[i];
+ }
+ else x->e_tmp[i] = 0;
+
+ //post("%d: in %f, d: %f, out: %f, e: %f, c1:%f, c2:%f", i, x_[i], d_[i], x->y_tmp[i], x->e_tmp[i], x->c[0], x->c[1]);
+ }
+
+ // outlet coefficients
+ for(i=0; i<x->N; i++)
+ SETFLOAT(&x->coef[i],x->c[i]);
+
+ outlet_list(x->c_out, &s_list, x->N, x->coef);
+
+ // store last samples for next audiobuffer
+ for(i=0; i<x->N-1; i++)
+ x->buf[i] = x_[n-1-i];
+
+ // now write tmps to outlets
+ while(n--)
+ {
+ y_[n] = x->y_tmp[n];
+ e_[n] = x->e_tmp[n];
+ }
+
+ return (w+7);
+}
+
+static void lms2_tilde_dsp(t_lms2_tilde *x, t_signal **sp)
+{
+ // allocate new temp buffer if buffersize changes
+ if(x->bufsize != sp[0]->s_n)
+ {
+ if(sp[0]->s_n < x->N)
+ post("lms2~ WARNING: buffersize must be bigger than N, you will get wrong results !!!");
+
+ if(x->y_tmp) freebytes(x->y_tmp, sizeof(t_sample) * x->bufsize);
+ x->y_tmp = (t_sample *)getbytes(sizeof(t_sample) * sp[0]->s_n);
+
+ if(x->e_tmp) freebytes(x->e_tmp, sizeof(t_sample) * x->bufsize);
+ x->e_tmp = (t_sample *)getbytes(sizeof(t_sample) * sp[0]->s_n);
+
+ x->bufsize = sp[0]->s_n;
+ }
+
+ dsp_add(lms2_tilde_perform, 6, sp[0]->s_vec, sp[1]->s_vec,
+ sp[2]->s_vec, sp[3]->s_vec, sp[0]->s_n, x);
+}
+
+static void lms2_tilde_helper(void)
+{
+ post("\nlms2~: Adaptive transversal filter using LMS (for algorithm analysis)");
+ post("INPUT:");
+ post("\tinlet1: input signal x[n]");
+ post("\tinlet2: desired output signal d[n]");
+ post("\tinit_arg1: number of coefficients of the adaptive system");
+ post("\tinit_arg2, mu: step-size parameter (learning rate)");
+ post("OUTPUT:");
+ post("\toutlet1: output signal y[n]");
+ post("\toutlet2: error signal e[n]");
+ post("\toutlet3: coefficients c[n] (only per block)\n");
+}
+
+static void *lms2_tilde_new(t_symbol *s, int argc, t_atom *argv)
+{
+ t_lms2_tilde *x = (t_lms2_tilde *)pd_new(lms2_tilde_class);
+ int i;
+
+ // default values:
+ x->N = 8;
+ x->mu = 0.05;
+ x->adapt = 0;
+ x->y_tmp = NULL;
+ x->e_tmp = NULL;
+ x->bufsize = 0;
+
+ switch(argc)
+ {
+ case 2:
+ x->mu = atom_getfloat(argv+1);
+ case 1:
+ x->N = atom_getint(argv);
+ x->N = (x->N<=0) ? 1 : x->N;
+ }
+
+ // allocate mem and init coefficients
+ x->c = (t_float *)getbytes(sizeof(t_float) * x->N);
+ for(i=0; i<x->N; i++)
+ x->c[i] = 0;
+
+ // allocate mem for temp buffer
+ x->buf = (t_float *)getbytes(sizeof(t_float) * x->N-1);
+ for(i=0; i<x->N-1; i++)
+ x->buf[i] = 0;
+
+ // for output atoms (coefficients):
+ x->coef = (t_atom *)getbytes(sizeof(t_atom) * x->N);
+
+ inlet_new(&x->x_obj, &x->x_obj.ob_pd, &s_signal, &s_signal);
+ outlet_new(&x->x_obj, &s_signal);
+ outlet_new(&x->x_obj, &s_signal);
+ x->c_out = outlet_new(&x->x_obj, 0);
+
+ x->x_canvas = canvas_getcurrent();
+
+
+ return (x);
+}
+
+static void lms2_tilde_free(t_lms2_tilde *x)
+{
+ if(x->c) freebytes(x->c, sizeof(t_float) * x->N);
+ if(x->buf) freebytes(x->buf, sizeof(t_sample) * x->N-1);
+ if(x->y_tmp) freebytes(x->y_tmp, sizeof(t_sample) * x->bufsize);
+ if(x->e_tmp) freebytes(x->e_tmp, sizeof(t_sample) * x->bufsize);
+ if(x->coef) freebytes(x->coef, sizeof(t_atom) * x->N);
+}
+
+void lms2_tilde_setup(void)
+{
+ lms2_tilde_class = class_new(gensym("lms2~"), (t_newmethod)lms2_tilde_new,
+ (t_method)lms2_tilde_free, sizeof(t_lms2_tilde),
+ CLASS_DEFAULT, A_GIMME, 0);
+
+ class_addmethod(lms2_tilde_class, (t_method)lms2_tilde_a,
+ gensym("adaptation"), A_DEFFLOAT, 0);
+ class_addmethod(lms2_tilde_class, (t_method)lms2_tilde_geta,
+ gensym("getadaptation"), 0);
+ class_addmethod(lms2_tilde_class, (t_method)lms2_tilde_mu,
+ gensym("mu"), A_DEFFLOAT, 0);
+ class_addmethod(lms2_tilde_class, (t_method)lms2_tilde_getmu,
+ gensym("getmu"), 0);
+ class_addmethod(lms2_tilde_class, (t_method)lms2_tilde_getN,
+ gensym("getN"), 0);
+ class_addmethod(lms2_tilde_class, (t_method)lms2_tilde_init,
+ gensym("init_unity"), 0);
+ class_addmethod(lms2_tilde_class, (t_method)lms2_tilde_clear,
+ gensym("clear"), 0);
+ class_addmethod(lms2_tilde_class, (t_method)lms2_tilde_print,
+ gensym("print"), 0);
+ class_addmethod(lms2_tilde_class, (t_method)lms2_tilde_write,
+ gensym("write"), A_DEFSYMBOL, 0);
+ class_addmethod(lms2_tilde_class, (t_method)lms2_tilde_read,
+ gensym("read"), A_DEFSYMBOL, 0);
+
+ class_addmethod(lms2_tilde_class, (t_method)lms2_tilde_dsp, gensym("dsp"), 0);
+ CLASS_MAINSIGNALIN(lms2_tilde_class, t_lms2_tilde, f);
+
+ class_addmethod(lms2_tilde_class, (t_method)lms2_tilde_helper, gensym("help"), 0);
+}