Windows2003-3790/enduser/netmeeting/av/codecs/intel/h261/e1mbenc.cpp
2020-09-30 16:53:55 +02:00

2004 lines
53 KiB
C++

/* *************************************************************************
** INTEL Corporation Proprietary Information
**
** This listing is supplied under the terms of a license
** agreement with INTEL Corporation and may not be copied
** nor disclosed except in accordance with the terms of
** that agreement.
**
** Copyright (c) 1995-1996 Intel Corporation.
** All Rights Reserved.
**
** *************************************************************************
*/
/******************************************************************************
* e1mbenc.cpp
*
* DESCRIPTION:
* Specific encoder compression functions.
*
* Routines: Prototypes in:
* GOB_Q_RLE_VLC_WriteBS
* MB_Quantize_RLE
* ComputeCheckSum
* WriteMBCheckSum
*/
/* $Header: S:\h26x\src\enc\e1mbenc.cpv 1.47 30 Oct 1996 09:58:46 MBODART $
* $Log: S:\h26x\src\enc\e1mbenc.cpv $
//
// Rev 1.47 30 Oct 1996 09:58:46 MBODART
// Fixed assertion failure. Need to reclamp unMQuant after adding delta.
//
// Rev 1.46 29 Oct 1996 11:18:18 RHAZRA
// Bug fix: in the IA code we previously modified MQuant on a MB basis
// even if we were operating with a fixed quantizer. Now we don't
//
// Rev 1.45 21 Oct 1996 09:05:16 RHAZRA
//
// MMX integration
//
// Rev 1.44 21 Aug 1996 19:06:02 RHAZRA
// Added RTP generatio code; fixed additional divide-by-zero possibilities.
//
// Rev 1.42 21 Jun 1996 10:08:34 AKASAI
// Changes to e1enc.cpp, e1mbenc.cpp, ex5me.asm to support "improved
// bit rate control", changing MacroBlock Quantization within a
// row of MB's in addition to changing the Quantization change
// between rows of macro blocks.
//
// ex5me.asm had a problem with SLF SWD. Brian updated asm code.
//
//
// Rev 1.41 14 May 1996 11:41:18 AKASAI
// Needed to test 0th and 1st coefficient to avoid clamping errors.
//
// Rev 1.40 14 May 1996 10:39:04 AKASAI
// Two files changed to hopefully eliminate Quantization clamping
// artifacts and to reduce the max buffer overflow case: e1enc.cpp
// and e1mbenc.cpp.
//
// In e1mbenc.cpp when the MQuant level is < 6 I test to see if
// the 0th coefficient is larger than the values representable
// at that Quant level if it is I increase the Quant level until
// the clamping artifact will not occur. Note: I am test only
// the Oth coefficient, there is the possibility that some other
// coefficient is larger but the performance trade off seems to
// indicate this is good for now and if we still see clamping
// artifacts we can add more testing later.
//
// In e1enc.cpp I modified when the Overflow types of warnings are
// turn on as well as changing the rate the Quantization level
// changes at.
//
// Rev 1.39 24 Apr 1996 12:18:22 AKASAI
// Added re-compression strategy to encoder. Had to change e1enc.cpp,
// e1enc.h and e1mbenc.cpp.
// Basic strategy is if spending too many bits in a GOB quantize the
// next GOB at a higher rate. If after compressing the frame too
// many bits have been used, re-compress the last GOB at a higher
// QUANT level if that still doesn't work send a "Skip" GOB.
// Needed to add extra parameter to GOB+Q_RLE_VLC_WriteBS because
// CalcMBQuant kept decreasing the QUANT when we were in trouble with
// possibly overflowing the buffer.
//
// Rev 1.38 22 Apr 1996 11:02:14 AKASAI
// Two files changed e1enc.cpp and e1mbenc.cpp to try and support
// allowing the Quantization values to go down to 2 instead of
// CLAMP to 6.
// This is part 1 of implementing the re-compression (what to do
// if exceed max compressed buffer size 8KBytes QCIF, 32KBytes FCIF).
// Also changed in e1enc was to limit request uFrameSize to 8KB or
// 32KB. Problem was if user specified too large of a datarate
// request frame size would be larger than the allowed buffer size.
// If you try to compress qnoise10.avi or fnoise5.avi you get an
// ASSERT error until rest of re-compression is implemented.
//
// Rev 1.37 19 Apr 1996 14:26:28 SCDAY
// Added adaptive bit usage profile (Karl's BRC changes)
//
// Rev 1.36 08 Jan 1996 10:11:16 DBRUCKS
// add an assert
//
// Rev 1.35 29 Dec 1995 18:11:42 DBRUCKS
//
// optimize walking pCurrMB and add CLAMP_N_TO(qp,6,31)
//
// Rev 1.34 27 Dec 1995 16:48:06 DBRUCKS
// moved incrementing InterCodeCnt from e1enc.cpp
//
// Rev 1.33 26 Dec 1995 17:45:18 DBRUCKS
// moved statistics to e1stat
//
// Rev 1.32 20 Dec 1995 14:56:52 DBRUCKS
// add timing stats
//
// Rev 1.31 18 Dec 1995 15:38:04 DBRUCKS
// improve stats
//
// Rev 1.30 15 Dec 1995 10:53:34 AKASAI
// Fixed bug that encoded the wrong type when spatial loop filter on
// bug 0 MV. Was incorrectly been encoded with no spatial loop filter.
// This seemed to have caused the "#" bug.
//
// Rev 1.29 13 Dec 1995 13:59:08 DBRUCKS
// added include exutil.h
// parameter change in call to cnvt_fdct_output - uses INTRA boolean instead
// of blocktype
//
// Rev 1.28 07 Dec 1995 12:50:54 DBRUCKS
// integrate Macroblock checksum fixes
//
// Rev 1.27 04 Dec 1995 12:12:30 DBRUCKS
// Unsigned compares using MQuant and lastencoded -2 yielded
// unexpected results when lastencoded was 1.
//
// Rev 1.26 01 Dec 1995 15:33:14 DBRUCKS
//
// Added the bit rate controller support. The one possibly confusing
// part is that the quantizer can change in the encoder on a macro block
// that is skipped or on one that does not have coefficients. In either
// case the decoder is not told of the change. The decoder is told of
// the change on the next macroblock that has coefficients.
//
// Rev 1.25 27 Nov 1995 17:53:40 DBRUCKS
// add spatial loop filtering
//
// Rev 1.24 22 Nov 1995 17:37:34 DBRUCKS
// cleanup me changes
//
// Rev 1.23 22 Nov 1995 15:34:52 DBRUCKS
//
// Motion Estimation works - but needs to be cleaned up
//
// Rev 1.22 17 Nov 1995 14:26:20 BECHOLS
//
// Made modifications so that this file can be made for ring 0.
//
// Rev 1.21 15 Nov 1995 14:40:34 AKASAI
// Union thing change ...
// (Integration point)
//
// Rev 1.20 01 Nov 1995 09:00:16 DBRUCKS
// cleanup
//
// Rev 1.19 27 Oct 1995 17:21:12 DBRUCKS
// fix MTYPE calc, improve var names and debug
//
// Rev 1.18 27 Oct 1995 15:06:30 DBRUCKS
// update cnvt_fdct_output
//
// Rev 1.17 27 Oct 1995 14:30:36 DBRUCKS
// delta frame support "coded", key frames tested
//
// Rev 1.15 17 Oct 1995 15:56:56 DBRUCKS
// cleanup debug message
//
// Rev 1.14 17 Oct 1995 15:52:10 DBRUCKS
//
// turn off a debug message
//
// Rev 1.13 16 Oct 1995 11:41:44 DBRUCKS
// fix the sign part of the checksum
//
// Rev 1.12 29 Sep 1995 10:31:02 DBRUCKS
// change to use e35qrle to get the latest
//
// Rev 1.11 27 Sep 1995 16:53:48 DBRUCKS
// move MB checksum before MB
//
// Rev 1.10 26 Sep 1995 13:33:14 DBRUCKS
// fixed TCOEFF table 4,1 and on used earlier values
//
// Rev 1.9 26 Sep 1995 09:29:46 DBRUCKS
// turn on MBEncodeVLC
//
// Rev 1.8 26 Sep 1995 09:09:24 DBRUCKS
// add checksum test code
//
// Rev 1.7 25 Sep 1995 10:23:16 DBRUCKS
//
// add checksum info AND
// fix the final param that is passed to MBEncodeVLC
//
// Rev 1.6 21 Sep 1995 20:37:56 BECHOLS
// Modified the VLC tables for H261. I included a placeholder for the
// sign bit, so that I can achieve an optimization in the code.
//
// Rev 1.5 21 Sep 1995 18:14:48 BECHOLS
// Changed the initialization of the VLC tables for efficient use of memory,
// and proper operation with VLC code. This is an intermediate step towards
// completion, but is not operable yet.
//
// Rev 1.4 20 Sep 1995 17:50:18 BECHOLS
//
// Removed VLC_TCOEF_LAST_TBL and changed the initialization code that
// use to assume 2 DWORDS to now make use of a single DWORD to conserve
// memory.
//
// Rev 1.3 20 Sep 1995 16:34:28 BECHOLS
// Moved the data declared in E1VLC.H to this module, where it is used.
//
// Rev 1.2 20 Sep 1995 12:39:38 DBRUCKS
// turn on complete mb processing and
// cleanup two routines
//
// Rev 1.1 18 Sep 1995 10:09:54 DBRUCKS
//
// activate more of the mb processing
//
// Rev 1.0 12 Sep 1995 18:57:16 BECHOLS
// Initial revision.
*/
#include "precomp.h"
#ifdef CHECKSUM_MACRO_BLOCK
static U32 ComputeCheckSum(I8 * pi8MBRunValTriplets, I8 * pi8EndAddress, I32 iBlockNumber);
static void WriteMBCheckSum(U32 uCheckSum, U8 * pu8PictureStart, U8 ** ppu8BitStream, U8 * pu8BitOffset, UN unCurrentMB);
#endif
static I8 * MB_Quantize_RLE(I32 **DCTCoefs, I8 *MBRunValPairs, U8 * CodedBlocks, U8 BlockType, I32 QP, U32 *puChecksum);
extern char string[128];
/*
* VLC table for TCOEFs
* Table entries are size INCLUDING PLACE HOLDER FOR SIGN BIT, code.
* Stored as (size, value)
*/
int VLC_TCOEF[102*2] = {
0X0003, 0x0006, // 0
0X0005, 0x0008,
0X0006, 0x000A,
0X0008, 0x000C,
0X0009, 0x004C,
0X0009, 0x0042,
0X000B, 0x0014,
0X000D, 0x003A,
0X000D, 0x0030,
0X000D, 0x0026,
0X000D, 0x0020,
0X000E, 0x0034,
0X000E, 0x0032,
0X000E, 0x0030,
0X000E, 0x002E,
0X0004, 0x0006, // 1
0X0007, 0x000C,
0X0009, 0x004A,
0X000B, 0x0018,
0X000D, 0x0036,
0X000E, 0x002C,
0X000E, 0x002A,
0X0005, 0x000A, // 2
0X0008, 0x0008,
0X000B, 0x0016,
0X000D, 0x0028,
0X000E, 0x0028,
0X0006, 0x000E, // 3
0X0009, 0x0048,
0X000D, 0x0038,
0X000E, 0x0026,
0X0006, 0x000C, // 4
0X000B, 0x001E,
0X000D, 0x0024,
0X0007, 0x000E, // 5
0X000B, 0x0012,
0X000E, 0x0024,
0X0007, 0x000A, // 6
0X000D, 0x003C,
0X0007, 0x0008, // 7
0X000D, 0x002A,
0X0008, 0x000E, // 8
0X000D, 0x0022,
0X0008, 0x000A, // 9
0X000E, 0x0022,
0X0009, 0x004E, // 10
0X000E, 0x0020,
0X0009, 0x0046, // 11
0X0009, 0x0044, // 12
0X0009, 0x0040, // 13
0X000B, 0x001C, // 14
0X000B, 0x001A, // 15
0X000B, 0x0010, // 16
0X000D, 0x003E, // 17
0X000D, 0x0034, // 18
0X000D, 0x0032, // 19
0X000D, 0x002E, // 20
0X000D, 0x002C, // 21
0X000E, 0x003E, // 22
0X000E, 0x003C, // 23
0X000E, 0x003A, // 24
0X000E, 0x0038, // 25
0X000E, 0x0036 // 26
};
/*
* This table lists the maximum level represented in the
* VLC table for a given run. If the level exceeds the
* max, then escape codes must be used to encode the
* run & level.
* The table entries are of the form {maxlevel, ptr to table for this run}.
*/
T_MAXLEVEL_PTABLE TCOEF_RUN_MAXLEVEL[65] = {
{15, &VLC_TCOEF[0]}, // run of 0
{ 7, &VLC_TCOEF[15*2]}, // run of 1
{ 5, &VLC_TCOEF[22*2]}, // run of 2
{ 4, &VLC_TCOEF[27*2]}, // run of 3
{ 3, &VLC_TCOEF[31*2]}, // run of 4
{ 3, &VLC_TCOEF[34*2]}, // run of 5
{ 2, &VLC_TCOEF[37*2]}, // run of 6
{ 2, &VLC_TCOEF[39*2]}, // run of 7
{ 2, &VLC_TCOEF[41*2]}, // run of 8
{ 2, &VLC_TCOEF[43*2]}, // run of 9
{ 2, &VLC_TCOEF[45*2]}, // run of 10
{ 1, &VLC_TCOEF[47*2]}, // run of 11
{ 1, &VLC_TCOEF[48*2]}, // run of 12
{ 1, &VLC_TCOEF[49*2]}, // run of 13
{ 1, &VLC_TCOEF[50*2]}, // run of 14
{ 1, &VLC_TCOEF[51*2]}, // run of 15
{ 1, &VLC_TCOEF[52*2]}, // run of 16
{ 1, &VLC_TCOEF[53*2]}, // run of 17
{ 1, &VLC_TCOEF[54*2]}, // run of 18
{ 1, &VLC_TCOEF[55*2]}, // run of 19
{ 1, &VLC_TCOEF[56*2]}, // run of 20
{ 1, &VLC_TCOEF[57*2]}, // run of 21
{ 1, &VLC_TCOEF[58*2]}, // run of 22
{ 1, &VLC_TCOEF[59*2]}, // run of 23
{ 1, &VLC_TCOEF[60*2]}, // run of 24
{ 1, &VLC_TCOEF[61*2]}, // run of 25
{ 1, &VLC_TCOEF[62*2]}, // run of 26
{ 0, 0}, // run of 27 not in VLC table
{ 0, 0}, // run of 28 not in VLC table
{ 0, 0}, // run of 29 not in VLC table
{ 0, 0}, // run of 30 not in VLC table
{ 0, 0}, // run of 31 not in VLC table
{ 0, 0}, // run of 32 not in VLC table
{ 0, 0}, // run of 33 not in VLC table
{ 0, 0}, // run of 34 not in VLC table
{ 0, 0}, // run of 35 not in VLC table
{ 0, 0}, // run of 36 not in VLC table
{ 0, 0}, // run of 37 not in VLC table
{ 0, 0}, // run of 38 not in VLC table
{ 0, 0}, // run of 39 not in VLC table
{ 0, 0}, // run of 40 not in VLC table
{ 0, 0}, // run of 41 not in VLC table
{ 0, 0}, // run of 42 not in VLC table
{ 0, 0}, // run of 43 not in VLC table
{ 0, 0}, // run of 44 not in VLC table
{ 0, 0}, // run of 45 not in VLC table
{ 0, 0}, // run of 46 not in VLC table
{ 0, 0}, // run of 47 not in VLC table
{ 0, 0}, // run of 48 not in VLC table
{ 0, 0}, // run of 49 not in VLC table
{ 0, 0}, // run of 50 not in VLC table
{ 0, 0}, // run of 51 not in VLC table
{ 0, 0}, // run of 52 not in VLC table
{ 0, 0}, // run of 53 not in VLC table
{ 0, 0}, // run of 54 not in VLC table
{ 0, 0}, // run of 55 not in VLC table
{ 0, 0}, // run of 56 not in VLC table
{ 0, 0}, // run of 57 not in VLC table
{ 0, 0}, // run of 58 not in VLC table
{ 0, 0}, // run of 59 not in VLC table
{ 0, 0}, // run of 60 not in VLC table
{ 0, 0}, // run of 61 not in VLC table
{ 0, 0}, // run of 62 not in VLC table
{ 0, 0}, // run of 63 not in VLC table
{ 0, 0} // run of 64 not in VLC table
};
/* VLC table for MBA
* Table is stored as {number of bits, code}.
* The index to the table should be the MBA value.
* The zero entry is not used.
*/
int VLC_MBA[34][2] =
{ {0, 0}, /* Not Used */
{1, 0x1}, /* 1 */
{3, 0x3}, /* 2 */
{3, 0x2}, /* 3 */
{4, 0x3}, /* 4 */
{4, 0x2}, /* 5 */
{5, 0x3}, /* 6 */
{5, 0x2}, /* 7 */
{7, 0x7}, /* 8 */
{7, 0x6}, /* 9 */
{8, 0xB}, /* 10 */
{8, 0xA}, /* 11 */
{8, 0x9}, /* 12 */
{8, 0x8}, /* 13 */
{8, 0x7}, /* 14 */
{8, 0x6}, /* 15 */
{10, 0x17}, /* 16 */
{10, 0x16}, /* 17 */
{10, 0x15}, /* 18 */
{10, 0x14}, /* 19 */
{10, 0x13}, /* 20 */
{10, 0x12}, /* 21 */
{11, 0x23}, /* 22 */
{11, 0x22}, /* 23 */
{11, 0x21}, /* 24 */
{11, 0x20}, /* 25 */
{11, 0x1F}, /* 26 */
{11, 0x1E}, /* 27 */
{11, 0x1D}, /* 28 */
{11, 0x1C}, /* 29 */
{11, 0x1B}, /* 30 */
{11, 0x1A}, /* 31 */
{11, 0x19}, /* 32 */
{11, 0x18} /* 33 */
};
/* VLC table for MTYPE
* Table is stored as {number of bits, code}.
*/
int VLC_MTYPE[10][2] =
{ {4, 0x1}, /* Intra : TCOEFF */
{7, 0x1}, /* Intra : MQUANT TCOEEF */
{1, 0x1}, /* Inter : CBP TCOEFF */
{5, 0x1}, /* Inter : MQUANT CBP TCOEFF */
{9, 0x1}, /* Inter MC : MVD */
{8, 0x1}, /* Inter MC : MVD CBP TCOEFF */
{10,0x1}, /* Inter MC : MQUANT MVD CBP TCOEFF */
{3, 0x1}, /* Inter MC FIL : MVD */
{2, 0x1}, /* Inter MC FIL : MVD CBP TCOEFF */
{6, 0x1} /* Inter MC FIL : MQUANT MVD CBP TCOEFF */
};
/* VLC table for CBP
* Table is stored as {number of bits, code}.
*/
int VLC_CBP[64][2] =
{ {0, 0}, /* Not Used - if zero it is not coded */
{5, 0x0B},/* 1 */
{5, 0x09},/* 2 */
{6, 0x0D},/* 3 */
{4, 0xD}, /* 4 */
{7, 0x17},/* 5 */
{7, 0x13},/* 6 */
{8, 0x1F},/* 7 */
{4, 0xC}, /* 8 */
{7, 0x16},/* 9 */
{7, 0x12},/* 10 */
{8, 0x1E},/* 11 */
{5, 0x13},/* 12 */
{8, 0x1B},/* 13 */
{8, 0x17},/* 14 */
{8, 0x13},/* 15 */
{4, 0xB}, /* 16 */
{7, 0x15},/* 17 */
{7, 0x11},/* 18 */
{8, 0x1D},/* 19 */
{5, 0x11},/* 20 */
{8, 0x19},/* 21 */
{8, 0x15},/* 22 */
{8, 0x11},/* 23 */
{6, 0x0F},/* 24 */
{8, 0x0F},/* 25 */
{8, 0x0D},/* 26 */
{9, 0x03},/* 27 */
{5, 0x0F},/* 28 */
{8, 0x0B},/* 29 */
{8, 0x07},/* 30 */
{9, 0x07},/* 31 */
{4, 0xA}, /* 32 */
{7, 0x14},/* 33 */
{7, 0x10},/* 34 */
{8, 0x1C},/* 35 */
{6, 0x0E},/* 36 */
{8, 0x0E},/* 37 */
{8, 0x0C},/* 38 */
{9, 0x02},/* 39 */
{5, 0x10},/* 40 */
{8, 0x18},/* 41 */
{8, 0x14},/* 42 */
{8, 0x10},/* 43 */
{5, 0x0E},/* 44 */
{8, 0x0A},/* 45 */
{8, 0x06},/* 46 */
{9, 0x06},/* 47 */
{5, 0x12},/* 48 */
{8, 0x1A},/* 49 */
{8, 0x16},/* 50 */
{8, 0x12},/* 51 */
{5, 0x0D},/* 52 */
{8, 0x09},/* 53 */
{8, 0x05},/* 54 */
{9, 0x05},/* 55 */
{5, 0x0C},/* 56 */
{8, 0x08},/* 57 */
{8, 0x04},/* 58 */
{9, 0x04},/* 59 */
{3, 0x7}, /* 60 */
{5, 0x0A},/* 61 */
{5, 0x08},/* 62 */
{6, 0x0C},/* 63 */
};
/* VLC table for MVD
* Table is stored as {number of bits, code}.
*/
int VLC_MVD[32][2] =
{ {11, 0x19}, /* -16 & 16 */
{11, 0x1B}, /* -15 & 17 */
{11, 0x1D}, /* -14 & 18 */
{11, 0x1F}, /* -13 & 19 */
{11, 0x21}, /* -12 & 20 */
{11, 0x23}, /* -11 & 21 */
{10, 0x13}, /* -10 & 22 */
{10, 0x15}, /* -9 & 23 */
{10, 0x17}, /* -8 & 24 */
{ 8, 0x07}, /* -7 & 25 */
{ 8, 0x09}, /* -6 & 26 */
{ 8, 0x0B}, /* -5 & 27 */
{ 7, 0x07}, /* -4 & 28 */
{ 5, 0x03}, /* -3 & 29 */
{ 4, 0x3}, /* -2 & 30 */
{ 3, 0x3}, /* -1 */
{ 1, 0x1}, /* 0 */
{ 3, 0x2}, /* 1 */
{ 4, 0x2}, /* 2 & -30 */
{ 5, 0x02}, /* 3 & -29 */
{ 7, 0x06}, /* 4 & -28 */
{ 8, 0x0A}, /* 5 & -27 */
{ 8, 0x08}, /* 6 & -26 */
{ 8, 0x06}, /* 7 & -25 */
{10, 0x16}, /* 8 & -24 */
{10, 0x14}, /* 9 & -23 */
{10, 0x12}, /* 10 & -22 */
{11, 0x22}, /* 11 & -21 */
{11, 0x20}, /* 12 & -20 */
{11, 0x1E}, /* 13 & -19 */
{11, 0x1C}, /* 14 & -18 */
{11, 0x1A} /* 15 & -17 */
};
/* Table to limit quant changes through out a row of Macro Blocks */
static U8 QPMaxTbl[32] =
{ 0, /* Not Used */
1, /* Not Used when clamp to (2,31) */
1, /* 2 */
1, /* 3 */
2, /* 4 */
2, /* 5 */
2, /* 6 */
2, /* 7 */
2, /* 8 */
2, /* 9 */
2, /* 10 */
2, /* 11 */
2, /* 12 */
2, /* 13 */
2, /* 14 */
2, /* 15 */
2, /* 16 */
2, /* 17 */
2, /* 18 */
2, /* 19 */
2, /* 20 */
2, /* 21 */
2, /* 22 */
2, /* 23 */
2, /* 24 */
2, /* 25 */
2, /* 26 */
2, /* 27 */
2, /* 28 */
2, /* 29 */
2, /* 30 */
2 /* 31 */
};
/* Table to limit Quant changes between Rows of Marco Blocks */
extern U8 MaxChangeRowMBTbl[32];
/*****************************************************************************
*
* GOB_Q_RLE_VLC_WriteBS
*
* Quantize and RLE each macroblock, then VLC and write to stream
*/
void GOB_Q_RLE_VLC_WriteBS(
T_H263EncoderCatalog * EC,
I32 *piDCTCoefs,
U8 **ppu8BitStream,
U8 *pu8BitOffset,
UN unStartingMB,
UN unGQuant,
BOOL bOverFlowWarningFlag,
BOOL bRTPHeader, //RTP: switch
U32 uGOBNumber, // RTP: info
U8 u8QPMin
)
{
T_MBlockActionStream *pCurrMB = NULL;
T_MBlockActionStream *pLastMB = NULL;
int iMBIndex;
int iLastMBIndex = -1;
UN unCurrentMB;
U32 uCheckSum;
UN unMBA;
UN unLastEncodedMBA=0; // RTP: information
UN unLastCodedMB = 0; // RTP: information
UN unCBP;
UN unMQuant;
UN unLastEncodedMQuant;
UN unMType;
UN bWriteTCOEFF;
UN bWriteMVD;
UN bWriteMQuant;
I8 MBRunValSign[65*3*6], *pi8EndAddress, *rvs;
T_MBlockActionStream *pMBActionStream = EC->pU8_MBlockActionStream;
int bIntraBlock;
int inPrecedingHMV;
int inPrecedingVMV;
int inHDelta;
int inVDelta;
U32 uCumFrmSize;
U32 uBlockCount;
ENC_BITSTREAM_INFO * pBSInfo = &EC->BSInfo;
UN unMQuantLast;
U32 SWDmax[3] = {0,0,0};
U32 SWDmin[3] = {65536,65536,65536};
U32 SWDrange[3] = {0,0,0};
U32 SWDSum[3] = {0,0,0};
U32 SWDNum[3] = {0,0,0};
double SWDAvg[3] = {0.0,0.0,0.0};
double Step, Delta;
int QPMax;
int NeedClamp=0;
int irow;
U8 SaveQuants[3];
UN unSaveMQuant;
unMQuant = unGQuant;
unMQuantLast = unMQuant; // save last MQuant so can reset if needed
/* initially it should be the same because the GOB header
* included the GQuant.
*/
unLastEncodedMQuant = unMQuant;
unSaveMQuant = unGQuant;
SaveQuants[0] = unSaveMQuant;
/* New code to modify Quant inside a row of MB based on SWD */
/* Loop through each macroblock of the GOB to find min and max SWD
*/
pCurrMB = &pMBActionStream[unStartingMB];
for(irow = 0; irow < 3; irow++)
{
for(iMBIndex = irow*11 ; iMBIndex < (irow+1)*11; iMBIndex++, pLastMB = pCurrMB++)
{
if (pCurrMB->BlockType != INTRABLOCK)
{
// ASSERT(pCurrMB->SWD >= 0); Always True
SWDSum[irow] += pCurrMB->SWD;
SWDNum[irow]++;
if (pCurrMB->SWD > SWDmax[irow])
SWDmax[irow] = pCurrMB->SWD;
if (pCurrMB->SWD < SWDmin[irow])
SWDmin[irow] = pCurrMB->SWD;
}
}
}
SWDrange[0] = SWDmax[0] - SWDmin[0];
SWDrange[1] = SWDmax[1] - SWDmin[1];
SWDrange[2] = SWDmax[2] - SWDmin[2];
if (SWDNum[0] != 0)
SWDAvg[0] = (double) SWDSum[0] / SWDNum[0];
else
SWDAvg[0] = 0.0;
if (SWDNum[1] != 0)
SWDAvg[1] = (double) SWDSum[1] / SWDNum[1];
else
SWDAvg[1] = 0.0;
if (SWDNum[2] != 0)
SWDAvg[2] = (double) SWDSum[2] / SWDNum[2];
else
SWDAvg[2] = 0.0;
QPMax = unGQuant + QPMaxTbl[unGQuant];
if (QPMax > 31)
QPMax = 32;
if ((SWDAvg[0] - SWDmin[0]) != 0)
Step = (double) (QPMax - unGQuant)/(SWDAvg[0] - SWDmin[0]);
else
Step = 0.0;
/* Loop through each macroblock of the GOB.
*/
pLastMB = NULL;
pCurrMB = &pMBActionStream[unStartingMB];
for(iMBIndex = 0 ; iMBIndex < 33; iMBIndex++, pLastMB = pCurrMB++)
{
unCurrentMB = unStartingMB + (unsigned int)iMBIndex;
#ifdef DEBUG_ENC
wsprintf(string, "MB #%d: QP=%d", unCurrentMB, unMQuant);
trace(string);
#endif
if (bRTPHeader)
{
H261RTP_MBUpdateBsInfo(EC,
pCurrMB,
unLastEncodedMQuant,
(U32 )unLastEncodedMBA,
uGOBNumber,
*ppu8BitStream,
(U32) *pu8BitOffset,
unCurrentMB,
unLastCodedMB
);
}
unMQuant = unMQuantLast; // reset MQuant in case needed to
// to raise on previous MB to avoid
// Quant clamping artifact.
/* Look to update the Quant on each new row.
*/
if (EC->bBitRateControl && ((iMBIndex == 11) || (iMBIndex == 22)))
{
/* Calculate number of bytes used in frame so far.
*/
uCumFrmSize = *ppu8BitStream - EC->pU8_BitStream;
unMQuant = CalcMBQUANT(&(EC->BRCState), EC->uBitUsageProfile[unCurrentMB], EC->uBitUsageProfile[EC->NumMBs], uCumFrmSize, EC->PictureHeader.PicCodType);
QPMax = unMQuant + QPMaxTbl[unMQuant];
if (QPMax > 31)
QPMax = 32;
if ((SWDAvg[iMBIndex/11] - SWDmin[iMBIndex/11]) != 0)
Step = (double) (QPMax - unMQuant)/(SWDAvg[iMBIndex/11] - SWDmin[iMBIndex/11]);
else
Step = 0.0;
EC->uBitUsageProfile[unCurrentMB] = uCumFrmSize;
if (bOverFlowWarningFlag)
{
DBOUT("DON'T CHANGE QUANT SET unMQuant = unGQuant");
unMQuant = unGQuant;
}
else if ((int)unMQuant > ((int)unLastEncodedMQuant + MaxChangeRowMBTbl[unGQuant]))
{
DBOUT("Slowing MQuant increase + [1-4]");
unMQuant = unLastEncodedMQuant + MaxChangeRowMBTbl[unMQuant];
}
else if ((int)unMQuant < ((int)unLastEncodedMQuant -2))
{
DBOUT("Slowing MQuant decrease to -2");
unMQuant = unLastEncodedMQuant -2;
}
//CLAMP_N_TO(unMQuant,6,31);
if (EC->BRCState.uTargetFrmSize == 0)
{
CLAMP_N_TO(unMQuant,6,31);
}
else
{
CLAMP_N_TO(unMQuant, u8QPMin, 31);
}
#ifdef DEBUG_BRC
wsprintf(string,"At MB %d MQuant=%d", unCurrentMB, unMQuant);
DBOUT(string);
#endif
#ifdef DEBUG_RECOMPRESS
wsprintf(string,"At MB %d MQuant=%d uCumFrmSize=%d", unCurrentMB, unMQuant,uCumFrmSize*8);
DBOUT(string);
//trace(string);
#endif
//EC->uQP_cumulative += unMQuant;
//EC->uQP_count++;
unSaveMQuant = unMQuant;
if (iMBIndex == 11)
SaveQuants[1] = unSaveMQuant;
else
SaveQuants[2] = unSaveMQuant;
}
/* new MB Quant code */
if (pCurrMB->BlockType != INTRABLOCK)
{
if (EC->BRCState.uTargetFrmSize != 0)
{
if (pCurrMB->SWD >= SWDAvg[iMBIndex/11])
{
Delta = (double) -1.0 * ((double) (pCurrMB->SWD - SWDAvg[iMBIndex/11]) * Step);
if (Delta < -2.0)
{
Delta = -2.0;
NeedClamp++;
}
}
else
{
Delta = (double) (SWDAvg[iMBIndex/11] - pCurrMB->SWD)*Step;
}
}
else
Delta = 0.0;
if (Delta > 0.0)
{
unMQuant = unSaveMQuant + (int) (Delta);
/* Need to clamp again, but only worry about upper limit */
if (unMQuant > 31)
unMQuant = 31;
}
else
{
unMQuant = unSaveMQuant + (int) (Delta - 0.5);
/* Need to clamp again, but only worry about lower limit */
if (EC->BRCState.uTargetFrmSize == 0)
{
if (unMQuant < 6)
unMQuant = 6;
}
else
{
if (unMQuant < 2)
unMQuant = 2;
}
}
}
/* end new stuff */
/* Quantize and RLE each block in the macroblock, skipping empty blocks as denoted by pu8CodedBlocks.
* If any more blocks are empty after quantization then the appropriate pu8CodedBlocks bit is cleared.
*/
//ASSERT(unMQuant >= 6 && unMQuant <= 31); /* CLAMP_N_TO(var,6,31) */
if (EC->BRCState.uTargetFrmSize == 0)
{
ASSERT(unMQuant >= 6 && unMQuant <= 31); /* CLAMP_N_TO(var,6,31) */
}
else
{
ASSERT(unMQuant >= 2 && unMQuant <= 31); /* CLAMP_N_TO(var,6,31) */
}
/* Check iDCTCoefs to see if need to raise quant level to avoid
* clamping artifacts.
*/
// first block is at piDCTCoefs
// second block is at piDCTCoefs + 0x80 and so on
// coefficients are unsigned shorts
// first coefficient is at 6 bytes, 3 words
// second coefficient is at 38 bytes, 19 words
// third coefficient is at 4 bytes, 2 words
// forth coefficient is at 36 bytes, 18 words
unMQuantLast = unMQuant;
if (unMQuant < 6)
{
I8 iBlockNum;
U8 u8Bitmask = 1;
I32 * ptmpiDCTCoefs = piDCTCoefs;
int coef0, coef1;
int biggestcoefval = -2048;
int smallestcoefval = 2048;
#ifdef DEBUG_QUANT
wsprintf(string,"At MB %d MQuant=%d", unCurrentMB, unMQuant);
DBOUT(string);
//trace(string);
#endif
for(iBlockNum = 0; iBlockNum < 6; iBlockNum++, u8Bitmask <<= 1)
{
/* Skip this block if not coded.
*/
if( (pCurrMB->CodedBlocks & u8Bitmask) == 0)
{
continue;
}
if(IsIntraBlock(pCurrMB->BlockType)) // if Intra
{
coef0 = ((int)*((U16*)ptmpiDCTCoefs+3)) >> 4;
}
else
{
coef0 = ((int)(*((U16*)ptmpiDCTCoefs+3) - 0x8000) ) >> 4;
}
coef1 = ((int)(*((U16*)ptmpiDCTCoefs+19) - 0x8000)) >> 4;
#ifdef DEBUG_QUANT
wsprintf(string,"At Block %d 0 = %x %d", iBlockNum,coef0,coef0);
//DBOUT(string);
//trace(string);
#endif
if (coef0 > biggestcoefval)
{
biggestcoefval = coef0;
}
if (coef1 > biggestcoefval)
{
biggestcoefval = coef1;
}
if (coef0 < smallestcoefval)
{
smallestcoefval = coef0;
}
if (coef1 < smallestcoefval)
{
smallestcoefval = coef1;
}
ptmpiDCTCoefs += 32;
}
#ifdef DEBUG_QUANT
wsprintf(string,"biggest = %x %d, smallest = %x %d",
biggestcoefval, biggestcoefval, smallestcoefval, smallestcoefval);
DBOUT(string);
// trace(string);
#endif
if (unMQuant == 5) {
if ((biggestcoefval > 1275) || (smallestcoefval < -1275))
unMQuant = 6;
}
else if (unMQuant == 4) {
if ((biggestcoefval > 1275) || (smallestcoefval < -1275))
unMQuant = 6;
else if ((biggestcoefval > 1019) || (smallestcoefval < -1019))
unMQuant = 5;
}
else if (unMQuant == 3) {
if ((biggestcoefval > 1275) || (smallestcoefval < -1275))
unMQuant = 6;
else if ((biggestcoefval > 1019) || (smallestcoefval < -1019))
unMQuant = 5;
else if ((biggestcoefval > 765) || (smallestcoefval < -765))
unMQuant = 4;
}
else {
if ((biggestcoefval > 1275) || (smallestcoefval < -1275))
unMQuant = 6;
else if ((biggestcoefval > 1019) || (smallestcoefval < -1019))
unMQuant = 5;
else if ((biggestcoefval > 765) || (smallestcoefval < -765))
unMQuant = 4;
else if ((biggestcoefval > 509) || (smallestcoefval < -509))
unMQuant = 3;
}
#ifdef DEBUG_QUANT
wsprintf(string,"At MB %d MQuant=%d", unCurrentMB, unMQuant);
DBOUT(string);
//trace(string);
#endif
}
/* This is the place to trace how Quant on a MB bases is varying
*/
EC->uQP_cumulative += unMQuant;
EC->uQP_count++;
pi8EndAddress = MB_Quantize_RLE(
&piDCTCoefs,
(I8 *) MBRunValSign,
&(pCurrMB->CodedBlocks),
pCurrMB->BlockType,
unMQuant,
&uCheckSum
);
pBSInfo->uQuantsUsedOnBlocks[unMQuant] += 6;
bWriteMVD = (pCurrMB->BlkY1.PHMV != 0) ||
(pCurrMB->BlkY1.PVMV != 0) ||
(IsSLFBlock(pCurrMB->BlockType)) ;
if (IsInterBlock(pCurrMB->BlockType))
{
/* Check if the Inter block is not coded?
*/
if ( ((pCurrMB->CodedBlocks & 0x3f) == 0) &&
(! bWriteMVD) )
{
#ifdef DEBUG_MBLK
wsprintf(string, "Inter MB (index=#%d) has neither Coeff nor MV - skipping", unCurrentMB);
DBOUT(string);
#endif
#ifdef FORCE_STUFFING
PutBits(FIELDVAL_MBA_STUFFING, FIELDLEN_MBA_STUFFING, ppu8CurBitStream, pu8BitOffset);
#endif
continue;
}
}
#ifdef CHECKSUM_MACRO_BLOCK
/* Write a checksum before all coded blocks
*/
WriteMBCheckSum(uCheckSum, EC->pU8_BitStream,ppu8BitStream, pu8BitOffset, unCurrentMB);
#endif
/* Calculate the MB header information
*/
unMBA = iMBIndex - iLastMBIndex;
iLastMBIndex = iMBIndex;
unLastEncodedMBA = unMBA;
unLastCodedMB = iMBIndex;
/* Note: The calculation of whether to write MQuant is done after
* skipping macro blocks in order to handle the case that the 11th
* or 22nd macro blocks are skipped. If they are skipped then
* the next macro block will be used to write the new quant value.
*/
if(IsIntraBlock(pCurrMB->BlockType))
{
ASSERT(pCurrMB->BlockType == INTRABLOCK);
if (EC->PictureHeader.PicCodType != INTRAPIC)
{
pCurrMB->InterCodeCnt = ((U8)unCurrentMB)&0x7;
}
bIntraBlock = 1;
unCBP = 0; /* Never write CBP for Intra blocks */
uBlockCount = 6;
bWriteTCOEFF = 1; /* Always include TCOEFF for Intra blocks */
/* Since we always have coefficients for Intra MBs we can always update
* the MQuant value.
*/
bWriteMQuant = (unMQuant != unLastEncodedMQuant);
unLastEncodedMQuant = unMQuant;
unMType = 0 + bWriteMQuant; /* Calculate MTYPE */
bWriteMVD = 0; /* No motion vectors for INTRA */
}
else
{
ASSERT(IsInterBlock(pCurrMB->BlockType));
bIntraBlock = 0;
unCBP = (pCurrMB->CodedBlocks & 0x1) << 5; /* x0 0000 */
unCBP |= (pCurrMB->CodedBlocks & 0x2) << 3; /* 0x 0000 */
unCBP |= (pCurrMB->CodedBlocks & 0x4) << 1; /* 00 x000 */
unCBP |= (pCurrMB->CodedBlocks & 0x8) >> 1; /* 00 0x00 */
unCBP |= (pCurrMB->CodedBlocks & 0x10) >> 3; /* 00 00x0 */
unCBP |= (pCurrMB->CodedBlocks & 0x20) >> 5; /* 00 000x */
uBlockCount = 0;
if (unCBP & 0x1) uBlockCount++;
if (unCBP & 0x2) uBlockCount++;
if (unCBP & 0x4) uBlockCount++;
if (unCBP & 0x8) uBlockCount++;
if (unCBP & 0x10) uBlockCount++;
if (unCBP & 0x20) uBlockCount++;
/* Increment the count if it is transmitted
* "should be forcibly updated at least once every
* 132 times it is transmitted" 3.4
*/
if (uBlockCount != 0 )
{
pCurrMB->InterCodeCnt++;
}
bWriteTCOEFF = (unCBP != 0);
if (bWriteTCOEFF)
{
/* We can only update the MQuant value when we have coefficients
*/
bWriteMQuant = (unMQuant != unLastEncodedMQuant);
unLastEncodedMQuant = unMQuant;
}
else
{
bWriteMQuant = 0;
}
#ifdef CHECKSUM_MACRO_BLOCK
/* Either there are coefficients or the checksum should equal zero
*/
ASSERT(bWriteTCOEFF || uCheckSum == 0);
#endif
/* Calculate MType
*/
unMType = 1;
if (bWriteMVD)
{
unMType += 3;
if (IsSLFBlock(pCurrMB->BlockType))
{
unMType += 3;
}
}
unMType += bWriteTCOEFF;
unMType += bWriteMQuant;
ASSERT(unMType > 1 && unMType < 10);
}
ASSERT(unMQuant >= 1 && unMQuant <= 31);
ASSERT(uBlockCount <= 6);
pBSInfo->uQuantsTransmittedOnBlocks[unMQuant] += uBlockCount;
if (bWriteMVD)
{
/* Find the preceding motion vectors
*/
if ( (unMBA != 1) || /* skipped one or more MB */
((unCurrentMB % 11) == 0) ) /* first MB in each row */
{
inPrecedingHMV = 0;
inPrecedingVMV = 0;
}
else
{
inPrecedingHMV = pLastMB->BlkY1.PHMV;
inPrecedingVMV = pLastMB->BlkY1.PVMV;
}
/* adjust vectors:
*/
inHDelta = pCurrMB->BlkY1.PHMV - inPrecedingHMV;
ASSERT((inHDelta & 0x1) == 0);
ASSERT((inHDelta >> 1) == (inHDelta / 2));
inHDelta >>= 1; /* Adjust to integer pels */
if(inHDelta > 15) /* Adjust to the range of -16...+15 */
inHDelta -= 32;
if(inHDelta < -16)
inHDelta += 32;
inHDelta = inHDelta + 16; /* 0 is at offset 16 */
inVDelta = pCurrMB->BlkY1.PVMV - inPrecedingVMV;
ASSERT((inVDelta & 0x1) == 0);
ASSERT((inVDelta >> 1) == (inVDelta / 2));
inVDelta >>= 1;
if(inVDelta > 15)
inVDelta -= 32;
if(inVDelta < -16)
inVDelta += 32;
inVDelta = inVDelta + 16;
#ifndef RING0
#ifdef DEBUG_PRINTMV
{
char buf132[132];
int iLength;
iLength = wsprintf(buf132, "MB # %d :: H MVD = %d; index = %d :: V MVD = %d; index = %d", unCurrentMB,
pCurrMB->BlkY1.PHMV / 2, inHDelta,
pCurrMB->BlkY1.PVMV / 2, inVDelta);
DBOUT(buf132);
ASSERT(iLength < 132);
}
#endif
#endif
}
else
{
/* MBs without MVD need to have zero motion vectors because of
* Rule 3) under 4.2.3.4
*/
pCurrMB->BlkY1.PHMV = 0;
pCurrMB->BlkY1.PVMV = 0;
}
/* we should only have MQuant if we have coefficients
*/
if (bWriteMQuant)
{
ASSERT(bWriteTCOEFF);
}
/* we should only have CBP if we have coefficients
*/
if (unCBP)
{
ASSERT(bWriteTCOEFF);
ASSERT(uBlockCount > 0);
}
/* Write the MacroBlock Header
*/
#ifndef RING0
#ifdef DEBUG_MBLK
{
int iLength;
char buf180[180];
iLength = wsprintf(buf180, "Enc #%d: MBType=%ld unNextMQuant=%d MQuant=%ld bWriteMVD=%d MVDH=%ld MVDV=%ld CBP=%ld",
(int) unCurrentMB,
unMType,
(int) bWriteMQuant,
unMQuant,
(int) bWriteMVD,
pCurrMB->BlkY1.PHMV / 2,
pCurrMB->BlkY1.PVMV / 2,
unCBP);
DBOUT(buf180);
ASSERT(iLength < 180);
}
#endif
#endif
/* MBA
*/
PutBits(VLC_MBA[unMBA][1], VLC_MBA[unMBA][0], ppu8BitStream, pu8BitOffset);
/* MTYPE
*/
pBSInfo->uMTypeCount[unMType]++;
pBSInfo->uBlockCount[unMType] += uBlockCount;
PutBits(VLC_MTYPE[unMType][1], VLC_MTYPE[unMType][0], ppu8BitStream, pu8BitOffset);
/* MQUANT
*/
if (bWriteMQuant)
{
ASSERT(unMQuant > 0 && unMQuant < 32); /* 4.2.2.3 */
PutBits((int)unMQuant, FIELDLEN_MQUANT, ppu8BitStream, pu8BitOffset);
}
/* MVD
*/
if (bWriteMVD)
{
ASSERT(inHDelta >= 0 && inHDelta < 32);
ASSERT(inVDelta >= 0 && inVDelta < 32);
PutBits(VLC_MVD[inHDelta][1], VLC_MVD[inHDelta][0], ppu8BitStream, pu8BitOffset);
PutBits(VLC_MVD[inVDelta][1], VLC_MVD[inVDelta][0], ppu8BitStream, pu8BitOffset);
}
/* CBP
*/
if (unCBP != 0)
{
PutBits(VLC_CBP[unCBP][1], VLC_CBP[unCBP][0], ppu8BitStream, pu8BitOffset);
}
/* TCOEFF
*/
if (bWriteTCOEFF)
{
/*
* Encode intra DC and all run/val pairs.
*/
rvs = MBRunValSign;
MBEncodeVLC(
&rvs,
NULL,
pCurrMB->CodedBlocks,
ppu8BitStream,
pu8BitOffset,
bIntraBlock,
FALSE);
}
} /* for iMBIndex */
} /* end of GOB_Q_RLE_VLC_WriteBS() */
void GOB_VLC_WriteBS(
T_H263EncoderCatalog * EC,
I8 *pMBRVS_Luma,
I8 *pMBRVS_Chroma,
U8 **ppu8BitStream,
U8 *pu8BitOffset,
UN unGQuant,
UN unStartingMB,
BOOL bRTPHeader, //RTP: switch
U32 uGOBNumber // RTP: info
)
{
T_MBlockActionStream *pCurrMB = NULL;
T_MBlockActionStream *pLastMB = NULL;
int iMBIndex;
int iLastMBIndex = -1;
UN unCurrentMB;
UN unMBA;
UN unLastEncodedMBA=0; // RTP: information
UN unLastCodedMB = 0; // RTP: information
UN unCBP;
UN unMQuant;
UN unLastEncodedMQuant;
UN unMType;
UN bWriteTCOEFF;
UN bWriteMVD;
UN bWriteMQuant;
// I8 MBRunValSign[65*3*6], *pi8EndAddress, *rvs;
T_MBlockActionStream *pMBActionStream = EC->pU8_MBlockActionStream;
int bIntraBlock;
int inPrecedingHMV;
int inPrecedingVMV;
int inHDelta;
int inVDelta;
// U32 uCumFrmSize;
U32 uBlockCount;
ENC_BITSTREAM_INFO * pBSInfo = &EC->BSInfo;
unMQuant = unGQuant;
unLastEncodedMQuant = unGQuant;
/* Loop through each macroblock of the GOB.
*/
pLastMB = NULL;
pCurrMB = &pMBActionStream[unStartingMB];
for(iMBIndex = 0 ; iMBIndex < 33; iMBIndex++, pLastMB = pCurrMB++)
{
unCurrentMB = unStartingMB + (unsigned int)iMBIndex;
#ifdef DEBUG_ENC
wsprintf(string, "MB #%d: QP=%d", unCurrentMB, unMQuant);
trace(string);
#endif
if (bRTPHeader)
{
H261RTP_MBUpdateBsInfo(EC,
pCurrMB,
unLastEncodedMQuant,
(U32 )unLastEncodedMBA,
uGOBNumber,
*ppu8BitStream,
(U32) *pu8BitOffset,
unCurrentMB,
unLastCodedMB
);
}
EC->uQP_cumulative += unMQuant;
EC->uQP_count++;
bWriteMVD = (pCurrMB->BlkY1.PHMV != 0) ||
(pCurrMB->BlkY1.PVMV != 0) ||
(IsSLFBlock(pCurrMB->BlockType)) ;
if (IsInterBlock(pCurrMB->BlockType))
{
/* Check if the Inter block is not coded?
*/
if ( ((pCurrMB->CodedBlocks & 0x3f) == 0) &&
(! bWriteMVD) )
{
#ifdef DEBUG_MBLK
wsprintf(string, "Inter MB (index=#%d) has neither Coeff nor MV - skipping", unCurrentMB);
DBOUT(string);
#endif
#ifdef FORCE_STUFFING
PutBits(FIELDVAL_MBA_STUFFING, FIELDLEN_MBA_STUFFING, ppu8CurBitStream, pu8BitOffset);
#endif
continue;
}
}
#ifdef CHECKSUM_MACRO_BLOCK
/* Write a checksum before all coded blocks
*/
WriteMBCheckSum(uCheckSum, EC->pU8_BitStream,ppu8BitStream, pu8BitOffset, unCurrentMB);
#endif
/* Calculate the MB header information
*/
unMBA = iMBIndex - iLastMBIndex;
iLastMBIndex = iMBIndex;
unLastEncodedMBA = unMBA;
unLastCodedMB = iMBIndex;
/* Note: The calculation of whether to write MQuant is done after
* skipping macro blocks in order to handle the case that the 11th
* or 22nd macro blocks are skipped. If they are skipped then
* the next macro block will be used to write the new quant value.
*/
if(IsIntraBlock(pCurrMB->BlockType))
{
ASSERT(pCurrMB->BlockType == INTRABLOCK);
if (EC->PictureHeader.PicCodType != INTRAPIC)
{
pCurrMB->InterCodeCnt = ((U8)unCurrentMB)&0x7;
}
bIntraBlock = 1;
unCBP = 0; /* Never write CBP for Intra blocks */
uBlockCount = 6;
bWriteTCOEFF = 1; /* Always include TCOEFF for Intra blocks */
/* Since we always have coefficients for Intra MBs we can always update
* the MQuant value.
*/
//bWriteMQuant = (unMQuant != unLastEncodedMQuant);
//unLastEncodedMQuant = unMQuant;
bWriteMQuant=0;
unMType = 0; // + bWriteMQuant; /* Calculate MTYPE */
bWriteMVD = 0; /* No motion vectors for INTRA */
}
else
{
ASSERT(IsInterBlock(pCurrMB->BlockType));
bIntraBlock = 0;
unCBP = (pCurrMB->CodedBlocks & 0x1) << 5; /* x0 0000 */
unCBP |= (pCurrMB->CodedBlocks & 0x2) << 3; /* 0x 0000 */
unCBP |= (pCurrMB->CodedBlocks & 0x4) << 1; /* 00 x000 */
unCBP |= (pCurrMB->CodedBlocks & 0x8) >> 1; /* 00 0x00 */
unCBP |= (pCurrMB->CodedBlocks & 0x10) >> 3; /* 00 00x0 */
unCBP |= (pCurrMB->CodedBlocks & 0x20) >> 5; /* 00 000x */
uBlockCount = 0;
if (unCBP & 0x1) uBlockCount++;
if (unCBP & 0x2) uBlockCount++;
if (unCBP & 0x4) uBlockCount++;
if (unCBP & 0x8) uBlockCount++;
if (unCBP & 0x10) uBlockCount++;
if (unCBP & 0x20) uBlockCount++;
/* Increment the count if it is transmitted
* "should be forcibly updated at least once every
* 132 times it is transmitted" 3.4
*/
if (uBlockCount != 0 )
{
pCurrMB->InterCodeCnt++;
}
bWriteTCOEFF = (unCBP != 0);
bWriteMQuant = 0;
#ifdef CHECKSUM_MACRO_BLOCK
/* Either there are coefficients or the checksum should equal zero
*/
ASSERT(bWriteTCOEFF || uCheckSum == 0);
#endif
/* Calculate MType
*/
unMType = 1;
if (bWriteMVD)
{
unMType += 3;
if (IsSLFBlock(pCurrMB->BlockType))
{
unMType += 3;
}
}
unMType += bWriteTCOEFF;
unMType += bWriteMQuant;
ASSERT(unMType > 1 && unMType < 10);
}
ASSERT(unMQuant >= 1 && unMQuant <= 31);
ASSERT(uBlockCount <= 6);
pBSInfo->uQuantsTransmittedOnBlocks[unMQuant] += uBlockCount;
if (bWriteMVD)
{
/* Find the preceding motion vectors
*/
if ( (unMBA != 1) || /* skipped one or more MB */
((unCurrentMB % 11) == 0) ) /* first MB in each row */
{
inPrecedingHMV = 0;
inPrecedingVMV = 0;
}
else
{
inPrecedingHMV = pLastMB->BlkY1.PHMV;
inPrecedingVMV = pLastMB->BlkY1.PVMV;
}
/* adjust vectors:
*/
inHDelta = pCurrMB->BlkY1.PHMV - inPrecedingHMV;
ASSERT((inHDelta & 0x1) == 0);
ASSERT((inHDelta >> 1) == (inHDelta / 2));
inHDelta >>= 1; /* Adjust to integer pels */
if(inHDelta > 15) /* Adjust to the range of -16...+15 */
inHDelta -= 32;
if(inHDelta < -16)
inHDelta += 32;
inHDelta = inHDelta + 16; /* 0 is at offset 16 */
inVDelta = pCurrMB->BlkY1.PVMV - inPrecedingVMV;
ASSERT((inVDelta & 0x1) == 0);
ASSERT((inVDelta >> 1) == (inVDelta / 2));
inVDelta >>= 1;
if(inVDelta > 15)
inVDelta -= 32;
if(inVDelta < -16)
inVDelta += 32;
inVDelta = inVDelta + 16;
#ifndef RING0
#ifdef DEBUG_PRINTMV
{
char buf132[132];
int iLength;
iLength = wsprintf(buf132, "MB # %d :: H MVD = %d; index = %d :: V MVD = %d; index = %d", unCurrentMB,
pCurrMB->BlkY1.PHMV / 2, inHDelta,
pCurrMB->BlkY1.PVMV / 2, inVDelta);
DBOUT(buf132);
ASSERT(iLength < 132);
}
#endif
#endif
}
else
{
/* MBs without MVD need to have zero motion vectors because of
* Rule 3) under 4.2.3.4
*/
pCurrMB->BlkY1.PHMV = 0;
pCurrMB->BlkY1.PVMV = 0;
}
/* we should only have MQuant if we have coefficients
*/
if (bWriteMQuant)
{
ASSERT(bWriteTCOEFF);
}
/* we should only have CBP if we have coefficients
*/
if (unCBP)
{
ASSERT(bWriteTCOEFF);
ASSERT(uBlockCount > 0);
}
/* Write the MacroBlock Header
*/
#ifndef RING0
#ifdef DEBUG_MBLK
{
int iLength;
char buf180[180];
iLength = wsprintf(buf180,
"Enc #%d: MBType=%ld bWriteMQuant=%ld MQuant=%ld bWriteMVD=%d MVDH=%ld MVDV=%ld CBP=%ld",
(int) unCurrentMB,
unMType,
(int) bWriteMQuant,
unMQuant,
(int) bWriteMVD,
pCurrMB->BlkY1.PHMV / 2,
pCurrMB->BlkY1.PVMV / 2,
unCBP);
DBOUT(buf180);
ASSERT(iLength < 180);
}
#endif
#endif
/* MBA
*/
PutBits(VLC_MBA[unMBA][1], VLC_MBA[unMBA][0], ppu8BitStream, pu8BitOffset);
/* MTYPE
*/
pBSInfo->uMTypeCount[unMType]++;
pBSInfo->uBlockCount[unMType] += uBlockCount;
PutBits(VLC_MTYPE[unMType][1], VLC_MTYPE[unMType][0], ppu8BitStream, pu8BitOffset);
/* MQUANT
*/
if (bWriteMQuant)
{
ASSERT(unMQuant > 0 && unMQuant < 32); /* 4.2.2.3 */
PutBits((int)unMQuant, FIELDLEN_MQUANT, ppu8BitStream, pu8BitOffset);
}
/* MVD
*/
if (bWriteMVD)
{
ASSERT(inHDelta >= 0 && inHDelta < 32);
ASSERT(inVDelta >= 0 && inVDelta < 32);
PutBits(VLC_MVD[inHDelta][1], VLC_MVD[inHDelta][0], ppu8BitStream, pu8BitOffset);
PutBits(VLC_MVD[inVDelta][1], VLC_MVD[inVDelta][0], ppu8BitStream, pu8BitOffset);
}
/* CBP
*/
if (unCBP != 0)
{
PutBits(VLC_CBP[unCBP][1], VLC_CBP[unCBP][0], ppu8BitStream, pu8BitOffset);
}
/* TCOEFF
*/
if (bWriteTCOEFF)
{
/*
* Encode intra DC and all run/val pairs.
*/
MBEncodeVLC(
&pMBRVS_Luma,
&pMBRVS_Chroma,
pCurrMB->CodedBlocks,
ppu8BitStream,
pu8BitOffset,
bIntraBlock,
1);
}
} /* for iMBIndex */
} /* end of GOB_VLC_WriteBS() */
/*****************************************************************************
*
* MB_Quantize_RLE
*
* Takes the list of coefficient pairs from the DCT routine and returns a list
* of Run/Level/Sign triples (each 1 byte). The end of the run/level/sign
* triples for a block is signalled by an illegal combination (TBD).
*/
static I8 * MB_Quantize_RLE(
I32 ** ppiDCTCoefs,
I8 * pi8MBRunValTriplets,
U8 * pu8CodedBlocks,
U8 u8BlockType,
I32 iQP,
U32 * puCheckSum
)
{
I32 iBlockNumber;
U8 u8Bitmask = 1;
I8 * pi8EndAddress;
U32 uCheckSum;
#ifdef DEBUG_DCT
int iDCTArray[64];
#endif
/*
* Loop through all 6 blocks of macroblock.
*/
uCheckSum = 0;
for(iBlockNumber = 0; iBlockNumber < 6; iBlockNumber++, u8Bitmask <<= 1)
{
#ifdef DEBUG_ENC
wsprintf(string, "Block #%d", iBlockNumber);
trace(string);
#endif
/* Skip this block if not coded.
*/
if( (*pu8CodedBlocks & u8Bitmask) == 0)
{
continue;
}
#ifdef DEBUG_DCT
cnvt_fdct_output((unsigned short *) *ppiDCTCoefs, iDCTArray, IsIntraBlock(u8BlockType));
#endif
/*
* Quantize and run-length encode a block.
*/
pi8EndAddress = QUANTRLE(*ppiDCTCoefs, pi8MBRunValTriplets, iQP, (I32)u8BlockType);
#ifdef DEBUG_ENC
I8 * pi8;
for(pi8 = pi8MBRunValTriplets; pi8 < pi8EndAddress; pi8+=3)
{
wsprintf(string, "(%u, %u, %d)", (unsigned char)*pi8, (unsigned char)*(pi8+1), (int)*(pi8+2) );
trace(string);
}
#endif
#ifdef CHECKSUM_MACRO_BLOCK
uCheckSum += ComputeCheckSum(pi8MBRunValTriplets, pi8EndAddress, iBlockNumber);
#endif
/* Clear coded block bit for this block.
*/
if ( pi8EndAddress == pi8MBRunValTriplets)
{
ASSERT(u8BlockType != INTRABLOCK) /* should have at least INTRADC in an INTRA blck */
*pu8CodedBlocks &= ~u8Bitmask;
}
else if ( (pi8EndAddress == (pi8MBRunValTriplets+3)) && (u8BlockType == INTRABLOCK) )
{
*pu8CodedBlocks &= ~u8Bitmask;
pi8MBRunValTriplets = pi8EndAddress;
}
else
{
pi8MBRunValTriplets = pi8EndAddress;
*pi8MBRunValTriplets = -1; /* Assign an illegal run to signal end of block. */
pi8MBRunValTriplets += 3; /* Increment to the next triple. */
}
/* Increment DCT Coefficient pointer to next block.
*/
*ppiDCTCoefs += 32;
}
*puCheckSum = uCheckSum;
return pi8MBRunValTriplets;
} /* end MB_Quantize_RLE() */
void InitVLC(void)
{
int i;
int run, level;
/*
* initialize INTRADC fixed length code table.
*/
for(i = 1; i < 254; i++)
{
FLC_INTRADC[i] = i;
}
FLC_INTRADC[0] = 1;
FLC_INTRADC[128] = 255;
FLC_INTRADC[254] = 254;
FLC_INTRADC[255] = 254;
/*
* Initialize tcoef tables.
*/
for(i = 0; i < (NUMBER_OF_TCOEF_ENTRIES); i++) {
VLC_TCOEF_TBL[i] = 0x0000FFFF;
}
for(run = 0; run < 64; run++) {
for(level = 1; level <= TCOEF_RUN_MAXLEVEL[run].maxlevel; level++) {
DWORD dwSize, dwCode;
dwSize = *(TCOEF_RUN_MAXLEVEL[run].ptable + (level - 1) * 2);
dwSize <<= 16;
dwCode = *(TCOEF_RUN_MAXLEVEL[run].ptable + (level - 1) * 2 + 1);
VLC_TCOEF_TBL[run + (level - 1) * 64] = dwCode;
VLC_TCOEF_TBL[run + (level - 1) * 64] |= dwSize;
} // end of for level
} // end of for run
} // InitVLC.
#ifdef CHECKSUM_MACRO_BLOCK
/*****************************************************************************
*
* ComputeCheckSum
*
* Compute the checksum for this block
*/
static U32 ComputeCheckSum(
I8 * pi8MBRunValTriplets,
I8 * pi8EndAddress,
I32 iBlockNumber)
{
I8 * pi8;
U32 uRun;
U32 uLevel;
I32 iSign;
U32 uSignBit;
U32 uCheckSum = 0;
#if CHECKSUM_MACRO_BLOCK_DETAIL
char buf80[80];
int iLength;
#endif
for (pi8 = pi8MBRunValTriplets; pi8 < pi8EndAddress; )
{
uRun = (U32)*pi8++;
uLevel = (U32)(U8)*pi8++;
iSign = (I32)*pi8++;
if (iSign == 0)
{
uSignBit = 0;
}
else
{
ASSERT(iSign == 0xFFFFFFFF);
uSignBit = 1;
}
uCheckSum += uRun << 24;
uCheckSum += uLevel << 8;
uCheckSum += uSignBit;
#ifdef CHECKSUM_MACRO_BLOCK_DETAIL
iLength = wsprintf(buf80,"Block=%d R=0x%x L=0x%x S=%d, CheckSum=0x%x", iBlockNumber, uRun, uLevel, uSignBit, uCheckSum);
DBOUT(buf80);
ASSERT(iLength < 80);
#endif
}
return uCheckSum;
} /* end ComputeCheckSum() */
/*****************************************************************************
*
* WriteMBCheckSum
*
* Write the macro block checksum information.
*/
static void WriteMBCheckSum(
U32 uCheckSum,
U8 * pu8PictureStart,
U8 ** ppu8BitStream,
U8 * pu8BitOffset,
UN unCurrentMB)
{
U32 uBytes;
U32 uTempBytes;
U8 u8Bits;
U8 u8TempBits;
UN unCount;
UN unKey;
UN unData;
uBytes = *ppu8BitStream - pu8PictureStart;
u8Bits = *pu8BitOffset;
/* Add in the space for the checksum info (eleven 8-bit fields + 12 "1"s + MBA stuffing)
*/
uBytes += 12;
u8Bits += 4 + FIELDLEN_MBA_STUFFING;
/* Adjust bits to < 7
*/
while (u8Bits > 7)
{
u8Bits -= 8;
uBytes++;
}
#if _DEBUG
#if CHECKSUM_MACRO_BLOCK_DETAIL
{
char buf80[80];
int iLength;
iLength = wsprintf(buf80,"MB=%d CHK=0x%x Bytes=%ld Bits=%d", unCurrentMB, uCheckSum, uBytes, (int) u8Bits);
DBOUT(buf80);
ASSERT(iLength < 80);
}
#endif
#endif
/* Write the MBASTUFFING value
*/
PutBits(FIELDVAL_MBA_STUFFING, FIELDLEN_MBA_STUFFING, ppu8BitStream, pu8BitOffset);
/* Write the data to the bitstream
*/
/* Key - a value of 1 in an 8-bit field following a "1"
*/
unKey = 1;
PutBits(1, 1, ppu8BitStream, pu8BitOffset);
PutBits(unKey, 8, ppu8BitStream, pu8BitOffset);
/* Count - number of bits after the Count field.
*/
unCount = 9*8 + 10*1; /* nine 8-bit value and 10 "1"s. */
PutBits(1, 1, ppu8BitStream, pu8BitOffset);
PutBits(unCount, 8, ppu8BitStream, pu8BitOffset);
/* Bytes - high to low bytes
*/
unData = (UN) ((uBytes >> 24) & 0xFF);
PutBits(1, 1, ppu8BitStream, pu8BitOffset);
PutBits(unData, 8, ppu8BitStream, pu8BitOffset);
unData = (UN) ((uBytes >> 16) & 0xFF);
PutBits(1, 1, ppu8BitStream, pu8BitOffset);
PutBits(unData, 8, ppu8BitStream, pu8BitOffset);
unData = (UN) ((uBytes >> 8) & 0xFF);
PutBits(1, 1, ppu8BitStream, pu8BitOffset);
PutBits(unData, 8, ppu8BitStream, pu8BitOffset);
unData = (UN) (uBytes & 0xFF);
PutBits(1, 1, ppu8BitStream, pu8BitOffset);
PutBits(unData, 8, ppu8BitStream, pu8BitOffset);
/* Bits
*/
unData = (UN) u8Bits;
PutBits(1, 1, ppu8BitStream, pu8BitOffset);
PutBits(unData, 8, ppu8BitStream, pu8BitOffset);
/* Checksum - high to low bytes
*/
unData = (UN) ((uCheckSum >> 24) & 0xFF);
PutBits(1, 1, ppu8BitStream, pu8BitOffset);
PutBits(unData, 8, ppu8BitStream, pu8BitOffset);
unData = (UN) ((uCheckSum >> 16) & 0xFF);
PutBits(1, 1, ppu8BitStream, pu8BitOffset);
PutBits(unData, 8, ppu8BitStream, pu8BitOffset);
unData = (UN) ((uCheckSum >> 8) & 0xFF);
PutBits(1, 1, ppu8BitStream, pu8BitOffset);
PutBits(unData, 8, ppu8BitStream, pu8BitOffset);
unData = (UN) (uCheckSum & 0xFF);
PutBits(1, 1, ppu8BitStream, pu8BitOffset);
PutBits(unData, 8, ppu8BitStream, pu8BitOffset);
/* Trailing 1 bit to avoid start code duplication.
*/
PutBits(1, 1, ppu8BitStream, pu8BitOffset);
/* Check that the pointers are correct
*/
uTempBytes = *ppu8BitStream - pu8PictureStart;
u8TempBits = *pu8BitOffset;
while (u8TempBits > 7)
{
u8TempBits -= 8;
uTempBytes++;
}
ASSERT(uTempBytes == uBytes);
ASSERT(u8TempBits == u8Bits);
} /* end WriteMBCheckSum() */
#endif