aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorFranz Zotter <fzotter@users.sourceforge.net>2008-05-07 05:42:25 +0000
committerFranz Zotter <fzotter@users.sourceforge.net>2008-05-07 05:42:25 +0000
commitc859f9f8b1474aabdc12b0b919e63669fcd75bc0 (patch)
tree70d9c85c3a3bca7ed9a0f52dc2dd40dca3c37c46
parent210205f55dcd8055bf0fa916202eb6e7eeaa4275 (diff)
added complex valued matrix fft routines
svn path=/trunk/externals/iem/iemmatrix/; revision=9768
-rw-r--r--src/mtx_fft.c208
-rw-r--r--src/mtx_ifft.c219
2 files changed, 427 insertions, 0 deletions
diff --git a/src/mtx_fft.c b/src/mtx_fft.c
new file mode 100644
index 0000000..94ebbd1
--- /dev/null
+++ b/src/mtx_fft.c
@@ -0,0 +1,208 @@
+/*
+ * iemmatrix
+ *
+ * objects for manipulating simple matrices
+ * mostly refering to matlab/octave matrix functions
+ *
+ * Copyright (c) 2005, Franz Zotter
+ * IEM, Graz, Austria
+ *
+ * For information on usage and redistribution, and for a DISCLAIMER OF ALL
+ * WARRANTIES, see the file, "LICENSE.txt," in this distribution.
+ *
+ */
+
+#include "iemmatrix.h"
+#include <stdlib.h>
+
+static t_class *mtx_fft_class;
+
+typedef struct _MtxFFT_ MtxFFT;
+struct _MtxFFT_
+{
+ t_object x_obj;
+ int size;
+
+ t_float *f_re;
+ t_float *f_im;
+
+ t_outlet *list_re_out;
+ t_outlet *list_im_out;
+
+ t_atom *list_re;
+ t_atom *list_im;
+};
+
+static void deleteMtxFFT (MtxFFT *x)
+{
+ if (x->f_re)
+ free (x->f_re);
+ if (x->f_im)
+ free (x->f_im);
+ if (x->list_re)
+ free (x->list_re);
+ if (x->list_im)
+ free (x->list_im);
+}
+
+static void *newMtxFFT (t_symbol *s, int argc, t_atom *argv)
+{
+ MtxFFT *x = (MtxFFT *) pd_new (mtx_fft_class);
+ inlet_new(&x->x_obj, &x->x_obj.ob_pd, gensym("matrix"),gensym(""));
+ x->list_re_out = outlet_new (&x->x_obj, gensym("matrix"));
+ x->list_im_out = outlet_new (&x->x_obj, gensym("matrix"));
+
+ x->size=0;
+ x->f_re=x->f_im=0;
+ x->list_re=x->list_im=0;
+
+ return ((void *) x);
+}
+
+static void mtxFFTBang (MtxFFT *x)
+{
+ if (x->list_im) {
+ outlet_anything(x->list_im_out, gensym("matrix"), x->size, x->list_im);
+ outlet_anything(x->list_re_out, gensym("matrix"), x->size, x->list_re);
+ }
+}
+
+static void zeroFloatArray (int n, t_float *f)
+{
+ while (n--)
+ *f++ = 0.0f;
+}
+
+static void writeFloatIntoList (int n, t_atom *l, t_float *f)
+{
+ for (;n--;f++, l++)
+ SETFLOAT (l, *f);
+}
+static void readFloatFromList (int n, t_atom *l, t_float *f)
+{
+ while (n--)
+ *f++ = atom_getfloat (l++);
+}
+
+static void mtxFFTMatrixCold (MtxFFT *x, t_symbol *s,
+ int argc, t_atom *argv)
+{
+ int rows = atom_getint (argv++);
+ int columns = atom_getint (argv++);
+ int size = rows * columns;
+ int in_size = argc-2;
+ t_atom *list_re = x->list_re;
+ t_atom *list_im = x->list_im;
+ t_float *f_re = x->f_re;
+ t_float *f_im = x->f_im;
+
+ /* fftsize check */
+ if (!size)
+ post("mtx_fft: invalid dimensions");
+ else if (in_size<size)
+ post("mtx_fft: sparse matrix not yet supported: use \"mtx_check\"");
+ else if (columns < 4){
+ post("mtx_fft: matrix must have at least 4 columns");
+ }
+ else if (columns == (1 << ilog2(columns))) {
+ /* ok, prepare real-part of FFT! */
+
+ /* memory things */
+ f_re=(t_float*)realloc(f_re, sizeof(t_float)*size);
+ f_im=(t_float*)realloc(f_im, sizeof(t_float)*size);
+ list_re=(t_atom*)realloc(list_re, sizeof(t_atom)*(size+2));
+ list_im=(t_atom*)realloc(list_im, sizeof(t_atom)*(size+2));
+
+ x->size = size;
+ x->list_im = list_im;
+ x->list_re = list_re;
+ x->f_re = f_re;
+ x->f_im = f_im;
+
+ /* main part */
+ readFloatFromList (size, argv, f_im);
+
+ }
+ else
+ post("mtx_rowfft: rowvector size no power of 2!");
+
+}
+
+
+static void mtxFFTMatrixHot (MtxFFT *x, t_symbol *s,
+ int argc, t_atom *argv)
+{
+ int rows = atom_getint (argv++);
+ int columns = atom_getint (argv++);
+ int size = rows * columns;
+ int in_size = argc-2;
+ int fft_count;
+ t_atom *list_re = x->list_re;
+ t_atom *list_im = x->list_im;
+ t_float *f_re = x->f_re;
+ t_float *f_im = x->f_im;
+
+ /* fftsize check */
+ if (!size)
+ post("mtx_fft: invalid dimensions");
+ else if (in_size<size)
+ post("mtx_fft: sparse matrix not yet supported: use \"mtx_check\"");
+ else if (size != x->size)
+ post("mtx_fft: left matrix has other dimensions than right matrix");
+ else if (columns < 4){
+ post("mtx_fft: matrix must have at least 4 columns");
+ }
+ else if (columns == (1 << ilog2(columns))) {
+ /* ok, do the FFT! */
+
+ /* main part */
+ readFloatFromList (size, argv, f_re);
+
+ fft_count = rows;
+ list_re += 2;
+ list_im += 2;
+ while (fft_count--){
+ mayer_fft (columns, f_re, f_im);
+ writeFloatIntoList (columns, list_re, f_re);
+ writeFloatIntoList (columns, list_im, f_im);
+ f_im += columns;
+ f_re += columns;
+ list_re += columns;
+ list_im += columns;
+ }
+
+ list_re = x->list_re;
+ list_im = x->list_im;
+
+ SETSYMBOL(list_re, gensym("matrix"));
+ SETSYMBOL(list_im, gensym("matrix"));
+ SETFLOAT(list_re, rows);
+ SETFLOAT(list_im, rows);
+ SETFLOAT(list_re+1, columns);
+ SETFLOAT(list_im+1, columns);
+ outlet_anything(x->list_im_out, gensym("matrix"),
+ x->size+2, list_im);
+ outlet_anything(x->list_re_out, gensym("matrix"),
+ x->size+2, list_re);
+ }
+ else
+ post("mtx_rowfft: rowvector size no power of 2!");
+
+}
+
+void mtx_fft_setup (void)
+{
+ mtx_fft_class = class_new
+ (gensym("mtx_fft"),
+ (t_newmethod) newMtxFFT,
+ (t_method) deleteMtxFFT,
+ sizeof (MtxFFT),
+ CLASS_DEFAULT, A_GIMME, 0);
+ class_addbang (mtx_fft_class, (t_method) mtxFFTBang);
+ class_addmethod (mtx_fft_class, (t_method) mtxFFTMatrixHot, gensym("matrix"), A_GIMME,0);
+ class_addmethod (mtx_fft_class, (t_method) mtxFFTMatrixCold, gensym(""), A_GIMME,0);
+}
+
+void iemtx_fft_setup(void){
+ mtx_fft_setup();
+}
diff --git a/src/mtx_ifft.c b/src/mtx_ifft.c
new file mode 100644
index 0000000..0c159ab
--- /dev/null
+++ b/src/mtx_ifft.c
@@ -0,0 +1,219 @@
+/*
+ * iemmatrix
+ *
+ * objects for manipulating simple matrices
+ * mostly refering to matlab/octave matrix functions
+ *
+ * Copyright (c) 2005, Franz Zotter
+ * IEM, Graz, Austria
+ *
+ * For information on usage and redistribution, and for a DISCLAIMER OF ALL
+ * WARRANTIES, see the file, "LICENSE.txt," in this distribution.
+ *
+ */
+
+#include "iemmatrix.h"
+#include <stdlib.h>
+
+static t_class *mtx_ifft_class;
+
+typedef struct _MtxIFFT_ MtxIFFT;
+struct _MtxIFFT_
+{
+ t_object x_obj;
+ int size;
+ float renorm_fac;
+
+ t_float *f_re;
+ t_float *f_im;
+
+ t_outlet *list_re_out;
+ t_outlet *list_im_out;
+
+ t_atom *list_re;
+ t_atom *list_im;
+};
+
+static void deleteMtxIFFT (MtxIFFT *x)
+{
+ if (x->f_re)
+ free (x->f_re);
+ if (x->f_im)
+ free (x->f_im);
+ if (x->list_re)
+ free (x->list_re);
+ if (x->list_im)
+ free (x->list_im);
+}
+
+static void *newMtxIFFT (t_symbol *s, int argc, t_atom *argv)
+{
+ MtxIFFT *x = (MtxIFFT *) pd_new (mtx_ifft_class);
+ inlet_new(&x->x_obj, &x->x_obj.ob_pd, gensym("matrix"),gensym(""));
+ x->list_re_out = outlet_new (&x->x_obj, gensym("matrix"));
+ x->list_im_out = outlet_new (&x->x_obj, gensym("matrix"));
+
+ x->size=0;
+ x->f_re=x->f_im=0;
+ x->list_re=x->list_im=0;
+
+ return ((void *) x);
+}
+
+static void mtxIFFTBang (MtxIFFT *x)
+{
+ if (x->list_im) {
+ outlet_anything(x->list_im_out, gensym("matrix"), x->size, x->list_im);
+ outlet_anything(x->list_re_out, gensym("matrix"), x->size, x->list_re);
+ }
+}
+
+static void zeroFloatArray (int n, t_float *f)
+{
+ while (n--)
+ *f++ = 0.0f;
+}
+
+static void writeFloatIntoList (int n, t_atom *l, t_float *f)
+{
+ for (;n--;f++, l++)
+ SETFLOAT (l, *f);
+}
+static void readFloatFromList (int n, t_atom *l, t_float *f)
+{
+ while (n--)
+ *f++ = atom_getfloat (l++);
+}
+
+static void multiplyVector (int n, t_float *f, t_float fac)
+{
+ while (n--)
+ *f++ *= fac;
+}
+
+static void mtxIFFTMatrixCold (MtxIFFT *x, t_symbol *s,
+ int argc, t_atom *argv)
+{
+ int rows = atom_getint (argv++);
+ int columns = atom_getint (argv++);
+ int size = rows * columns;
+ int in_size = argc-2;
+ t_atom *list_re = x->list_re;
+ t_atom *list_im = x->list_im;
+ t_float *f_re = x->f_re;
+ t_float *f_im = x->f_im;
+
+ /* fftsize check */
+ if (!size)
+ post("mtx_ifft: invalid dimensions");
+ else if (in_size<size)
+ post("mtx_ifft: sparse matrix not yet supported: use \"mtx_check\"");
+ else if (columns < 4){
+ post("mtx_ifft: matrix must have at least 4 columns");
+ }
+ else if (columns == (1 << ilog2(columns))) {
+ /* ok, prepare real-part of FFT! */
+
+ x->renorm_fac = 1.0f / columns;
+
+ /* memory things */
+ f_re=(t_float*)realloc(f_re, sizeof(t_float)*size);
+ f_im=(t_float*)realloc(f_im, sizeof(t_float)*size);
+ list_re=(t_atom*)realloc(list_re, sizeof(t_atom)*(size+2));
+ list_im=(t_atom*)realloc(list_im, sizeof(t_atom)*(size+2));
+
+ x->size = size;
+ x->list_im = list_im;
+ x->list_re = list_re;
+ x->f_re = f_re;
+ x->f_im = f_im;
+
+ /* main part */
+ readFloatFromList (size, argv, f_im);
+
+ }
+ else
+ post("mtx_rowfft: rowvector size no power of 2!");
+
+}
+
+
+static void mtxIFFTMatrixHot (MtxIFFT *x, t_symbol *s,
+ int argc, t_atom *argv)
+{
+ int rows = atom_getint (argv++);
+ int columns = atom_getint (argv++);
+ int size = rows * columns;
+ int in_size = argc-2;
+ int fft_count;
+ t_atom *list_re = x->list_re;
+ t_atom *list_im = x->list_im;
+ t_float *f_re = x->f_re;
+ t_float *f_im = x->f_im;
+
+ /* fftsize check */
+ if (!size)
+ post("mtx_ifft: invalid dimensions");
+ else if (in_size<size)
+ post("mtx_ifft: sparse matrix not yet supported: use \"mtx_check\"");
+ else if (size != x->size)
+ post("mtx_ifft: left matrix has other dimensions than right matrix");
+ else if (columns < 4){
+ post("mtx_ifft: matrix must have at least 4 columns");
+ }
+ else if (columns == (1 << ilog2(columns))) {
+ /* ok, do the FFT! */
+
+ /* main part */
+ readFloatFromList (size, argv, f_re);
+
+ fft_count = rows;
+ list_re += 2;
+ list_im += 2;
+ while (fft_count--){
+ mayer_ifft (columns, f_re, f_im);
+ multiplyVector (columns, f_re, x->renorm_fac);
+ multiplyVector (columns, f_im, x->renorm_fac);
+ writeFloatIntoList (columns, list_re, f_re);
+ writeFloatIntoList (columns, list_im, f_im);
+ f_im += columns;
+ f_re += columns;
+ list_re += columns;
+ list_im += columns;
+ }
+
+ list_re = x->list_re;
+ list_im = x->list_im;
+
+ SETSYMBOL(list_re, gensym("matrix"));
+ SETSYMBOL(list_im, gensym("matrix"));
+ SETFLOAT(list_re, rows);
+ SETFLOAT(list_im, rows);
+ SETFLOAT(list_re+1, columns);
+ SETFLOAT(list_im+1, columns);
+ outlet_anything(x->list_im_out, gensym("matrix"),
+ x->size+2, list_im);
+ outlet_anything(x->list_re_out, gensym("matrix"),
+ x->size+2, list_re);
+ }
+ else
+ post("mtx_rowfft: rowvector size no power of 2!");
+
+}
+
+void mtx_ifft_setup (void)
+{
+ mtx_ifft_class = class_new
+ (gensym("mtx_ifft"),
+ (t_newmethod) newMtxIFFT,
+ (t_method) deleteMtxIFFT,
+ sizeof (MtxIFFT),
+ CLASS_DEFAULT, A_GIMME, 0);
+ class_addbang (mtx_ifft_class, (t_method) mtxIFFTBang);
+ class_addmethod (mtx_ifft_class, (t_method) mtxIFFTMatrixHot, gensym("matrix"), A_GIMME,0);
+ class_addmethod (mtx_ifft_class, (t_method) mtxIFFTMatrixCold, gensym(""), A_GIMME,0);
+}
+
+void iemtx_ifft_setup(void){
+ mtx_ifft_setup();
+}