23 |
* |
* |
24 |
* History: |
* History: |
25 |
* |
* |
26 |
* ... ??? |
* 15.04.2002 updated cbr support |
27 |
|
* 04.04.2002 separated 2-pass code to 2pass.c |
28 |
|
* interlacing support |
29 |
|
* hinted ME support |
30 |
|
* 23.03.2002 daniel smith <danielsmith@astroboymail.com> |
31 |
|
* changed inter4v to only be in modes 5 or 6 |
32 |
|
* fixed null mode crash ? |
33 |
|
* merged foxer's alternative 2-pass code |
34 |
|
* added DEBUGERR output on errors instead of returning |
35 |
|
* 16.03.2002 daniel smith <danielsmith@astroboymail.com> |
36 |
|
* changed BITMAPV4HEADER to BITMAPINFOHEADER |
37 |
|
* - prevents memcpy crash in compress_get_format() |
38 |
|
* credits are processed in external 2pass mode |
39 |
|
* motion search precision = 0 now effective in 2-pass |
40 |
|
* modulated quantization |
41 |
|
* added DX50 fourcc |
42 |
* 01.12.2001 inital version; (c)2001 peter ross <suxen_drol@hotmail.com> |
* 01.12.2001 inital version; (c)2001 peter ross <suxen_drol@hotmail.com> |
43 |
* |
* |
44 |
*************************************************************************/ |
*************************************************************************/ |
47 |
#include <vfw.h> |
#include <vfw.h> |
48 |
|
|
49 |
#include "codec.h" |
#include "codec.h" |
50 |
#include <math.h> // fabs() |
#include "2pass.h" |
51 |
|
|
52 |
int pmvfast_presets[7] = { |
int pmvfast_presets[7] = { |
53 |
0, PMV_QUICKSTOP16, PMV_EARLYSTOP16, PMV_EARLYSTOP16 | PMV_EARLYSTOP8, |
0, PMV_QUICKSTOP16, PMV_EARLYSTOP16, PMV_EARLYSTOP16 | PMV_EARLYSTOP8, |
61 |
or XVID_CSP_NULL if failure |
or XVID_CSP_NULL if failure |
62 |
*/ |
*/ |
63 |
|
|
64 |
int get_colorspace(BITMAPV4HEADER * hdr) |
int get_colorspace(BITMAPINFOHEADER * hdr) |
65 |
{ |
{ |
66 |
if (hdr->bV4Height < 0) |
if (hdr->biHeight < 0) |
67 |
{ |
{ |
68 |
DEBUG("colorspace: inverted input format not supported"); |
DEBUGERR("colorspace: inverted input format not supported"); |
69 |
return XVID_CSP_NULL; |
return XVID_CSP_NULL; |
70 |
} |
} |
71 |
|
|
72 |
switch(hdr->bV4V4Compression) |
switch(hdr->biCompression) |
73 |
{ |
{ |
74 |
case BI_RGB : |
case BI_RGB : |
75 |
if (hdr->bV4BitCount == 16) |
if (hdr->biBitCount == 16) |
76 |
{ |
{ |
77 |
DEBUG("RGB16 (RGB555)"); |
DEBUG("RGB16 (RGB555)"); |
78 |
return XVID_CSP_VFLIP | XVID_CSP_RGB555; |
return XVID_CSP_VFLIP | XVID_CSP_RGB555; |
79 |
} |
} |
80 |
if (hdr->bV4BitCount == 24) |
if (hdr->biBitCount == 24) |
81 |
{ |
{ |
82 |
DEBUG("RGB24"); |
DEBUG("RGB24"); |
83 |
return XVID_CSP_VFLIP | XVID_CSP_RGB24; |
return XVID_CSP_VFLIP | XVID_CSP_RGB24; |
84 |
} |
} |
85 |
if (hdr->bV4BitCount == 32) |
if (hdr->biBitCount == 32) |
86 |
{ |
{ |
87 |
DEBUG("RGB32"); |
DEBUG("RGB32"); |
88 |
return XVID_CSP_VFLIP | XVID_CSP_RGB32; |
return XVID_CSP_VFLIP | XVID_CSP_RGB32; |
89 |
} |
} |
90 |
|
|
91 |
DEBUG1("BI_RGB unsupported", hdr->bV4BitCount); |
DEBUG1("BI_RGB unsupported", hdr->biBitCount); |
92 |
return XVID_CSP_NULL; |
return XVID_CSP_NULL; |
93 |
|
|
94 |
case BI_BITFIELDS : |
// how do these work in BITMAPINFOHEADER ??? |
95 |
if(hdr->bV4BitCount == 16 && |
/* case BI_BITFIELDS : |
96 |
|
if (hdr->biBitCount == 16 |
97 |
|
if(hdr->biBitCount == 16 && |
98 |
hdr->bV4RedMask == 0x7c00 && |
hdr->bV4RedMask == 0x7c00 && |
99 |
hdr->bV4GreenMask == 0x3e0 && |
hdr->bV4GreenMask == 0x3e0 && |
100 |
hdr->bV4BlueMask == 0x1f) |
hdr->bV4BlueMask == 0x1f) |
113 |
|
|
114 |
DEBUG1("BI_FIELDS unsupported", hdr->bV4BitCount); |
DEBUG1("BI_FIELDS unsupported", hdr->bV4BitCount); |
115 |
return XVID_CSP_NULL; |
return XVID_CSP_NULL; |
116 |
|
*/ |
117 |
case FOURCC_I420: |
case FOURCC_I420: |
118 |
case FOURCC_IYUV: |
case FOURCC_IYUV: |
119 |
DEBUG("IYUY"); |
DEBUG("IYUY"); |
138 |
return XVID_CSP_UYVY; |
return XVID_CSP_UYVY; |
139 |
|
|
140 |
} |
} |
141 |
DEBUGFOURCC("colorspace: unknown", hdr->bV4V4Compression); |
DEBUGFOURCC("colorspace: unknown", hdr->biCompression); |
142 |
return XVID_CSP_NULL; |
return XVID_CSP_NULL; |
143 |
} |
} |
144 |
|
|
150 |
|
|
151 |
LRESULT compress_query(CODEC * codec, BITMAPINFO * lpbiInput, BITMAPINFO * lpbiOutput) |
LRESULT compress_query(CODEC * codec, BITMAPINFO * lpbiInput, BITMAPINFO * lpbiOutput) |
152 |
{ |
{ |
153 |
BITMAPV4HEADER * inhdr = (BITMAPV4HEADER *)&lpbiInput->bmiHeader; |
BITMAPINFOHEADER * inhdr = &lpbiInput->bmiHeader; |
154 |
BITMAPV4HEADER * outhdr = (BITMAPV4HEADER *)&lpbiOutput->bmiHeader; |
BITMAPINFOHEADER * outhdr = &lpbiOutput->bmiHeader; |
155 |
|
|
156 |
if (get_colorspace(inhdr) == XVID_CSP_NULL) |
if (get_colorspace(inhdr) == XVID_CSP_NULL) |
157 |
{ |
{ |
163 |
return ICERR_OK; |
return ICERR_OK; |
164 |
} |
} |
165 |
|
|
166 |
if (inhdr->bV4Width != outhdr->bV4Width || inhdr->bV4Height != outhdr->bV4Height || |
if (inhdr->biWidth != outhdr->biWidth || inhdr->biHeight != outhdr->biHeight || |
167 |
(outhdr->bV4V4Compression != FOURCC_XVID && outhdr->bV4V4Compression != FOURCC_DIVX)) |
(outhdr->biCompression != FOURCC_XVID && outhdr->biCompression != FOURCC_DIVX)) |
168 |
{ |
{ |
169 |
return ICERR_BADFORMAT; |
return ICERR_BADFORMAT; |
170 |
} |
} |
175 |
|
|
176 |
LRESULT compress_get_format(CODEC * codec, BITMAPINFO * lpbiInput, BITMAPINFO * lpbiOutput) |
LRESULT compress_get_format(CODEC * codec, BITMAPINFO * lpbiInput, BITMAPINFO * lpbiOutput) |
177 |
{ |
{ |
178 |
BITMAPV4HEADER * inhdr = (BITMAPV4HEADER *)&lpbiInput->bmiHeader; |
BITMAPINFOHEADER * inhdr = &lpbiInput->bmiHeader; |
179 |
BITMAPV4HEADER * outhdr = (BITMAPV4HEADER *)&lpbiOutput->bmiHeader; |
BITMAPINFOHEADER * outhdr = &lpbiOutput->bmiHeader; |
180 |
|
|
181 |
if (get_colorspace(inhdr) == XVID_CSP_NULL) |
if (get_colorspace(inhdr) == XVID_CSP_NULL) |
182 |
{ |
{ |
188 |
return sizeof(BITMAPV4HEADER); |
return sizeof(BITMAPV4HEADER); |
189 |
} |
} |
190 |
|
|
191 |
memcpy(outhdr, inhdr, sizeof(BITMAPV4HEADER)); |
memcpy(outhdr, inhdr, sizeof(BITMAPINFOHEADER)); |
192 |
outhdr->bV4Size = sizeof(BITMAPV4HEADER); |
outhdr->biSize = sizeof(BITMAPINFOHEADER); |
193 |
outhdr->bV4BitCount = 24; // or 16 |
outhdr->biBitCount = 24; // or 16 |
194 |
outhdr->bV4SizeImage = compress_get_size(codec, lpbiInput, lpbiOutput); |
outhdr->biSizeImage = compress_get_size(codec, lpbiInput, lpbiOutput); |
195 |
outhdr->bV4XPelsPerMeter = 0; |
outhdr->biXPelsPerMeter = 0; |
196 |
outhdr->bV4YPelsPerMeter = 0; |
outhdr->biYPelsPerMeter = 0; |
197 |
outhdr->bV4ClrUsed = 0; |
outhdr->biClrUsed = 0; |
198 |
outhdr->bV4ClrImportant = 0; |
outhdr->biClrImportant = 0; |
199 |
|
|
200 |
if (codec->config.fourcc_used == 0) |
if (codec->config.fourcc_used == 0) |
201 |
{ |
{ |
202 |
outhdr->bV4V4Compression = FOURCC_XVID; |
outhdr->biCompression = FOURCC_XVID; |
203 |
|
} |
204 |
|
else if (codec->config.fourcc_used == 1) |
205 |
|
{ |
206 |
|
outhdr->biCompression = FOURCC_DIVX; |
207 |
} |
} |
208 |
else |
else |
209 |
{ |
{ |
210 |
outhdr->bV4V4Compression = FOURCC_DIVX; |
outhdr->biCompression = FOURCC_DX50; |
211 |
} |
} |
212 |
|
|
213 |
return ICERR_OK; |
return ICERR_OK; |
216 |
|
|
217 |
LRESULT compress_get_size(CODEC * codec, BITMAPINFO * lpbiInput, BITMAPINFO * lpbiOutput) |
LRESULT compress_get_size(CODEC * codec, BITMAPINFO * lpbiInput, BITMAPINFO * lpbiOutput) |
218 |
{ |
{ |
219 |
BITMAPV4HEADER * outhdr = (BITMAPV4HEADER *)&lpbiOutput->bmiHeader; |
return lpbiOutput->bmiHeader.biWidth * lpbiOutput->bmiHeader.biHeight * 3; |
|
return outhdr->bV4Width * outhdr->bV4Height * 3; |
|
220 |
} |
} |
221 |
|
|
222 |
|
|
231 |
|
|
232 |
LRESULT compress_begin(CODEC * codec, BITMAPINFO * lpbiInput, BITMAPINFO * lpbiOutput) |
LRESULT compress_begin(CODEC * codec, BITMAPINFO * lpbiInput, BITMAPINFO * lpbiOutput) |
233 |
{ |
{ |
|
BITMAPV4HEADER * inhdr = (BITMAPV4HEADER *)&lpbiInput->bmiHeader; |
|
234 |
XVID_ENC_PARAM param; |
XVID_ENC_PARAM param; |
235 |
XVID_INIT_PARAM init_param; |
XVID_INIT_PARAM init_param; |
236 |
|
|
237 |
switch (codec->config.mode) |
switch (codec->config.mode) |
238 |
{ |
{ |
239 |
case DLG_MODE_CBR : |
case DLG_MODE_CBR : |
240 |
param.bitrate = codec->config.bitrate; |
param.rc_bitrate = codec->config.rc_bitrate; |
241 |
param.rc_buffersize = codec->config.rc_buffersize; |
param.rc_reaction_delay_factor = codec->config.rc_reaction_delay_factor; |
242 |
|
param.rc_averaging_period = codec->config.rc_averaging_period; |
243 |
|
param.rc_buffer = codec->config.rc_buffer; |
244 |
break; |
break; |
245 |
|
|
246 |
case DLG_MODE_VBR_QUAL : |
case DLG_MODE_VBR_QUAL : |
247 |
codec->config.fquant = 0; |
codec->config.fquant = 0; |
248 |
param.bitrate = 0; |
param.rc_bitrate = 0; |
249 |
break; |
break; |
250 |
|
|
251 |
case DLG_MODE_VBR_QUANT : |
case DLG_MODE_VBR_QUANT : |
252 |
codec->config.fquant = (float) codec->config.quant; |
codec->config.fquant = (float) codec->config.quant; |
253 |
param.bitrate = 0; |
param.rc_bitrate = 0; |
254 |
break; |
break; |
255 |
|
|
256 |
case DLG_MODE_2PASS_1: |
case DLG_MODE_2PASS_1: |
257 |
case DLG_MODE_2PASS_2_INT: |
case DLG_MODE_2PASS_2_INT: |
258 |
case DLG_MODE_2PASS_2_EXT: |
case DLG_MODE_2PASS_2_EXT: |
259 |
param.bitrate = 0; |
param.rc_bitrate = 0; |
260 |
|
codec->twopass.max_framesize = (int)((double)codec->config.twopass_max_bitrate / 8.0 / ((double)codec->fbase / (double)codec->fincr)); |
261 |
break; |
break; |
262 |
|
|
263 |
|
case DLG_MODE_NULL : |
264 |
|
return ICERR_OK; |
265 |
|
|
266 |
default : |
default : |
267 |
break; |
break; |
268 |
} |
} |
278 |
if((codec->config.cpu & XVID_CPU_FORCE) <= 0) |
if((codec->config.cpu & XVID_CPU_FORCE) <= 0) |
279 |
codec->config.cpu = init_param.cpu_flags; |
codec->config.cpu = init_param.cpu_flags; |
280 |
|
|
281 |
param.width = inhdr->bV4Width; |
param.width = lpbiInput->bmiHeader.biWidth; |
282 |
param.height = inhdr->bV4Height; |
param.height = lpbiInput->bmiHeader.biHeight; |
283 |
param.fincr = codec->fincr; |
param.fincr = codec->fincr; |
284 |
param.fbase = codec->fbase; |
param.fbase = codec->fbase; |
285 |
|
|
286 |
param.rc_buffersize = codec->config.rc_buffersize; |
param.min_quantizer = codec->config.min_pquant; |
287 |
|
param.max_quantizer = codec->config.max_pquant; |
|
param.min_quantizer = codec->config.min_quant; |
|
|
param.max_quantizer = codec->config.max_quant; |
|
288 |
param.max_key_interval = codec->config.max_key_interval; |
param.max_key_interval = codec->config.max_key_interval; |
289 |
|
|
290 |
switch(xvid_encore(0, XVID_ENC_CREATE, ¶m, NULL)) |
switch(xvid_encore(0, XVID_ENC_CREATE, ¶m, NULL)) |
303 |
codec->framenum = 0; |
codec->framenum = 0; |
304 |
codec->keyspacing = 0; |
codec->keyspacing = 0; |
305 |
|
|
306 |
codec->twopass.stats1 = codec->twopass.stats2 = INVALID_HANDLE_VALUE; |
codec->twopass.hints = codec->twopass.stats1 = codec->twopass.stats2 = INVALID_HANDLE_VALUE; |
307 |
|
codec->twopass.hintstream = NULL; |
308 |
|
|
309 |
return ICERR_OK; |
return ICERR_OK; |
310 |
} |
} |
315 |
if (codec->ehandle != NULL) |
if (codec->ehandle != NULL) |
316 |
{ |
{ |
317 |
xvid_encore(codec->ehandle, XVID_ENC_DESTROY, NULL, NULL); |
xvid_encore(codec->ehandle, XVID_ENC_DESTROY, NULL, NULL); |
318 |
|
|
319 |
|
if (codec->twopass.hints != INVALID_HANDLE_VALUE) |
320 |
|
{ |
321 |
|
CloseHandle(codec->twopass.hints); |
322 |
|
} |
323 |
if (codec->twopass.stats1 != INVALID_HANDLE_VALUE) |
if (codec->twopass.stats1 != INVALID_HANDLE_VALUE) |
324 |
{ |
{ |
325 |
CloseHandle(codec->twopass.stats1); |
CloseHandle(codec->twopass.stats1); |
328 |
{ |
{ |
329 |
CloseHandle(codec->twopass.stats2); |
CloseHandle(codec->twopass.stats2); |
330 |
} |
} |
331 |
|
if (codec->twopass.hintstream != NULL) |
332 |
|
{ |
333 |
|
free(codec->twopass.hintstream); |
334 |
|
} |
335 |
|
|
336 |
codec->ehandle = NULL; |
codec->ehandle = NULL; |
337 |
|
|
338 |
|
codec_2pass_finish(codec); |
339 |
} |
} |
340 |
|
|
341 |
return ICERR_OK; |
return ICERR_OK; |
344 |
|
|
345 |
LRESULT compress(CODEC * codec, ICCOMPRESS * icc) |
LRESULT compress(CODEC * codec, ICCOMPRESS * icc) |
346 |
{ |
{ |
347 |
BITMAPV4HEADER * inhdr = (BITMAPV4HEADER *)icc->lpbiInput; |
BITMAPINFOHEADER * inhdr = icc->lpbiInput; |
348 |
BITMAPV4HEADER * outhdr = (BITMAPV4HEADER *)icc->lpbiOutput; |
BITMAPINFOHEADER * outhdr = icc->lpbiOutput; |
349 |
XVID_ENC_FRAME frame; |
XVID_ENC_FRAME frame; |
350 |
XVID_ENC_STATS stats; |
XVID_ENC_STATS stats; |
351 |
|
|
360 |
|
|
361 |
frame.general = 0; |
frame.general = 0; |
362 |
frame.motion = 0; |
frame.motion = 0; |
363 |
|
frame.intra = -1; |
|
if(codec->config.motion_search == 0) |
|
|
frame.intra = 1; |
|
364 |
|
|
365 |
frame.general |= XVID_HALFPEL; |
frame.general |= XVID_HALFPEL; |
366 |
|
|
367 |
if(codec->config.motion_search > 3) |
if(codec->config.motion_search > 4) |
368 |
frame.general |= XVID_INTER4V; |
frame.general |= XVID_INTER4V; |
369 |
|
|
370 |
// we actually need "default/custom" selectbox for both inter/intra |
if(((codec->config.mode == DLG_MODE_2PASS_1) ? 0 : codec->config.lum_masking) == 1) |
371 |
// this will do for now |
frame.general |= XVID_LUMIMASKING; |
372 |
|
|
373 |
if (codec->config.quant_type == QUANT_MODE_CUSTOM) |
if (codec->config.interlacing) |
374 |
|
frame.general |= XVID_INTERLACING; |
375 |
|
|
376 |
|
if (codec->config.hinted_me && codec->config.mode == DLG_MODE_2PASS_1) |
377 |
{ |
{ |
378 |
frame.general |= XVID_CUSTOM_QMATRIX; |
frame.hint.hintstream = codec->twopass.hintstream; |
379 |
frame.quant_intra_matrix = codec->config.qmatrix_intra; |
frame.hint.rawhints = 0; |
380 |
frame.quant_inter_matrix = codec->config.qmatrix_inter; |
frame.general |= XVID_HINTEDME_GET; |
381 |
} |
} |
382 |
else |
else if (codec->config.hinted_me && (codec->config.mode == DLG_MODE_2PASS_2_EXT || codec->config.mode == DLG_MODE_2PASS_2_INT)) |
383 |
{ |
{ |
384 |
frame.quant_intra_matrix = NULL; |
DWORD read; |
385 |
frame.quant_inter_matrix = NULL; |
DWORD blocksize; |
|
} |
|
386 |
|
|
387 |
if(codec->config.quant_type == 0) |
frame.hint.hintstream = codec->twopass.hintstream; |
388 |
frame.general |= XVID_H263QUANT; |
frame.hint.rawhints = 0; |
389 |
else |
frame.general |= XVID_HINTEDME_SET; |
|
frame.general |= XVID_MPEGQUANT; |
|
390 |
|
|
391 |
if(((codec->config.mode == DLG_MODE_2PASS_1) ? 0 : codec->config.lum_masking) == 1) |
if (codec->twopass.hints == INVALID_HANDLE_VALUE) |
392 |
frame.general |= XVID_LUMIMASKING; |
{ |
393 |
|
codec->twopass.hints = CreateFile(codec->config.hintfile, GENERIC_READ, 0, 0, OPEN_EXISTING, 0, 0); |
394 |
|
if (codec->twopass.hints == INVALID_HANDLE_VALUE) |
395 |
|
{ |
396 |
|
DEBUGERR("couldn't open hints file"); |
397 |
|
return ICERR_ERROR; |
398 |
|
} |
399 |
|
} |
400 |
|
if (!ReadFile(codec->twopass.hints, &blocksize, sizeof(DWORD), &read, 0) || read != sizeof(DWORD) || |
401 |
|
!ReadFile(codec->twopass.hints, frame.hint.hintstream, blocksize, &read, 0) || read != blocksize) |
402 |
|
{ |
403 |
|
DEBUGERR("couldn't read from hints file"); |
404 |
|
return ICERR_ERROR; |
405 |
|
} |
406 |
|
} |
407 |
|
|
408 |
frame.motion = pmvfast_presets[codec->config.motion_search]; |
frame.motion = pmvfast_presets[codec->config.motion_search]; |
409 |
|
|
410 |
frame.image = icc->lpInput; |
frame.image = icc->lpInput; |
411 |
|
|
412 |
if ((frame.colorspace = get_colorspace(inhdr)) == XVID_CSP_NULL) { |
if ((frame.colorspace = get_colorspace(inhdr)) == XVID_CSP_NULL) |
413 |
return ICERR_BADFORMAT; |
return ICERR_BADFORMAT; |
|
} |
|
414 |
|
|
415 |
frame.bitstream = icc->lpOutput; |
frame.bitstream = icc->lpOutput; |
416 |
frame.length = icc->lpbiOutput->biSizeImage; |
frame.length = icc->lpbiOutput->biSizeImage; |
|
frame.intra = -1; |
|
417 |
|
|
418 |
switch (codec->config.mode) |
switch (codec->config.mode) |
419 |
{ |
{ |
438 |
} |
} |
439 |
if (codec->config.dummy2pass) |
if (codec->config.dummy2pass) |
440 |
{ |
{ |
441 |
outhdr->bV4SizeImage = codec->twopass.bytes2; |
outhdr->biSizeImage = codec->twopass.bytes2; |
442 |
*icc->lpdwFlags = (codec->twopass.nns1.quant & NNSTATS_KEYFRAME) ? AVIIF_KEYFRAME : 0; |
*icc->lpdwFlags = (codec->twopass.nns1.quant & NNSTATS_KEYFRAME) ? AVIIF_KEYFRAME : 0; |
443 |
return ICERR_OK; |
return ICERR_OK; |
444 |
} |
} |
445 |
break; |
break; |
446 |
|
|
447 |
case DLG_MODE_NULL : |
case DLG_MODE_NULL : |
448 |
outhdr->bV4SizeImage = 0; |
outhdr->biSizeImage = 0; |
449 |
*icc->lpdwFlags = AVIIF_KEYFRAME; |
*icc->lpdwFlags = AVIIF_KEYFRAME; |
450 |
return ICERR_OK; |
return ICERR_OK; |
451 |
|
|
452 |
default : |
default : |
453 |
DEBUG("Invalid encoding mode"); |
DEBUGERR("Invalid encoding mode"); |
454 |
return ICERR_ERROR; |
return ICERR_ERROR; |
455 |
} |
} |
456 |
|
|
457 |
// force keyframe spacing in 2-pass modes |
if (codec->config.quant_type == QUANT_MODE_H263) |
458 |
if ((codec->keyspacing < codec->config.min_key_interval && codec->framenum) && |
{ |
459 |
(codec->config.mode == DLG_MODE_2PASS_1 || codec->config.mode == DLG_MODE_2PASS_2_INT || |
frame.general |= XVID_H263QUANT; |
460 |
codec->config.mode == DLG_MODE_2PASS_2_EXT)) |
} |
461 |
|
else |
462 |
|
{ |
463 |
|
frame.general |= XVID_MPEGQUANT; |
464 |
|
|
465 |
|
// we actually need "default/custom" selectbox for both inter/intra |
466 |
|
// this will do for now |
467 |
|
if (codec->config.quant_type == QUANT_MODE_CUSTOM) |
468 |
|
{ |
469 |
|
frame.general |= XVID_CUSTOM_QMATRIX; |
470 |
|
frame.quant_intra_matrix = codec->config.qmatrix_intra; |
471 |
|
frame.quant_inter_matrix = codec->config.qmatrix_inter; |
472 |
|
} |
473 |
|
else |
474 |
|
{ |
475 |
|
frame.quant_intra_matrix = NULL; |
476 |
|
frame.quant_inter_matrix = NULL; |
477 |
|
} |
478 |
|
} |
479 |
|
|
480 |
|
// force keyframe spacing in 2-pass 1st pass |
481 |
|
if (codec->config.motion_search == 0) |
482 |
|
{ |
483 |
|
frame.intra = 1; |
484 |
|
} |
485 |
|
else if (codec->keyspacing < codec->config.min_key_interval && codec->framenum) |
486 |
{ |
{ |
487 |
DEBUG("current frame forced to p-frame"); |
DEBUG("current frame forced to p-frame"); |
488 |
frame.intra = 0; |
frame.intra = 0; |
510 |
*icc->lpdwFlags = 0; |
*icc->lpdwFlags = 0; |
511 |
} |
} |
512 |
|
|
513 |
outhdr->bV4SizeImage = frame.length; |
outhdr->biSizeImage = frame.length; |
514 |
|
|
515 |
|
if (codec->config.mode == DLG_MODE_2PASS_1 && codec->config.discard1pass) |
516 |
|
{ |
517 |
|
outhdr->biSizeImage = 0; |
518 |
|
} |
519 |
|
|
520 |
|
if (frame.general & XVID_HINTEDME_GET) |
521 |
|
{ |
522 |
|
DWORD wrote; |
523 |
|
DWORD blocksize = frame.hint.hintlength; |
524 |
|
|
525 |
if (codec->config.mode == DLG_MODE_2PASS_1) |
if (codec->twopass.hints == INVALID_HANDLE_VALUE) |
526 |
{ |
{ |
527 |
if (codec->config.discard1pass) |
codec->twopass.hints = CreateFile(codec->config.hintfile, GENERIC_WRITE, 0, 0, CREATE_ALWAYS, 0, 0); |
528 |
|
if (codec->twopass.hints == INVALID_HANDLE_VALUE) |
529 |
|
{ |
530 |
|
DEBUGERR("couldn't create hints file"); |
531 |
|
return ICERR_ERROR; |
532 |
|
} |
533 |
|
} |
534 |
|
if (!WriteFile(codec->twopass.hints, &frame.hint.hintlength, sizeof(int), &wrote, 0) || wrote != sizeof(int) || |
535 |
|
!WriteFile(codec->twopass.hints, frame.hint.hintstream, blocksize, &wrote, 0) || wrote != blocksize) |
536 |
{ |
{ |
537 |
outhdr->bV4SizeImage = 0; |
DEBUGERR("couldn't write to hints file"); |
538 |
|
return ICERR_ERROR; |
539 |
} |
} |
540 |
} |
} |
541 |
|
|
553 |
|
|
554 |
LRESULT decompress_query(CODEC * codec, BITMAPINFO *lpbiInput, BITMAPINFO *lpbiOutput) |
LRESULT decompress_query(CODEC * codec, BITMAPINFO *lpbiInput, BITMAPINFO *lpbiOutput) |
555 |
{ |
{ |
556 |
BITMAPV4HEADER * inhdr = (BITMAPV4HEADER *)&lpbiInput->bmiHeader; |
BITMAPINFOHEADER * inhdr = &lpbiInput->bmiHeader; |
557 |
BITMAPV4HEADER * outhdr = (BITMAPV4HEADER *)&lpbiOutput->bmiHeader; |
BITMAPINFOHEADER * outhdr = &lpbiOutput->bmiHeader; |
558 |
|
|
559 |
if (lpbiInput == NULL) |
if (lpbiInput == NULL) |
560 |
{ |
{ |
561 |
return ICERR_ERROR; |
return ICERR_ERROR; |
562 |
} |
} |
563 |
|
|
564 |
if (inhdr->bV4V4Compression != FOURCC_XVID && inhdr->bV4V4Compression != FOURCC_DIVX) |
if (inhdr->biCompression != FOURCC_XVID && inhdr->biCompression != FOURCC_DIVX) |
565 |
{ |
{ |
566 |
return ICERR_BADFORMAT; |
return ICERR_BADFORMAT; |
567 |
} |
} |
571 |
return ICERR_OK; |
return ICERR_OK; |
572 |
} |
} |
573 |
|
|
574 |
if (inhdr->bV4Width != outhdr->bV4Width || |
if (inhdr->biWidth != outhdr->biWidth || |
575 |
inhdr->bV4Height != outhdr->bV4Height || |
inhdr->biHeight != outhdr->biHeight || |
576 |
get_colorspace(outhdr) == XVID_CSP_NULL) |
get_colorspace(outhdr) == XVID_CSP_NULL) |
577 |
{ |
{ |
578 |
return ICERR_BADFORMAT; |
return ICERR_BADFORMAT; |
584 |
|
|
585 |
LRESULT decompress_get_format(CODEC * codec, BITMAPINFO * lpbiInput, BITMAPINFO * lpbiOutput) |
LRESULT decompress_get_format(CODEC * codec, BITMAPINFO * lpbiInput, BITMAPINFO * lpbiOutput) |
586 |
{ |
{ |
587 |
BITMAPV4HEADER * inhdr = (BITMAPV4HEADER *)&lpbiInput->bmiHeader; |
BITMAPINFOHEADER * inhdr = &lpbiInput->bmiHeader; |
588 |
BITMAPV4HEADER * outhdr = (BITMAPV4HEADER *)&lpbiOutput->bmiHeader; |
BITMAPINFOHEADER * outhdr = &lpbiOutput->bmiHeader; |
589 |
LRESULT result; |
LRESULT result; |
590 |
|
|
591 |
if (lpbiOutput == NULL) |
if (lpbiOutput == NULL) |
592 |
{ |
{ |
593 |
return sizeof(BITMAPV4HEADER); |
return sizeof(BITMAPINFOHEADER); |
594 |
} |
} |
595 |
|
|
596 |
result = decompress_query(codec, lpbiInput, lpbiOutput); |
result = decompress_query(codec, lpbiInput, lpbiOutput); |
599 |
return result; |
return result; |
600 |
} |
} |
601 |
|
|
602 |
memcpy(outhdr, inhdr, sizeof(BITMAPV4HEADER)); |
memcpy(outhdr, inhdr, sizeof(BITMAPINFOHEADER)); |
603 |
outhdr->bV4Size = sizeof(BITMAPV4HEADER); |
outhdr->biSize = sizeof(BITMAPINFOHEADER); |
604 |
outhdr->bV4V4Compression = FOURCC_YUY2; |
outhdr->biCompression = FOURCC_YUY2; |
605 |
outhdr->bV4SizeImage = outhdr->bV4Width * outhdr->bV4Height * outhdr->bV4BitCount; |
outhdr->biSizeImage = outhdr->biWidth * outhdr->biHeight * outhdr->biBitCount; |
606 |
outhdr->bV4XPelsPerMeter = 0; |
outhdr->biXPelsPerMeter = 0; |
607 |
outhdr->bV4YPelsPerMeter = 0; |
outhdr->biYPelsPerMeter = 0; |
608 |
outhdr->bV4ClrUsed = 0; |
outhdr->biClrUsed = 0; |
609 |
outhdr->bV4ClrImportant = 0; |
outhdr->biClrImportant = 0; |
610 |
|
|
611 |
return ICERR_OK; |
return ICERR_OK; |
612 |
} |
} |
614 |
|
|
615 |
LRESULT decompress_begin(CODEC * codec, BITMAPINFO * lpbiInput, BITMAPINFO * lpbiOutput) |
LRESULT decompress_begin(CODEC * codec, BITMAPINFO * lpbiInput, BITMAPINFO * lpbiOutput) |
616 |
{ |
{ |
|
BITMAPV4HEADER * inhdr = (BITMAPV4HEADER *)&lpbiInput->bmiHeader; |
|
|
BITMAPV4HEADER * outhdr = (BITMAPV4HEADER *)&lpbiOutput->bmiHeader; |
|
617 |
XVID_DEC_PARAM param; |
XVID_DEC_PARAM param; |
618 |
XVID_INIT_PARAM init_param; |
XVID_INIT_PARAM init_param; |
619 |
|
|
624 |
codec->config.cpu = init_param.cpu_flags; |
codec->config.cpu = init_param.cpu_flags; |
625 |
} |
} |
626 |
|
|
627 |
param.width = inhdr->bV4Width; |
param.width = lpbiInput->bmiHeader.biWidth; |
628 |
param.height = inhdr->bV4Height; |
param.height = lpbiInput->bmiHeader.biHeight; |
629 |
|
|
630 |
switch(xvid_decore(0, XVID_DEC_CREATE, ¶m, NULL)) |
switch(xvid_decore(0, XVID_DEC_CREATE, ¶m, NULL)) |
631 |
{ |
{ |
658 |
|
|
659 |
LRESULT decompress(CODEC * codec, ICDECOMPRESS * icd) |
LRESULT decompress(CODEC * codec, ICDECOMPRESS * icd) |
660 |
{ |
{ |
|
BITMAPV4HEADER * inhdr = (BITMAPV4HEADER *)icd->lpbiInput; |
|
|
BITMAPV4HEADER * outhdr = (BITMAPV4HEADER *)icd->lpbiOutput; |
|
661 |
XVID_DEC_FRAME frame; |
XVID_DEC_FRAME frame; |
662 |
|
|
663 |
frame.bitstream = icd->lpInput; |
frame.bitstream = icd->lpInput; |
668 |
|
|
669 |
if (~((icd->dwFlags & ICDECOMPRESS_HURRYUP) | (icd->dwFlags & ICDECOMPRESS_UPDATE))) |
if (~((icd->dwFlags & ICDECOMPRESS_HURRYUP) | (icd->dwFlags & ICDECOMPRESS_UPDATE))) |
670 |
{ |
{ |
671 |
if ((frame.colorspace = get_colorspace(outhdr)) == XVID_CSP_NULL) |
if ((frame.colorspace = get_colorspace(icd->lpbiOutput)) == XVID_CSP_NULL) |
672 |
{ |
{ |
673 |
return ICERR_BADFORMAT; |
return ICERR_BADFORMAT; |
674 |
} |
} |
693 |
return ICERR_OK; |
return ICERR_OK; |
694 |
} |
} |
695 |
|
|
|
int codec_2pass_init(CODEC* codec) |
|
|
{ |
|
|
TWOPASS *twopass = &codec->twopass; |
|
|
DWORD version = -20; |
|
|
DWORD read, wrote; |
|
|
|
|
|
int frames = 0, credits_frames = 0, i_frames = 0; |
|
|
__int64 total = 0, i_total = 0, i_boost_total = 0, start = 0, end = 0, start_curved = 0, end_curved = 0; |
|
|
__int64 desired = codec->config.desired_size * 1024; |
|
|
|
|
|
double total1 = 0.0; |
|
|
double total2 = 0.0; |
|
|
|
|
|
switch (codec->config.mode) |
|
|
{ |
|
|
case DLG_MODE_2PASS_1 : |
|
|
twopass->stats1 = CreateFile(codec->config.stats1, GENERIC_WRITE, FILE_SHARE_READ, 0, CREATE_ALWAYS, FILE_ATTRIBUTE_NORMAL, 0); |
|
|
if (twopass->stats1 == INVALID_HANDLE_VALUE) |
|
|
{ |
|
|
DEBUG("2pass init error - couldn't create stats1"); |
|
|
return ICERR_ERROR; |
|
|
} |
|
|
if (WriteFile(twopass->stats1, &version, sizeof(DWORD), &wrote, 0) == 0 || wrote != sizeof(DWORD)) |
|
|
{ |
|
|
CloseHandle(twopass->stats1); |
|
|
twopass->stats1 = INVALID_HANDLE_VALUE; |
|
|
DEBUG("2pass init error - couldn't write to stats1"); |
|
|
return ICERR_ERROR; |
|
|
} |
|
|
break; |
|
|
|
|
|
case DLG_MODE_2PASS_2_INT : |
|
|
case DLG_MODE_2PASS_2_EXT : |
|
|
twopass->stats1 = CreateFile(codec->config.stats1, GENERIC_READ, FILE_SHARE_READ, 0, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0); |
|
|
if (twopass->stats1 == INVALID_HANDLE_VALUE) |
|
|
{ |
|
|
DEBUG("2pass init error - couldn't open stats1"); |
|
|
return ICERR_ERROR; |
|
|
} |
|
|
if (ReadFile(twopass->stats1, &version, sizeof(DWORD), &read, 0) == 0 || read != sizeof(DWORD)) |
|
|
{ |
|
|
CloseHandle(twopass->stats1); |
|
|
twopass->stats1 = INVALID_HANDLE_VALUE; |
|
|
DEBUG("2pass init error - couldn't read from stats1"); |
|
|
return ICERR_ERROR; |
|
|
} |
|
|
if (version != -20) |
|
|
{ |
|
|
CloseHandle(twopass->stats1); |
|
|
twopass->stats1 = INVALID_HANDLE_VALUE; |
|
|
DEBUG("2pass init error - wrong .stats version"); |
|
|
return ICERR_ERROR; |
|
|
} |
|
|
|
|
|
if (codec->config.mode == DLG_MODE_2PASS_2_EXT) |
|
|
{ |
|
|
if (twopass->stats2 != INVALID_HANDLE_VALUE) |
|
|
{ |
|
|
CloseHandle(twopass->stats2); |
|
|
} |
|
|
|
|
|
twopass->stats2 = CreateFile(codec->config.stats2, GENERIC_READ, FILE_SHARE_READ, 0, OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, 0); |
|
|
|
|
|
if (twopass->stats2 == INVALID_HANDLE_VALUE) |
|
|
{ |
|
|
CloseHandle(twopass->stats1); |
|
|
twopass->stats1 = INVALID_HANDLE_VALUE; |
|
|
DEBUG("2pass init error - couldn't open stats2"); |
|
|
return ICERR_ERROR; |
|
|
} |
|
|
if (ReadFile(twopass->stats2, &version, sizeof(DWORD), &read, 0) == 0 || read != sizeof(DWORD)) |
|
|
{ |
|
|
CloseHandle(twopass->stats1); |
|
|
twopass->stats1 = INVALID_HANDLE_VALUE; |
|
|
CloseHandle(twopass->stats2); |
|
|
twopass->stats2 = INVALID_HANDLE_VALUE; |
|
|
DEBUG("2pass init error - couldn't read from stats2"); |
|
|
return ICERR_ERROR; |
|
|
} |
|
|
if (version != -20) |
|
|
{ |
|
|
CloseHandle(twopass->stats1); |
|
|
twopass->stats1 = INVALID_HANDLE_VALUE; |
|
|
CloseHandle(twopass->stats2); |
|
|
twopass->stats2 = INVALID_HANDLE_VALUE; |
|
|
DEBUG("2pass init error - wrong .stats version"); |
|
|
return ICERR_ERROR; |
|
|
} |
|
|
|
|
|
while (1) |
|
|
{ |
|
|
if (!ReadFile(twopass->stats1, &twopass->nns1, sizeof(NNSTATS), &read, NULL) || read != sizeof(NNSTATS) || |
|
|
!ReadFile(twopass->stats2, &twopass->nns2, sizeof(NNSTATS), &read, NULL) || read != sizeof(NNSTATS)) |
|
|
{ |
|
|
DWORD err = GetLastError(); |
|
|
|
|
|
if (err == ERROR_HANDLE_EOF || err == ERROR_SUCCESS) |
|
|
{ |
|
|
break; |
|
|
} |
|
|
else |
|
|
{ |
|
|
CloseHandle(twopass->stats1); |
|
|
CloseHandle(twopass->stats2); |
|
|
twopass->stats1 = INVALID_HANDLE_VALUE; |
|
|
twopass->stats2 = INVALID_HANDLE_VALUE; |
|
|
DEBUG("2pass init error - incomplete stats1/stats2 record?"); |
|
|
return ICERR_ERROR; |
|
|
} |
|
|
} |
|
|
|
|
|
if (!codec_is_in_credits(&codec->config, frames)) |
|
|
{ |
|
|
if (twopass->nns1.quant & NNSTATS_KEYFRAME) |
|
|
{ |
|
|
i_boost_total = twopass->nns2.bytes * codec->config.keyframe_boost / 100; |
|
|
i_total += twopass->nns2.bytes; |
|
|
++i_frames; |
|
|
} |
|
|
|
|
|
total += twopass->nns2.bytes; |
|
|
} |
|
|
else |
|
|
++credits_frames; |
|
|
|
|
|
++frames; |
|
|
} |
|
|
|
|
|
twopass->movie_curve = ((float)(total + i_boost_total) / total); |
|
|
twopass->average_frame = ((double)(total - i_total) / (frames - credits_frames - i_frames) / twopass->movie_curve); |
|
|
|
|
|
SetFilePointer(twopass->stats1, sizeof(DWORD), 0, FILE_BEGIN); |
|
|
SetFilePointer(twopass->stats2, sizeof(DWORD), 0, FILE_BEGIN); |
|
|
|
|
|
// perform prepass to compensate for over/undersizing |
|
|
frames = 0; |
|
|
|
|
|
while (1) |
|
|
{ |
|
|
if (!ReadFile(twopass->stats1, &twopass->nns1, sizeof(NNSTATS), &read, NULL) || read != sizeof(NNSTATS) || |
|
|
!ReadFile(twopass->stats2, &twopass->nns2, sizeof(NNSTATS), &read, NULL) || read != sizeof(NNSTATS)) |
|
|
{ |
|
|
DWORD err = GetLastError(); |
|
|
|
|
|
if (err == ERROR_HANDLE_EOF || err == ERROR_SUCCESS) |
|
|
{ |
|
|
break; |
|
|
} |
|
|
else |
|
|
{ |
|
|
CloseHandle(twopass->stats1); |
|
|
CloseHandle(twopass->stats2); |
|
|
twopass->stats1 = INVALID_HANDLE_VALUE; |
|
|
twopass->stats2 = INVALID_HANDLE_VALUE; |
|
|
DEBUG("2pass init error - incomplete stats1/stats2 record?"); |
|
|
return ICERR_ERROR; |
|
|
} |
|
|
} |
|
|
|
|
|
if (!codec_is_in_credits(&codec->config, frames) && |
|
|
!(twopass->nns1.quant & NNSTATS_KEYFRAME)) |
|
|
{ |
|
|
double dbytes = twopass->nns2.bytes / twopass->movie_curve; |
|
|
total1 += dbytes; |
|
|
|
|
|
if (dbytes > twopass->average_frame) |
|
|
{ |
|
|
total2 += ((double)dbytes + (twopass->average_frame - dbytes) * |
|
|
codec->config.curve_compression_high / 100.0); |
|
|
} |
|
|
else |
|
|
{ |
|
|
total2 += ((double)dbytes + (twopass->average_frame - dbytes) * |
|
|
codec->config.curve_compression_low / 100.0); |
|
|
} |
|
|
} |
|
|
|
|
|
++frames; |
|
|
} |
|
|
|
|
|
twopass->curve_comp_scale = total1 / total2; |
|
|
|
|
|
{ |
|
|
int asymmetric_average_frame; |
|
|
char s[100]; |
|
|
|
|
|
asymmetric_average_frame = (int)(twopass->average_frame * twopass->curve_comp_scale); |
|
|
wsprintf(s, "middle frame size for asymmetric curve compression: %i", asymmetric_average_frame); |
|
|
DEBUG2P(s); |
|
|
} |
|
|
|
|
|
SetFilePointer(twopass->stats1, sizeof(DWORD), 0, FILE_BEGIN); |
|
|
SetFilePointer(twopass->stats2, sizeof(DWORD), 0, FILE_BEGIN); |
|
|
} |
|
|
else // DLG_MODE_2PASS_2_INT |
|
|
{ |
|
|
while (1) |
|
|
{ |
|
|
if (!ReadFile(twopass->stats1, &twopass->nns1, sizeof(NNSTATS), &read, NULL) || read != sizeof(NNSTATS)) |
|
|
{ |
|
|
DWORD err = GetLastError(); |
|
|
|
|
|
if (err == ERROR_HANDLE_EOF || err == ERROR_SUCCESS) |
|
|
{ |
|
|
break; |
|
|
} |
|
|
else |
|
|
{ |
|
|
CloseHandle(twopass->stats1); |
|
|
twopass->stats1 = INVALID_HANDLE_VALUE; |
|
|
DEBUG("2pass init error - incomplete stats2 record?"); |
|
|
return ICERR_ERROR; |
|
|
} |
|
|
} |
|
|
|
|
|
if (codec_is_in_credits(&codec->config, frames) == CREDITS_START) |
|
|
{ |
|
|
start += twopass->nns1.bytes; |
|
|
++credits_frames; |
|
|
} |
|
|
else if (codec_is_in_credits(&codec->config, frames) == CREDITS_END) |
|
|
{ |
|
|
end += twopass->nns1.bytes; |
|
|
++credits_frames; |
|
|
} |
|
|
else if (twopass->nns1.quant & NNSTATS_KEYFRAME) |
|
|
{ |
|
|
i_total += twopass->nns1.bytes + twopass->nns1.bytes * codec->config.keyframe_boost / 100; |
|
|
total += twopass->nns1.bytes * codec->config.keyframe_boost / 100; |
|
|
++i_frames; |
|
|
} |
|
|
|
|
|
total += twopass->nns1.bytes; |
|
|
|
|
|
++frames; |
|
|
} |
|
|
|
|
|
// compensate for avi frame overhead |
|
|
desired -= frames * 24; |
|
|
|
|
|
switch (codec->config.credits_mode) |
|
|
{ |
|
|
case CREDITS_MODE_RATE : |
|
|
|
|
|
// credits curve = (total / desired_size) * (100 / credits_rate) |
|
|
twopass->credits_start_curve = twopass->credits_end_curve = |
|
|
((float)total / desired) * ((float)100 / codec->config.credits_rate); |
|
|
|
|
|
start_curved = (__int64)(start / twopass->credits_start_curve); |
|
|
end_curved = (__int64)(end / twopass->credits_end_curve); |
|
|
|
|
|
// movie curve = (total - credits) / (desired_size - curved credits) |
|
|
twopass->movie_curve = (float) |
|
|
(total - start - end) / |
|
|
(desired - start_curved - end_curved); |
|
|
|
|
|
break; |
|
|
|
|
|
case CREDITS_MODE_QUANT : |
|
|
|
|
|
// movie curve = (total - credits) / (desired_size - credits) |
|
|
twopass->movie_curve = (float) |
|
|
(total - start - end) / (desired - start - end); |
|
|
|
|
|
// aid the average asymmetric frame calculation below |
|
|
start_curved = start; |
|
|
end_curved = end; |
|
|
|
|
|
break; |
|
|
|
|
|
case CREDITS_MODE_SIZE : |
|
|
|
|
|
// start curve = (start / start desired size) |
|
|
twopass->credits_start_curve = (float) |
|
|
(start / 1024) / codec->config.credits_start_size; |
|
|
|
|
|
// end curve = (end / end desired size) |
|
|
twopass->credits_end_curve = (float) |
|
|
(end / 1024) / codec->config.credits_end_size; |
|
|
|
|
|
start_curved = (__int64)(start / twopass->credits_start_curve); |
|
|
end_curved = (__int64)(end / twopass->credits_end_curve); |
|
|
|
|
|
// movie curve = (total - credits) / (desired_size - curved credits) |
|
|
twopass->movie_curve = (float) |
|
|
(total - start - end) / |
|
|
(desired - start_curved - end_curved); |
|
|
|
|
|
break; |
|
|
} |
|
|
|
|
|
// average frame size = (desired - curved credits - curved keyframes) / |
|
|
// (frames - credits frames - keyframes) |
|
|
twopass->average_frame = (double) |
|
|
(desired - start_curved - end_curved - (i_total / twopass->movie_curve)) / |
|
|
(frames - credits_frames - i_frames); |
|
|
|
|
|
SetFilePointer(twopass->stats1, sizeof(DWORD), 0, FILE_BEGIN); |
|
|
|
|
|
// perform prepass to compensate for over/undersizing |
|
|
frames = 0; |
|
|
|
|
|
while (1) |
|
|
{ |
|
|
if (!ReadFile(twopass->stats1, &twopass->nns1, sizeof(NNSTATS), &read, NULL) || read != sizeof(NNSTATS)) |
|
|
{ |
|
|
DWORD err = GetLastError(); |
|
|
|
|
|
if (err == ERROR_HANDLE_EOF || err == ERROR_SUCCESS) |
|
|
{ |
|
|
break; |
|
|
} |
|
|
else |
|
|
{ |
|
|
CloseHandle(twopass->stats1); |
|
|
twopass->stats1 = INVALID_HANDLE_VALUE; |
|
|
DEBUG("2pass init error - incomplete stats2 record?"); |
|
|
return ICERR_ERROR; |
|
|
} |
|
|
} |
|
|
|
|
|
if (!codec_is_in_credits(&codec->config, frames) && |
|
|
!(twopass->nns1.quant & NNSTATS_KEYFRAME)) |
|
|
{ |
|
|
double dbytes = twopass->nns1.bytes / twopass->movie_curve; |
|
|
total1 += dbytes; |
|
|
|
|
|
if (dbytes > twopass->average_frame) |
|
|
{ |
|
|
total2 += ((double)dbytes + (twopass->average_frame - dbytes) * |
|
|
codec->config.curve_compression_high / 100.0); |
|
|
} |
|
|
else |
|
|
{ |
|
|
total2 += ((double)dbytes + (twopass->average_frame - dbytes) * |
|
|
codec->config.curve_compression_low / 100.0); |
|
|
} |
|
|
} |
|
|
|
|
|
++frames; |
|
|
} |
|
|
|
|
|
twopass->curve_comp_scale = total1 / total2; |
|
|
|
|
|
{ |
|
|
int asymmetric_average_frame; |
|
|
char s[100]; |
|
|
|
|
|
asymmetric_average_frame = (int)(twopass->average_frame * twopass->curve_comp_scale); |
|
|
wsprintf(s, "middle frame size for asymmetric curve compression: %i", asymmetric_average_frame); |
|
|
DEBUG2P(s); |
|
|
} |
|
|
|
|
|
SetFilePointer(twopass->stats1, sizeof(DWORD), 0, FILE_BEGIN); |
|
|
} |
|
|
|
|
|
twopass->overflow = 0; |
|
|
|
|
|
break; |
|
|
} |
|
|
|
|
|
return ICERR_OK; |
|
|
} |
|
|
|
|
|
|
|
696 |
int codec_get_quant(CODEC* codec, XVID_ENC_FRAME* frame) |
int codec_get_quant(CODEC* codec, XVID_ENC_FRAME* frame) |
697 |
{ |
{ |
698 |
switch (codec->config.mode) |
switch (codec->config.mode) |
711 |
break; |
break; |
712 |
|
|
713 |
default : |
default : |
714 |
DEBUG("Can't use credits size mode in quality mode"); |
DEBUGERR("Can't use credits size mode in quality mode"); |
715 |
return ICERR_ERROR; |
return ICERR_ERROR; |
716 |
} |
} |
717 |
} |
} |
728 |
{ |
{ |
729 |
case CREDITS_MODE_RATE : |
case CREDITS_MODE_RATE : |
730 |
frame->quant = |
frame->quant = |
731 |
codec->config.max_quant - |
codec->config.max_pquant - |
732 |
((codec->config.max_quant - codec->config.quant) * codec->config.credits_rate / 100); |
((codec->config.max_pquant - codec->config.quant) * codec->config.credits_rate / 100); |
733 |
break; |
break; |
734 |
|
|
735 |
case CREDITS_MODE_QUANT : |
case CREDITS_MODE_QUANT : |
737 |
break; |
break; |
738 |
|
|
739 |
default : |
default : |
740 |
DEBUG("Can't use credits size mode in quantizer mode"); |
DEBUGERR("Can't use credits size mode in quantizer mode"); |
741 |
return ICERR_ERROR; |
return ICERR_ERROR; |
742 |
} |
} |
743 |
} |
} |
766 |
return ICERR_OK; |
return ICERR_OK; |
767 |
|
|
768 |
default: |
default: |
769 |
DEBUG("get quant: invalid mode"); |
DEBUGERR("get quant: invalid mode"); |
770 |
return ICERR_ERROR; |
return ICERR_ERROR; |
771 |
} |
} |
772 |
} |
} |
773 |
|
|
774 |
|
|
|
int codec_2pass_get_quant(CODEC* codec, XVID_ENC_FRAME* frame) |
|
|
{ |
|
|
static double quant_error[32]; |
|
|
static double curve_comp_error; |
|
|
static int last_quant; |
|
|
|
|
|
TWOPASS * twopass = &codec->twopass; |
|
|
|
|
|
DWORD read; |
|
|
int bytes1, bytes2; |
|
|
int overflow; |
|
|
int credits_pos; |
|
|
|
|
|
if (codec->framenum == 0) |
|
|
{ |
|
|
int i; |
|
|
|
|
|
for (i=0 ; i<32 ; ++i) |
|
|
{ |
|
|
quant_error[i] = 0.0; |
|
|
} |
|
|
|
|
|
curve_comp_error = 0.0; |
|
|
last_quant = 0; |
|
|
} |
|
|
|
|
|
if (ReadFile(twopass->stats1, &twopass->nns1, sizeof(NNSTATS), &read, 0) == 0 || read != sizeof(NNSTATS)) |
|
|
{ |
|
|
DEBUG("2ndpass quant: couldn't read from stats1"); |
|
|
return ICERR_ERROR; |
|
|
} |
|
|
if (codec->config.mode == DLG_MODE_2PASS_2_EXT) |
|
|
{ |
|
|
if (ReadFile(twopass->stats2, &twopass->nns2, sizeof(NNSTATS), &read, 0) == 0 || read != sizeof(NNSTATS)) |
|
|
{ |
|
|
DEBUG("2ndpass quant: couldn't read from stats2"); |
|
|
return ICERR_ERROR; |
|
|
} |
|
|
} |
|
|
|
|
|
bytes1 = twopass->nns1.bytes; |
|
|
overflow = twopass->overflow / 8; |
|
|
|
|
|
// override codec i-frame choice (reenable in credits) |
|
|
frame->intra = (twopass->nns1.quant & NNSTATS_KEYFRAME); |
|
|
|
|
|
if (frame->intra) |
|
|
{ |
|
|
overflow = 0; |
|
|
} |
|
|
|
|
|
credits_pos = codec_is_in_credits(&codec->config, codec->framenum); |
|
|
|
|
|
if (credits_pos) |
|
|
{ |
|
|
if (codec->config.mode == DLG_MODE_2PASS_2_INT) |
|
|
{ |
|
|
switch (codec->config.credits_mode) |
|
|
{ |
|
|
case CREDITS_MODE_RATE : |
|
|
case CREDITS_MODE_SIZE : |
|
|
if (credits_pos == CREDITS_START) |
|
|
{ |
|
|
bytes2 = (int)(bytes1 / twopass->credits_start_curve); |
|
|
} |
|
|
else // CREDITS_END |
|
|
{ |
|
|
bytes2 = (int)(bytes1 / twopass->credits_end_curve); |
|
|
} |
|
|
|
|
|
frame->intra = -1; |
|
|
break; |
|
|
|
|
|
case CREDITS_MODE_QUANT : |
|
|
if (codec->config.credits_quant_i != codec->config.credits_quant_p) |
|
|
{ |
|
|
frame->quant = frame->intra ? |
|
|
codec->config.credits_quant_i : |
|
|
codec->config.credits_quant_p; |
|
|
} |
|
|
else |
|
|
{ |
|
|
frame->quant = codec->config.credits_quant_p; |
|
|
frame->intra = -1; |
|
|
} |
|
|
|
|
|
twopass->bytes1 = bytes1; |
|
|
twopass->bytes2 = bytes1; |
|
|
twopass->desired_bytes2 = bytes1; |
|
|
return ICERR_OK; |
|
|
} |
|
|
} |
|
|
else // DLG_MODE_2PASS_2_EXT |
|
|
{ |
|
|
bytes2 = twopass->nns2.bytes; |
|
|
} |
|
|
} |
|
|
else // Foxer: apply curve compression outside credits |
|
|
{ |
|
|
double dbytes, curve_temp; |
|
|
|
|
|
bytes2 = (codec->config.mode == DLG_MODE_2PASS_2_INT) ? bytes1 : twopass->nns2.bytes; |
|
|
|
|
|
if (frame->intra) |
|
|
{ |
|
|
dbytes = ((int)(bytes2 + bytes2 * codec->config.keyframe_boost / 100)) / |
|
|
twopass->movie_curve; |
|
|
} |
|
|
else |
|
|
{ |
|
|
dbytes = bytes2 / twopass->movie_curve; |
|
|
} |
|
|
|
|
|
// spread the compression error across payback_delay frames |
|
|
if (codec->config.bitrate_payback_method == 0) |
|
|
{ |
|
|
bytes2 = (int)(curve_comp_error / codec->config.bitrate_payback_delay); |
|
|
} |
|
|
else |
|
|
{ |
|
|
bytes2 = (int)(curve_comp_error * dbytes / |
|
|
twopass->average_frame / codec->config.bitrate_payback_delay); |
|
|
|
|
|
if (labs(bytes2) > fabs(curve_comp_error)) |
|
|
{ |
|
|
bytes2 = (int)curve_comp_error; |
|
|
} |
|
|
} |
|
|
|
|
|
curve_comp_error -= bytes2; |
|
|
|
|
|
if ((codec->config.curve_compression_high + codec->config.curve_compression_low) && |
|
|
!frame->intra) |
|
|
{ |
|
|
if (dbytes > twopass->average_frame) |
|
|
{ |
|
|
curve_temp = twopass->curve_comp_scale * |
|
|
((double)dbytes + (twopass->average_frame - dbytes) * |
|
|
codec->config.curve_compression_high / 100.0); |
|
|
} |
|
|
else |
|
|
{ |
|
|
curve_temp = twopass->curve_comp_scale * |
|
|
((double)dbytes + (twopass->average_frame - dbytes) * |
|
|
codec->config.curve_compression_low / 100.0); |
|
|
} |
|
|
|
|
|
bytes2 += ((int)curve_temp); |
|
|
curve_comp_error += curve_temp - ((int)curve_temp); |
|
|
} |
|
|
else |
|
|
{ |
|
|
curve_comp_error += dbytes - ((int)dbytes); |
|
|
bytes2 += ((int)dbytes); |
|
|
} |
|
|
|
|
|
// cap bytes2 to first pass size, lowers number of quant=1 frames |
|
|
if (bytes2 > bytes1) |
|
|
{ |
|
|
curve_comp_error += bytes2 - bytes1; |
|
|
bytes2 = bytes1; |
|
|
} |
|
|
else if (bytes2 < 1) |
|
|
{ |
|
|
curve_comp_error += --bytes2; |
|
|
bytes2 = 1; |
|
|
} |
|
|
} |
|
|
|
|
|
twopass->desired_bytes2 = bytes2; |
|
|
|
|
|
// Foxer: scale overflow in relation to average size, so smaller frames don't get |
|
|
// too much/little bitrate |
|
|
overflow = (int)((double)overflow * bytes2 / twopass->average_frame); |
|
|
|
|
|
// Foxer: reign in overflow with huge frames |
|
|
if (labs(overflow) > labs(twopass->overflow)) |
|
|
{ |
|
|
overflow = twopass->overflow; |
|
|
} |
|
|
|
|
|
// Foxer: make sure overflow doesn't run away |
|
|
if (overflow > bytes2 * 6 / 10) |
|
|
{ |
|
|
bytes2 += (overflow <= bytes2) ? bytes2 * 6 / 10 : overflow * 6 / 10; |
|
|
} |
|
|
else if (overflow < bytes2 * -6 / 10) |
|
|
{ |
|
|
bytes2 += bytes2 * -6 / 10; |
|
|
} |
|
|
else |
|
|
{ |
|
|
bytes2 += overflow; |
|
|
} |
|
|
|
|
|
if (bytes2 < 1) |
|
|
{ |
|
|
bytes2 = 1; |
|
|
} |
|
|
|
|
|
twopass->bytes1 = bytes1; |
|
|
twopass->bytes2 = bytes2; |
|
|
|
|
|
// very 'simple' quant<->filesize relationship |
|
|
frame->quant = ((twopass->nns1.quant & ~NNSTATS_KEYFRAME) * bytes1) / bytes2; |
|
|
|
|
|
if (frame->quant < 1) |
|
|
{ |
|
|
frame->quant = 1; |
|
|
} |
|
|
else if (frame->quant > 31) |
|
|
{ |
|
|
frame->quant = 31; |
|
|
} |
|
|
else if (!frame->intra) |
|
|
{ |
|
|
// Foxer: aid desired quantizer precision by accumulating decision error |
|
|
quant_error[frame->quant] += ((double)((twopass->nns1.quant & ~NNSTATS_KEYFRAME) * |
|
|
bytes1) / bytes2) - frame->quant; |
|
|
|
|
|
if (quant_error[frame->quant] >= 1.0) |
|
|
{ |
|
|
quant_error[frame->quant] -= 1.0; |
|
|
++frame->quant; |
|
|
} |
|
|
} |
|
|
|
|
|
// we're done with credits |
|
|
if (codec_is_in_credits(&codec->config, codec->framenum)) |
|
|
{ |
|
|
return ICERR_OK; |
|
|
} |
|
|
|
|
|
if (frame->intra) |
|
|
{ |
|
|
if (frame->quant < codec->config.min_iquant) |
|
|
{ |
|
|
frame->quant = codec->config.min_iquant; |
|
|
DEBUG2P("I-frame quantizer raised"); |
|
|
} |
|
|
if (frame->quant > codec->config.max_iquant) |
|
|
{ |
|
|
frame->quant = codec->config.max_iquant; |
|
|
DEBUG2P("I-frame quantizer lowered"); |
|
|
} |
|
|
} |
|
|
else |
|
|
{ |
|
|
if (frame->quant > codec->config.max_quant) |
|
|
{ |
|
|
frame->quant = codec->config.max_quant; |
|
|
} |
|
|
if (frame->quant < codec->config.min_quant) |
|
|
{ |
|
|
frame->quant = codec->config.min_quant; |
|
|
} |
|
|
|
|
|
// subsequent frame quants can only be +- 2 |
|
|
if (last_quant) |
|
|
{ |
|
|
if (frame->quant > last_quant + 2) |
|
|
{ |
|
|
frame->quant = last_quant + 2; |
|
|
DEBUG2P("P-frame quantizer prevented from rising too steeply"); |
|
|
} |
|
|
if (frame->quant < last_quant - 2) |
|
|
{ |
|
|
frame->quant = last_quant - 2; |
|
|
DEBUG2P("P-frame quantizer prevented from falling too steeply"); |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
last_quant = frame->quant; |
|
|
|
|
|
return ICERR_OK; |
|
|
} |
|
|
|
|
|
|
|
|
int codec_2pass_update(CODEC* codec, XVID_ENC_FRAME* frame, XVID_ENC_STATS* stats) |
|
|
{ |
|
|
static __int64 total_size; |
|
|
|
|
|
NNSTATS nns1; |
|
|
DWORD wrote; |
|
|
|
|
|
if (codec->framenum == 0) |
|
|
{ |
|
|
total_size = 0; |
|
|
} |
|
|
|
|
|
switch (codec->config.mode) |
|
|
{ |
|
|
case DLG_MODE_2PASS_1 : |
|
|
nns1.bytes = frame->length; // total bytes |
|
|
nns1.dd_v = stats->hlength; // header bytes |
|
|
|
|
|
nns1.dd_u = nns1.dd_y = 0; |
|
|
nns1.dk_v = nns1.dk_u = nns1.dk_y = 0; |
|
|
nns1.md_u = nns1.md_y = 0; |
|
|
nns1.mk_u = nns1.mk_y = 0; |
|
|
|
|
|
nns1.quant = stats->quant; |
|
|
if (frame->intra) |
|
|
{ |
|
|
nns1.quant |= NNSTATS_KEYFRAME; |
|
|
} |
|
|
nns1.kblk = stats->kblks; |
|
|
nns1.mblk = stats->mblks; |
|
|
nns1.ublk = stats->ublks; |
|
|
nns1.lum_noise[0] = nns1.lum_noise[1] = 1; |
|
|
|
|
|
total_size += frame->length; |
|
|
DEBUG1ST(frame->length, (int)total_size/1024, frame->intra, frame->quant, stats->kblks, stats->mblks) |
|
|
|
|
|
if (WriteFile(codec->twopass.stats1, &nns1, sizeof(NNSTATS), &wrote, 0) == 0 || wrote != sizeof(NNSTATS)) |
|
|
{ |
|
|
DEBUG("stats1: WriteFile error"); |
|
|
return ICERR_ERROR; |
|
|
} |
|
|
break; |
|
|
|
|
|
case DLG_MODE_2PASS_2_INT : |
|
|
case DLG_MODE_2PASS_2_EXT : |
|
|
codec->twopass.overflow += codec->twopass.desired_bytes2 - frame->length; |
|
|
DEBUG2ND(frame->quant, frame->intra, codec->twopass.bytes1, codec->twopass.desired_bytes2, frame->length, codec->twopass.overflow, codec_is_in_credits(&codec->config, codec->framenum)) |
|
|
break; |
|
|
|
|
|
default: |
|
|
break; |
|
|
} |
|
|
|
|
|
return ICERR_OK; |
|
|
} |
|
|
|
|
|
|
|
775 |
int codec_is_in_credits(CONFIG* config, int framenum) |
int codec_is_in_credits(CONFIG* config, int framenum) |
776 |
{ |
{ |
|
if (config->mode == DLG_MODE_2PASS_2_EXT) |
|
|
{ |
|
|
return 0; |
|
|
} |
|
|
|
|
777 |
if (config->credits_start) |
if (config->credits_start) |
778 |
{ |
{ |
779 |
if (framenum >= config->credits_start_begin && |
if (framenum >= config->credits_start_begin && |
814 |
if (!config->fquant) |
if (!config->fquant) |
815 |
{ |
{ |
816 |
config->fquant = |
config->fquant = |
817 |
((float) (config->max_quant - config->min_quant) / 100) * |
((float) (config->max_pquant - config->min_pquant) / 100) * |
818 |
(100 - quality) + |
(100 - quality) + |
819 |
(float) config->min_quant; |
(float) config->min_pquant; |
820 |
|
|
821 |
fquant_running = config->fquant; |
fquant_running = config->fquant; |
822 |
} |
} |
823 |
|
|
824 |
if (fquant_running < config->min_quant) |
if (fquant_running < config->min_pquant) |
825 |
{ |
{ |
826 |
fquant_running = (float) config->min_quant; |
fquant_running = (float) config->min_pquant; |
827 |
} |
} |
828 |
else if(fquant_running > config->max_quant) |
else if(fquant_running > config->max_pquant) |
829 |
{ |
{ |
830 |
fquant_running = (float) config->max_quant; |
fquant_running = (float) config->max_pquant; |
831 |
} |
} |
832 |
|
|
833 |
quant = (int) fquant_running; |
quant = (int) fquant_running; |