diff options
author | Thibaut Mattern <tmattern@users.sourceforge.net> | 2003-09-14 12:30:36 +0000 |
---|---|---|
committer | Thibaut Mattern <tmattern@users.sourceforge.net> | 2003-09-14 12:30:36 +0000 |
commit | 6c7c3f336c619065fba9fce7e876ec86bce0b838 (patch) | |
tree | b9b741131db827c398107782cc1ebc4e04f7cdfe /src | |
parent | df62f1a3465a223ad8e229b35db874b8311ed946 (diff) | |
download | xine-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.c | 210 | ||||
-rw-r--r-- | src/post/visualizations/fft.h | 50 | ||||
-rw-r--r-- | src/post/visualizations/fftgraph.c | 533 |
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; +} |