892 lines
27 KiB
C++
892 lines
27 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.
|
||
|
**
|
||
|
** *************************************************************************
|
||
|
*/
|
||
|
|
||
|
/*****************************************************************************
|
||
|
*
|
||
|
* d1mblk.cpp
|
||
|
*
|
||
|
* DESCRIPTION:
|
||
|
* Decoder macro block functions
|
||
|
*
|
||
|
* Routines: Prototypes in:
|
||
|
* H263DecodeMBHeader d1dec.h
|
||
|
* H263DecodeMBData d1dec.h
|
||
|
*/
|
||
|
|
||
|
// $Header: S:\h26x\src\dec\d1mblk.cpv 1.23 20 Dec 1996 16:58:06 RHAZRA $
|
||
|
// $Log: S:\h26x\src\dec\d1mblk.cpv $
|
||
|
//
|
||
|
// Rev 1.23 20 Dec 1996 16:58:06 RHAZRA
|
||
|
// Fixed bitstream docoding for the case where MB stuffing is inserted
|
||
|
// between MBs. This was identified by a PTEL bitstream. This fix needs
|
||
|
// to be verified with our other tests.
|
||
|
//
|
||
|
// Rev 1.22 16 Dec 1996 14:41:46 RHAZRA
|
||
|
//
|
||
|
// Changed a bitstream error ASSERT to a bonafide error
|
||
|
//
|
||
|
// Rev 1.21 18 Nov 1996 17:12:22 MBODART
|
||
|
// Replaced all debug message invocations with Active Movie's DbgLog.
|
||
|
//
|
||
|
// Rev 1.20 07 Nov 1996 15:44:08 SCDAY
|
||
|
//
|
||
|
// Added MMX_ClipAndScale to replace Raj's glue code
|
||
|
//
|
||
|
// Rev 1.19 04 Nov 1996 10:28:10 RHAZRA
|
||
|
// Changed the IDCT scaling table to be a DWORD table (with rounding
|
||
|
// factored in) that is declared as a static.
|
||
|
//
|
||
|
// Rev 1.18 31 Oct 1996 08:58:28 SCDAY
|
||
|
// Raj added support for MMX decoder
|
||
|
//
|
||
|
// Rev 1.17 26 Sep 1996 12:35:06 RHAZRA
|
||
|
// Forced the decoder to use the IA version of VLD_RLD_IQ routine even
|
||
|
// when MMX is on (since we don't have a corresponding MMX routine ... yet)
|
||
|
//
|
||
|
// Rev 1.16 05 Aug 1996 11:00:26 MBODART
|
||
|
//
|
||
|
// H.261 decoder rearchitecture:
|
||
|
// Files changed: d1gob.cpp, d1mblk.{cpp,h}, d1dec.{cpp,h},
|
||
|
// filelist.261, h261_32.mak
|
||
|
// New files: d1bvriq.cpp, d1idct.cpp
|
||
|
// Obsolete files: d1block.cpp
|
||
|
// Work still to be done:
|
||
|
// Update h261_mf.mak
|
||
|
// Optimize uv pairing in d1bvriq.cpp and d1idct.cpp
|
||
|
// Fix checksum code (it doesn't work now)
|
||
|
// Put back in decoder stats
|
||
|
//
|
||
|
// Rev 1.15 18 Mar 1996 17:02:12 AKASAI
|
||
|
//
|
||
|
// Added pragma code_seg("IACODE2") and changed the timing statistics.
|
||
|
// At one point changed GET_VAR_BITS into subroutine to save code
|
||
|
// space but it didn't so left it as a macro.
|
||
|
//
|
||
|
// Rev 1.14 26 Dec 1995 17:42:14 DBRUCKS
|
||
|
// changed bTimerIsOn to bTimingThisFrame
|
||
|
//
|
||
|
// Rev 1.13 26 Dec 1995 12:50:00 DBRUCKS
|
||
|
//
|
||
|
// fix copyright
|
||
|
// add timing code
|
||
|
// comment out define of DEBUG_MBLK
|
||
|
//
|
||
|
// Rev 1.12 05 Dec 1995 10:19:46 SCDAY
|
||
|
//
|
||
|
// Added assembler version of Spatial Loop Filter
|
||
|
//
|
||
|
// Rev 1.11 03 Nov 1995 11:44:30 AKASAI
|
||
|
//
|
||
|
// Changed the processing of MB checksum and MBA stuffing. Changed
|
||
|
// GET_VAR_BITS & GET_GT8_BITS for how to detect MBA stuffing code.
|
||
|
//
|
||
|
// Rev 1.10 01 Nov 1995 13:43:48 AKASAI
|
||
|
//
|
||
|
// Added support for loop filter. New routines call LpFilter,
|
||
|
// BlockAddSpecial and BlockCopySpecial.
|
||
|
//
|
||
|
// Rev 1.9 27 Oct 1995 18:17:20 AKASAI
|
||
|
//
|
||
|
// Put in fix "hack" to keep the block action stream pointers
|
||
|
// in sync between d1dec and d1mblk. With skip macro blocks some
|
||
|
// macroblocks were being processed multiple times. Still a problem
|
||
|
// when gob ends with a skip macroblock.
|
||
|
//
|
||
|
// Rev 1.8 26 Oct 1995 15:36:28 SCDAY
|
||
|
//
|
||
|
// Delta frames partially working -- changed main loops to accommodate
|
||
|
// skipped macroblocks by detecting next startcode
|
||
|
//
|
||
|
// Rev 1.7 17 Oct 1995 11:28:56 SCDAY
|
||
|
// Added error message if (MBA stuffing code found && Checksum not enabled)
|
||
|
//
|
||
|
// Rev 1.6 16 Oct 1995 16:28:02 AKASAI
|
||
|
// Fixed bug when CHECKSUM_MACRO_BLOCK_DETAIL & CHECKSUM_MACRO_BLOCK are
|
||
|
// both defined.
|
||
|
//
|
||
|
// Rev 1.5 16 Oct 1995 13:53:24 SCDAY
|
||
|
//
|
||
|
// Added macroblock level checksum
|
||
|
//
|
||
|
// Rev 1.4 06 Oct 1995 15:32:54 SCDAY
|
||
|
//
|
||
|
// Integrated with latest AKK d1block
|
||
|
//
|
||
|
// Rev 1.3 22 Sep 1995 14:48:46 SCDAY
|
||
|
//
|
||
|
// added more mblock header and data decoding
|
||
|
//
|
||
|
// Rev 1.2 20 Sep 1995 09:52:22 SCDAY
|
||
|
//
|
||
|
// eliminated a warning
|
||
|
//
|
||
|
// Rev 1.1 19 Sep 1995 15:24:10 SCDAY
|
||
|
//
|
||
|
// added H261 MBA parsing
|
||
|
//
|
||
|
// Rev 1.0 11 Sep 1995 13:51:52 SCDAY
|
||
|
// Initial revision.
|
||
|
//
|
||
|
// Rev 1.11 25 Aug 1995 09:16:32 DBRUCKS
|
||
|
// add ifdef DEBUG_MBLK
|
||
|
//
|
||
|
// Rev 1.10 23 Aug 1995 19:12:02 AKASAI
|
||
|
// Fixed gNewTAB_CBPY table building. Was using 8 as mask instead of 0xf.
|
||
|
//
|
||
|
// Rev 1.9 18 Aug 1995 15:03:22 CZHU
|
||
|
//
|
||
|
// Output more error message when DecodeBlock returns error.
|
||
|
//
|
||
|
// Rev 1.8 16 Aug 1995 14:26:54 CZHU
|
||
|
//
|
||
|
// Changed DWORD adjustment back to byte oriented reading.
|
||
|
//
|
||
|
// Rev 1.7 15 Aug 1995 09:54:18 DBRUCKS
|
||
|
// improve stuffing handling and add debug msg
|
||
|
//
|
||
|
// Rev 1.6 14 Aug 1995 18:00:40 DBRUCKS
|
||
|
// add chroma parsing
|
||
|
//
|
||
|
// Rev 1.5 11 Aug 1995 17:47:58 DBRUCKS
|
||
|
// cleanup
|
||
|
//
|
||
|
// Rev 1.4 11 Aug 1995 16:12:28 DBRUCKS
|
||
|
// add ptr check to MB data
|
||
|
//
|
||
|
// Rev 1.3 11 Aug 1995 15:10:58 DBRUCKS
|
||
|
// finish INTRA mb header parsing and callblock
|
||
|
//
|
||
|
// Rev 1.2 03 Aug 1995 14:30:26 CZHU
|
||
|
// Take block level operations out to d3block.cpp
|
||
|
//
|
||
|
// Rev 1.1 02 Aug 1995 10:21:12 CZHU
|
||
|
// Added asm codes for VLD of TCOEFF, inverse quantization, run-length decode.
|
||
|
//
|
||
|
// Rev 1.0 31 Jul 1995 13:00:08 DBRUCKS
|
||
|
// Initial revision.
|
||
|
//
|
||
|
// Rev 1.2 31 Jul 1995 11:45:42 CZHU
|
||
|
// changed the parameter list
|
||
|
//
|
||
|
// Rev 1.1 28 Jul 1995 16:25:52 CZHU
|
||
|
//
|
||
|
// Added per block decoding framework.
|
||
|
//
|
||
|
// Rev 1.0 28 Jul 1995 15:20:16 CZHU
|
||
|
// Initial revision.
|
||
|
|
||
|
//Block level decoding for H.26x decoder
|
||
|
|
||
|
#include "precomp.h" // rearch idct
|
||
|
|
||
|
/*****************************************************************************
|
||
|
*
|
||
|
* GET_VAR_BITS
|
||
|
*
|
||
|
* Read a variable number of bits using a lookup table.
|
||
|
*
|
||
|
* The input count should be the number of bits used to index the table.
|
||
|
* The output count is the number of bits in that symbol.
|
||
|
*
|
||
|
* The table should be initialized such that all don't care symbols match to
|
||
|
* the same value. Thus if the table is indexed by 6-bits a two bit symbol
|
||
|
* 01XX XX will be used to initialize all entries 0100 00 -> 0111 11. These
|
||
|
* entries will include an 8-bit length in the least significant byte.
|
||
|
*
|
||
|
* uCount - IN
|
||
|
* fpu8 - IN and OUT
|
||
|
* uWork - IN and OUT
|
||
|
* uBitsReady - IN and OUT
|
||
|
* uResult - OUT
|
||
|
* uCode - OUT
|
||
|
* fpTable - IN
|
||
|
*/
|
||
|
|
||
|
#define GET_VAR_BITS(uCount, fpu8, uWork, uBitsReady, uResult, uCode, uBitCount, fpTable) { \
|
||
|
while (uBitsReady < uCount) { \
|
||
|
uWork <<= 8; \
|
||
|
uBitsReady += 8; \
|
||
|
uWork |= *fpu8++; \
|
||
|
} \
|
||
|
/* calculate how much to shift off */ \
|
||
|
/* and get the code */ \
|
||
|
uCode = uBitsReady - uCount; \
|
||
|
uCode = (uWork >> uCode); \
|
||
|
/* read the data */ \
|
||
|
uResult = fpTable[uCode]; \
|
||
|
/* count of bits used */ \
|
||
|
/* !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! */ \
|
||
|
/* H.261 tables are reverse order from H.263 */ \
|
||
|
/* !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! */ \
|
||
|
uBitCount = uResult & 0xff00; \
|
||
|
uBitCount >>= 8; \
|
||
|
/* bits remaining */ \
|
||
|
uBitsReady = uBitsReady - uBitCount; \
|
||
|
/* special case for stuffing processing */ \
|
||
|
/* if (uBitsReady < 0) */ \
|
||
|
/* kluged to test for negative */ \
|
||
|
if (uBitsReady > 33) \
|
||
|
/* if (bStuffing) */ \
|
||
|
{ \
|
||
|
uWork <<= 8; \
|
||
|
uBitsReady += 8; \
|
||
|
uWork |= *fpu8++; \
|
||
|
} \
|
||
|
/* end special case for stuffing */ \
|
||
|
uWork &= GetBitsMask[uBitsReady]; \
|
||
|
}
|
||
|
|
||
|
#define GET_GT8_BITS(uCount, fpu8, uWork, uBitsReady, uResult, uCode, uBitCount, fpTable) { \
|
||
|
while (uBitsReady < uCount) { \
|
||
|
uWork <<= 8; \
|
||
|
uBitsReady += 8; \
|
||
|
uWork |= *fpu8++; \
|
||
|
} \
|
||
|
/* calculate how much to shift off */ \
|
||
|
/* and get the code */ \
|
||
|
uCode = uBitsReady - uCount; \
|
||
|
uCode = (uWork >> uCode); \
|
||
|
/* read the data */ \
|
||
|
uResult = fpTable[uCode]; \
|
||
|
/* count of bits used */ \
|
||
|
/* !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! */ \
|
||
|
/* H.261 tables are reverse order from H.263 */ \
|
||
|
/* !!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!!! */ \
|
||
|
uBitCount = uResult & 0xff00; \
|
||
|
if ((uBitCount & 0x8000) == 0) /* if not negative */ \
|
||
|
{ \
|
||
|
uBitCount >>= 8; \
|
||
|
/* bits remaining */ \
|
||
|
uBitsReady = uBitsReady - uBitCount; \
|
||
|
/* special case for stuffing processing */ \
|
||
|
/* if (uBitsReady < 0) */ \
|
||
|
/* kluged to test for negative */ \
|
||
|
if (uBitsReady > 33) \
|
||
|
/* if (bStuffing) */ \
|
||
|
{ \
|
||
|
uWork <<= 8; \
|
||
|
uBitsReady += 8; \
|
||
|
uWork |= *fpu8++; \
|
||
|
} \
|
||
|
/* end special case for stuffing */ \
|
||
|
uWork &= GetBitsMask[uBitsReady]; \
|
||
|
} \
|
||
|
else \
|
||
|
uWork &= GetBitsMask[uBitsReady-8]; \
|
||
|
}
|
||
|
|
||
|
extern void BlockCopy(
|
||
|
U32 uDstBlock,
|
||
|
U32 uSrcBlock);
|
||
|
|
||
|
extern void BlockCopySpecial(
|
||
|
U32 uDstBlock,
|
||
|
U32 uSrcBlock);
|
||
|
|
||
|
extern void BlockAdd (
|
||
|
U32 uResidual,
|
||
|
U32 uRefBlock,
|
||
|
U32 uDstBlock);
|
||
|
|
||
|
extern void BlockAddSpecial (
|
||
|
U32 uResidual,
|
||
|
U32 uRefBlock,
|
||
|
U32 uDstBlock);
|
||
|
|
||
|
T_pFunc_VLD_RLD_IQ_Block pFunc_VLD_RLD_IQ_Block[2] = {VLD_RLD_IQ_Block,VLD_RLD_IQ_Block}; // New rearch
|
||
|
//T_pFunc_VLD_RLD_IQ_Block pFunc_VLD_RLD_IQ_Block[2] = {VLD_RLD_IQ_Block, MMX_VLD_RLD_IQ_Block}; // New rearch
|
||
|
|
||
|
/*****************************************************************************
|
||
|
*
|
||
|
* H263DecodeMBHeader
|
||
|
*
|
||
|
* Decode the MB header
|
||
|
*/
|
||
|
#pragma code_seg("IACODE1")
|
||
|
I32 H263DecodeMBHeader(
|
||
|
T_H263DecoderCatalog FAR * DC,
|
||
|
BITSTREAM_STATE FAR * fpbsState,
|
||
|
U32 * uReadChecksum)
|
||
|
{
|
||
|
I32 iReturn = ICERR_ERROR;
|
||
|
U8 FAR * fpu8;
|
||
|
U32 uBitsReady;
|
||
|
U32 uWork;
|
||
|
U32 uResult;
|
||
|
U32 uCode;
|
||
|
U32 uBitCount;
|
||
|
int bStuffing;
|
||
|
|
||
|
#define START_CODE 0xff18
|
||
|
#define STUFFING_CODE 0x0b22
|
||
|
//#define DEBUG_MBLK -- Turn this on with a define in the makefile.
|
||
|
|
||
|
#ifndef RING0
|
||
|
#ifdef DEBUG_MBLK
|
||
|
char buf120[120];
|
||
|
int iLength;
|
||
|
#endif
|
||
|
#endif
|
||
|
|
||
|
GET_BITS_RESTORE_STATE(fpu8, uWork, uBitsReady, fpbsState)
|
||
|
|
||
|
/* MBA --------------- */
|
||
|
/* ********************************************* */
|
||
|
/* minor table decode (>8 bits) not fully tested */
|
||
|
/* to do note: */
|
||
|
/* this is hacked */
|
||
|
/* change >8 bit processing to use major/minor */
|
||
|
/* tables and ONE GET_BITS routine */
|
||
|
/* ********************************************* */
|
||
|
|
||
|
ReadMBA:
|
||
|
bStuffing = 0;
|
||
|
GET_GT8_BITS(8, fpu8, uWork, uBitsReady, uResult,
|
||
|
uCode, uBitCount, gTAB_MBA_MAJOR);
|
||
|
|
||
|
if (uResult == STUFFING_CODE)
|
||
|
{ /* is stuffing code */
|
||
|
bStuffing = 1;
|
||
|
/* do MB checksum stuff here */
|
||
|
#ifdef CHECKSUM_MACRO_BLOCK
|
||
|
GET_BITS_SAVE_STATE(fpu8, uWork, uBitsReady, fpbsState)
|
||
|
/* might want to move this to a separate function for readability */
|
||
|
GET_ONE_BIT(fpu8, uWork, uBitsReady, uResult);
|
||
|
if (uResult == 1)
|
||
|
{
|
||
|
GET_FIXED_BITS(8, fpu8, uWork, uBitsReady, uResult);
|
||
|
if (uResult == 1)
|
||
|
{ /* indicates TCOEFF checksum processing */
|
||
|
/* read off all but the real checksum data */
|
||
|
GET_FIXED_BITS(9, fpu8, uWork, uBitsReady, uResult);
|
||
|
GET_FIXED_BITS(9, fpu8, uWork, uBitsReady, uResult);
|
||
|
GET_FIXED_BITS(9, fpu8, uWork, uBitsReady, uResult);
|
||
|
GET_FIXED_BITS(9, fpu8, uWork, uBitsReady, uResult);
|
||
|
GET_FIXED_BITS(9, fpu8, uWork, uBitsReady, uResult);
|
||
|
GET_FIXED_BITS(9, fpu8, uWork, uBitsReady, uResult);
|
||
|
|
||
|
/* now get real checksum data */
|
||
|
/* run */
|
||
|
GET_FIXED_BITS(9, fpu8, uWork, uBitsReady, uResult);
|
||
|
*uReadChecksum = ((uResult & 0xff) << 24);
|
||
|
/* level */
|
||
|
GET_FIXED_BITS(9, fpu8, uWork, uBitsReady, uResult);
|
||
|
*uReadChecksum = (*uReadChecksum | ((uResult & 0xff) << 16));
|
||
|
GET_FIXED_BITS(9, fpu8, uWork, uBitsReady, uResult);
|
||
|
*uReadChecksum = (*uReadChecksum | ((uResult & 0xff) << 8));
|
||
|
/* sign */
|
||
|
GET_FIXED_BITS(9, fpu8, uWork, uBitsReady, uResult);
|
||
|
*uReadChecksum = (*uReadChecksum | (uResult & 0xff));
|
||
|
GET_ONE_BIT(fpu8, uWork, uBitsReady, uResult);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
DBOUT("ERROR :: H261MBChecksum :: Invalid Checksum Data :: ERROR");
|
||
|
iReturn = ICERR_ERROR;
|
||
|
goto done;
|
||
|
}
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
GET_BITS_RESTORE_STATE(fpu8, uWork, uBitsReady, fpbsState)
|
||
|
goto ReadMBA;
|
||
|
}
|
||
|
|
||
|
#else /* is MBA stuffing, but checksum not enabled */
|
||
|
GET_BITS_SAVE_STATE(fpu8, uWork, uBitsReady, fpbsState)
|
||
|
|
||
|
// GET_ONE_BIT(fpu8, uWork, uBitsReady, uResult);
|
||
|
/*if (uResult == 1) {
|
||
|
DbgLog((LOG_ERROR, HDBG_ALWAYS, TEXT("ERROR :: Stuffing code found, Checksum not enabled :: ERROR")));
|
||
|
iReturn = ICERR_ERROR;
|
||
|
goto done;
|
||
|
} */
|
||
|
// if (uResult == 1)
|
||
|
// {
|
||
|
// GET_BITS_RESTORE_STATE(fpu8, uWork, uBitsReady, fpbsState)
|
||
|
|
||
|
// }
|
||
|
// else {
|
||
|
//GET_BITS_RESTORE_STATE(fpu8, uWork, uBitsReady, fpbsState)
|
||
|
// }
|
||
|
#endif
|
||
|
} /* end if (uResult == STUFFING_CODE) */
|
||
|
/* try this for now */
|
||
|
else
|
||
|
{
|
||
|
if (uResult == START_CODE)
|
||
|
{
|
||
|
I8 temp;
|
||
|
temp = (I8)(uResult & 0xff);
|
||
|
GET_VAR_BITS(16, fpu8, uWork, uBitsReady, uResult, uCode, uBitCount, (gTAB_MBA_MINOR + temp));
|
||
|
if (uResult != 0x1023)
|
||
|
{
|
||
|
DBOUT("ERROR :: Invalid startcode :: ERROR");
|
||
|
iReturn = ICERR_ERROR;
|
||
|
}
|
||
|
else
|
||
|
iReturn = START_CODE;
|
||
|
|
||
|
GET_BITS_SAVE_STATE(fpu8, uWork, uBitsReady, fpbsState)
|
||
|
goto done;
|
||
|
} /* end if (uResult == START_CODE) */
|
||
|
else /* is not stuffing */
|
||
|
{ /* if uResult negative, get more bits */
|
||
|
if (uResult & 0x8000)
|
||
|
{
|
||
|
I8 temp;
|
||
|
temp = (I8)(uResult & 0xff);
|
||
|
GET_VAR_BITS(11, fpu8, uWork, uBitsReady, uResult, uCode, uBitCount, (gTAB_MBA_MINOR + temp));
|
||
|
}
|
||
|
DC->uMBA = (uResult & 0xff);
|
||
|
}/* end else is not stuffing */
|
||
|
}
|
||
|
|
||
|
/* When MBA==Stuffing, we jump back to the start to look for MBA */
|
||
|
|
||
|
if (bStuffing)
|
||
|
goto ReadMBA;
|
||
|
|
||
|
|
||
|
/* MTYPE ---------------------------------------- */
|
||
|
GET_GT8_BITS(8, fpu8, uWork, uBitsReady, uResult,
|
||
|
uCode, uBitCount, gTAB_MTYPE_MAJOR);
|
||
|
if (uResult & 0x8000)
|
||
|
{
|
||
|
I8 temp;
|
||
|
temp = (I8)(uResult & 0xff);
|
||
|
GET_VAR_BITS(10, fpu8, uWork, uBitsReady, uResult,
|
||
|
uCode, uBitCount, (gTAB_MTYPE_MINOR + temp));
|
||
|
}
|
||
|
DC->uMBType = (uResult & 0xff);
|
||
|
|
||
|
/* MQUANT ---------------------------------------- */
|
||
|
if (DC->uMBType == 1 || DC->uMBType == 3 || DC->uMBType == 6 || DC->uMBType == 9)
|
||
|
{ /* get 5-bit MQuant */
|
||
|
GET_FIXED_BITS(5, fpu8, uWork, uBitsReady, uResult);
|
||
|
DC->uMQuant = (uResult & 0xff);
|
||
|
}
|
||
|
|
||
|
/* MVD ------------------------------------------- */
|
||
|
/* reset previous motion vectors */
|
||
|
/* if MB 0,11,22 */
|
||
|
/* if MBA != 1 or */
|
||
|
/* if previous MB was not MC */
|
||
|
|
||
|
if (DC->uMBType >3)
|
||
|
{
|
||
|
if ((DC->uMBA != 1) || (DC->i16LastMBA == 10) || (DC->i16LastMBA == 21))
|
||
|
DC->i8MVDH = DC->i8MVDV = 0;
|
||
|
/* get X motion vector */
|
||
|
GET_GT8_BITS(8, fpu8, uWork, uBitsReady, uResult,
|
||
|
uCode, uBitCount, gTAB_MVD_MAJOR);
|
||
|
if (uResult & 0x8000)
|
||
|
{
|
||
|
I8 temp;
|
||
|
temp = (I8)(uResult & 0xff);
|
||
|
GET_VAR_BITS(11, fpu8, uWork, uBitsReady, uResult,
|
||
|
uCode, uBitCount, (gTAB_MVD_MINOR + temp));
|
||
|
}
|
||
|
/* convert and make incremental */
|
||
|
DC->i8MVDH = gTAB_MV_ADJUST[DC->i8MVDH + (I8)(uResult & 0xff) + 32];
|
||
|
/* get Y motion vector */
|
||
|
GET_GT8_BITS(8, fpu8, uWork, uBitsReady, uResult,
|
||
|
uCode, uBitCount, gTAB_MVD_MAJOR);
|
||
|
if (uResult & 0x8000)
|
||
|
{
|
||
|
I8 temp;
|
||
|
temp = (I8)(uResult & 0xff);
|
||
|
GET_VAR_BITS(11, fpu8, uWork, uBitsReady, uResult,
|
||
|
uCode, uBitCount, (gTAB_MVD_MINOR + temp));
|
||
|
}
|
||
|
/* convert and make incremental */
|
||
|
DC->i8MVDV = gTAB_MV_ADJUST[DC->i8MVDV + (I8)(uResult & 0xff) + 32];
|
||
|
} /* end if (DC->MBType > 3) */
|
||
|
else
|
||
|
DC->i8MVDH = DC->i8MVDV = 0;
|
||
|
|
||
|
/* CBP --------------------------------------------- */
|
||
|
/* brute force method */
|
||
|
DC->uCBP = 0; /* for MType = 4 or 7 */
|
||
|
if (DC->uMBType == 2 || DC->uMBType == 3 || DC->uMBType == 5 || DC->uMBType == 6 || DC->uMBType == 8 || DC->uMBType == 9)
|
||
|
{ /* get CBP */
|
||
|
GET_VAR_BITS(9, fpu8, uWork, uBitsReady, uResult,
|
||
|
uCode, uBitCount, gTAB_CBP);
|
||
|
DC->uCBP = (uResult & 0xff);
|
||
|
} /* end get CBP */
|
||
|
else
|
||
|
if (DC->uMBType < 2) /* is intra */
|
||
|
DC->uCBP = 63; /* force CBP to 63 */
|
||
|
|
||
|
GET_BITS_SAVE_STATE(fpu8, uWork, uBitsReady, fpbsState)
|
||
|
|
||
|
iReturn = ICERR_OK;
|
||
|
|
||
|
#ifndef RING0
|
||
|
#ifdef DEBUG_MBLK
|
||
|
iLength = wsprintf(buf120, "MBType=%ld MQuant=%ld MVDH=%ld MVDV=%ld CBP=%ld",
|
||
|
DC->uMBType,
|
||
|
DC->uMQuant,
|
||
|
DC->i8MVDH,
|
||
|
DC->i8MVDV,
|
||
|
DC->uCBP);
|
||
|
DBOUT(buf120);
|
||
|
ASSERT(iLength < 120);
|
||
|
#endif
|
||
|
#endif
|
||
|
|
||
|
done:
|
||
|
return iReturn;
|
||
|
} /* end H263DecodeMBHeader() */
|
||
|
|
||
|
#pragma code_seg()
|
||
|
/*****************************************************************************
|
||
|
*
|
||
|
* H263DecodeMBData
|
||
|
*
|
||
|
* Decode each of the blocks in this macro block
|
||
|
*/
|
||
|
#pragma code_seg("IACODE1")
|
||
|
I32 H263DecodeMBData(
|
||
|
T_H263DecoderCatalog FAR * DC,
|
||
|
T_BlkAction FAR * fpBlockAction,
|
||
|
I32 iBlockNumber,
|
||
|
BITSTREAM_STATE FAR * fpbsState,
|
||
|
U8 FAR * fpu8MaxPtr,
|
||
|
U32 * uReadChecksum,
|
||
|
U32 **pN, // New rearch
|
||
|
T_IQ_INDEX ** pRUN_INVERSE_Q) // New rearch
|
||
|
{
|
||
|
|
||
|
I32 iResult = ICERR_ERROR;
|
||
|
int iCBP = (int) DC->uCBP;
|
||
|
int i;
|
||
|
U32 uBitsReady;
|
||
|
U32 uBitsReadIn;
|
||
|
U32 uBitsReadOut;
|
||
|
U8 u8Quant; /* quantization level for this block */
|
||
|
U8 FAR * fpu8;
|
||
|
U32 uByteCnt;
|
||
|
I8 mvx, mvy, mvx2, mvy2;
|
||
|
|
||
|
T_pFunc_VLD_RLD_IQ_Block pFunc_VLD =pFunc_VLD_RLD_IQ_Block[0];
|
||
|
|
||
|
U32 uCheckSum; /* checksum data returned from DecodeBlock */
|
||
|
#ifdef CHECKSUM_MACRO_BLOCK_DETAIL
|
||
|
char buf80[80];
|
||
|
int iMBDLength;
|
||
|
#endif
|
||
|
|
||
|
#ifdef CHECKSUM_MACRO_BLOCK
|
||
|
char buff80[80];
|
||
|
int iLength;
|
||
|
#endif
|
||
|
|
||
|
#ifdef DECODE_STATS
|
||
|
U32 uStartLow = DC->uStartLow;
|
||
|
U32 uStartHigh = DC->uStartHigh;
|
||
|
U32 uElapsed;
|
||
|
U32 uBefore;
|
||
|
U32 uDecodeBlockSum = 0;
|
||
|
U32 uLoopFilterSum = 0;
|
||
|
U32 uBlockCopySum = 0;
|
||
|
U32 uBlockCopySpSum = 0;
|
||
|
U32 uBlockAddSum = 0;
|
||
|
U32 uBlockAddSpSum = 0;
|
||
|
int bTimingThisFrame = DC->bTimingThisFrame;
|
||
|
DEC_TIMING_INFO * pDecTimingInfo = NULL;
|
||
|
#endif
|
||
|
|
||
|
/* On input the pointer points to the next byte. */
|
||
|
/* We need to change it to */
|
||
|
/* point to the current word on a 32-bit boundary. */
|
||
|
|
||
|
fpu8 = fpbsState->fpu8 - 1; /* point to the current byte */
|
||
|
uBitsReady = fpbsState->uBitsReady;
|
||
|
while (uBitsReady >= 8) {
|
||
|
fpu8--;
|
||
|
uBitsReady -= 8;
|
||
|
}
|
||
|
uBitsReadIn = 8 - uBitsReady;
|
||
|
|
||
|
u8Quant = (U8) (DC->uMQuant);
|
||
|
|
||
|
if (DC->uMBType > 1)
|
||
|
{
|
||
|
/* calculate motion vectors */
|
||
|
mvx = DC->i8MVDH;
|
||
|
mvy = DC->i8MVDV;
|
||
|
// calculate UV blocks MV
|
||
|
mvx2 = mvx / 2;
|
||
|
mvy2 = mvy / 2;
|
||
|
|
||
|
fpBlockAction->i8MVX = mvx;
|
||
|
fpBlockAction->i8MVY = mvy;
|
||
|
// duplicate other 3 Y blocks
|
||
|
fpBlockAction[1].i8MVX = mvx;
|
||
|
fpBlockAction[1].i8MVY = mvy;
|
||
|
fpBlockAction[2].i8MVX = mvx;
|
||
|
fpBlockAction[2].i8MVY = mvy;
|
||
|
fpBlockAction[3].i8MVX = mvx;
|
||
|
fpBlockAction[3].i8MVY = mvy;
|
||
|
// init UV blocks
|
||
|
fpBlockAction[4].i8MVX = mvx2;
|
||
|
fpBlockAction[4].i8MVY = mvy2;
|
||
|
fpBlockAction[5].i8MVX = mvx2;
|
||
|
fpBlockAction[5].i8MVY = mvy2;
|
||
|
}
|
||
|
|
||
|
uCheckSum = 0; /* Init MB Checksum */
|
||
|
|
||
|
for (i = 0; i < 6; i++)
|
||
|
{
|
||
|
if (DC->uMBType <= 1) /* is intra */
|
||
|
fpBlockAction->u8BlkType = BT_INTRA;
|
||
|
else
|
||
|
if (iCBP & 0x20) /* if coded */
|
||
|
fpBlockAction->u8BlkType = BT_INTER;
|
||
|
else
|
||
|
fpBlockAction->u8BlkType = BT_EMPTY;
|
||
|
|
||
|
if (fpBlockAction->u8BlkType != BT_EMPTY)
|
||
|
{
|
||
|
fpBlockAction->u8Quant = u8Quant;
|
||
|
ASSERT(fpBlockAction->pCurBlock != NULL);
|
||
|
ASSERT(fpBlockAction->uBlkNumber == (U32)iBlockNumber);
|
||
|
|
||
|
/*----- DecodeBlock ----*/
|
||
|
#ifdef DECODE_STATS
|
||
|
TIMER_BEFORE(bTimingThisFrame,uStartLow,uStartHigh,uBefore);
|
||
|
#endif
|
||
|
#ifdef CHECKSUM_MACRO_BLOCK
|
||
|
// uBitsReadOut = DecodeBlock(fpBlockAction, fpu8, uBitsReadIn, (U32)DC+DC->uMBBuffer+i*256, fpBlockAction->pCurBlock, &uCheckSum);
|
||
|
#else
|
||
|
// rearch
|
||
|
uBitsReadOut = (*pFunc_VLD) ( fpBlockAction,
|
||
|
fpu8,
|
||
|
uBitsReadIn,
|
||
|
(U32 *) *pN,
|
||
|
(U32 *) *pRUN_INVERSE_Q);
|
||
|
// rearch
|
||
|
#endif
|
||
|
#ifdef DECODE_STATS
|
||
|
TIMER_AFTER_P5(bTimingThisFrame,uStartLow,uStartHigh,uBefore,uElapsed,uDecodeBlockSum)
|
||
|
#endif
|
||
|
|
||
|
if (uBitsReadOut == 0)
|
||
|
{
|
||
|
DBOUT("ERROR :: H263DecodeMBData :: Error decoding a Y block :: ERROR");
|
||
|
DBOUT("ERROR :: DecodeBlock return 0 bits read....");
|
||
|
goto done;
|
||
|
}
|
||
|
uByteCnt = uBitsReadOut >> 3; /* divide by 8 */
|
||
|
uBitsReadIn = uBitsReadOut & 0x7; /* mod 8 */
|
||
|
fpu8 += uByteCnt;
|
||
|
|
||
|
/* New for rearch */
|
||
|
ASSERT ( **pN < 65 );
|
||
|
////////////////////////////////////////////////
|
||
|
// End hack //
|
||
|
////////////////////////////////////////////////
|
||
|
|
||
|
*pRUN_INVERSE_Q += **pN;
|
||
|
if ((0xf & fpBlockAction->u8BlkType) != BT_INTER)
|
||
|
**pN += 65;
|
||
|
(*pN)++;
|
||
|
/* end of new rearch */
|
||
|
|
||
|
/* allow the pointer to address up to four beyond */
|
||
|
/* the end reading by DWORD using postincrement. */
|
||
|
/* changed for detection of stuffing code at */
|
||
|
/* end of frame */
|
||
|
// ASSERT(fpu8 <= (fpu8MaxPtr+14));
|
||
|
|
||
|
if (fpu8 > (fpu8MaxPtr+14))
|
||
|
{
|
||
|
iResult = ICERR_ERROR; // probably not needed
|
||
|
goto done;
|
||
|
}
|
||
|
|
||
|
} /* end if not empty */
|
||
|
else /* is empty */
|
||
|
{ /* zero out intermediate data structure and advance pointers */
|
||
|
|
||
|
/* New for rearch */
|
||
|
**pN = 0;
|
||
|
(*pN)++;
|
||
|
/* end of new rearch */
|
||
|
}
|
||
|
|
||
|
fpBlockAction++;
|
||
|
iCBP <<= 1;
|
||
|
iBlockNumber++;
|
||
|
} /* end for each block in macroblock */
|
||
|
|
||
|
#ifdef CHECKSUM_MACRO_BLOCK
|
||
|
/* Compare checksum */
|
||
|
if ((uCheckSum != *uReadChecksum) && (*uReadChecksum != 0))
|
||
|
{
|
||
|
iLength = wsprintf(buff80,"WARNING:MB CheckSum miss match, Enc Checksum=0x%x Dec Checksum=0x%x",
|
||
|
*uReadChecksum, uCheckSum);
|
||
|
DBOUT(buff80);
|
||
|
ASSERT(iLength < 80);
|
||
|
}
|
||
|
#ifdef CHECKSUM_MACRO_BLOCK_DETAIL
|
||
|
iMBDLength = wsprintf(buf80,"Block=%d CheckSum=0x%x", i, uCheckSum);
|
||
|
DBOUT(buf80);
|
||
|
ASSERT(iMBDLength < 80);
|
||
|
#endif
|
||
|
#endif
|
||
|
|
||
|
/* restore the scanning pointers to point to the next byte */
|
||
|
/* and set the uWork and uBitsReady values. */
|
||
|
while (uBitsReadIn > 8)
|
||
|
{
|
||
|
fpu8++;
|
||
|
uBitsReadIn -= 8;
|
||
|
}
|
||
|
fpbsState->uBitsReady = 8 - uBitsReadIn;
|
||
|
fpbsState->uWork = *fpu8++; /* store the data and point to next byte */
|
||
|
fpbsState->uWork &= GetBitsMask[fpbsState->uBitsReady];
|
||
|
fpbsState->fpu8 = fpu8;
|
||
|
|
||
|
#ifdef DECODE_STATS
|
||
|
if (bTimingThisFrame)
|
||
|
{
|
||
|
pDecTimingInfo = DC->pDecTimingInfo + DC->uStatFrameCount;
|
||
|
pDecTimingInfo->uDecodeBlock += uDecodeBlockSum;
|
||
|
pDecTimingInfo->uLoopFilter += uLoopFilterSum;
|
||
|
pDecTimingInfo->uBlockCopy += uBlockCopySum;
|
||
|
pDecTimingInfo->uBlockCopySp += uBlockCopySpSum;
|
||
|
pDecTimingInfo->uBlockAdd += uBlockAddSum;
|
||
|
pDecTimingInfo->uBlockAddSp += uBlockAddSpSum;
|
||
|
}
|
||
|
#endif
|
||
|
|
||
|
iResult = ICERR_OK;
|
||
|
|
||
|
done:
|
||
|
return iResult;
|
||
|
} /* H263DecodeMBData() */
|
||
|
#pragma code_seg()
|
||
|
|
||
|
/*****************************************************************************
|
||
|
*
|
||
|
* H263IDCTandMC
|
||
|
*
|
||
|
* Inverse Discrete Cosine Transform and
|
||
|
* Motion Compensation for each block
|
||
|
*
|
||
|
*/
|
||
|
|
||
|
#pragma code_seg("IACODE2")
|
||
|
void H263IDCTandMC(
|
||
|
T_H263DecoderCatalog FAR *DC,
|
||
|
T_BlkAction FAR *fpBlockAction,
|
||
|
int iBlock,
|
||
|
int iMBNum, // AP-NEW
|
||
|
int iGOBNum, // AP-NEW
|
||
|
U32 *pN,
|
||
|
T_IQ_INDEX *pRUN_INVERSE_Q,
|
||
|
T_MBInfo *fpMBInfo, // AP-NEW
|
||
|
int iEdgeFlag
|
||
|
)
|
||
|
{
|
||
|
I32 pRef;
|
||
|
I32 mvx, mvy;
|
||
|
|
||
|
ASSERT(*pN != 65);
|
||
|
|
||
|
if (*pN < 65) // Inter block
|
||
|
{
|
||
|
|
||
|
// first do motion compensation
|
||
|
// result will be pointed to by pRef
|
||
|
|
||
|
mvx = fpBlockAction[iBlock].i8MVX;
|
||
|
mvy = fpBlockAction[iBlock].i8MVY;
|
||
|
|
||
|
pRef = fpBlockAction[iBlock].pRefBlock + (I32) mvx + PITCH * (I32) mvy;
|
||
|
|
||
|
|
||
|
// now do the inverse transform (where appropriate) & combine
|
||
|
if (*pN > 0) // and, of course, < 65.
|
||
|
{
|
||
|
// Get residual block; output at DC+DC->uMBBuffer+BLOCK_BUFFER_OFFSET
|
||
|
// Finally add the residual to the reference block
|
||
|
// TODO
|
||
|
|
||
|
DecodeBlock_IDCT(
|
||
|
(U32)pRUN_INVERSE_Q,
|
||
|
*pN,
|
||
|
fpBlockAction[iBlock].pCurBlock, // not used here
|
||
|
(U32) DC + DC->uMBBuffer + BLOCK_BUFFER_OFFSET);// inter output
|
||
|
|
||
|
if (fpMBInfo->i8MBType >=7)
|
||
|
{
|
||
|
// do spatial loop filter
|
||
|
LoopFilter((U8 *)pRef, (U8*)DC+DC->uFilterBBuffer, PITCH);
|
||
|
|
||
|
BlockAddSpecial((U32)DC+DC->uMBBuffer + BLOCK_BUFFER_OFFSET,
|
||
|
(U32)DC+DC->uFilterBBuffer,
|
||
|
fpBlockAction[iBlock].pCurBlock);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
BlockAdd(
|
||
|
(U32) DC + DC->uMBBuffer + BLOCK_BUFFER_OFFSET, // output
|
||
|
pRef, // prediction
|
||
|
fpBlockAction[iBlock].pCurBlock); // destination
|
||
|
}
|
||
|
|
||
|
}
|
||
|
else // *pN == 0, so no transform coefficients for this block
|
||
|
{
|
||
|
// Just copy motion compensated reference block
|
||
|
|
||
|
if (fpMBInfo->i8MBType >=7)
|
||
|
{
|
||
|
// do spatial loop filter
|
||
|
LoopFilter((U8 *)pRef, (U8*)DC+DC->uFilterBBuffer, PITCH);
|
||
|
//MMX_LoopFilter((U8 *)pRef, (U8*)DC+DC->uFilterBBuffer, 8);
|
||
|
|
||
|
BlockCopySpecial(fpBlockAction[iBlock].pCurBlock,
|
||
|
(U32)DC+DC->uFilterBBuffer);
|
||
|
}
|
||
|
else
|
||
|
|
||
|
BlockCopy(
|
||
|
fpBlockAction[iBlock].pCurBlock, // destination
|
||
|
pRef); // prediction
|
||
|
|
||
|
}
|
||
|
|
||
|
}
|
||
|
else // *pN >= 65, hence intRA
|
||
|
{
|
||
|
// TODO
|
||
|
|
||
|
DecodeBlock_IDCT(
|
||
|
(U32)pRUN_INVERSE_Q,
|
||
|
*pN,
|
||
|
fpBlockAction[iBlock].pCurBlock, // intRA transform output
|
||
|
(U32) DC + DC->uMBBuffer + BLOCK_BUFFER_OFFSET);
|
||
|
} // end if (*pN < 65) ... else ...
|
||
|
|
||
|
}
|
||
|
// End IDCTandMC
|
||
|
////////////////////////////////////////////////////////////////////////////////
|
||
|
#pragma code_seg()
|
||
|
|