changeset 11437:7826e4e376c7

XviD Api4 driver from http://ed.gomez.free.fr/
author iive
date Wed, 12 Nov 2003 00:50:43 +0000
parents 6afc2d6f5a08
children b98d1a3e1ea1
files libmpcodecs/vd_xvid4.c libmpcodecs/ve_xvid4.c
diffstat 2 files changed, 1277 insertions(+), 0 deletions(-) [+]
line wrap: on
line diff
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/libmpcodecs/vd_xvid4.c	Wed Nov 12 00:50:43 2003 +0000
@@ -0,0 +1,247 @@
+/*****************************************************************************
+ *
+ *  - XviD 1.0 decoder module for mplayer/mencoder -
+ *
+ *  Copyright(C) 2003 Marco Belli <elcabesa@inwind.it>
+ *               2003 Edouard Gomez <ed.gomez@free.fr>
+ *
+ *  This program 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.
+ *
+ *  This program 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
+ *
+ *****************************************************************************/
+
+/*****************************************************************************
+ * Includes
+ ****************************************************************************/
+
+#include <stdio.h>
+#include <stdlib.h>
+
+#include "config.h"
+#include "mp_msg.h"
+
+#ifdef HAVE_XVID4
+
+#include "vd_internal.h"
+#include "m_option.h"
+
+#include <xvid.h>
+
+/*****************************************************************************
+ * Configuration options
+ ****************************************************************************/
+
+static int do_dr2 = 1;
+
+m_option_t xvid_dec_opts[] = {
+	{ "dr2", &do_dr2, CONF_TYPE_FLAG, 0, 0, 1, NULL},
+	{ "nodr2", &do_dr2, CONF_TYPE_FLAG, 0, 1, 0, NULL},
+	{NULL, NULL, 0, 0, 0, 0, NULL}
+};
+
+/*****************************************************************************
+ * Module private data
+ ****************************************************************************/
+
+typedef struct {
+	int cs;
+	unsigned char img_type;
+	void* hdl;
+	mp_image_t* mpi;
+} priv_t;
+
+/*****************************************************************************
+ * Video decoder API function definitions
+ ****************************************************************************/
+
+/*============================================================================
+ * control - to set/get/query special features/parameters
+ *==========================================================================*/
+
+static int control(sh_video_t *sh,int cmd,void* arg,...)
+{
+	return(CONTROL_UNKNOWN);
+}
+
+/*============================================================================
+ * init - initialize the codec
+ *==========================================================================*/
+
+static int init(sh_video_t *sh)
+{
+	xvid_gbl_init_t xvid_ini;
+	xvid_dec_create_t dec_p;
+	priv_t* p;
+	int cs;
+
+	memset(&xvid_ini, 0, sizeof(xvid_gbl_init_t));
+	xvid_ini.version = XVID_VERSION;
+	memset(&dec_p, 0, sizeof(xvid_dec_create_t));
+	dec_p.version = XVID_VERSION;
+
+	if(!mpcodecs_config_vo(sh, sh->disp_w, sh->disp_h, IMGFMT_YV12))
+		return(0);
+
+	switch(sh->codec->outfmt[sh->outfmtidx]){
+	case IMGFMT_YV12:
+		/* We will use our own buffers, this speeds decoding avoiding
+		 * frame memcpy's overhead */
+		cs = (do_dr2)?XVID_CSP_INTERNAL:XVID_CSP_USER;
+		break;
+	case IMGFMT_YUY2:
+		cs = XVID_CSP_YUY2;
+		break;
+	case IMGFMT_UYVY:
+		cs = XVID_CSP_UYVY;
+		break;
+	case IMGFMT_I420: 
+	case IMGFMT_IYUV:
+		/* We will use our own buffers, this speeds decoding avoiding
+		 * frame memcpy's overhead */
+		cs = (do_dr2)?XVID_CSP_INTERNAL:XVID_CSP_USER;
+		break;
+	case IMGFMT_BGR15: 
+		cs = XVID_CSP_RGB555;
+		break;
+	case IMGFMT_BGR16: 
+		cs = XVID_CSP_RGB565;
+		break;
+	case IMGFMT_BGR32:
+		cs = XVID_CSP_BGRA;
+		break;
+	case IMGFMT_YVYU:
+		cs = XVID_CSP_YVYU;
+		break;
+	default:
+		mp_msg(MSGT_DECVIDEO, MSGL_ERR, "Unsupported out_fmt: 0x%X\n",
+		       sh->codec->outfmt[sh->outfmtidx]);
+		return(0);
+	}
+
+	if(xvid_global(NULL, XVID_GBL_INIT, &xvid_ini, NULL))
+		return(0);
+
+	dec_p.width = sh->disp_w;
+	dec_p.height =  sh->disp_h;
+
+	if(xvid_decore(0, XVID_DEC_CREATE, &dec_p, NULL)<0) {
+		mp_msg(MSGT_DECVIDEO, MSGL_ERR, "XviD init failed\n");
+		return(0);
+	}
+
+	p = (priv_t*)malloc(sizeof(priv_t));
+	p->cs = cs;
+	p->hdl = dec_p.handle;
+	sh->context = p;
+
+	switch(cs) {
+	case XVID_CSP_INTERNAL:
+		p->img_type = MP_IMGTYPE_EXPORT;
+		break;
+	case XVID_CSP_USER:
+		p->img_type = MP_IMGTYPE_STATIC;
+		break;
+	default:
+		p->img_type = MP_IMGTYPE_TEMP;
+		break;
+	}
+
+	return(1);
+}
+
+/*============================================================================
+ * uninit - close the codec
+ *==========================================================================*/
+
+static void uninit(sh_video_t *sh){
+	priv_t* p = sh->context;
+	if(!p)
+		return;
+	xvid_decore(p->hdl,XVID_DEC_DESTROY, NULL, NULL);
+	free(p);
+}
+
+/*============================================================================
+ * decode - decode a frame from stream
+ *==========================================================================*/
+
+static mp_image_t* decode(sh_video_t *sh, void* data, int len, int flags)
+{
+	xvid_dec_frame_t dec;
+	priv_t* p = sh->context;
+
+	mp_image_t* mpi = mpcodecs_get_image(sh, p->img_type,
+					     MP_IMGFLAG_ACCEPT_STRIDE,
+					     sh->disp_w,sh->disp_h);
+
+	if(!data || !mpi || len <= 0)
+		return(NULL);
+
+	memset(&dec,0,sizeof(xvid_dec_frame_t));
+	dec.version = XVID_VERSION;
+
+	dec.bitstream = data;
+	dec.length = len;
+
+	dec.general |= XVID_LOWDELAY;
+
+	dec.output.csp = p->cs;   
+
+	if(p->cs != XVID_CSP_INTERNAL) {
+		dec.output.plane[0] = mpi->planes[0];
+		dec.output.plane[1] = mpi->planes[1];
+		dec.output.plane[2] = mpi->planes[2];
+
+		dec.output.stride[0] = mpi->stride[0]; 
+		dec.output.stride[1] = mpi->stride[1]; 
+		dec.output.stride[2] = mpi->stride[2];
+	}
+
+	if(xvid_decore(p->hdl, XVID_DEC_DECODE, &dec, NULL) < 0) {
+		mp_msg(MSGT_DECVIDEO, MSGL_ERR, "Decoding error\n");
+		return(NULL);
+	}
+
+	if(p->cs == XVID_CSP_INTERNAL) {
+		mpi->planes[0] = dec.output.plane[0];
+		mpi->planes[1] = dec.output.plane[1];
+		mpi->planes[2] = dec.output.plane[2];
+
+		mpi->stride[0] = dec.output.stride[0];
+		mpi->stride[1] = dec.output.stride[1];
+		mpi->stride[2] = dec.output.stride[2];
+	}
+
+	return(mpi);
+}
+
+/*****************************************************************************
+ * Module structure definition
+ ****************************************************************************/
+
+static vd_info_t info = 
+{
+	"XviD 1.0 decoder",
+	"xvid",
+	"Marco Belli <elcabesa@inwind.it>, Edouard Gomez <ed.gomez@free.fr>",
+	"Marco Belli <elcabesa@inwind.it>, Edouard Gomez <ed.gomez@free.fr>",
+	"No Comment"
+};
+
+LIBVD_EXTERN(xvid)
+
+#endif  /* HAVE_XVID4 */
+
+/* Please do not change that tag comment.
+ * arch-tag: b7d654a5-76ea-4768-9713-2c791567fe7d mplayer xvid decoder module */
--- /dev/null	Thu Jan 01 00:00:00 1970 +0000
+++ b/libmpcodecs/ve_xvid4.c	Wed Nov 12 00:50:43 2003 +0000
@@ -0,0 +1,1030 @@
+/*****************************************************************************
+ *
+ *  - XviD 1.0 export module for mplayer/mencoder -
+ *
+ *  Copyright(C) 2003 Marco Belli <elcabesa@inwind.it>
+ *               2003 Edouard Gomez <ed.gomez@free.fr>
+ *
+ *  This program 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.
+ *
+ *  This program 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
+ *
+ ****************************************************************************/
+
+/*****************************************************************************
+ * Includes
+ ****************************************************************************/
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <errno.h>
+#include <math.h>
+#include <limits.h>
+
+#include "../config.h"
+#include "../mp_msg.h"
+
+#ifdef HAVE_XVID4
+
+#include "codec-cfg.h"
+#include "stream.h"
+#include "demuxer.h"
+#include "stheader.h"
+
+#include "muxer.h"
+
+#include "img_format.h"
+#include "mp_image.h"
+#include "vf.h"
+
+#include <xvid.h>
+#include <stdio.h>
+#include <stdarg.h>
+#include <limits.h>
+
+#include "m_option.h"
+
+#define XVID_FIRST_PASS_FILENAME "xvid-twopass.stats"
+#define FINE (!0)
+#define BAD (!FINE)
+
+/*****************************************************************************
+ * Configuration options
+ ****************************************************************************/
+
+static int xvidenc_bitrate = 0;
+static int xvidenc_pass = 0;
+static float xvidenc_quantizer = 0;
+
+static int xvidenc_packed = 0;
+static int xvidenc_closed_gop = 1;
+static int xvidenc_interlaced = 0;
+static int xvidenc_quarterpel = 0;
+static int xvidenc_gmc = 0;
+static int xvidenc_trellis = 0;
+static int xvidenc_cartoon = 0;
+static int xvidenc_hqacpred = 1;
+static int xvidenc_chromame = 0;
+static int xvidenc_vhq = 0;
+static int xvidenc_motion = 6;
+static int xvidenc_stats = 0;
+static int xvidenc_max_key_interval = 0;
+static int xvidenc_frame_drop_ratio = 0;
+static int xvidenc_greyscale = 0;
+
+static int xvidenc_max_bframes = 0;
+static int xvidenc_bquant_ratio = 150;
+static int xvidenc_bquant_offset = 100;
+static int xvidenc_bframe_threshold = 0;
+
+static int xvidenc_min_quant[3] = {2, 2, 2};
+static int xvidenc_max_quant[3] = {31, 31, 31};
+static char *xvidenc_intra_matrix_file = NULL;
+static char *xvidenc_inter_matrix_file = NULL;
+static char *xvidenc_quant_method = NULL;
+
+static int xvidenc_cbr_reaction_delay_factor = 0;
+static int xvidenc_cbr_averaging_period = 0;
+static int xvidenc_cbr_buffer = 0;
+
+static int xvidenc_vbr_keyframe_boost = 0;
+static int xvidenc_vbr_overflow_control_strength = 0;
+static int xvidenc_vbr_curve_compression_high = 0;
+static int xvidenc_vbr_curve_compression_low = 0;
+static int xvidenc_vbr_max_overflow_improvement = 0;
+static int xvidenc_vbr_max_overflow_degradation = 0;
+static int xvidenc_vbr_kfreduction = 0;
+static int xvidenc_vbr_min_key_interval = 0;
+static int xvidenc_vbr_container_frame_overhead = 0;
+
+static char *xvidenc_par = NULL;
+static int xvidenc_par_width = 0;
+static int xvidenc_par_height = 0;
+
+m_option_t xvidencopts_conf[] =
+{
+	/* Standard things mencoder should be able to treat directly */
+	{"bitrate", &xvidenc_bitrate, CONF_TYPE_INT, 0, 0, 0, NULL},
+	{"pass", &xvidenc_pass, CONF_TYPE_INT, CONF_RANGE, 1, 2, NULL},
+	{"fixed_quant", &xvidenc_quantizer, CONF_TYPE_FLOAT, CONF_RANGE, 1, 31, NULL},
+
+	/* Features */
+	{"quant_type", &xvidenc_quant_method, CONF_TYPE_STRING, 0, 0, 0, NULL},
+	{"me_quality", &xvidenc_motion, CONF_TYPE_INT, CONF_RANGE, 0, 6, NULL},
+	{"chroma_me", &xvidenc_chromame, CONF_TYPE_FLAG, 0, 0, 1, NULL},
+	{"vhq", &xvidenc_vhq, CONF_TYPE_INT, CONF_RANGE, 0, 4, NULL},
+	{"max_bframes", &xvidenc_max_bframes, CONF_TYPE_INT, CONF_RANGE, 0, 20, NULL},
+	{"bquant_ratio", &xvidenc_bquant_ratio, CONF_TYPE_INT, CONF_RANGE, 0, 200, NULL},
+	{"bquant_offset", &xvidenc_bquant_offset, CONF_TYPE_INT, CONF_RANGE, 0, 200, NULL},
+	{"bf_threshold", &xvidenc_bframe_threshold, CONF_TYPE_INT, CONF_RANGE, -255, 255, NULL},
+	{"qpel", &xvidenc_quarterpel, CONF_TYPE_FLAG, 0, 0, 1, NULL},
+	{"gmc", &xvidenc_gmc, CONF_TYPE_FLAG, 0, 0, 1, NULL},
+	{"trellis", &xvidenc_trellis, CONF_TYPE_FLAG, 0, 0, 1, NULL},
+	{"packed", &xvidenc_packed, CONF_TYPE_FLAG, 0, 0, 1, NULL},
+	{"closed_gop", &xvidenc_closed_gop, CONF_TYPE_FLAG, 0, 0, 1, NULL},
+	{"interlaced", &xvidenc_interlaced, CONF_TYPE_FLAG, 0, 0, 1, NULL},
+	{"cartoon", &xvidenc_cartoon, CONF_TYPE_FLAG, 0, 0, 1, NULL},
+	{"hq_ac", &xvidenc_hqacpred, CONF_TYPE_FLAG, 0, 0, 1, NULL},
+	{"frame_drop_ratio", &xvidenc_frame_drop_ratio, CONF_TYPE_INT, CONF_RANGE, 0, 100, NULL},
+	{"max_key_interval", &xvidenc_max_key_interval, CONF_TYPE_INT, CONF_MIN, 0, 0, NULL},
+	{"greyscale", &xvidenc_greyscale, CONF_TYPE_FLAG, 0, 0, 1, NULL},
+	{"stats", &xvidenc_stats, CONF_TYPE_FLAG, 0, 0, 1, NULL},
+
+
+	/* section [quantizer] */
+	{"min_iquant", &xvidenc_min_quant[0], CONF_TYPE_INT, CONF_RANGE, 1, 31, NULL},
+	{"max_iquant", &xvidenc_max_quant[0], CONF_TYPE_INT, CONF_RANGE, 1, 31, NULL},
+	{"min_pquant", &xvidenc_min_quant[1], CONF_TYPE_INT, CONF_RANGE, 1, 31, NULL},
+	{"max_pquant", &xvidenc_max_quant[1], CONF_TYPE_INT, CONF_RANGE, 1, 31, NULL},
+	{"min_bquant", &xvidenc_min_quant[2], CONF_TYPE_INT, CONF_RANGE, 1, 31, NULL},
+	{"max_bquant", &xvidenc_max_quant[2], CONF_TYPE_INT, CONF_RANGE, 1, 31, NULL},
+	{"quant_intra_matrix", &xvidenc_intra_matrix_file, CONF_TYPE_STRING, 0, 0, 100, NULL},
+	{"quant_inter_matrix", &xvidenc_inter_matrix_file, CONF_TYPE_STRING, 0, 0, 100, NULL},
+
+	/* section [cbr] */
+	{"cbr_reaction_delay_factor", &xvidenc_cbr_reaction_delay_factor, CONF_TYPE_INT, CONF_RANGE, 0, 100, NULL},
+	{"cbr_averaging_period", &xvidenc_cbr_averaging_period, CONF_TYPE_INT, CONF_MIN, 0, 0, NULL},
+	{"cbr_buffer", &xvidenc_cbr_buffer, CONF_TYPE_INT, CONF_MIN, 0, 0, NULL},
+
+	/* section [vbr] */
+	{"keyframe_boost", &xvidenc_vbr_keyframe_boost, CONF_TYPE_INT, CONF_RANGE, 0, 100, NULL},
+	{"curve_compression_high", &xvidenc_vbr_curve_compression_high, CONF_TYPE_INT, CONF_RANGE, 0, 100, NULL},
+	{"curve_compression_low", &xvidenc_vbr_curve_compression_low, CONF_TYPE_INT, CONF_RANGE, 0, 100, NULL},
+	{"overflow_control_strength", &xvidenc_vbr_overflow_control_strength, CONF_TYPE_INT, CONF_RANGE, 0, 100, NULL},
+	{"max_overflow_improvement", &xvidenc_vbr_max_overflow_improvement, CONF_TYPE_INT, CONF_RANGE, 0, 100, NULL},
+	{"max_overflow_degradation", &xvidenc_vbr_max_overflow_degradation, CONF_TYPE_INT, CONF_RANGE, 0, 100, NULL},
+	{"kfreduction", &xvidenc_vbr_kfreduction, CONF_TYPE_INT, CONF_RANGE, 0, 100, NULL},
+	{"min_key_interval", &xvidenc_vbr_min_key_interval, CONF_TYPE_INT, CONF_MIN, 0, 0, NULL},
+	{"container_frame_overhead", &xvidenc_vbr_container_frame_overhead, CONF_TYPE_INT, CONF_MIN, 0, 0, NULL},
+
+	/* Section Aspect Ratio */
+	{"par", &xvidenc_par, CONF_TYPE_STRING, 0, 0, 0, NULL},
+	{"par_width", &xvidenc_par_width, CONF_TYPE_INT, CONF_RANGE, 0, 255, NULL},
+	{"par_height", &xvidenc_par_height, CONF_TYPE_INT, CONF_RANGE, 0, 255, NULL},
+
+	/* End of the config array */
+	{NULL, 0, 0, 0, 0, 0, NULL}
+};
+
+/*****************************************************************************
+ * Module private data
+ ****************************************************************************/
+
+typedef struct _xvid_mplayer_module_t
+{
+	/* Instance related global vars */
+	void *instance;
+	xvid_gbl_init_t      init;
+	xvid_enc_create_t    create;
+	xvid_enc_frame_t     frame;
+	xvid_plugin_single_t onepass;
+	xvid_plugin_2pass1_t pass1;
+	xvid_plugin_2pass2_t pass2;
+
+	/* This data must survive local block scope, so here it is */
+	xvid_enc_plugin_t    plugins[7];
+	xvid_enc_zone_t      zones[1];
+
+	/* MPEG4 stream buffer */
+	muxer_stream_t *mux;
+
+	/* Stats accumulators */
+	int frames;
+	long long sse_y;
+	long long sse_u;
+	long long sse_v;
+
+	/* Min & Max PSNR */
+	int min_sse_y;
+	int min_sse_u;
+	int min_sse_v;
+	int max_sse_y;
+	int max_sse_u;
+	int max_sse_v;
+} xvid_mplayer_module_t;
+
+static void dispatch_settings(xvid_mplayer_module_t *mod);
+static int set_create_struct(xvid_mplayer_module_t *mod);
+static int set_frame_struct(xvid_mplayer_module_t *mod, mp_image_t *mpi);
+static const char *errorstring(int err);
+
+/*****************************************************************************
+ * Video Filter API function definitions
+ ****************************************************************************/
+
+/*============================================================================
+ * config
+ *==========================================================================*/
+
+static int
+config(struct vf_instance_s* vf,
+       int width, int height, int d_width, int d_height,
+       unsigned int flags, unsigned int outfmt)
+{
+	int err;
+	xvid_mplayer_module_t *mod = (xvid_mplayer_module_t *)vf->priv;
+	
+	/* Complete the muxer initialization */
+	mod->mux->bih->biWidth = width;
+	mod->mux->bih->biHeight = height;
+	mod->mux->bih->biSizeImage = 
+		mod->mux->bih->biWidth * mod->mux->bih->biHeight * 3;
+
+	/* Message the FourCC type */
+	mp_msg(MSGT_MENCODER, MSGL_INFO,
+	       "videocodec: XviD (%dx%d fourcc=%x [%.4s])\n",
+	       width, height, mod->mux->bih->biCompression,
+	       (char *)&mod->mux->bih->biCompression);
+
+	/*--------------------------------------------------------------------
+	 * Dispatch all module settings to XviD structures
+	 *------------------------------------------------------------------*/
+
+	dispatch_settings(mod);
+
+	/*--------------------------------------------------------------------
+	 * Set remaining information in the xvid_enc_create_t structure
+	 *------------------------------------------------------------------*/
+
+	if(set_create_struct(mod) == BAD)
+		return(BAD);
+
+	/*--------------------------------------------------------------------
+	 * Encoder instance creation
+	 *------------------------------------------------------------------*/
+
+	err = xvid_encore(NULL, XVID_ENC_CREATE, &mod->create, NULL);
+
+	if(err<0) {
+		mp_msg(MSGT_MENCODER, MSGL_ERR,
+		       "xvid: xvidcore returned a '%s' error\n", errorstring(err));
+		return(BAD);
+	}
+	
+	/* Store the encoder instance into the private data */
+	mod->instance = mod->create.handle;
+
+	return(FINE);
+}
+
+/*============================================================================
+ * uninit
+ *==========================================================================*/
+
+static void
+uninit(struct vf_instance_s* vf)
+{
+
+	xvid_mplayer_module_t *mod = (xvid_mplayer_module_t *)vf->priv;
+
+	/* Destroy xvid instance */
+	xvid_encore(mod->instance, XVID_ENC_DESTROY, NULL, NULL);
+
+	/* Display stats */
+	if(mod->frames) {
+		int pixels;
+
+		mod->sse_y /= mod->frames;
+		mod->sse_u /= mod->frames;
+		mod->sse_v /= mod->frames;
+
+		pixels = mod->create.width*mod->create.height;
+
+#define SSE2PSNR(sse, nbpixels) \
+((!(sse)) ? 99.99f : 48.131f - 10*(double)log10((double)(sse)/(double)((nbpixels))))
+		mp_msg(MSGT_MENCODER, MSGL_INFO,
+		       "The value 99.99dB is a special value and represents "
+		       "the upper range limit\n");
+		mp_msg(MSGT_MENCODER, MSGL_INFO,
+		       "xvid:     Min PSNR y : %.2f dB, u : %.2f dB, v : %.2f dB\n",
+		       SSE2PSNR(mod->max_sse_y, pixels),
+		       SSE2PSNR(mod->max_sse_u, pixels/4),
+		       SSE2PSNR(mod->max_sse_v, pixels/4));
+		mp_msg(MSGT_MENCODER, MSGL_INFO,
+		       "xvid: Average PSNR y : %.2f dB, u : %.2f dB, v : %.2f dB\n",
+		       SSE2PSNR(mod->sse_y, pixels),
+		       SSE2PSNR(mod->sse_u, pixels/4),
+		       SSE2PSNR(mod->sse_v, pixels/4));
+		mp_msg(MSGT_MENCODER, MSGL_INFO,
+		       "xvid:     Max PSNR y : %.2f dB, u : %.2f dB, v : %.2f dB\n",
+		       SSE2PSNR(mod->min_sse_y, pixels),
+		       SSE2PSNR(mod->min_sse_u, pixels/4),
+		       SSE2PSNR(mod->min_sse_v, pixels/4));
+	}
+#undef SSE2PSNR
+
+	/* ToDo: free matrices, and some string settings (quant method, matrix
+	 * filenames...) */
+
+	return;
+}
+
+/*============================================================================
+ * control
+ *==========================================================================*/
+
+static int
+control(struct vf_instance_s* vf, int request, void* data)
+{
+	return(CONTROL_UNKNOWN);
+}
+
+/*============================================================================
+ * query_format
+ *==========================================================================*/
+
+static int
+query_format(struct vf_instance_s* vf, unsigned int fmt)
+{
+	switch(fmt){
+	case IMGFMT_YV12:
+	case IMGFMT_IYUV:
+	case IMGFMT_I420:
+		return(VFCAP_CSP_SUPPORTED | VFCAP_CSP_SUPPORTED_BY_HW);
+	case IMGFMT_YUY2:
+	case IMGFMT_UYVY:
+		return(VFCAP_CSP_SUPPORTED);
+	}
+	return(BAD);
+}
+
+/*============================================================================
+ * put_image
+ *==========================================================================*/
+
+static int
+put_image(struct vf_instance_s* vf, mp_image_t *mpi)
+{
+	int size;
+	xvid_enc_stats_t stats; 
+	xvid_mplayer_module_t *mod = (xvid_mplayer_module_t *)vf->priv;
+
+	/* Prepare the stats */
+	memset(&stats,0,sizeof( xvid_enc_stats_t));
+	stats.version = XVID_VERSION;
+
+	/* -------------------------------------------------------------------
+	 * Set remaining information in the xvid_enc_frame_t structure
+	 * NB: all the other struct members were initialized by
+	 *     dispatch_settings
+	 * -----------------------------------------------------------------*/
+
+	if(set_frame_struct(mod, mpi) == BAD)
+		return(BAD);
+
+	/* -------------------------------------------------------------------
+	 * Encode the frame
+	 * ---------------------------------------------------------------- */
+
+	size = xvid_encore(mod->instance, XVID_ENC_ENCODE, &mod->frame, &stats);
+
+	/* Analyse the returned value */
+	if(size<0) {
+		mp_msg(MSGT_MENCODER, MSGL_ERR,
+		       "xvid: xvidcore returned a '%s' error\n", errorstring(size));
+		return(BAD);
+	}
+
+	/* If size is == 0, we're done with that frame */
+	if(size == 0) return(FINE);
+
+	/* Did xvidcore returned stats about an ecoded frame ? (asynchrone) */
+	if(xvidenc_stats && stats.type > 0) {
+		mod->frames++;
+		mod->sse_y += stats.sse_y;
+		mod->sse_u += stats.sse_u;
+		mod->sse_v += stats.sse_v;
+
+		if(mod->min_sse_y > stats.sse_y) {
+			mod->min_sse_y = stats.sse_y;
+			mod->min_sse_u = stats.sse_u;
+			mod->min_sse_v = stats.sse_v;
+		}
+
+		if(mod->max_sse_y < stats.sse_y) {
+			mod->max_sse_y = stats.sse_y;
+			mod->max_sse_u = stats.sse_u;
+			mod->max_sse_v = stats.sse_v;
+		}
+	}
+
+	/* xvidcore outputed bitstream -- mux it */
+	muxer_write_chunk(mod->mux,
+			  size,
+			  (mod->frame.out_flags & XVID_KEYFRAME)?0x10:0);
+
+	return(FINE);
+}
+
+/*============================================================================
+ * vf_open
+ *==========================================================================*/
+
+static int
+vf_open(vf_instance_t *vf, char* args)
+{
+	xvid_mplayer_module_t *mod;
+	xvid_gbl_init_t xvid_gbl_init;
+	xvid_gbl_info_t xvid_gbl_info;
+
+	/* Setting libmpcodec module API pointers */
+	vf->config       = config;
+	vf->control      = control;
+	vf->uninit       = uninit;
+	vf->query_format = query_format;
+	vf->put_image    = put_image;
+
+	/* Allocate the private part of the codec module */
+	vf->priv = malloc(sizeof(xvid_mplayer_module_t));
+	mod = (xvid_mplayer_module_t*)vf->priv;
+
+	if(mod == NULL) {
+		mp_msg(MSGT_MENCODER,MSGL_ERR,
+		       "xvid: memory allocation failure (private data)\n");
+		return(BAD);
+	}
+
+	/* Initialize the module to zeros */
+	memset(mod, 0, sizeof(xvid_mplayer_module_t));
+	mod->min_sse_y = mod->min_sse_u = mod->min_sse_v = INT_MAX;
+	mod->max_sse_y = mod->max_sse_u = mod->max_sse_v = INT_MIN;
+
+	/* Bind the Muxer */
+	mod->mux = (muxer_stream_t*)args;
+
+	/* Initialize muxer BITMAP header */
+	mod->mux->bih = malloc(sizeof(BITMAPINFOHEADER));
+
+	if(mod->mux->bih  == NULL) {
+		mp_msg(MSGT_MENCODER,MSGL_ERR,
+		       "xvid: memory allocation failure (BITMAP header)\n");
+		return(BAD);
+	}
+
+	mod->mux->bih->biSize = sizeof(BITMAPINFOHEADER);
+	mod->mux->bih->biWidth = 0;
+	mod->mux->bih->biHeight = 0;
+	mod->mux->bih->biPlanes = 1;
+	mod->mux->bih->biBitCount = 24;
+	mod->mux->bih->biCompression = mmioFOURCC('X','V','I','D');
+
+	/* Retrieve information about the host XviD library */
+	memset(&xvid_gbl_info, 0, sizeof(xvid_gbl_info_t));
+	xvid_gbl_info.version = XVID_VERSION;
+
+	if (xvid_global(NULL, XVID_GBL_INFO, &xvid_gbl_info, NULL) < 0) {
+		mp_msg(MSGT_MENCODER,MSGL_INFO, "xvid: could not get information about the library\n");
+	} else {
+		mp_msg(MSGT_MENCODER,MSGL_INFO, "xvid: using library version %d.%d.%d (build %s)\n",
+		       XVID_VERSION_MAJOR(xvid_gbl_info.actual_version),
+		       XVID_VERSION_MINOR(xvid_gbl_info.actual_version),
+		       XVID_VERSION_PATCH(xvid_gbl_info.actual_version),
+		       xvid_gbl_info.build);
+	}
+		
+	/* Initialize the xvid_gbl_init structure */
+	memset(&xvid_gbl_init, 0, sizeof(xvid_gbl_init_t));
+	xvid_gbl_init.version = XVID_VERSION;
+
+	/* Initialize the xvidcore library */
+	if (xvid_global(NULL, XVID_GBL_INIT, &xvid_gbl_init, NULL) < 0) {
+		mp_msg(MSGT_MENCODER,MSGL_ERR, "xvid: initialisation failure\n");
+		return(BAD);
+	}
+
+	return(FINE);
+}
+
+/*****************************************************************************
+ * Helper functions
+ ****************************************************************************/
+
+static void *read_matrix(unsigned char *filename);
+
+static void dispatch_settings(xvid_mplayer_module_t *mod)
+{
+	xvid_enc_create_t *create     = &mod->create;
+	xvid_enc_frame_t  *frame      = &mod->frame;
+	xvid_plugin_single_t *onepass = &mod->onepass;
+	xvid_plugin_2pass2_t *pass2   = &mod->pass2;
+
+	const int motion_presets[7] =
+		{
+			0,
+			0,
+			0,
+			0,
+			XVID_ME_HALFPELREFINE16,
+			XVID_ME_HALFPELREFINE16 | XVID_ME_ADVANCEDDIAMOND16,
+			XVID_ME_HALFPELREFINE16 | XVID_ME_EXTSEARCH16 |
+			XVID_ME_HALFPELREFINE8  | XVID_ME_USESQUARES16
+		};
+
+	
+	/* -------------------------------------------------------------------
+	 * Dispatch all settings having an impact on the "create" structure
+	 * This includes plugins as they are passed to encore through the
+	 * create structure
+	 * -----------------------------------------------------------------*/
+
+	/* -------------------------------------------------------------------
+	 * The create structure
+	 * ---------------------------------------------------------------- */
+
+	create->global = 0;
+
+	if(xvidenc_packed)
+		create->global |= XVID_GLOBAL_PACKED;
+
+	if(xvidenc_closed_gop)
+		create->global |= XVID_GLOBAL_CLOSED_GOP;
+
+	if(xvidenc_stats)
+		create->global |= XVID_GLOBAL_EXTRASTATS_ENABLE;
+
+	create->num_zones = 0;
+	create->zones = NULL;
+	create->num_plugins = 0;
+	create->plugins = NULL;
+	create->num_threads = 0;
+	create->max_bframes = xvidenc_max_bframes;
+	create->bquant_ratio = xvidenc_bquant_ratio;
+	create->bquant_offset = xvidenc_bquant_offset;
+	create->max_key_interval = xvidenc_max_key_interval;
+	create->frame_drop_ratio = xvidenc_frame_drop_ratio;
+	create->min_quant[0] = xvidenc_min_quant[0];
+	create->min_quant[1] = xvidenc_min_quant[1];
+	create->min_quant[2] = xvidenc_min_quant[2];
+	create->max_quant[0] = xvidenc_max_quant[0];
+	create->max_quant[1] = xvidenc_max_quant[1];
+	create->max_quant[2] = xvidenc_max_quant[2];
+
+
+	/* -------------------------------------------------------------------
+	 * The single pass plugin
+	 * ---------------------------------------------------------------- */
+
+	onepass->bitrate = xvidenc_bitrate;
+	onepass->reaction_delay_factor = xvidenc_cbr_reaction_delay_factor;
+	onepass->averaging_period = xvidenc_cbr_averaging_period;
+	onepass->buffer = xvidenc_cbr_buffer;
+
+	/* -------------------------------------------------------------------
+	 * The pass2 plugin
+	 * ---------------------------------------------------------------- */
+
+	pass2->keyframe_boost = xvidenc_vbr_keyframe_boost;
+	pass2->overflow_control_strength = xvidenc_vbr_overflow_control_strength;
+	pass2->curve_compression_high = xvidenc_vbr_curve_compression_high;
+	pass2->curve_compression_low = xvidenc_vbr_curve_compression_low;
+	pass2->max_overflow_improvement = xvidenc_vbr_max_overflow_improvement;
+	pass2->max_overflow_degradation = xvidenc_vbr_max_overflow_degradation;
+	pass2->kfreduction = xvidenc_vbr_kfreduction;
+	pass2->min_key_interval = xvidenc_vbr_min_key_interval;
+	pass2->container_frame_overhead = xvidenc_vbr_container_frame_overhead;
+
+	/* -------------------------------------------------------------------
+	 * The frame structure
+	 * ---------------------------------------------------------------- */
+	frame->vol_flags = 0;
+	frame->vop_flags = 0;
+	frame->motion    = 0;
+
+	frame->vop_flags |= XVID_VOP_HALFPEL;
+	frame->motion    |= motion_presets[xvidenc_motion];
+
+	if(xvidenc_stats)
+		frame->vol_flags |= XVID_VOL_EXTRASTATS;
+
+	if(xvidenc_greyscale)
+		frame->vop_flags |= XVID_VOP_GREYSCALE;
+
+	if(xvidenc_cartoon) {
+		frame->vop_flags |= XVID_VOP_CARTOON;
+		frame->motion |= XVID_ME_DETECT_STATIC_MOTION;
+	}
+
+	if(xvidenc_intra_matrix_file != NULL) {
+		frame->quant_intra_matrix = (unsigned char*)read_matrix(xvidenc_intra_matrix_file);
+		if(frame->quant_intra_matrix != NULL) {
+			fprintf(stderr, "xvid: Loaded Intra matrix (switching to mpeg quantization type)\n");
+			if(xvidenc_quant_method) free(xvidenc_quant_method);
+			xvidenc_quant_method = strdup("mpeg");
+		}
+	}
+	if(xvidenc_inter_matrix_file != NULL) {
+		frame->quant_inter_matrix = read_matrix(xvidenc_inter_matrix_file);
+		if(frame->quant_inter_matrix) {
+			fprintf(stderr, "\nxvid: Loaded Inter matrix (switching to mpeg quantization type)\n");
+			if(xvidenc_quant_method) free(xvidenc_quant_method);
+			xvidenc_quant_method = strdup("mpeg");
+		}
+	}
+	if(xvidenc_quant_method != NULL && !strcasecmp(xvidenc_quant_method, "mpeg")) {
+		frame->vol_flags |= XVID_VOL_MPEGQUANT;
+	}
+	if(xvidenc_quarterpel) {
+		frame->vol_flags |= XVID_VOL_QUARTERPEL;
+		frame->motion    |= XVID_ME_QUARTERPELREFINE16;
+		frame->motion    |= XVID_ME_QUARTERPELREFINE8;
+	}
+	if(xvidenc_gmc) {
+		frame->vol_flags |= XVID_VOL_GMC;
+		frame->motion    |= XVID_ME_GME_REFINE;
+	}
+	if(xvidenc_interlaced) {
+		frame->vol_flags |= XVID_VOL_INTERLACING;
+	}
+	if(xvidenc_trellis) {
+		frame->vop_flags |= XVID_VOP_TRELLISQUANT;
+	}
+	if(xvidenc_hqacpred) {
+		frame->vop_flags |= XVID_VOP_HQACPRED;
+	}
+	if(xvidenc_motion > 4) {
+		frame->vop_flags |= XVID_VOP_INTER4V;
+	}
+	if(xvidenc_chromame) {
+		frame->motion |= XVID_ME_CHROMA_PVOP;
+		frame->motion |= XVID_ME_CHROMA_BVOP;
+	}
+	if(xvidenc_vhq >= 1) {
+		frame->vop_flags |= XVID_VOP_MODEDECISION_RD;
+	}
+	if(xvidenc_vhq >= 2) {
+		frame->motion |= XVID_ME_HALFPELREFINE16_RD;
+		frame->motion |= XVID_ME_QUARTERPELREFINE16_RD;
+	}
+	if(xvidenc_vhq >= 3) {
+		frame->motion |= XVID_ME_HALFPELREFINE8_RD;
+		frame->motion |= XVID_ME_QUARTERPELREFINE8_RD;
+		frame->motion |= XVID_ME_CHECKPREDICTION_RD;
+	}
+	if(xvidenc_vhq >= 4) {
+		frame->motion |= XVID_ME_EXTSEARCH_RD;
+	}
+
+	/* motion level == 0 means no motion search which is equivalent to
+	 * intra coding only */
+	if(xvidenc_motion == 0) {
+		frame->type = XVID_TYPE_IVOP;
+	} else {
+		frame->type = XVID_TYPE_AUTO;
+	}
+
+	frame->bframe_threshold = xvidenc_bframe_threshold;
+
+	frame->par = 0;
+	if(xvidenc_par != NULL) {
+		if(strcasecmp(xvidenc_par, "pal43") == 0)
+			frame->par = XVID_PAR_43_PAL;
+		if(strcasecmp(xvidenc_par, "pal169") == 0)
+			frame->par = XVID_PAR_169_PAL;
+		if(strcasecmp(xvidenc_par, "ntsc43") == 0)
+			frame->par = XVID_PAR_43_NTSC;
+		if(strcasecmp(xvidenc_par, "ntsc169") == 0)
+			frame->par = XVID_PAR_169_NTSC;
+		if(strcasecmp(xvidenc_par, "ext") == 0)
+			frame->par = XVID_PAR_EXT;
+	}
+	if(frame->par == 0) {
+		frame->par = XVID_PAR_11_VGA;
+	}
+	if(frame->par == XVID_PAR_EXT) {
+		if(xvidenc_par_width)
+			frame->par_width = xvidenc_par_width;
+		else
+			frame->par_width = 1;
+
+		if(xvidenc_par_height)
+			frame->par_height = xvidenc_par_height;
+		else
+			frame->par_height = 1;
+	}
+
+	return;
+}
+
+static int set_create_struct(xvid_mplayer_module_t *mod)
+{
+	int pass;
+	xvid_enc_create_t *create    = &mod->create;
+
+	/* Most of the structure is initialized by dispatch settings, only a
+	 * few things are missing  */
+	create->version = XVID_VERSION;
+
+	/* Width and Height */
+	create->width  = mod->mux->bih->biWidth;
+	create->height = mod->mux->bih->biHeight;
+
+	/* FPS */
+	create->fincr = mod->mux->h.dwScale;
+	create->fbase = mod->mux->h.dwRate;
+
+	/* Encodings zones */
+	memset(mod->zones, 0, sizeof(mod->zones));
+	create->zones     = mod->zones;
+	create->num_zones = 0;
+
+	/* Plugins */
+	memset(mod->plugins, 0, sizeof(mod->plugins));
+	create->plugins     = mod->plugins;
+	create->num_plugins = 0;
+
+	/* -------------------------------------------------------------------
+	 * Initialize and bind the right rate controller plugin
+	 * ---------------------------------------------------------------- */
+
+	/* First we try to sort out configuration conflicts */
+	if(xvidenc_quantizer != 0 && (xvidenc_bitrate || xvidenc_pass)) {
+		mp_msg(MSGT_MENCODER, MSGL_ERR,
+		       "xvid: you can't mix Fixed Quantizer Rate Control"
+		       " with other Rate Control mechanisms\n");
+		return(BAD);
+	}
+
+	if(xvidenc_bitrate != 0 && xvidenc_pass == 1) {
+		mp_msg(MSGT_MENCODER, MSGL_ERR,
+		       "xvid: bitrate setting is ignored during first pass\n");
+	}
+
+	/* Sort out which sort of pass we are supposed to do
+	 * pass == 1<<0 CBR
+	 * pass == 1<<1 Two pass first pass
+	 * pass == 1<<2 Two pass second pass
+	 * pass == 1<<3 Constant quantizer
+	 */
+#define MODE_CBR    (1<<0)
+#define MODE_2PASS1 (1<<1)
+#define MODE_2PASS2 (1<<2)
+#define MODE_QUANT  (1<<3)
+
+	pass = 0;
+
+	if(xvidenc_bitrate != 0 && xvidenc_pass == 0)
+		pass |= MODE_CBR;
+
+	if(xvidenc_pass == 1)
+		pass |= MODE_2PASS1;
+
+	if(xvidenc_bitrate != 0 && xvidenc_pass == 2)
+		pass |= MODE_2PASS2;
+
+	if(xvidenc_quantizer != 0  && xvidenc_pass == 0)
+		pass |= MODE_QUANT;
+
+	/* We must be in at least one RC mode */
+	if(pass == 0) {
+		mp_msg(MSGT_MENCODER, MSGL_ERR,
+		       "xvid: you must specify one or a valid combination of "
+		       "'bitrate', 'pass', 'quantizer' settings\n");
+		return(BAD);
+	}
+
+	/* Sanity checking */
+	if(pass != MODE_CBR    && pass != MODE_QUANT &&
+	   pass != MODE_2PASS1 && pass != MODE_2PASS2) {
+		mp_msg(MSGT_MENCODER, MSGL_ERR,
+		       "xvid: this code should not be reached - fill a bug "
+		       "report\n");
+		return(BAD);
+	}
+
+	/* This is a single pass encoding: either a CBR pass or a constant
+	 * quantizer pass */
+	if(pass == MODE_CBR  || pass == MODE_QUANT) {
+		xvid_plugin_single_t *onepass = &mod->onepass;
+
+		/* There is not much left to initialize after dispatch settings */
+		onepass->version = XVID_VERSION;
+		onepass->bitrate = xvidenc_bitrate*1000;
+
+		/* Quantizer mode uses the same plugin, we have only to define
+		 * a constant quantizer zone beginning at frame 0 */
+		if(pass == MODE_QUANT) {
+			int base, incr;
+
+			base = 100;
+			incr = (int)xvidenc_quantizer*base;
+
+			create->zones[create->num_zones].mode      = XVID_ZONE_QUANT;
+			create->zones[create->num_zones].frame     = 0;
+			create->zones[create->num_zones].base      = base;
+			create->zones[create->num_zones].increment = incr;
+			create->num_zones++;
+
+			mp_msg(MSGT_MENCODER, MSGL_INFO,
+			       "xvid: Fixed Quant Rate Control -- quantizer=%d/%d=%2.2f\n",
+			       incr,
+			       base,
+			       (float)(incr)/(float)(base));
+			
+		} else {
+			mp_msg(MSGT_MENCODER, MSGL_INFO,
+			       "xvid: CBR Rate Control -- bitrate=%dkbit/s\n",
+			       xvidenc_bitrate);
+		}
+
+		create->plugins[create->num_plugins].func  = xvid_plugin_single;
+		create->plugins[create->num_plugins].param = onepass;
+		create->num_plugins++;
+	}
+
+	/* This is the first pass of a Two pass process */
+	if(pass == MODE_2PASS1) {
+		xvid_plugin_2pass1_t *pass1 = &mod->pass1;
+
+		/* There is not much to initialize for this plugin */
+		pass1->version  = XVID_VERSION;
+		pass1->filename = XVID_FIRST_PASS_FILENAME;
+
+		create->plugins[create->num_plugins].func  = xvid_plugin_2pass1;
+		create->plugins[create->num_plugins].param = pass1;
+		create->num_plugins++;
+
+		mp_msg(MSGT_MENCODER, MSGL_INFO,
+		       "xvid: 2Pass Rate Control -- 1st pass\n");
+	}
+
+	/* This is the second pass of a Two pass process */
+	if(pass == MODE_2PASS2) {
+		xvid_plugin_2pass2_t *pass2 = &mod->pass2;
+
+		/* There is not much left to initialize after dispatch settings */
+		pass2->version  = XVID_VERSION;
+		pass2->filename =  XVID_FIRST_PASS_FILENAME;
+
+		/* Positive bitrate values are bitrates as usual but if the
+		 * value is negative it is considered as being a total size
+		 * to reach (in kilobytes) */
+		if(xvidenc_bitrate > 0) {
+			pass2->bitrate  = xvidenc_bitrate*1000;
+			mp_msg(MSGT_MENCODER, MSGL_INFO,
+			       "xvid: 2Pass Rate Control -- 2nd pass -- bitrate=%dkbit/s\n",
+			       xvidenc_bitrate);
+		} else {
+			pass2->bitrate  = xvidenc_bitrate;
+			mp_msg(MSGT_MENCODER, MSGL_INFO,
+			       "xvid: 2Pass Rate Control -- 2nd pass -- total size=%dkB\n",
+			       -xvidenc_bitrate);
+		}
+
+		create->plugins[create->num_plugins].func  = xvid_plugin_2pass2;
+		create->plugins[create->num_plugins].param = pass2;
+		create->num_plugins++;
+	}
+
+	return(FINE);
+}
+
+static int set_frame_struct(xvid_mplayer_module_t *mod, mp_image_t *mpi)
+{
+	xvid_enc_frame_t *frame = &mod->frame;
+
+	/* Most of the initialization is done during dispatch_settings */
+	frame->version = XVID_VERSION;
+
+	/* Bind output buffer */
+	frame->bitstream = mod->mux->buffer;
+	frame->length    = -1;
+
+	/* Frame format */
+	switch(mpi->imgfmt) {
+	case IMGFMT_YV12:
+	case IMGFMT_IYUV:
+	case IMGFMT_I420:
+		frame->input.csp = XVID_CSP_USER;
+		break;
+	case IMGFMT_YUY2:
+		frame->input.csp = XVID_CSP_YUY2;
+		break;
+	case IMGFMT_UYVY:
+		frame->input.csp = XVID_CSP_UYVY;
+		break;
+	default:
+		mp_msg(MSGT_MENCODER, MSGL_ERR,
+		       "xvid: unsupported picture format (%s)!\n",
+		       vo_format_name(mpi->imgfmt));
+		return(BAD);
+	}
+
+	/* Bind source frame */
+	frame->input.plane[0]  = mpi->planes[0];
+	frame->input.plane[1]  = mpi->planes[1];
+	frame->input.plane[2]  = mpi->planes[2];
+	frame->input.stride[0] = mpi->stride[0];
+	frame->input.stride[1] = mpi->stride[1];
+	frame->input.stride[2] = mpi->stride[2];
+
+	/* Force the right quantizer -- It is internally managed by RC
+	 * plugins */
+	frame->quant = 0;
+
+	return(FINE);
+}
+
+static void *read_matrix(unsigned char *filename)
+{
+	int i;
+	unsigned char *matrix;
+	FILE *input;
+	
+	/* Allocate matrix space */
+	if((matrix = malloc(64*sizeof(unsigned char))) == NULL)
+	   return(NULL);
+
+	/* Open the matrix file */
+	if((input = fopen(filename, "rb")) == NULL) {
+		fprintf(stderr,
+			"xvid: Error opening the matrix file %s\n",
+			filename);
+		free(matrix);
+		return(NULL);
+	}
+
+	/* Read the matrix */
+	for(i=0; i<64; i++) {
+
+		int value;
+
+		/* If fscanf fails then get out of the loop */
+		if(fscanf(input, "%d", &value) != 1) {
+			fprintf(stderr,
+				"xvid: Error reading the matrix file %s\n",
+				filename);
+			free(matrix);
+			fclose(input);
+			return(NULL);
+		}
+
+		/* Clamp the value to safe range */
+		value     = (value<  1)?1  :value;
+		value     = (value>255)?255:value;
+		matrix[i] = value;
+	}
+
+	/* Fills the rest with 1 */
+	while(i<64) matrix[i++] = 1;
+
+	/* We're done */
+	fclose(input);
+
+	return(matrix);
+	
+}
+
+static const char *errorstring(int err)
+{
+	char *error;
+	switch(err) {
+	case XVID_ERR_FAIL:
+		error = "General fault";
+		break;
+	case XVID_ERR_MEMORY:
+		error =  "Memory allocation error";
+		break;
+	case XVID_ERR_FORMAT:
+		error =  "File format error";
+		break;
+	case XVID_ERR_VERSION:
+		error =  "Structure version not supported";
+		break;
+	case XVID_ERR_END:
+		error =  "End of stream reached";
+		break;
+	default:
+		error = "Unknown";
+	}
+
+	return((const char *)error);
+}
+
+/*****************************************************************************
+ * Module structure definition
+ ****************************************************************************/
+
+vf_info_t ve_info_xvid = {
+	"XviD 1.0 encoder",
+	"xvid",
+	"Marco Belli <elcabesa@inwind.it>, Edouard Gomez <ed.gomez@free.fr>",
+	"No comment",
+	vf_open
+};
+
+
+#endif /* HAVE_XVID4 */
+
+/* Please do not change that tag comment.
+ * arch-tag: 42ccc257-0548-4a3e-9617-2876c4e8ac88 mplayer xvid encoder module */