1 /*****************************************************************************
3 * - XviD 1.x export module for mplayer/mencoder -
5 * Copyright(C) 2003 Marco Belli <elcabesa@inwind.it>
6 * 2003-2004 Edouard Gomez <ed.gomez@free.fr>
8 * This program is free software; you can redistribute it and/or modify
9 * it under the terms of the GNU General Public License as published by
10 * the Free Software Foundation; either version 2 of the License, or
11 * (at your option) any later version.
13 * This program is distributed in the hope that it will be useful,
14 * but WITHOUT ANY WARRANTY; without even the implied warranty of
15 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 * GNU General Public License for more details.
18 * You should have received a copy of the GNU General Public License
19 * along with this program; if not, write to the Free Software
20 * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
22 ****************************************************************************/
24 /*****************************************************************************
26 ****************************************************************************/
39 #include "codec-cfg.h"
40 #include "stream/stream.h"
41 #include "libmpdemux/demuxer.h"
42 #include "libmpdemux/stheader.h"
44 #include "stream/stream.h"
45 #include "libmpdemux/muxer.h"
47 #include "img_format.h"
59 #ifdef USE_LIBAVUTIL_SO
60 #include <ffmpeg/avutil.h>
70 // Profile flag definitions
71 #define PROFILE_ADAPTQUANT 0x00000001
72 #define PROFILE_BVOP 0x00000002
73 #define PROFILE_MPEGQUANT 0x00000004
74 #define PROFILE_INTERLACE 0x00000008
75 #define PROFILE_QPEL 0x00000010
76 #define PROFILE_GMC 0x00000020
77 #define PROFILE_4MV 0x00000040
78 #define PROFILE_DXN 0x00000080
80 // Reduce code duplication in profiles[] array
81 #define PROFILE_S (PROFILE_4MV)
82 #define PROFILE_AS (PROFILE_4MV|PROFILE_ADAPTQUANT|PROFILE_BVOP|PROFILE_MPEGQUANT|PROFILE_INTERLACE|PROFILE_QPEL|PROFILE_GMC)
86 char *name
; ///< profile name
87 int id
; ///< mpeg-4 profile id; iso/iec 14496-2:2001 table G-1
88 int width
; ///< profile width restriction
89 int height
; ///< profile height restriction
90 int fps
; ///< profile frame rate restriction
91 int max_objects
; ///< ??????
92 int total_vmv_buffer_sz
; ///< macroblock memory; when BVOPS=false, vmv = 2*vcv; when BVOPS=true, vmv = 3*vcv
93 int max_vmv_buffer_sz
; ///< max macroblocks per vop
94 int vcv_decoder_rate
; ///< macroblocks decoded per second
95 int max_acpred_mbs
; ///< percentage
96 int max_vbv_size
; ///< max vbv size (bits) 16368 bits
97 int max_video_packet_length
; ///< bits
98 int max_bitrate
; ///< bits per second
99 int vbv_peakrate
; ///< max bits over anyone second period; 0=don't care
100 int dxn_max_bframes
; ///< dxn: max consecutive bframes
101 unsigned int flags
; ///< flags for allowed options/dxn note the definitions for PROFILE_S and PROFILE_AS
104 // Code taken from XviD VfW source for profile support
106 /* default vbv_occupancy is (64/170)*vbv_buffer_size */
108 static profile_t profiles
[] =
110 /* name p@l w h fps obj Tvmv vmv vcv ac% vbv pkt bps vbv_peak dbf flags */
111 /* unrestricted profile (default) */
112 { "unrestricted", 0x00, 0, 0, 0, 0, 0, 0, 0, 100, 0*16368, -1, 0, 0, -1, 0xffffffff & ~PROFILE_DXN
},
114 { "sp0", 0x08, 176, 144, 15, 1, 198, 99, 1485, 100, 10*16368, 2048, 64000, 0, -1, PROFILE_S
},
115 /* simple@l0: max f_code=1, intra_dc_vlc_threshold=0 */
116 /* if ac preidition is used, adaptive quantization must not be used */
117 /* <=qcif must be used */
118 { "sp1", 0x01, 176, 144, 15, 4, 198, 99, 1485, 100, 10*16368, 2048, 64000, 0, -1, PROFILE_S
|PROFILE_ADAPTQUANT
},
119 { "sp2", 0x02, 352, 288, 15, 4, 792, 396, 5940, 100, 40*16368, 4096, 128000, 0, -1, PROFILE_S
|PROFILE_ADAPTQUANT
},
120 { "sp3", 0x03, 352, 288, 15, 4, 792, 396, 11880, 100, 40*16368, 8192, 384000, 0, -1, PROFILE_S
|PROFILE_ADAPTQUANT
},
122 { "asp0", 0xf0, 176, 144, 30, 1, 297, 99, 2970, 100, 10*16368, 2048, 128000, 0, -1, PROFILE_AS
},
123 { "asp1", 0xf1, 176, 144, 30, 4, 297, 99, 2970, 100, 10*16368, 2048, 128000, 0, -1, PROFILE_AS
},
124 { "asp2", 0xf2, 352, 288, 15, 4, 1188, 396, 5940, 100, 40*16368, 4096, 384000, 0, -1, PROFILE_AS
},
125 { "asp3", 0xf3, 352, 288, 30, 4, 1188, 396, 11880, 100, 40*16368, 4096, 768000, 0, -1, PROFILE_AS
},
126 /* ISMA Profile 1, (ASP) @ L3b (CIF, 1.5 Mb/s) CIF(352x288), 30fps, 1.5Mbps max ??? */
127 { "asp4", 0xf4, 352, 576, 30, 4, 2376, 792, 23760, 50, 80*16368, 8192, 3000000, 0, -1, PROFILE_AS
},
128 { "asp5", 0xf5, 720, 576, 30, 4, 4860, 1620, 48600, 25, 112*16368, 16384, 8000000, 0, -1, PROFILE_AS
},
130 // information provided by DivXNetworks, USA.
131 // "DivX Certified Profile Compatibility v1.1", February 2005
132 { "dxnhandheld", 0x00, 176, 144, 15, 1, 198, 99, 1485, 100, 32*8192, -1, 537600, 800000, 0, PROFILE_ADAPTQUANT
|PROFILE_DXN
},
133 { "dxnportntsc", 0x00, 352, 240, 30, 1, 990, 330, 36000, 100, 384*8192, -1, 4854000, 8000000, 1, PROFILE_4MV
|PROFILE_ADAPTQUANT
|PROFILE_BVOP
|PROFILE_DXN
},
134 { "dxnportpal", 0x00, 352, 288, 25, 1, 1188, 396, 36000, 100, 384*8192, -1, 4854000, 8000000, 1, PROFILE_4MV
|PROFILE_ADAPTQUANT
|PROFILE_BVOP
|PROFILE_DXN
},
135 { "dxnhtntsc", 0x00, 720, 480, 30, 1, 4050, 1350, 40500, 100, 384*8192, -1, 4854000, 8000000, 1, PROFILE_4MV
|PROFILE_ADAPTQUANT
|PROFILE_BVOP
|PROFILE_INTERLACE
|PROFILE_DXN
},
136 { "dxnhtpal", 0x00, 720, 576, 25, 1, 4860, 1620, 40500, 100, 384*8192, -1, 4854000, 8000000, 1, PROFILE_4MV
|PROFILE_ADAPTQUANT
|PROFILE_BVOP
|PROFILE_INTERLACE
|PROFILE_DXN
},
137 { "dxnhdtv", 0x00, 1280, 720, 30, 1,10800, 3600, 108000, 100, 768*8192, -1, 9708400, 16000000, 2, PROFILE_4MV
|PROFILE_ADAPTQUANT
|PROFILE_BVOP
|PROFILE_INTERLACE
|PROFILE_DXN
},
139 { NULL
, 0x00, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0x00000000 },
143 * \brief return the pointer to a chosen profile
144 * \param str the profile name
145 * \return pointer of the appropriate profiles array entry or NULL for a mistyped profile name
147 static profile_t
*profileFromName(char *str
)
149 profile_t
*cur
= profiles
;
150 while (cur
->name
&& strcasecmp(cur
->name
, str
)) cur
++;
151 if(!cur
->name
) return NULL
;
155 /*****************************************************************************
156 * Configuration options
157 ****************************************************************************/
159 extern char* passtmpfile
;
161 static int xvidenc_bitrate
= 0;
162 static int xvidenc_pass
= 0;
163 static float xvidenc_quantizer
= 0;
165 static int xvidenc_packed
= 0;
166 static int xvidenc_closed_gop
= 1;
167 static int xvidenc_interlaced
= 0;
168 static int xvidenc_quarterpel
= 0;
169 static int xvidenc_gmc
= 0;
170 static int xvidenc_trellis
= 1;
171 static int xvidenc_cartoon
= 0;
172 static int xvidenc_hqacpred
= 1;
173 static int xvidenc_chromame
= 1;
174 static int xvidenc_chroma_opt
= 0;
175 static int xvidenc_vhq
= 1;
176 static int xvidenc_bvhq
= 1;
177 static int xvidenc_motion
= 6;
178 static int xvidenc_turbo
= 0;
179 static int xvidenc_stats
= 0;
180 static int xvidenc_max_key_interval
= 0; /* Let xvidcore set a 10s interval by default */
181 static int xvidenc_frame_drop_ratio
= 0;
182 static int xvidenc_greyscale
= 0;
183 static int xvidenc_luminance_masking
= 0;
184 static int xvidenc_debug
= 0;
185 static int xvidenc_psnr
= 0;
187 static int xvidenc_max_bframes
= 2;
188 static int xvidenc_num_threads
= 0;
189 static int xvidenc_bquant_ratio
= 150;
190 static int xvidenc_bquant_offset
= 100;
191 static int xvidenc_bframe_threshold
= 0;
193 static int xvidenc_min_quant
[3] = {2, 2, 2};
194 static int xvidenc_max_quant
[3] = {31, 31, 31};
195 static char *xvidenc_intra_matrix_file
= NULL
;
196 static char *xvidenc_inter_matrix_file
= NULL
;
197 static char *xvidenc_quant_method
= NULL
;
199 static int xvidenc_cbr_reaction_delay_factor
= 0;
200 static int xvidenc_cbr_averaging_period
= 0;
201 static int xvidenc_cbr_buffer
= 0;
203 static int xvidenc_vbr_keyframe_boost
= 0;
204 static int xvidenc_vbr_overflow_control_strength
= 5;
205 static int xvidenc_vbr_curve_compression_high
= 0;
206 static int xvidenc_vbr_curve_compression_low
= 0;
207 static int xvidenc_vbr_max_overflow_improvement
= 5;
208 static int xvidenc_vbr_max_overflow_degradation
= 5;
209 static int xvidenc_vbr_kfreduction
= 0;
210 static int xvidenc_vbr_kfthreshold
= 0;
211 static int xvidenc_vbr_container_frame_overhead
= 24; /* mencoder uses AVI container */
213 // commandline profile option string - default to unrestricted
214 static char *xvidenc_profile
= "unrestricted";
216 static char *xvidenc_par
= NULL
;
217 static int xvidenc_par_width
= 0;
218 static int xvidenc_par_height
= 0;
219 static float xvidenc_dar_aspect
= 0.0f
;
220 static int xvidenc_autoaspect
= 0;
222 static char *xvidenc_zones
= NULL
; // zones string
224 m_option_t xvidencopts_conf
[] =
226 /* Standard things mencoder should be able to treat directly */
227 {"bitrate", &xvidenc_bitrate
, CONF_TYPE_INT
, 0, 0, 0, NULL
},
228 {"pass", &xvidenc_pass
, CONF_TYPE_INT
, CONF_RANGE
, 1, 2, NULL
},
229 {"fixed_quant", &xvidenc_quantizer
, CONF_TYPE_FLOAT
, CONF_RANGE
, 1, 31, NULL
},
232 {"quant_type", &xvidenc_quant_method
, CONF_TYPE_STRING
, 0, 0, 0, NULL
},
233 {"me_quality", &xvidenc_motion
, CONF_TYPE_INT
, CONF_RANGE
, 0, 6, NULL
},
234 {"chroma_me", &xvidenc_chromame
, CONF_TYPE_FLAG
, 0, 0, 1, NULL
},
235 {"nochroma_me", &xvidenc_chromame
, CONF_TYPE_FLAG
, 0, 1, 0, NULL
},
236 {"chroma_opt", &xvidenc_chroma_opt
, CONF_TYPE_FLAG
, 0, 0, 1, NULL
},
237 {"nochroma_opt", &xvidenc_chroma_opt
, CONF_TYPE_FLAG
, 0, 1, 0, NULL
},
238 {"vhq", &xvidenc_vhq
, CONF_TYPE_INT
, CONF_RANGE
, 0, 4, NULL
},
239 {"bvhq", &xvidenc_bvhq
, CONF_TYPE_INT
, CONF_RANGE
, 0, 1, NULL
},
240 {"max_bframes", &xvidenc_max_bframes
, CONF_TYPE_INT
, CONF_RANGE
, 0, 20, NULL
},
241 {"threads", &xvidenc_num_threads
, CONF_TYPE_INT
, 0, 0, 0, NULL
},
242 {"bquant_ratio", &xvidenc_bquant_ratio
, CONF_TYPE_INT
, CONF_RANGE
, 0, 200, NULL
},
243 {"bquant_offset", &xvidenc_bquant_offset
, CONF_TYPE_INT
, CONF_RANGE
, 0, 200, NULL
},
244 {"bf_threshold", &xvidenc_bframe_threshold
, CONF_TYPE_INT
, CONF_RANGE
, -255, 255, NULL
},
245 {"qpel", &xvidenc_quarterpel
, CONF_TYPE_FLAG
, 0, 0, 1, NULL
},
246 {"noqpel", &xvidenc_quarterpel
, CONF_TYPE_FLAG
, 0, 1, 0, NULL
},
247 {"gmc", &xvidenc_gmc
, CONF_TYPE_FLAG
, 0, 0, 1, NULL
},
248 {"nogmc", &xvidenc_gmc
, CONF_TYPE_FLAG
, 0, 1, 0, NULL
},
249 {"trellis", &xvidenc_trellis
, CONF_TYPE_FLAG
, 0, 0, 1, NULL
},
250 {"notrellis", &xvidenc_trellis
, CONF_TYPE_FLAG
, 0, 1, 0, NULL
},
251 {"packed", &xvidenc_packed
, CONF_TYPE_FLAG
, 0, 0, 1, NULL
},
252 {"nopacked", &xvidenc_packed
, CONF_TYPE_FLAG
, 0, 1, 0, NULL
},
253 {"closed_gop", &xvidenc_closed_gop
, CONF_TYPE_FLAG
, 0, 0, 1, NULL
},
254 {"noclosed_gop", &xvidenc_closed_gop
, CONF_TYPE_FLAG
, 0, 1, 0, NULL
},
255 {"interlacing", &xvidenc_interlaced
, CONF_TYPE_FLAG
, 0, 0, 1, NULL
},
256 {"nointerlacing", &xvidenc_interlaced
, CONF_TYPE_FLAG
, 0, 1, 0, NULL
},
257 {"cartoon", &xvidenc_cartoon
, CONF_TYPE_FLAG
, 0, 0, 1, NULL
},
258 {"nocartoon", &xvidenc_cartoon
, CONF_TYPE_FLAG
, 0, 1, 0, NULL
},
259 {"hq_ac", &xvidenc_hqacpred
, CONF_TYPE_FLAG
, 0, 0, 1, NULL
},
260 {"nohq_ac", &xvidenc_hqacpred
, CONF_TYPE_FLAG
, 0, 1, 0, NULL
},
261 {"frame_drop_ratio", &xvidenc_frame_drop_ratio
, CONF_TYPE_INT
, CONF_RANGE
, 0, 100, NULL
},
262 {"max_key_interval", &xvidenc_max_key_interval
, CONF_TYPE_INT
, CONF_MIN
, 0, 0, NULL
},
263 {"greyscale", &xvidenc_greyscale
, CONF_TYPE_FLAG
, 0, 0, 1, NULL
}, /* kept for backward compatibility */
264 {"grayscale", &xvidenc_greyscale
, CONF_TYPE_FLAG
, 0, 0, 1, NULL
},
265 {"nogreyscale", &xvidenc_greyscale
, CONF_TYPE_FLAG
, 0, 1, 0, NULL
},
266 {"lumi_mask", &xvidenc_luminance_masking
, CONF_TYPE_FLAG
, 0, 0, 1, NULL
},
267 {"nolumi_mask", &xvidenc_luminance_masking
, CONF_TYPE_FLAG
, 0, 1, 0, NULL
},
268 {"turbo", &xvidenc_turbo
, CONF_TYPE_FLAG
, 0, 0, 1, NULL
},
269 {"debug", &xvidenc_debug
, CONF_TYPE_INT
, 0 ,0,-1,NULL
},
270 {"stats", &xvidenc_stats
, CONF_TYPE_FLAG
, 0, 0, 1, NULL
},
271 {"psnr", &xvidenc_psnr
, CONF_TYPE_FLAG
, 0, 0, 1, NULL
},
274 /* section [quantizer] */
275 {"min_iquant", &xvidenc_min_quant
[0], CONF_TYPE_INT
, CONF_RANGE
, 1, 31, NULL
},
276 {"max_iquant", &xvidenc_max_quant
[0], CONF_TYPE_INT
, CONF_RANGE
, 1, 31, NULL
},
277 {"min_pquant", &xvidenc_min_quant
[1], CONF_TYPE_INT
, CONF_RANGE
, 1, 31, NULL
},
278 {"max_pquant", &xvidenc_max_quant
[1], CONF_TYPE_INT
, CONF_RANGE
, 1, 31, NULL
},
279 {"min_bquant", &xvidenc_min_quant
[2], CONF_TYPE_INT
, CONF_RANGE
, 1, 31, NULL
},
280 {"max_bquant", &xvidenc_max_quant
[2], CONF_TYPE_INT
, CONF_RANGE
, 1, 31, NULL
},
281 {"quant_intra_matrix", &xvidenc_intra_matrix_file
, CONF_TYPE_STRING
, 0, 0, 100, NULL
},
282 {"quant_inter_matrix", &xvidenc_inter_matrix_file
, CONF_TYPE_STRING
, 0, 0, 100, NULL
},
285 {"rc_reaction_delay_factor", &xvidenc_cbr_reaction_delay_factor
, CONF_TYPE_INT
, CONF_RANGE
, 0, 100, NULL
},
286 {"rc_averaging_period", &xvidenc_cbr_averaging_period
, CONF_TYPE_INT
, CONF_MIN
, 0, 0, NULL
},
287 {"rc_buffer", &xvidenc_cbr_buffer
, CONF_TYPE_INT
, CONF_MIN
, 0, 0, NULL
},
290 {"keyframe_boost", &xvidenc_vbr_keyframe_boost
, CONF_TYPE_INT
, CONF_RANGE
, 0, 100, NULL
},
291 {"curve_compression_high", &xvidenc_vbr_curve_compression_high
, CONF_TYPE_INT
, CONF_RANGE
, 0, 100, NULL
},
292 {"curve_compression_low", &xvidenc_vbr_curve_compression_low
, CONF_TYPE_INT
, CONF_RANGE
, 0, 100, NULL
},
293 {"overflow_control_strength", &xvidenc_vbr_overflow_control_strength
, CONF_TYPE_INT
, CONF_RANGE
, 0, 100, NULL
},
294 {"max_overflow_improvement", &xvidenc_vbr_max_overflow_improvement
, CONF_TYPE_INT
, CONF_RANGE
, 0, 100, NULL
},
295 {"max_overflow_degradation", &xvidenc_vbr_max_overflow_degradation
, CONF_TYPE_INT
, CONF_RANGE
, 0, 100, NULL
},
296 {"kfreduction", &xvidenc_vbr_kfreduction
, CONF_TYPE_INT
, CONF_RANGE
, 0, 100, NULL
},
297 {"kfthreshold", &xvidenc_vbr_kfthreshold
, CONF_TYPE_INT
, CONF_MIN
, 0, 0, NULL
},
298 {"container_frame_overhead", &xvidenc_vbr_container_frame_overhead
, CONF_TYPE_INT
, CONF_MIN
, 0, 0, NULL
},
300 /* Section Aspect Ratio */
301 {"par", &xvidenc_par
, CONF_TYPE_STRING
, 0, 0, 0, NULL
},
302 {"par_width", &xvidenc_par_width
, CONF_TYPE_INT
, CONF_RANGE
, 0, 255, NULL
},
303 {"par_height", &xvidenc_par_height
, CONF_TYPE_INT
, CONF_RANGE
, 0, 255, NULL
},
304 {"aspect", &xvidenc_dar_aspect
, CONF_TYPE_FLOAT
, CONF_RANGE
, 0.1, 9.99, NULL
},
305 {"autoaspect", &xvidenc_autoaspect
, CONF_TYPE_FLAG
, 0, 0, 1, NULL
},
306 {"noautoaspect", &xvidenc_autoaspect
, CONF_TYPE_FLAG
, 0, 1, 0, NULL
},
309 {"zones", &xvidenc_zones
, CONF_TYPE_STRING
, 0, 0, 0, NULL
},
311 /* section profiles */
312 {"profile", &xvidenc_profile
, CONF_TYPE_STRING
, 0, 0, 0, NULL
},
314 /* End of the config array */
315 {NULL
, 0, 0, 0, 0, 0, NULL
}
318 /*****************************************************************************
319 * Module private data
320 ****************************************************************************/
322 typedef struct _xvid_mplayer_module_t
324 /* Instance related global vars */
326 xvid_gbl_init_t init
;
327 xvid_enc_create_t create
;
328 xvid_enc_frame_t frame
;
329 xvid_plugin_single_t onepass
;
330 xvid_plugin_2pass1_t pass1
;
331 xvid_plugin_2pass2_t pass2
;
333 /* This data must survive local block scope, so here it is */
334 xvid_enc_plugin_t plugins
[7];
335 xvid_enc_zone_t zones
[MAX_ZONES
];
337 /* MPEG4 stream buffer */
340 /* Stats accumulators */
358 /* DAR/PAR and all that thingies */
362 } xvid_mplayer_module_t
;
364 static int dispatch_settings(xvid_mplayer_module_t
*mod
);
365 static int set_create_struct(xvid_mplayer_module_t
*mod
);
366 static int set_frame_struct(xvid_mplayer_module_t
*mod
, mp_image_t
*mpi
);
367 static void update_stats(xvid_mplayer_module_t
*mod
, xvid_enc_stats_t
*stats
);
368 static void print_stats(xvid_mplayer_module_t
*mod
);
369 static void flush_internal_buffers(xvid_mplayer_module_t
*mod
);
370 static const char *par_string(int parcode
);
371 static const char *errorstring(int err
);
373 /*****************************************************************************
374 * Video Filter API function definitions
375 ****************************************************************************/
377 /*============================================================================
379 *==========================================================================*/
382 config(struct vf_instance_s
* vf
,
383 int width
, int height
, int d_width
, int d_height
,
384 unsigned int flags
, unsigned int outfmt
)
387 xvid_mplayer_module_t
*mod
= (xvid_mplayer_module_t
*)vf
->priv
;
389 /* Complete the muxer initialization */
390 mod
->mux
->bih
->biWidth
= width
;
391 mod
->mux
->bih
->biHeight
= height
;
392 mod
->mux
->bih
->biSizeImage
=
393 mod
->mux
->bih
->biWidth
* mod
->mux
->bih
->biHeight
* 3 / 2;
394 mod
->mux
->aspect
= (float)d_width
/d_height
;
396 /* Message the FourCC type */
397 mp_msg(MSGT_MENCODER
, MSGL_INFO
,
398 "videocodec: XviD (%dx%d fourcc=%x [%.4s])\n",
399 width
, height
, mod
->mux
->bih
->biCompression
,
400 (char *)&mod
->mux
->bih
->biCompression
);
402 /* Total number of pixels per frame required for PSNR */
403 mod
->pixels
= mod
->mux
->bih
->biWidth
*mod
->mux
->bih
->biHeight
;
405 /*--------------------------------------------------------------------
406 * Dispatch all module settings to XviD structures
407 *------------------------------------------------------------------*/
409 mod
->d_width
= d_width
;
410 mod
->d_height
= d_height
;
412 if(dispatch_settings(mod
) == BAD
)
415 /*--------------------------------------------------------------------
416 * Set remaining information in the xvid_enc_create_t structure
417 *------------------------------------------------------------------*/
419 if(set_create_struct(mod
) == BAD
)
422 /*--------------------------------------------------------------------
423 * Encoder instance creation
424 *------------------------------------------------------------------*/
426 err
= xvid_encore(NULL
, XVID_ENC_CREATE
, &mod
->create
, NULL
);
429 mp_msg(MSGT_MENCODER
, MSGL_ERR
,
430 "xvid: xvidcore returned a '%s' error\n", errorstring(err
));
434 /* Store the encoder instance into the private data */
435 mod
->instance
= mod
->create
.handle
;
437 mod
->mux
->decoder_delay
= mod
->create
.max_bframes
? 1 : 0;
442 /*============================================================================
444 *==========================================================================*/
447 uninit(struct vf_instance_s
* vf
)
450 xvid_mplayer_module_t
*mod
= (xvid_mplayer_module_t
*)vf
->priv
;
452 /* Destroy xvid instance */
453 xvid_encore(mod
->instance
, XVID_ENC_DESTROY
, NULL
, NULL
);
455 /* Display stats (if any) */
458 /* Close PSNR file if ever opened */
460 fclose(mod
->fvstats
);
464 /* Free allocated memory */
465 if(mod
->frame
.quant_intra_matrix
)
466 free(mod
->frame
.quant_intra_matrix
);
468 if(mod
->frame
.quant_inter_matrix
)
469 free(mod
->frame
.quant_inter_matrix
);
480 /*============================================================================
482 *==========================================================================*/
485 control(struct vf_instance_s
* vf
, int request
, void* data
)
487 xvid_mplayer_module_t
*mod
= (xvid_mplayer_module_t
*)vf
->priv
;
490 case VFCTRL_FLUSH_FRAMES
:
492 flush_internal_buffers(mod
);
495 return(CONTROL_UNKNOWN
);
498 /*============================================================================
500 *==========================================================================*/
503 query_format(struct vf_instance_s
* vf
, unsigned int fmt
)
509 return(VFCAP_CSP_SUPPORTED
| VFCAP_CSP_SUPPORTED_BY_HW
);
512 return(VFCAP_CSP_SUPPORTED
);
517 /*============================================================================
519 *==========================================================================*/
522 put_image(struct vf_instance_s
* vf
, mp_image_t
*mpi
, double pts
)
525 xvid_enc_stats_t stats
;
526 xvid_mplayer_module_t
*mod
= (xvid_mplayer_module_t
*)vf
->priv
;
528 /* Prepare the stats */
529 memset(&stats
,0,sizeof( xvid_enc_stats_t
));
530 stats
.version
= XVID_VERSION
;
532 /* -------------------------------------------------------------------
533 * Set remaining information in the xvid_enc_frame_t structure
534 * NB: all the other struct members were initialized by
536 * -----------------------------------------------------------------*/
538 if(set_frame_struct(mod
, mpi
) == BAD
)
541 /* -------------------------------------------------------------------
543 * ---------------------------------------------------------------- */
545 size
= xvid_encore(mod
->instance
, XVID_ENC_ENCODE
, &mod
->frame
, &stats
);
547 /* Analyse the returned value */
549 mp_msg(MSGT_MENCODER
, MSGL_ERR
,
550 "xvid: xvidcore returned a '%s' error\n", errorstring(size
));
554 /* If size is == 0, we're done with that frame */
556 ++mod
->mux
->encoder_delay
;
560 /* xvidcore returns stats about encoded frame in an asynchronous way
561 * accumulate these stats */
562 update_stats(mod
, &stats
);
564 /* xvidcore outputed bitstream -- mux it */
565 muxer_write_chunk(mod
->mux
,
567 (mod
->frame
.out_flags
& XVID_KEYFRAME
)?0x10:0, MP_NOPTS_VALUE
, MP_NOPTS_VALUE
);
572 /*============================================================================
574 *==========================================================================*/
577 vf_open(vf_instance_t
*vf
, char* args
)
579 xvid_mplayer_module_t
*mod
;
580 xvid_gbl_init_t xvid_gbl_init
;
581 xvid_gbl_info_t xvid_gbl_info
;
583 /* Setting libmpcodec module API pointers */
585 vf
->default_caps
= VFCAP_CONSTANT
;
586 vf
->control
= control
;
588 vf
->query_format
= query_format
;
589 vf
->put_image
= put_image
;
591 /* Allocate the private part of the codec module */
592 vf
->priv
= malloc(sizeof(xvid_mplayer_module_t
));
593 mod
= (xvid_mplayer_module_t
*)vf
->priv
;
596 mp_msg(MSGT_MENCODER
,MSGL_ERR
,
597 "xvid: memory allocation failure (private data)\n");
601 /* Initialize the module to zeros */
602 memset(mod
, 0, sizeof(xvid_mplayer_module_t
));
603 mod
->min_sse_y
= mod
->min_sse_u
= mod
->min_sse_v
= INT_MAX
;
604 mod
->max_sse_y
= mod
->max_sse_u
= mod
->max_sse_v
= INT_MIN
;
607 mod
->mux
= (muxer_stream_t
*)args
;
609 /* Initialize muxer BITMAP header */
610 mod
->mux
->bih
= calloc(1, sizeof(BITMAPINFOHEADER
));
612 if(mod
->mux
->bih
== NULL
) {
613 mp_msg(MSGT_MENCODER
,MSGL_ERR
,
614 "xvid: memory allocation failure (BITMAP header)\n");
618 mod
->mux
->bih
->biSize
= sizeof(BITMAPINFOHEADER
);
619 mod
->mux
->bih
->biWidth
= 0;
620 mod
->mux
->bih
->biHeight
= 0;
621 mod
->mux
->bih
->biPlanes
= 1;
622 mod
->mux
->bih
->biBitCount
= 12;
623 mod
->mux
->bih
->biCompression
= mmioFOURCC('X','V','I','D');
625 /* Retrieve information about the host XviD library */
626 memset(&xvid_gbl_info
, 0, sizeof(xvid_gbl_info_t
));
627 xvid_gbl_info
.version
= XVID_VERSION
;
629 if (xvid_global(NULL
, XVID_GBL_INFO
, &xvid_gbl_info
, NULL
) < 0) {
630 mp_msg(MSGT_MENCODER
,MSGL_WARN
, "xvid: could not get information about the library\n");
632 mp_msg(MSGT_MENCODER
,MSGL_INFO
, "xvid: using library version %d.%d.%d (build %s)\n",
633 XVID_VERSION_MAJOR(xvid_gbl_info
.actual_version
),
634 XVID_VERSION_MINOR(xvid_gbl_info
.actual_version
),
635 XVID_VERSION_PATCH(xvid_gbl_info
.actual_version
),
636 xvid_gbl_info
.build
);
639 /* Initialize the xvid_gbl_init structure */
640 memset(&xvid_gbl_init
, 0, sizeof(xvid_gbl_init_t
));
641 xvid_gbl_init
.version
= XVID_VERSION
;
642 xvid_gbl_init
.debug
= xvidenc_debug
;
644 /* Initialize the xvidcore library */
645 if (xvid_global(NULL
, XVID_GBL_INIT
, &xvid_gbl_init
, NULL
) < 0) {
646 mp_msg(MSGT_MENCODER
,MSGL_ERR
, "xvid: initialisation failure\n");
653 /*****************************************************************************
655 ****************************************************************************/
657 static void *read_matrix(unsigned char *filename
);
659 static int dispatch_settings(xvid_mplayer_module_t
*mod
)
661 xvid_enc_create_t
*create
= &mod
->create
;
662 xvid_enc_frame_t
*frame
= &mod
->frame
;
663 xvid_plugin_single_t
*onepass
= &mod
->onepass
;
664 xvid_plugin_2pass2_t
*pass2
= &mod
->pass2
;
667 const int motion_presets
[7] =
673 XVID_ME_HALFPELREFINE16
,
674 XVID_ME_HALFPELREFINE16
| XVID_ME_ADVANCEDDIAMOND16
,
675 XVID_ME_HALFPELREFINE16
| XVID_ME_EXTSEARCH16
|
676 XVID_ME_HALFPELREFINE8
| XVID_ME_USESQUARES16
679 //profile is unrestricted as default
680 profile_t
*selected_profile
= profileFromName("unrestricted");
682 selected_profile
= profileFromName(xvidenc_profile
);
683 if(!selected_profile
)
685 mp_msg(MSGT_MENCODER
,MSGL_ERR
,
686 "xvid:[ERROR] \"%s\" is an invalid profile name\n", xvidenc_profile
);
690 /* -------------------------------------------------------------------
691 * Dispatch all settings having an impact on the "create" structure
692 * This includes plugins as they are passed to encore through the
694 * -----------------------------------------------------------------*/
696 /* -------------------------------------------------------------------
697 * The create structure
698 * ---------------------------------------------------------------- */
706 create
->global
|= XVID_GLOBAL_EXTRASTATS_ENABLE
;
708 create
->num_zones
= 0;
709 create
->zones
= NULL
;
710 create
->num_plugins
= 0;
711 create
->plugins
= NULL
;
712 create
->num_threads
= xvidenc_num_threads
;
714 if( (selected_profile
->flags
& PROFILE_BVOP
) &&
715 /* dxn: prevent bframes usage if interlacing is selected */
716 !((selected_profile
->flags
& PROFILE_DXN
) && xvidenc_interlaced
) )
718 create
->max_bframes
= xvidenc_max_bframes
;
719 create
->bquant_ratio
= xvidenc_bquant_ratio
;
720 create
->bquant_offset
= xvidenc_bquant_offset
;
722 create
->global
|= XVID_GLOBAL_PACKED
;
723 if(xvidenc_closed_gop
)
724 create
->global
|= XVID_GLOBAL_CLOSED_GOP
;
726 /* dxn: restrict max bframes, require closed gop
727 and require packed b-frames */
728 if(selected_profile
->flags
& PROFILE_DXN
)
730 if(create
->max_bframes
> selected_profile
->dxn_max_bframes
)
731 create
->max_bframes
= selected_profile
->dxn_max_bframes
;
732 create
->global
|= XVID_GLOBAL_CLOSED_GOP
;
733 create
->global
|= XVID_GLOBAL_PACKED
;
737 create
->max_bframes
= 0;
739 #if XVID_API >= XVID_MAKE_API(4,1)
740 /* dxn: always write divx5 userdata */
741 if(selected_profile
->flags
& PROFILE_DXN
)
742 create
->global
|= XVID_GLOBAL_DIVX5_USERDATA
;
745 create
->max_key_interval
= xvidenc_max_key_interval
;
746 create
->frame_drop_ratio
= xvidenc_frame_drop_ratio
;
747 create
->min_quant
[0] = xvidenc_min_quant
[0];
748 create
->min_quant
[1] = xvidenc_min_quant
[1];
749 create
->min_quant
[2] = xvidenc_min_quant
[2];
750 create
->max_quant
[0] = xvidenc_max_quant
[0];
751 create
->max_quant
[1] = xvidenc_max_quant
[1];
752 create
->max_quant
[2] = xvidenc_max_quant
[2];
755 /* -------------------------------------------------------------------
756 * The single pass plugin
757 * ---------------------------------------------------------------- */
759 if (xvidenc_bitrate
> 16000) onepass
->bitrate
= xvidenc_bitrate
;
760 else onepass
->bitrate
= xvidenc_bitrate
*1000;
761 onepass
->reaction_delay_factor
= xvidenc_cbr_reaction_delay_factor
;
762 onepass
->averaging_period
= xvidenc_cbr_averaging_period
;
763 onepass
->buffer
= xvidenc_cbr_buffer
;
765 /* -------------------------------------------------------------------
767 * ---------------------------------------------------------------- */
769 pass2
->keyframe_boost
= xvidenc_vbr_keyframe_boost
;
770 pass2
->overflow_control_strength
= xvidenc_vbr_overflow_control_strength
;
771 pass2
->curve_compression_high
= xvidenc_vbr_curve_compression_high
;
772 pass2
->curve_compression_low
= xvidenc_vbr_curve_compression_low
;
773 pass2
->max_overflow_improvement
= xvidenc_vbr_max_overflow_improvement
;
774 pass2
->max_overflow_degradation
= xvidenc_vbr_max_overflow_degradation
;
775 pass2
->kfreduction
= xvidenc_vbr_kfreduction
;
776 pass2
->kfthreshold
= xvidenc_vbr_kfthreshold
;
777 pass2
->container_frame_overhead
= xvidenc_vbr_container_frame_overhead
;
781 #if XVID_API >= XVID_MAKE_API(4,1)
782 pass2
->vbv_size
= selected_profile
->max_vbv_size
;
783 pass2
->vbv_initial
= (selected_profile
->max_vbv_size
*3)>>2; /* 75% */
784 pass2
->vbv_maxrate
= selected_profile
->max_bitrate
;
785 pass2
->vbv_peakrate
= selected_profile
->vbv_peakrate
*3;
787 // XXX: xvidcore currently provides a "peak bits over 3 seconds" constraint.
788 // according to the latest dxn literature, a 1 second constraint is now used
790 create
->profile
= selected_profile
->id
;
792 /* -------------------------------------------------------------------
793 * The frame structure
794 * ---------------------------------------------------------------- */
795 frame
->vol_flags
= 0;
796 frame
->vop_flags
= 0;
799 frame
->vop_flags
|= XVID_VOP_HALFPEL
;
800 frame
->motion
|= motion_presets
[xvidenc_motion
];
803 frame
->vol_flags
|= XVID_VOL_EXTRASTATS
;
805 if(xvidenc_greyscale
)
806 frame
->vop_flags
|= XVID_VOP_GREYSCALE
;
808 if(xvidenc_cartoon
) {
809 frame
->vop_flags
|= XVID_VOP_CARTOON
;
810 frame
->motion
|= XVID_ME_DETECT_STATIC_MOTION
;
813 // MPEG quantisation is only supported in ASP and unrestricted profiles
814 if((selected_profile
->flags
& PROFILE_MPEGQUANT
) &&
815 (xvidenc_quant_method
!= NULL
) &&
816 !strcasecmp(xvidenc_quant_method
, "mpeg"))
818 frame
->vol_flags
|= XVID_VOL_MPEGQUANT
;
819 if(xvidenc_intra_matrix_file
!= NULL
) {
820 frame
->quant_intra_matrix
= (unsigned char*)read_matrix(xvidenc_intra_matrix_file
);
821 if(frame
->quant_intra_matrix
!= NULL
) {
822 mp_msg(MSGT_MENCODER
, MSGL_INFO
, "xvid: Loaded Intra matrix (switching to mpeg quantization type)\n");
823 if(xvidenc_quant_method
) free(xvidenc_quant_method
);
824 xvidenc_quant_method
= strdup("mpeg");
827 if(xvidenc_inter_matrix_file
!= NULL
) {
828 frame
->quant_inter_matrix
= read_matrix(xvidenc_inter_matrix_file
);
829 if(frame
->quant_inter_matrix
) {
830 mp_msg(MSGT_MENCODER
, MSGL_INFO
, "\nxvid: Loaded Inter matrix (switching to mpeg quantization type)\n");
831 if(xvidenc_quant_method
) free(xvidenc_quant_method
);
832 xvidenc_quant_method
= strdup("mpeg");
836 if(xvidenc_quarterpel
&& (selected_profile
->flags
& PROFILE_QPEL
)) {
837 frame
->vol_flags
|= XVID_VOL_QUARTERPEL
;
838 frame
->motion
|= XVID_ME_QUARTERPELREFINE16
;
839 frame
->motion
|= XVID_ME_QUARTERPELREFINE8
;
841 if(xvidenc_gmc
&& (selected_profile
->flags
& PROFILE_GMC
)) {
842 frame
->vol_flags
|= XVID_VOL_GMC
;
843 frame
->motion
|= XVID_ME_GME_REFINE
;
845 if(xvidenc_interlaced
&& (selected_profile
->flags
& PROFILE_INTERLACE
)) {
846 frame
->vol_flags
|= XVID_VOL_INTERLACING
;
848 if(xvidenc_trellis
) {
849 frame
->vop_flags
|= XVID_VOP_TRELLISQUANT
;
851 if(xvidenc_hqacpred
) {
852 frame
->vop_flags
|= XVID_VOP_HQACPRED
;
854 if(xvidenc_chroma_opt
) {
855 frame
->vop_flags
|= XVID_VOP_CHROMAOPT
;
857 if((xvidenc_motion
> 4) && (selected_profile
->flags
& PROFILE_4MV
)) {
858 frame
->vop_flags
|= XVID_VOP_INTER4V
;
860 if(xvidenc_chromame
) {
861 frame
->motion
|= XVID_ME_CHROMA_PVOP
;
862 frame
->motion
|= XVID_ME_CHROMA_BVOP
;
864 if(xvidenc_vhq
>= 1) {
865 frame
->vop_flags
|= XVID_VOP_MODEDECISION_RD
;
867 if(xvidenc_vhq
>= 2) {
868 frame
->motion
|= XVID_ME_HALFPELREFINE16_RD
;
869 frame
->motion
|= XVID_ME_QUARTERPELREFINE16_RD
;
871 if(xvidenc_vhq
>= 3) {
872 frame
->motion
|= XVID_ME_HALFPELREFINE8_RD
;
873 frame
->motion
|= XVID_ME_QUARTERPELREFINE8_RD
;
874 frame
->motion
|= XVID_ME_CHECKPREDICTION_RD
;
876 if(xvidenc_vhq
>= 4) {
877 frame
->motion
|= XVID_ME_EXTSEARCH_RD
;
879 if(xvidenc_bvhq
>= 1) {
880 #if XVID_API >= XVID_MAKE_API(4,1)
881 frame
->vop_flags
|= XVID_VOP_RD_BVOP
;
885 frame
->motion
|= XVID_ME_FASTREFINE16
;
886 frame
->motion
|= XVID_ME_FASTREFINE8
;
887 frame
->motion
|= XVID_ME_SKIP_DELTASEARCH
;
888 frame
->motion
|= XVID_ME_FAST_MODEINTERPOLATE
;
889 frame
->motion
|= XVID_ME_BFRAME_EARLYSTOP
;
892 /* motion level == 0 means no motion search which is equivalent to
893 * intra coding only */
894 if(xvidenc_motion
== 0) {
895 frame
->type
= XVID_TYPE_IVOP
;
897 frame
->type
= XVID_TYPE_AUTO
;
900 frame
->bframe_threshold
= xvidenc_bframe_threshold
;
902 /* PAR related initialization */
903 frame
->par
= XVID_PAR_11_VGA
; /* Default */
905 if( !(selected_profile
->flags
& PROFILE_DXN
) )
907 if(xvidenc_dar_aspect
> 0)
908 ar
= av_d2q(xvidenc_dar_aspect
* mod
->mux
->bih
->biHeight
/ mod
->mux
->bih
->biWidth
, 255);
909 else if(xvidenc_autoaspect
)
910 ar
= av_d2q((float)mod
->d_width
/ mod
->d_height
* mod
->mux
->bih
->biHeight
/ mod
->mux
->bih
->biWidth
, 255);
911 else ar
.num
= ar
.den
= 0;
914 if(ar
.num
== 12 && ar
.den
== 11)
915 frame
->par
= XVID_PAR_43_PAL
;
916 else if(ar
.num
== 10 && ar
.den
== 11)
917 frame
->par
= XVID_PAR_43_NTSC
;
918 else if(ar
.num
== 16 && ar
.den
== 11)
919 frame
->par
= XVID_PAR_169_PAL
;
920 else if(ar
.num
== 40 && ar
.den
== 33)
921 frame
->par
= XVID_PAR_169_NTSC
;
924 frame
->par
= XVID_PAR_EXT
;
925 frame
->par_width
= ar
.num
;
926 frame
->par_height
= ar
.den
;
929 } else if(xvidenc_par
!= NULL
) {
930 if(strcasecmp(xvidenc_par
, "pal43") == 0)
931 frame
->par
= XVID_PAR_43_PAL
;
932 else if(strcasecmp(xvidenc_par
, "pal169") == 0)
933 frame
->par
= XVID_PAR_169_PAL
;
934 else if(strcasecmp(xvidenc_par
, "ntsc43") == 0)
935 frame
->par
= XVID_PAR_43_NTSC
;
936 else if(strcasecmp(xvidenc_par
, "ntsc169") == 0)
937 frame
->par
= XVID_PAR_169_NTSC
;
938 else if(strcasecmp(xvidenc_par
, "ext") == 0)
939 frame
->par
= XVID_PAR_EXT
;
941 if(frame
->par
== XVID_PAR_EXT
) {
942 if(xvidenc_par_width
)
943 frame
->par_width
= xvidenc_par_width
;
945 frame
->par_width
= 1;
947 if(xvidenc_par_height
)
948 frame
->par_height
= xvidenc_par_height
;
950 frame
->par_height
= 1;
954 /* Display par information */
955 mp_msg(MSGT_MENCODER
, MSGL_INFO
, "xvid: par=%d/%d (%s), displayed=%dx%d, sampled=%dx%d\n",
956 ar
.num
, ar
.den
, par_string(frame
->par
),
957 mod
->d_width
, mod
->d_height
, mod
->mux
->bih
->biWidth
, mod
->mux
->bih
->biHeight
);
960 mp_msg(MSGT_MENCODER
, MSGL_INFO
,
961 "xvid: par=0/0 (vga11) forced by choosing a DXN profile\n");
965 static int set_create_struct(xvid_mplayer_module_t
*mod
)
969 xvid_enc_create_t
*create
= &mod
->create
;
971 // profile is unrestricted as default
972 profile_t
*selected_profile
= profileFromName("unrestricted");
974 selected_profile
= profileFromName(xvidenc_profile
);
975 if(!selected_profile
)
978 /* Most of the structure is initialized by dispatch settings, only a
979 * few things are missing */
980 create
->version
= XVID_VERSION
;
982 /* Width and Height */
983 create
->width
= mod
->mux
->bih
->biWidth
;
984 create
->height
= mod
->mux
->bih
->biHeight
;
986 /* Check resolution of video to be coded is within profile width/height
988 if( ((selected_profile
->width
!= 0) &&
989 (mod
->mux
->bih
->biWidth
> selected_profile
->width
)) ||
990 ((selected_profile
->height
!= 0) &&
991 (mod
->mux
->bih
->biHeight
> selected_profile
->height
)) )
993 mp_msg(MSGT_MENCODER
,MSGL_ERR
,
994 "xvid:[ERROR] resolution must be <= %dx%d for the chosen profile\n",
995 selected_profile
->width
, selected_profile
->height
);
1000 create
->fincr
= mod
->mux
->h
.dwScale
;
1001 create
->fbase
= mod
->mux
->h
.dwRate
;
1003 // Check frame rate is within profile restrictions
1004 if( ((float)mod
->mux
->h
.dwRate
/(float)mod
->mux
->h
.dwScale
> (float)selected_profile
->fps
) &&
1005 (selected_profile
->fps
!= 0))
1007 mp_msg(MSGT_MENCODER
,MSGL_ERR
,
1008 "xvid:[ERROR] frame rate must be <= %d for the chosen profile\n",
1009 selected_profile
->fps
);
1013 /* Encodings zones */
1014 memset(mod
->zones
, 0, sizeof(mod
->zones
));
1015 create
->zones
= mod
->zones
;
1016 create
->num_zones
= 0;
1019 memset(mod
->plugins
, 0, sizeof(mod
->plugins
));
1020 create
->plugins
= mod
->plugins
;
1021 create
->num_plugins
= 0;
1023 /* -------------------------------------------------------------------
1024 * Initialize and bind the right rate controller plugin
1025 * ---------------------------------------------------------------- */
1027 /* First we try to sort out configuration conflicts */
1028 if(xvidenc_quantizer
!= 0 && (xvidenc_bitrate
|| xvidenc_pass
)) {
1029 mp_msg(MSGT_MENCODER
, MSGL_ERR
,
1030 "xvid: you can't mix Fixed Quantizer Rate Control"
1031 " with other Rate Control mechanisms\n");
1035 if(xvidenc_bitrate
!= 0 && xvidenc_pass
== 1) {
1036 mp_msg(MSGT_MENCODER
, MSGL_WARN
,
1037 "xvid: bitrate setting is ignored during first pass\n");
1040 /* Sort out which sort of pass we are supposed to do
1042 * pass == 1<<1 Two pass first pass
1043 * pass == 1<<2 Two pass second pass
1044 * pass == 1<<3 Constant quantizer
1046 #define MODE_CBR (1<<0)
1047 #define MODE_2PASS1 (1<<1)
1048 #define MODE_2PASS2 (1<<2)
1049 #define MODE_QUANT (1<<3)
1053 if(xvidenc_bitrate
!= 0 && xvidenc_pass
== 0)
1056 if(xvidenc_pass
== 1)
1057 pass
|= MODE_2PASS1
;
1059 if(xvidenc_bitrate
!= 0 && xvidenc_pass
== 2)
1060 pass
|= MODE_2PASS2
;
1062 if(xvidenc_quantizer
!= 0 && xvidenc_pass
== 0)
1065 /* We must be in at least one RC mode */
1067 mp_msg(MSGT_MENCODER
, MSGL_ERR
,
1068 "xvid: you must specify one or a valid combination of "
1069 "'bitrate', 'pass', 'quantizer' settings\n");
1073 /* Sanity checking */
1074 if(pass
!= MODE_CBR
&& pass
!= MODE_QUANT
&&
1075 pass
!= MODE_2PASS1
&& pass
!= MODE_2PASS2
) {
1076 mp_msg(MSGT_MENCODER
, MSGL_ERR
,
1077 "xvid: this code should not be reached - fill a bug "
1082 /* This is a single pass encoding: either a CBR pass or a constant
1084 if(pass
== MODE_CBR
|| pass
== MODE_QUANT
) {
1085 xvid_plugin_single_t
*onepass
= &mod
->onepass
;
1087 /* There is not much left to initialize after dispatch settings */
1088 onepass
->version
= XVID_VERSION
;
1089 if (xvidenc_bitrate
> 16000) onepass
->bitrate
= xvidenc_bitrate
;
1090 else onepass
->bitrate
= xvidenc_bitrate
*1000;
1092 /* Quantizer mode uses the same plugin, we have only to define
1093 * a constant quantizer zone beginning at frame 0 */
1094 if(pass
== MODE_QUANT
) {
1096 squant
= av_d2q(xvidenc_quantizer
,128);
1098 create
->zones
[create
->num_zones
].mode
= XVID_ZONE_QUANT
;
1099 create
->zones
[create
->num_zones
].frame
= 0;
1100 create
->zones
[create
->num_zones
].increment
= squant
.num
;
1101 create
->zones
[create
->num_zones
].base
= squant
.den
;
1102 create
->num_zones
++;
1104 mp_msg(MSGT_MENCODER
, MSGL_INFO
,
1105 "xvid: Fixed Quant Rate Control -- quantizer=%d/%d=%2.2f\n",
1108 (float)(squant
.num
)/(float)(squant
.den
));
1111 mp_msg(MSGT_MENCODER
, MSGL_INFO
,
1112 "xvid: CBR Rate Control -- bitrate=%dkbit/s\n",
1113 xvidenc_bitrate
>16000?xvidenc_bitrate
/1000:xvidenc_bitrate
);
1117 create
->plugins
[create
->num_plugins
].func
= xvid_plugin_single
;
1118 create
->plugins
[create
->num_plugins
].param
= onepass
;
1119 create
->num_plugins
++;
1122 /* This is the first pass of a Two pass process */
1123 if(pass
== MODE_2PASS1
) {
1124 xvid_plugin_2pass1_t
*pass1
= &mod
->pass1
;
1126 /* There is not much to initialize for this plugin */
1127 pass1
->version
= XVID_VERSION
;
1128 pass1
->filename
= passtmpfile
;
1130 create
->plugins
[create
->num_plugins
].func
= xvid_plugin_2pass1
;
1131 create
->plugins
[create
->num_plugins
].param
= pass1
;
1132 create
->num_plugins
++;
1134 mp_msg(MSGT_MENCODER
, MSGL_INFO
,
1135 "xvid: 2Pass Rate Control -- 1st pass\n");
1138 /* This is the second pass of a Two pass process */
1139 if(pass
== MODE_2PASS2
) {
1140 xvid_plugin_2pass2_t
*pass2
= &mod
->pass2
;
1142 /* There is not much left to initialize after dispatch settings */
1143 pass2
->version
= XVID_VERSION
;
1144 pass2
->filename
= passtmpfile
;
1146 /* Positive bitrate values are bitrates as usual but if the
1147 * value is negative it is considered as being a total size
1148 * to reach (in kilobytes) */
1149 if(xvidenc_bitrate
> 0) {
1150 if(xvidenc_bitrate
> 16000) pass2
->bitrate
= xvidenc_bitrate
;
1151 else pass2
->bitrate
= xvidenc_bitrate
*1000;
1152 mp_msg(MSGT_MENCODER
, MSGL_INFO
,
1153 "xvid: 2Pass Rate Control -- 2nd pass -- bitrate=%dkbit/s\n",
1154 xvidenc_bitrate
>16000?xvidenc_bitrate
/1000:xvidenc_bitrate
);
1156 pass2
->bitrate
= xvidenc_bitrate
;
1157 mp_msg(MSGT_MENCODER
, MSGL_INFO
,
1158 "xvid: 2Pass Rate Control -- 2nd pass -- total size=%dkB\n",
1162 create
->plugins
[create
->num_plugins
].func
= xvid_plugin_2pass2
;
1163 create
->plugins
[create
->num_plugins
].param
= pass2
;
1164 create
->num_plugins
++;
1168 if(xvidenc_luminance_masking
&& (selected_profile
->flags
& PROFILE_ADAPTQUANT
)) {
1169 create
->plugins
[create
->num_plugins
].func
= xvid_plugin_lumimasking
;
1170 create
->plugins
[create
->num_plugins
].param
= NULL
;
1171 create
->num_plugins
++;
1175 if (xvidenc_zones
!= NULL
&& doZones
> 0) // do not apply zones in CQ, and first pass mode (xvid vfw doesn't allow them in those modes either)
1180 create
->num_zones
= 0; // set the number of zones back to zero, this overwrites the zone defined for CQ - desired because each zone has to be specified on the commandline even in cq mode
1187 int e
= sscanf(p
, "%d,%c,%lf", &start
, &mode
, &value
); // start,mode(q = constant quant, w = weight),value
1190 mp_msg(MSGT_MENCODER
,MSGL_ERR
, "error parsing zones\n");
1193 q
= (int)(value
* 100);
1196 if (q
< 200 || q
> 3100) // make sure that quantizer is in allowable range
1198 mp_msg(MSGT_MENCODER
, MSGL_ERR
, "zone quantizer must be between 2 and 31\n");
1203 create
->zones
[create
->num_zones
].mode
= XVID_ZONE_QUANT
;
1208 if (q
< 1 || q
> 200)
1210 mp_msg(MSGT_MENCODER
, MSGL_ERR
, "zone weight must be between 1 and 200\n");
1215 create
->zones
[create
->num_zones
].mode
= XVID_ZONE_WEIGHT
;
1218 create
->zones
[create
->num_zones
].frame
= start
;
1219 create
->zones
[create
->num_zones
].increment
= q
;
1220 create
->zones
[create
->num_zones
].base
= 100; // increment is 100 times the actual value
1221 create
->num_zones
++;
1222 if (create
->num_zones
> MAX_ZONES
) // show warning if we have too many zones
1224 mp_msg(MSGT_MENCODER
, MSGL_ERR
, "too many zones, zones will be ignored\n");
1233 static int set_frame_struct(xvid_mplayer_module_t
*mod
, mp_image_t
*mpi
)
1235 xvid_enc_frame_t
*frame
= &mod
->frame
;
1237 /* Most of the initialization is done during dispatch_settings */
1238 frame
->version
= XVID_VERSION
;
1240 /* Bind output buffer */
1241 frame
->bitstream
= mod
->mux
->buffer
;
1245 switch(mpi
->imgfmt
) {
1249 frame
->input
.csp
= XVID_CSP_USER
;
1252 frame
->input
.csp
= XVID_CSP_YUY2
;
1255 frame
->input
.csp
= XVID_CSP_UYVY
;
1258 mp_msg(MSGT_MENCODER
, MSGL_ERR
,
1259 "xvid: unsupported picture format (%s)!\n",
1260 vo_format_name(mpi
->imgfmt
));
1264 /* Bind source frame */
1265 frame
->input
.plane
[0] = mpi
->planes
[0];
1266 frame
->input
.plane
[1] = mpi
->planes
[1];
1267 frame
->input
.plane
[2] = mpi
->planes
[2];
1268 frame
->input
.stride
[0] = mpi
->stride
[0];
1269 frame
->input
.stride
[1] = mpi
->stride
[1];
1270 frame
->input
.stride
[2] = mpi
->stride
[2];
1272 /* Force the right quantizer -- It is internally managed by RC
1280 flush_internal_buffers(xvid_mplayer_module_t
*mod
)
1283 xvid_enc_frame_t
*frame
= &mod
->frame
;
1285 if (mod
->instance
== NULL
)
1286 return;/*encoder not inited*/
1288 /* Init a fake frame to force flushing */
1289 frame
->version
= XVID_VERSION
;
1290 frame
->bitstream
= mod
->mux
->buffer
;
1292 frame
->input
.csp
= XVID_CSP_NULL
;
1293 frame
->input
.plane
[0] = NULL
;
1294 frame
->input
.plane
[1] = NULL
;
1295 frame
->input
.plane
[2] = NULL
;
1296 frame
->input
.stride
[0] = 0;
1297 frame
->input
.stride
[1] = 0;
1298 frame
->input
.stride
[2] = 0;
1301 /* Flush encoder buffers caused by bframes usage */
1303 xvid_enc_stats_t stats
;
1304 memset(&stats
, 0, sizeof(xvid_enc_stats_t
));
1305 stats
.version
= XVID_VERSION
;
1307 /* Encode internal buffer */
1308 size
= xvid_encore(mod
->instance
, XVID_ENC_ENCODE
, &mod
->frame
, &stats
);
1312 update_stats(mod
, &stats
);
1314 /* xvidcore outputed bitstream -- mux it */
1315 muxer_write_chunk(mod
->mux
, size
,
1316 (mod
->frame
.out_flags
& XVID_KEYFRAME
)?0x10:0, MP_NOPTS_VALUE
, MP_NOPTS_VALUE
);
1321 #define SSE2PSNR(sse, nbpixels) \
1322 ((!(sse)) ? 99.99f : 48.131f - 10*(double)log10((double)(sse)/(double)((nbpixels))))
1324 update_stats(xvid_mplayer_module_t
*mod
, xvid_enc_stats_t
*stats
)
1326 if(xvidenc_stats
&& stats
->type
> 0) {
1327 mod
->sse_y
+= stats
->sse_y
;
1328 mod
->sse_u
+= stats
->sse_u
;
1329 mod
->sse_v
+= stats
->sse_v
;
1331 if(mod
->min_sse_y
> stats
->sse_y
) {
1332 mod
->min_sse_y
= stats
->sse_y
;
1333 mod
->min_sse_u
= stats
->sse_u
;
1334 mod
->min_sse_v
= stats
->sse_v
;
1335 mod
->min_framenum
= mod
->frames
;
1338 if(mod
->max_sse_y
< stats
->sse_y
) {
1339 mod
->max_sse_y
= stats
->sse_y
;
1340 mod
->max_sse_u
= stats
->sse_u
;
1341 mod
->max_sse_v
= stats
->sse_v
;
1342 mod
->max_framenum
= mod
->frames
;
1346 if (!mod
->fvstats
) {
1350 today2
= time (NULL
);
1351 today
= localtime (&today2
);
1352 sprintf (filename
, "psnr_%02d%02d%02d.log", today
->tm_hour
, today
->tm_min
, today
->tm_sec
);
1353 mod
->fvstats
= fopen (filename
,"w");
1354 if (!mod
->fvstats
) {
1356 /* Disable PSNR file output so we don't get here again */
1360 fprintf (mod
->fvstats
, "%6d, %2d, %6d, %2.2f, %2.2f, %2.2f, %2.2f %c\n",
1364 SSE2PSNR (stats
->sse_y
, mod
->pixels
),
1365 SSE2PSNR (stats
->sse_u
, mod
->pixels
/ 4),
1366 SSE2PSNR (stats
->sse_v
, mod
->pixels
/ 4),
1367 SSE2PSNR (stats
->sse_y
+ stats
->sse_u
+ stats
->sse_v
,(double)mod
->pixels
* 1.5),
1368 stats
->type
==1?'I':stats
->type
==2?'P':stats
->type
==3?'B':stats
->type
?'S':'?'
1376 print_stats(xvid_mplayer_module_t
*mod
)
1379 mod
->sse_y
/= mod
->frames
;
1380 mod
->sse_u
/= mod
->frames
;
1381 mod
->sse_v
/= mod
->frames
;
1383 mp_msg(MSGT_MENCODER
, MSGL_INFO
,
1384 "The value 99.99dB is a special value and represents "
1385 "the upper range limit\n");
1386 mp_msg(MSGT_MENCODER
, MSGL_INFO
,
1387 "xvid: Min PSNR Y:%.2f, Cb:%.2f, Cr:%.2f, All:%.2f in frame %d\n",
1388 SSE2PSNR(mod
->max_sse_y
, mod
->pixels
),
1389 SSE2PSNR(mod
->max_sse_u
, mod
->pixels
/4),
1390 SSE2PSNR(mod
->max_sse_v
, mod
->pixels
/4),
1391 SSE2PSNR(mod
->max_sse_y
+ mod
->max_sse_u
+ mod
->max_sse_v
, mod
->pixels
*1.5),
1393 mp_msg(MSGT_MENCODER
, MSGL_INFO
,
1394 "xvid: Average PSNR Y:%.2f, Cb:%.2f, Cr:%.2f, All:%.2f for %d frames\n",
1395 SSE2PSNR(mod
->sse_y
, mod
->pixels
),
1396 SSE2PSNR(mod
->sse_u
, mod
->pixels
/4),
1397 SSE2PSNR(mod
->sse_v
, mod
->pixels
/4),
1398 SSE2PSNR(mod
->sse_y
+ mod
->sse_u
+ mod
->sse_v
, mod
->pixels
*1.5),
1400 mp_msg(MSGT_MENCODER
, MSGL_INFO
,
1401 "xvid: Max PSNR Y:%.2f, Cb:%.2f, Cr:%.2f, All:%.2f in frame %d\n",
1402 SSE2PSNR(mod
->min_sse_y
, mod
->pixels
),
1403 SSE2PSNR(mod
->min_sse_u
, mod
->pixels
/4),
1404 SSE2PSNR(mod
->min_sse_v
, mod
->pixels
/4),
1405 SSE2PSNR(mod
->min_sse_y
+ mod
->min_sse_u
+ mod
->min_sse_v
, mod
->pixels
*1.5),
1411 static void *read_matrix(unsigned char *filename
)
1414 unsigned char *matrix
;
1417 /* Allocate matrix space */
1418 if((matrix
= malloc(64*sizeof(unsigned char))) == NULL
)
1421 /* Open the matrix file */
1422 if((input
= fopen(filename
, "rb")) == NULL
) {
1423 mp_msg(MSGT_MENCODER
, MSGL_ERR
,
1424 "xvid: Error opening the matrix file %s\n",
1430 /* Read the matrix */
1431 for(i
=0; i
<64; i
++) {
1435 /* If fscanf fails then get out of the loop */
1436 if(fscanf(input
, "%d", &value
) != 1) {
1437 mp_msg(MSGT_MENCODER
, MSGL_ERR
,
1438 "xvid: Error reading the matrix file %s\n",
1445 /* Clamp the value to safe range */
1446 value
= (value
< 1)?1 :value
;
1447 value
= (value
>255)?255:value
;
1451 /* Fills the rest with 1 */
1452 while(i
<64) matrix
[i
++] = 1;
1463 par_string(int parcode
)
1465 const char *par_string
;
1467 case XVID_PAR_11_VGA
:
1468 par_string
= "vga11";
1470 case XVID_PAR_43_PAL
:
1471 par_string
= "pal43";
1473 case XVID_PAR_43_NTSC
:
1474 par_string
= "ntsc43";
1476 case XVID_PAR_169_PAL
:
1477 par_string
= "pal169";
1479 case XVID_PAR_169_NTSC
:
1480 par_string
= "ntsc69";
1486 par_string
= "unknown";
1489 return (par_string
);
1492 static const char *errorstring(int err
)
1497 error
= "General fault";
1499 case XVID_ERR_MEMORY
:
1500 error
= "Memory allocation error";
1502 case XVID_ERR_FORMAT
:
1503 error
= "File format error";
1505 case XVID_ERR_VERSION
:
1506 error
= "Structure version not supported";
1509 error
= "End of stream reached";
1518 /*****************************************************************************
1519 * Module structure definition
1520 ****************************************************************************/
1522 vf_info_t ve_info_xvid
= {
1525 "Marco Belli <elcabesa@inwind.it>, Edouard Gomez <ed.gomez@free.fr>",
1531 /* Please do not change that tag comment.
1532 * arch-tag: 42ccc257-0548-4a3e-9617-2876c4e8ac88 mplayer xvid encoder module */