summaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorThibaut Mattern <tmattern@users.sourceforge.net>2003-09-14 12:30:36 +0000
committerThibaut Mattern <tmattern@users.sourceforge.net>2003-09-14 12:30:36 +0000
commit6c7c3f336c619065fba9fce7e876ec86bce0b838 (patch)
treeb9b741131db827c398107782cc1ebc4e04f7cdfe /src
parentdf62f1a3465a223ad8e229b35db874b8311ed946 (diff)
downloadxine-lib-6c7c3f336c619065fba9fce7e876ec86bce0b838.tar.gz
xine-lib-6c7c3f336c619065fba9fce7e876ec86bce0b838.tar.bz2
New viz plugin.
CVS patchset: 5371 CVS date: 2003/09/14 12:30:36
Diffstat (limited to 'src')
-rw-r--r--src/post/visualizations/fft.c210
-rw-r--r--src/post/visualizations/fft.h50
-rw-r--r--src/post/visualizations/fftgraph.c533
3 files changed, 793 insertions, 0 deletions
diff --git a/src/post/visualizations/fft.c b/src/post/visualizations/fft.c
new file mode 100644
index 000000000..5dfc40eab
--- /dev/null
+++ b/src/post/visualizations/fft.c
@@ -0,0 +1,210 @@
+/*
+ * Copyright (C) 2000-2002 the xine project
+ *
+ * This file is part of xine, a free video player.
+ *
+ * xine is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * xine is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
+ *
+ * FFT code by Steve Haehnichen, originally licensed under GPL v1
+ * modified by Thibaut Mattern (tmattern@noos.fr) to remove global vars
+ */
+#include <stdio.h>
+#include <stdlib.h>
+#include <math.h>
+
+#include "fft.h"
+
+/**************************************************************************
+ * fft specific decode functions
+ *************************************************************************/
+
+# define SINE(x) (fft->SineTable[(x)])
+# define COSINE(x) (fft->CosineTable[(x)])
+# define WINDOW(x) (fft->WinTable[(x)])
+
+#define PERMUTE(x, y) reverse((x), (y))
+
+/* Number of samples in one "frame" */
+#define SAMPLES (1 << bits)
+
+#define REAL(x) wave[(x)].re
+#define IMAG(x) wave[(x)].im
+#define ALPHA 0.54
+
+/*
+ * Bit reverser for unsigned ints
+ * Reverses 'bits' bits.
+ */
+static inline const unsigned int
+reverse (unsigned int val, int bits)
+{
+ unsigned int retn = 0;
+
+ while (bits--)
+ {
+ retn <<= 1;
+ retn |= (val & 1);
+ val >>= 1;
+ }
+ return (retn);
+}
+
+/*
+ * Here is the real work-horse.
+ * It's a generic FFT, so nothing is lost or approximated.
+ * The samples in wave[] should be in order, and they
+ * will be decimated when fft() returns.
+ */
+void fft_compute (fft_t *fft, complex_t wave[])
+{
+ register int loop, loop1, loop2;
+ unsigned i1; /* going to right shift this */
+ int i2, i3, i4, y;
+ double a1, a2, b1, b2, z1, z2;
+ int bits = fft->bits;
+
+ i1 = SAMPLES / 2;
+ i2 = 1;
+
+ /* perform the butterflys */
+
+ for (loop = 0; loop < bits; loop++)
+ {
+ i3 = 0;
+ i4 = i1;
+
+ for (loop1 = 0; loop1 < i2; loop1++)
+ {
+ y = PERMUTE(i3 / (int)i1, bits);
+ z1 = COSINE(y);
+ z2 = -SINE(y);
+
+ for (loop2 = i3; loop2 < i4; loop2++)
+ {
+ a1 = REAL(loop2);
+ a2 = IMAG(loop2);
+
+ b1 = z1 * REAL(loop2+i1) - z2 * IMAG(loop2+i1);
+ b2 = z2 * REAL(loop2+i1) + z1 * IMAG(loop2+i1);
+
+ REAL(loop2) = a1 + b1;
+ IMAG(loop2) = a2 + b2;
+
+ REAL(loop2+i1) = a1 - b1;
+ IMAG(loop2+i1) = a2 - b2;
+ }
+
+ i3 += (i1 << 1);
+ i4 += (i1 << 1);
+ }
+
+ i1 >>= 1;
+ i2 <<= 1;
+ }
+}
+
+/*
+ * Initializer for FFT routines. Currently only sets up tables.
+ * - Generates scaled lookup tables for sin() and cos()
+ * - Fills a table for the Hamming window function
+ */
+fft_t *fft_new (int bits)
+{
+ fft_t *fft;
+ int i;
+ const double TWOPIoN = (atan(1.0) * 8.0) / (double)SAMPLES;
+ const double TWOPIoNm1 = (atan(1.0) * 8.0) / (double)(SAMPLES - 1);
+
+ printf("fft_new: bits=%d\n", bits);
+
+ fft = (fft_t*)malloc(sizeof(fft_t));
+ if (!fft)
+ return NULL;
+
+ fft->bits = bits;
+ fft->SineTable = malloc (sizeof(double) * SAMPLES);
+ fft->CosineTable = malloc (sizeof(double) * SAMPLES);
+ fft->WinTable = malloc (sizeof(double) * SAMPLES);
+ for (i=0; i < SAMPLES; i++)
+ {
+ fft->SineTable[i] = sin((double) i * TWOPIoN);
+ fft->CosineTable[i] = cos((double) i * TWOPIoN);
+ /*
+ * Generalized Hamming window function.
+ * Set ALPHA to 0.54 for a hanning window. (Good idea)
+ */
+ fft->WinTable[i] = ALPHA + ((1.0 - ALPHA)
+ * cos (TWOPIoNm1 * (i - SAMPLES/2)));
+ }
+ return fft;
+}
+
+void fft_dispose(fft_t *fft)
+{
+ free(fft->SineTable);
+ free(fft->CosineTable);
+ free(fft->WinTable);
+ free(fft);
+}
+
+/*
+ * Apply some windowing function to the samples.
+ */
+void fft_window (fft_t *fft, complex_t wave[])
+{
+ int i;
+ int bits = fft->bits;
+
+ for (i = 0; i < SAMPLES; i++)
+ {
+ REAL(i) *= WINDOW(i);
+ IMAG(i) *= WINDOW(i);
+ }
+}
+
+/*
+ * Calculate amplitude of component n in the decimated wave[] array.
+ */
+double fft_amp (int n, complex_t wave[], int bits)
+{
+ n = PERMUTE (n, bits);
+ return (hypot (REAL(n), IMAG(n)));
+}
+
+/*
+ * Calculate phase of component n in the decimated wave[] array.
+ */
+double fft_phase (int n, complex_t wave[], int bits)
+{
+ n = PERMUTE (n, bits);
+ if (REAL(n) != 0.0)
+ return (atan (IMAG(n) / REAL(n)));
+ else
+ return (0.0);
+}
+
+/*
+ * Scale sampled values.
+ * Do this *before* the fft.
+ */
+void fft_scale (complex_t wave[], int bits)
+{
+ int i;
+
+ for (i = 0; i < SAMPLES; i++) {
+ wave[i].re /= SAMPLES;
+ wave[i].im /= SAMPLES;
+ }
+}
diff --git a/src/post/visualizations/fft.h b/src/post/visualizations/fft.h
new file mode 100644
index 000000000..707879641
--- /dev/null
+++ b/src/post/visualizations/fft.h
@@ -0,0 +1,50 @@
+/*
+ * Copyright (C) 2000-2002 the xine project
+ *
+ * This file is part of xine, a free video player.
+ *
+ * xine is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * xine is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
+ *
+ */
+#ifndef FFT_H
+#define FFT_H
+
+struct complex_s
+{
+ double re;
+ double im;
+};
+typedef struct complex_s complex_t;
+
+
+struct fft_s {
+ int bits;
+ double *SineTable;
+ double *CosineTable;
+ double *WinTable;
+};
+typedef struct fft_s fft_t;
+
+fft_t *fft_new (int bits);
+void fft_dispose(fft_t *fft);
+
+void fft_compute (fft_t *fft, complex_t wave[]);
+void fft_window (fft_t *fft, complex_t wave[]);
+
+double fft_amp (int n, complex_t wave[], int bits);
+double fft_phase (int n, complex_t wave[], int bits);
+void fft_scale (complex_t wave[], int bits);
+
+#endif /* FFT_H */
diff --git a/src/post/visualizations/fftgraph.c b/src/post/visualizations/fftgraph.c
new file mode 100644
index 000000000..c9e92cb12
--- /dev/null
+++ b/src/post/visualizations/fftgraph.c
@@ -0,0 +1,533 @@
+/*
+ * Copyright (C) 2000-2002 the xine project
+ *
+ * This file is part of xine, a free video player.
+ *
+ * xine is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * xine is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA
+ *
+ * FftGraph Visualization Post Plugin For xine
+ * by Thibaut Mattern (tmattern@noos.fr)
+ *
+ * $Id: fftgraph.c,v 1.1 2003/09/14 12:30:36 tmattern Exp $
+ *
+ */
+
+#include <stdio.h>
+#include <math.h>
+
+#include <assert.h>
+
+#include "xine_internal.h"
+#include "xineutils.h"
+#include "post.h"
+#include "bswap.h"
+#include "fft.h"
+
+#define FPS 20
+
+#define FFTGRAPH_WIDTH 512
+#define FFTGRAPH_HEIGHT 256
+
+#define FFT_BITS 11
+#define NUMSAMPLES (1 << FFT_BITS)
+
+#define MAXCHANNELS 6
+
+typedef struct post_plugin_fftgraph_s post_plugin_fftgraph_t;
+
+struct post_plugin_fftgraph_s {
+ post_plugin_t post;
+
+ /* private data */
+ xine_video_port_t *vo_port;
+ xine_stream_t *stream;
+
+ double ratio;
+
+ int data_idx;
+ complex_t wave[MAXCHANNELS][NUMSAMPLES];
+ audio_buffer_t buf; /* dummy buffer just to hold a copy of audio data */
+
+ int bits;
+ int mode;
+ int channels;
+ int sample_rate;
+ int sample_counter;
+ int samples_per_frame;
+
+ fft_t *fft;
+ uint32_t map[FFTGRAPH_HEIGHT][FFTGRAPH_WIDTH / 2];
+ int cur_line;
+ int lines_per_channel;
+
+ uint32_t yuy2_colors[8192];
+};
+
+/*
+ * fade function
+ */
+static void fade(int r1, int g1, int b1,
+ int r2, int g2, int b2,
+ uint32_t *yuy2_colors, int steps) {
+ int i, r, g, b, y, u, v;
+
+ for (i = 0; i < steps; i++) {
+ r = r1 + (r2 - r1) * i / steps;
+ g = g1 + (g2 - g1) * i / steps;
+ b = b1 + (b2 - b1) * i / steps;
+
+ y = COMPUTE_Y(r, g, b);
+ u = COMPUTE_U(r, g, b);
+ v = COMPUTE_V(r, g, b);
+
+ *(yuy2_colors + i) = be2me_32((y << 24) |
+ (u << 16) |
+ (y << 8) |
+ v);
+
+ }
+}
+
+static void draw_fftgraph(post_plugin_fftgraph_t *this, vo_frame_t *frame) {
+
+ int i, c, y;
+ int map_ptr;
+ int amp_int;
+ float amp_float;
+ uint32_t yuy2_white;
+ int line, line_min, line_max;
+
+ yuy2_white = be2me_32((0xFF << 24) |
+ (0x80 << 16) |
+ (0xFF << 8) |
+ 0x80);
+
+ for (c = 0; c < this->channels; c++){
+ /* perform FFT for channel data */
+ fft_window(this->fft, this->wave[c]);
+ fft_scale(this->wave[c], this->fft->bits);
+ fft_compute(this->fft, this->wave[c]);
+
+ /* plot the FFT points for the channel */
+ line = this->cur_line + c * this->lines_per_channel;
+
+ for (i = 0; i < FFTGRAPH_WIDTH / 2; i++) {
+ amp_float = fft_amp(i, this->wave[c], this->fft->bits);
+ amp_int = (int)(amp_float);
+ if (amp_int > 8191)
+ amp_int = 8191;
+ if (amp_int < 0)
+ amp_int = 0;
+
+ this->map[line][i] = this->yuy2_colors[amp_int];
+ }
+ }
+
+ this->cur_line = (this->cur_line + 1) % this->lines_per_channel;
+
+ /* scrolling map */
+ map_ptr = 0;
+
+ for(c = 0; c < this->channels; c++) {
+ line = this->cur_line + c * this->lines_per_channel;
+ line_min = c * this->lines_per_channel;
+ line_max = (c + 1) * this->lines_per_channel;
+
+ for(y = line; y < line_max; y++) {
+ xine_fast_memcpy(((uint32_t *)frame->base[0]) + map_ptr,
+ this->map[y],
+ FFTGRAPH_WIDTH * 2);
+ map_ptr += (FFTGRAPH_WIDTH / 2);
+ }
+
+ for(y = line_min; y < line; y++) {
+ xine_fast_memcpy(((uint32_t *)frame->base[0]) + map_ptr,
+ this->map[y],
+ FFTGRAPH_WIDTH * 2);
+ map_ptr += (FFTGRAPH_WIDTH / 2);
+ }
+ }
+
+ /* top line */
+ for (map_ptr = 0; map_ptr < FFTGRAPH_WIDTH / 2; map_ptr++)
+ ((uint32_t *)frame->base[0])[map_ptr] = yuy2_white;
+
+ /* lines under each channel */
+ for (c = 0; c < this->channels; c++){
+ for (i = 0, map_ptr = ((FFTGRAPH_HEIGHT * (c+1) / this->channels -1 ) * FFTGRAPH_WIDTH) / 2;
+ i < FFTGRAPH_WIDTH / 2; i++, map_ptr++)
+ ((uint32_t *)frame->base[0])[map_ptr] = yuy2_white;
+ }
+
+}
+
+/**************************************************************************
+ * xine video post plugin functions
+ *************************************************************************/
+
+typedef struct post_fftgraph_out_s post_fftgraph_out_t;
+struct post_fftgraph_out_s {
+ xine_post_out_t out;
+ post_plugin_fftgraph_t *post;
+};
+
+static int fftgraph_rewire_audio(xine_post_out_t *output_gen, void *data)
+{
+ post_fftgraph_out_t *output = (post_fftgraph_out_t *)output_gen;
+ xine_audio_port_t *old_port = *(xine_audio_port_t **)output_gen->data;
+ xine_audio_port_t *new_port = (xine_audio_port_t *)data;
+ post_plugin_fftgraph_t *this = (post_plugin_fftgraph_t *)output->post;
+
+ if (!data)
+ return 0;
+ if (this->stream) {
+ /* register our stream at the new output port */
+ old_port->close(old_port, this->stream);
+ new_port->open(new_port, this->stream, this->bits, this->sample_rate, this->mode);
+ }
+ /* reconnect ourselves */
+ *(xine_audio_port_t **)output_gen->data = new_port;
+ return 1;
+}
+
+static int fftgraph_rewire_video(xine_post_out_t *output_gen, void *data)
+{
+ post_fftgraph_out_t *output = (post_fftgraph_out_t *)output_gen;
+ xine_video_port_t *old_port = *(xine_video_port_t **)output_gen->data;
+ xine_video_port_t *new_port = (xine_video_port_t *)data;
+ post_plugin_fftgraph_t *this = (post_plugin_fftgraph_t *)output->post;
+
+ if (!data)
+ return 0;
+ if (this->stream) {
+ /* register our stream at the new output port */
+ old_port->close(old_port, this->stream);
+ new_port->open(new_port, this->stream);
+ }
+ /* reconnect ourselves */
+ *(xine_video_port_t **)output_gen->data = new_port;
+ return 1;
+}
+
+static int mode_channels( int mode ) {
+ switch( mode ) {
+ case AO_CAP_MODE_MONO:
+ return 1;
+ case AO_CAP_MODE_STEREO:
+ return 2;
+ case AO_CAP_MODE_4CHANNEL:
+ return 4;
+ case AO_CAP_MODE_5CHANNEL:
+ return 5;
+ case AO_CAP_MODE_5_1CHANNEL:
+ return 6;
+ }
+ return 0;
+}
+
+static int fftgraph_port_open(xine_audio_port_t *port_gen, xine_stream_t *stream,
+ uint32_t bits, uint32_t rate, int mode) {
+
+ post_audio_port_t *port = (post_audio_port_t *)port_gen;
+ post_plugin_fftgraph_t *this = (post_plugin_fftgraph_t *)port->post;
+ int i,j;
+ uint32_t *color_ptr;
+ uint32_t last_color, yuy2_black;
+
+ printf("fftgraph_port_open, port_gen=%p, stream=%p, this=%p\n", port_gen, stream, this);
+
+ this->ratio = (double)FFTGRAPH_WIDTH / (double)FFTGRAPH_HEIGHT;
+
+ this->bits = bits;
+ this->mode = mode;
+ this->channels = mode_channels(mode);
+ if( this->channels > MAXCHANNELS )
+ this->channels = MAXCHANNELS;
+ this->lines_per_channel = FFTGRAPH_HEIGHT / this->channels;
+ this->samples_per_frame = rate / FPS;
+ this->sample_rate = rate;
+ this->stream = stream;
+ this->data_idx = 0;
+
+ this->fft = fft_new(FFT_BITS);
+
+ this->cur_line = 0;
+
+ /* compute colors */
+ color_ptr = this->yuy2_colors;
+ /* black -> red */
+ fade(0, 0, 0,
+ 128, 0, 0,
+ color_ptr, 128);
+ color_ptr += 128;
+
+ /* red -> blue */
+ fade(128, 0, 0,
+ 40, 0, 160,
+ color_ptr, 256);
+ color_ptr += 256;
+
+ /* blue -> green */
+ fade(40, 0, 160,
+ 40, 160, 70,
+ color_ptr, 1024);
+ color_ptr += 1024;
+
+ /* green -> white */
+ fade(40, 160, 70,
+ 255, 255, 255,
+ color_ptr, 2048);
+ color_ptr += 2048;
+
+ last_color = *(color_ptr - 1);
+
+ /* white */
+ for (i = 0; i < 8192 - 128 - 256 - 1024 - 2048; i++) {
+ *color_ptr = last_color;
+ color_ptr++;
+ }
+
+ /* clear the map */
+ yuy2_black = be2me_32((0x00 << 24) |
+ (0x80 << 16) |
+ (0x00 << 8) |
+ 0x80);
+ for (i = 0; i < FFTGRAPH_HEIGHT; i++) {
+ for (j = 0; j < (FFTGRAPH_WIDTH / 2); j++) {
+ this->map[i][j] = yuy2_black;
+ }
+ }
+
+ return port->original_port->open(port->original_port, stream, bits, rate, mode );
+}
+
+static void fftgraph_port_close(xine_audio_port_t *port_gen, xine_stream_t *stream ) {
+
+ post_audio_port_t *port = (post_audio_port_t *)port_gen;
+ post_plugin_fftgraph_t *this = (post_plugin_fftgraph_t *)port->post;
+
+ this->stream = NULL;
+ fft_dispose(this->fft);
+ this->fft = NULL;
+
+ port->original_port->close(port->original_port, stream );
+}
+
+static void fftgraph_port_put_buffer (xine_audio_port_t *port_gen,
+ audio_buffer_t *buf, xine_stream_t *stream) {
+
+ post_audio_port_t *port = (post_audio_port_t *)port_gen;
+ post_plugin_fftgraph_t *this = (post_plugin_fftgraph_t *)port->post;
+ vo_frame_t *frame;
+ int16_t *data;
+ int8_t *data8;
+ int samples_used = 0;
+ uint64_t vpts = buf->vpts;
+ int i, c;
+
+ /* HACK: compute a pts using metronom internals */
+ if (!vpts) {
+ metronom_t *metronom = this->stream->metronom;
+ pthread_mutex_lock(&metronom->lock);
+ vpts = metronom->audio_vpts - metronom->vpts_offset;
+ pthread_mutex_unlock(&metronom->lock);
+ }
+
+ /* make a copy of buf data for private use */
+ if( this->buf.mem_size < buf->mem_size ) {
+ this->buf.mem = realloc(this->buf.mem, buf->mem_size);
+ this->buf.mem_size = buf->mem_size;
+ }
+ memcpy(this->buf.mem, buf->mem,
+ buf->num_frames*this->channels*((this->bits == 8)?1:2));
+ this->buf.num_frames = buf->num_frames;
+
+ /* pass data to original port */
+ port->original_port->put_buffer(port->original_port, buf, stream );
+
+ /* we must not use original data anymore, it should have already being moved
+ * to the fifo of free audio buffers. just use our private copy instead.
+ */
+ buf = &this->buf;
+
+ this->sample_counter += buf->num_frames;
+
+ do {
+
+ if( this->bits == 8 ) {
+ data8 = (int8_t *)buf->mem;
+ data8 += samples_used * this->channels;
+
+ /* scale 8 bit data to 16 bits and convert to signed as well */
+ for( i = 0; i < buf->num_frames && this->data_idx < NUMSAMPLES;
+ i++, this->data_idx++, data8 += this->channels ) {
+ for( c = 0; c < this->channels; c++){
+ this->wave[c][this->data_idx].re = (double)(data8[c] << 8) - 0x8000;
+ this->wave[c][this->data_idx].im = 0;
+ }
+ }
+ } else {
+ data = buf->mem;
+ data += samples_used * this->channels;
+
+ for( i = 0; i < buf->num_frames && this->data_idx < NUMSAMPLES;
+ i++, this->data_idx++, data += this->channels ) {
+ for( c = 0; c < this->channels; c++){
+ this->wave[c][this->data_idx].re = (double)data[c];
+ this->wave[c][this->data_idx].im = 0;
+ }
+ }
+ }
+
+ if( this->sample_counter >= this->samples_per_frame &&
+ this->data_idx == NUMSAMPLES ) {
+
+ this->data_idx = 0;
+ samples_used += this->samples_per_frame;
+
+ frame = this->vo_port->get_frame (this->vo_port, FFTGRAPH_WIDTH, FFTGRAPH_HEIGHT,
+ this->ratio, XINE_IMGFMT_YUY2,
+ VO_BOTH_FIELDS);
+ frame->extra_info->invalid = 1;
+ frame->bad_frame = 0;
+ frame->pts = vpts;
+ vpts = 0;
+
+ frame->duration = 90000 * this->samples_per_frame / this->sample_rate;
+ this->sample_counter -= this->samples_per_frame;
+
+ draw_fftgraph(this, frame);
+
+ frame->draw(frame, stream);
+ frame->free(frame);
+ }
+ } while( this->sample_counter >= this->samples_per_frame );
+}
+
+static void fftgraph_dispose(post_plugin_t *this_gen)
+{
+ post_plugin_fftgraph_t *this = (post_plugin_fftgraph_t *)this_gen;
+ xine_post_out_t *output = (xine_post_out_t *)xine_list_last_content(this_gen->output);
+ xine_video_port_t *port = *(xine_video_port_t **)output->data;
+
+ if (this->stream)
+ port->close(port, this->stream);
+
+ free(this->post.xine_post.audio_input);
+ free(this->post.xine_post.video_input);
+ free(xine_list_first_content(this->post.input));
+ free(xine_list_first_content(this->post.output));
+ xine_list_free(this->post.input);
+ xine_list_free(this->post.output);
+ if(this->buf.mem)
+ free(this->buf.mem);
+ free(this);
+}
+
+/* plugin class functions */
+static post_plugin_t *fftgraph_open_plugin(post_class_t *class_gen, int inputs,
+ xine_audio_port_t **audio_target,
+ xine_video_port_t **video_target)
+{
+ post_plugin_fftgraph_t *this = (post_plugin_fftgraph_t *)malloc(sizeof(post_plugin_fftgraph_t));
+ xine_post_in_t *input = (xine_post_in_t *)malloc(sizeof(xine_post_in_t));
+ post_fftgraph_out_t *output = (post_fftgraph_out_t *)malloc(sizeof(post_fftgraph_out_t));
+ post_fftgraph_out_t *outputv = (post_fftgraph_out_t *)malloc(sizeof(post_fftgraph_out_t));
+ post_audio_port_t *port;
+
+ if (!this || !input || !output || !outputv || !video_target || !video_target[0] ||
+ !audio_target || !audio_target[0] ) {
+ free(this);
+ free(input);
+ free(output);
+ free(outputv);
+ return NULL;
+ }
+
+ this->sample_counter = 0;
+ this->stream = NULL;
+ this->vo_port = video_target[0];
+ this->buf.mem = NULL;
+ this->buf.mem_size = 0;
+
+ port = post_intercept_audio_port(&this->post, audio_target[0]);
+ port->port.open = fftgraph_port_open;
+ port->port.close = fftgraph_port_close;
+ port->port.put_buffer = fftgraph_port_put_buffer;
+
+ input->name = "audio in";
+ input->type = XINE_POST_DATA_AUDIO;
+ input->data = (xine_audio_port_t *)&port->port;
+
+ output->out.name = "audio out";
+ output->out.type = XINE_POST_DATA_AUDIO;
+ output->out.data = (xine_audio_port_t **)&port->original_port;
+ output->out.rewire = fftgraph_rewire_audio;
+ output->post = this;
+
+ outputv->out.name = "generated video";
+ outputv->out.type = XINE_POST_DATA_VIDEO;
+ outputv->out.data = (xine_video_port_t **)&this->vo_port;
+ outputv->out.rewire = fftgraph_rewire_video;
+ outputv->post = this;
+
+ this->post.xine_post.audio_input = (xine_audio_port_t **)malloc(sizeof(xine_audio_port_t *) * 2);
+ this->post.xine_post.audio_input[0] = &port->port;
+ this->post.xine_post.audio_input[1] = NULL;
+ this->post.xine_post.video_input = (xine_video_port_t **)malloc(sizeof(xine_video_port_t *) * 1);
+ this->post.xine_post.video_input[0] = NULL;
+
+ this->post.input = xine_list_new();
+ this->post.output = xine_list_new();
+
+ xine_list_append_content(this->post.input, input);
+ xine_list_append_content(this->post.output, output);
+ xine_list_append_content(this->post.output, outputv);
+
+ this->post.dispose = fftgraph_dispose;
+
+ return &this->post;
+}
+
+static char *fftgraph_get_identifier(post_class_t *class_gen)
+{
+ return "fftgraph";
+}
+
+static char *fftgraph_get_description(post_class_t *class_gen)
+{
+ return "fftgraph Visualization Post Plugin";
+}
+
+static void fftgraph_class_dispose(post_class_t *class_gen)
+{
+ free(class_gen);
+}
+
+/* plugin class initialization function */
+void *fftgraph_init_plugin(xine_t *xine, void *data)
+{
+ post_class_t *class = (post_class_t *)malloc(sizeof(post_class_t));
+
+ if (!class)
+ return NULL;
+
+ class->open_plugin = fftgraph_open_plugin;
+ class->get_identifier = fftgraph_get_identifier;
+ class->get_description = fftgraph_get_description;
+ class->dispose = fftgraph_class_dispose;
+
+ return class;
+}