2020-09-30 16:53:55 +02:00

3138 lines
102 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.
**
** *************************************************************************
*/
/*****************************************************************************
*
* d1dec.cpp
*
* DESCRIPTION:
* H261 decoder top level functions
*
* Routines: Prototypes in:
* H263InitDecoderGlobal d1dec.h
* H263InitDecoderInstance d1dec.h
* H263Decompress d1dec.h
* H263TermDecoderInstance d1dec.h
*/
// $Header: S:\h26x\src\dec\d1dec.cpv 1.69 24 Mar 1997 11:34:36 mbodart $
// $Log: S:\h26x\src\dec\d1dec.cpv $
//
// Rev 1.69 24 Mar 1997 11:34:36 mbodart
// Added check for PREROLL, if so don't display.
//
// Rev 1.68 19 Mar 1997 16:24:36 mbodart
// Fixed potential problem where aspect ratio adjustment to uNewOffsetToLine0
// should not occur for positive pitches.
//
// Rev 1.67 19 Mar 1997 15:01:46 mbodart
// Changes to DibXY to support RGB output with a negative bitmap height.
//
// Rev 1.66 24 Jan 1997 17:05:16 RHAZRA
// RTP change: we now look for an EBS for every frame. If there is one
// then we copy the H.261 bits and the EBS separately into our local
// bitstream buffer, inserting two zero bytes between the H261 bits and
// the EBS. We need the two zero bytes to mark the end of the frame for
// the pass 1 code. If there is no EBS, then we proceed as before by
// copying the bitstream and then adding two zero bytes at the end.
//
// Rev 1.65 22 Jan 1997 13:33:40 RHAZRA
// Since PPM now fills in the source format even for a PSC packet loss,
// the check for format change has been moved back into d1pict.cpp. This
// was how the check was initially designed in the pre-RTP era.
//
// Rev 1.64 23 Dec 1996 16:32:38 MBODART
// Fixed a bug where we allowed more than 33 macro blocks to be present
// in a GOB. Now we return an error in this case.
// Also removed some dead code involving mb_start.
//
// Rev 1.63 16 Dec 1996 14:41:08 RHAZRA
// Changed a bitstream error ASSERT to a bonafide error.
//
// Rev 1.62 16 Dec 1996 09:09:42 RHAZRA
// Now LOSS_RECOVERY mode is turned on by default in Pass 1
//
// Rev 1.61 12 Dec 1996 09:36:04 SCDAY
//
// Changed size of a couple of data structures in H263InitDecoderInstance
// to improve memory footprint
//
// Rev 1.60 18 Nov 1996 17:12:38 MBODART
// Replaced all debug message invocations with Active Movie's DbgLog.
//
// Rev 1.59 13 Nov 1996 11:35:56 RHAZRA
// Added MMX_autosensing.
//
// Rev 1.58 11 Nov 1996 11:03:28 MBODART
// Fixed bug where block action block type was not explicitly initialized for
// skipped macro blocks. This led to the block edge filter being used more
// often than needed.
//
// Rev 1.57 04 Nov 1996 08:43:18 RHAZRA
// Fixed setting MMX on or off via the INI file when the MMX key
// has an illegal value (<0 or > 1) assigned to it.
//
// Rev 1.56 31 Oct 1996 08:58:34 SCDAY
// Raj added support for MMX decoder
//
// Rev 1.55 30 Oct 1996 09:59:46 MBODART
// Fixed mirroring. Need to use absolute value of dst biWidth in most context
// Also made cosmetic changes to DibXY. It's identical to H.263's DibXY, we
// should probably put it into a common file.
//
// Rev 1.54 27 Sep 1996 14:59:32 MBODART
// DECODE_STATS enabled build will now compile, but numbers aren't accurate.
//
// Rev 1.53 26 Sep 1996 12:30:00 RHAZRA
// Added (i) MMX sensing in the decoder and ini file reading (requires a new
// "MMX" section in h263test.ini to turn off MMX on a MMX CPU) and (ii)
// MMX & PentiumPro CCs.
//
// Rev 1.52 25 Sep 1996 17:35:20 BECHOLS
//
// Added code just prior to color conversion that will perform the
// Snapshot copy on request.
//
// Rev 1.51 24 Sep 1996 13:52:24 RHAZRA
// Changed fpBlockAction synchronization to deal with MBAP being biased
// by -1 in the RTP extension.
//
// Rev 1.50 17 Sep 1996 22:08:36 RHAZRA
// Added code in RTP packet loss recovery to read GOB number from the
// bitstream when the packet following the lost packet starts with a
// GOB start code.
//
// Rev 1.49 16 Sep 1996 09:28:56 RHAZRA
// Fixed a bug in MB-level fragmentation recovery.
//
// Rev 1.48 12 Sep 1996 14:23:12 MBODART
// Replaced GlobalAlloc family with HeapAlloc in the H.261 decoder.
//
// Rev 1.47 10 Sep 1996 15:51:42 RHAZRA
// Bug fixes in RTP packet loss recovery when bad GBSC or MBA is
// detected in the PPM generated lost packet.
//
// Rev 1.45 04 Sep 1996 09:52:32 RHAZRA
// Added a new pass 1 function to enable RTp decoder resiliency when
// LOSS_RECOVERY is defined.
//
// Rev 1.44 14 Aug 1996 08:41:04 RHAZRA
//
// Added support for YUV12 and YUY2 color convertors
//
// Rev 1.43 09 Aug 1996 17:23:10 MBODART
// Fixed uninitialized variable bugs: one in decoder rearchitecture, where
// MB type needed to be defined for skipped blocks; and one previously
// existing bug where the block action u8BlkType needed to be defined
// for skip blocks, in order to suppress the BEF on those blocks.
// These bugs render build 027 of H.261 broken.
//
// Rev 1.42 05 Aug 1996 11:00:30 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.41 10 Jul 1996 08:20:44 SCDAY
// Increased memory allocation for I420
//
// Rev 1.40 03 Jun 1996 12:21:52 AKASAI
// Initialized DC = NULL and added tests so that don't try to free
// and unlock if DC == NULL. This effected the "done" return area
// of H263Decompress and one other place.
//
// Also added checking of return status from reading GOB start code.
//
// Rev 1.39 03 May 1996 15:54:26 AKASAI
// Eliminate allocating space for B frame in decoder. This frame is
// not used.
//
// Rev 1.38 17 Apr 1996 18:36:30 AKASAI
// Updates to use non-distructive color convertors.
// Color Convertor has modified parameter list.
// FrameCopy is called only when BlockEdgeFilter is enabled or
// AdjustPels is enabled or when mirroring is enabled.
// For H.261 bitstreams.
// A frame copy is used for YUV12 when mirroring is enabled or
// AdjustPels is enabled.
//
// Basically normal processing without BEF you don't have to do
// a frame copy which saves ~2msec per frame QCIF.
//
// Rev 1.37 05 Apr 1996 14:22:18 AKASAI
//
// Added support for BlockEdgeFilter.
// Need to change where ReInitializeBlockActionStream was called.
//
// Rev 1.36 21 Mar 1996 16:59:54 AKASAI
// Needed to move location of picture checksum calculation because
// of the swap of Previous and Current Frames.
//
// Rev 1.35 18 Mar 1996 15:52:06 AKASAI
// Many, many changes.
// 1) To optimize for performance eliminated memcpy of current to
// previous frame. Now switch the pointers and re-initialize
// block Action stream. New routine H263ReInitializeBlockActionStream
// written and called after each frame is compressed. This
// change accounted to 3-4 of the 4-5 msec improvment.
// 2) Needed to add call to BlockCopy (NOTE: maybe BlockCopySpecial would
// be faster) to copy any skip blocks at the end of a GOB from
// previous to current. Change was necessary after 1).
// 3) Deleted some dead code
// 4) Changed timing statistic code some.
//
// Rev 1.34 29 Feb 1996 09:20:30 SCDAY
// Added support for mirroring
//
// Rev 1.33 14 Feb 1996 11:54:26 AKASAI
// Update to use new color convertors that fix palette flash.
// Also corrected data alignment problem which improves performance
// of decoder.
//
// Rev 1.32 09 Feb 1996 13:33:36 AKASAI
//
// Updated interface to call new AdjustPels routine. CustomChange
// Brightness, Saturation and Contrast seem to be working but very
// little testing has been done.
//
// Rev 1.31 12 Jan 1996 15:12:34 AKASAI
// Fixed pink blocks in RING0 QCIF TO FCIF by fixing static initialzation
// of GOBUpdate arrays. Was based on input parameter but now on constant.
//
// Rev 1.30 11 Jan 1996 16:57:00 DBRUCKS
//
// added GetDecoderOptions
// added use of bUseBlockEdgeFilter
// added use of bForceOnAspectRatioCorrection
// Changed to do aspect ratio correction for both I420 and H261 if either
// forced or specified by result of the DecompressQuery
//
// Rev 1.29 26 Dec 1995 17:40:54 DBRUCKS
//
// changed bTimerIsOn to bTimingThisFrame because it is used after STOP_TIMER
// fixed YUV12 decode when timer ifdefs are defined
//
// Rev 1.28 26 Dec 1995 12:48:18 DBRUCKS
// remove TIMING code
// add general purpose timing code using d1stat.*
//
// Rev 1.26 21 Dec 1995 17:49:06 AKASAI
// Replaced an uninitialized variable to AdjustPels with the correct on.
// Change of Contrast, Brightness and Saturation is not working correctly.
//
// Rev 1.25 13 Dec 1995 14:23:52 AKASAI
// Deleted setting of Initialized to False; Added calling of H263TermDecoderIn
// if Initialized == True.
//
// Rev 1.24 05 Dec 1995 10:20:12 SCDAY
// Cleaned up warnings
//
// Rev 1.23 17 Nov 1995 15:21:48 BECHOLS
//
// Added ring 0 stuff.
//
// Rev 1.22 17 Nov 1995 15:13:18 SCDAY
//
// Added key field to picture checksum data
//
// Rev 1.21 16 Nov 1995 18:11:42 AGANTZX
// Added p5 timing code (#define TIMING)
//
// Rev 1.20 15 Nov 1995 19:04:12 AKASAI
// Should now be able to play raw YUV12 files. Note: funny white stop
// when I play downriv4.avi.
//
// Rev 1.19 15 Nov 1995 14:27:22 AKASAI
// Added support for YUV12 "if 0" old code with aspec correction and
// 8 to 7 bit conversion. Added FrameCopy calls and DispFrame into structure.
// (Integration point)
//
// Rev 1.18 08 Nov 1995 14:58:02 SCDAY
// Added picture layer checksums
//
// Rev 1.17 03 Nov 1995 11:42:54 AKASAI
//
// Added and changed code to handle MB checksum hopefully better.
//
// Rev 1.16 01 Nov 1995 13:46:02 AKASAI
//
// Added allocation of temporary buffer for loop filter. uFilterBBuffer
// right after uMBBuffer.
//
// Rev 1.15 30 Oct 1995 16:20:26 AKASAI
// Fixed up extra bytes some more. Doug and Sylvia had already decided
// on 2 extra bytes for the decoder instead of 4. We now copy 2 zeros
// at the end of the biSizeImage.
//
// Rev 1.14 30 Oct 1995 15:38:22 AKASAI
// Frame 94 of grouch read past the end of the bit stream finding junk.
// Enabled code Sylvia had put in to copy 4 bytes of zero after biSizeImage.
// This seems to fix the problem playing grouch.avi.
//
// Rev 1.13 27 Oct 1995 19:11:26 AKASAI
// Added some special case code to handle when skip macroblock is last
// in a gob.
//
// Rev 1.12 27 Oct 1995 18:17:22 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.11 26 Oct 1995 15:33:10 SCDAY
//
// Delta frames partially working -- changed main loops to accommodate
// skipped macroblocks by detecting next startcode
//
// Rev 1.10 16 Oct 1995 13:53:46 SCDAY
//
// Added macroblock level checksum
//
// Rev 1.9 10 Oct 1995 15:44:02 SCDAY
// clean up
//
// Rev 1.8 10 Oct 1995 14:58:10 SCDAY
//
// added support for FCIF
//
// Rev 1.7 06 Oct 1995 15:32:28 SCDAY
//
// Integrated with latest AKK d1block
//
// Rev 1.6 04 Oct 1995 15:24:46 SCDAY
// changed test pattern stuff
//
// Rev 1.5 22 Sep 1995 15:07:02 SCDAY
// Doug fixed ASSERT bug, scd debug changes
//
// Rev 1.2 19 Sep 1995 15:25:32 SCDAY
//
// added H261 pict, GOB, MB/MBA parsing
//
// Rev 1.1 12 Sep 1995 15:52:24 DBRUCKS
// add SKIP_DECODE option for encoder work
//
// Rev 1.0 11 Sep 1995 13:51:48 SCDAY
// Initial revision.
//
// Rev 1.18 05 Sep 1995 17:22:12 DBRUCKS
// u & v are offset by 8 from Y in YVU12ForEnc
//
// Rev 1.17 01 Sep 1995 17:13:52 DBRUCKS
// add adjustpels
//
// Rev 1.16 01 Sep 1995 09:49:34 DBRUCKS
// checkin partial ajdust pels changes
//
// Rev 1.15 29 Aug 1995 16:50:40 DBRUCKS
// add support for YVU9 playback
//
// Rev 1.14 28 Aug 1995 17:45:58 DBRUCKS
// add yvu12forenc
//
// Rev 1.13 28 Aug 1995 10:15:14 DBRUCKS
// update to 5 July Spec and 8/25 Errata
//
// Rev 1.12 24 Aug 1995 08:51:30 CZHU
// Turned off apsect ratio correction.
//
// Rev 1.11 23 Aug 1995 12:25:10 DBRUCKS
// Turn on the color converters
//
// Rev 1.10 14 Aug 1995 16:40:34 DBRUCKS
// initialize block action stream
//
// Rev 1.9 11 Aug 1995 17:47:58 DBRUCKS
// cleanup
//
// Rev 1.8 11 Aug 1995 17:30:00 DBRUCKS
// copy source to bitstream
//
// Rev 1.7 11 Aug 1995 16:12:14 DBRUCKS
// add ptr check to MB data and add #ifndef early exit
//
// Rev 1.6 11 Aug 1995 15:10:18 DBRUCKS
// get ready to integrate with block level code and hook up macro block level code
//
// Rev 1.5 03 Aug 1995 14:57:56 DBRUCKS
// Add ASSERT macro
//
// Rev 1.4 02 Aug 1995 15:31:34 DBRUCKS
// added GOB header parsing
//
// Rev 1.3 01 Aug 1995 12:27:38 DBRUCKS
// add PSC parsing
//
// Rev 1.2 31 Jul 1995 16:28:00 DBRUCKS
// move loacl BITS defs to D3DEC.CPP
//
// Rev 1.1 31 Jul 1995 15:32:22 CZHU
// Moved global tables to d3tables.h
//
// Rev 1.0 31 Jul 1995 13:00:04 DBRUCKS
// Initial revision.
//
// Rev 1.3 28 Jul 1995 13:57:36 CZHU
// Started to add picture level decoding of fixed length codes.
//
// Rev 1.2 24 Jul 1995 14:57:52 CZHU
// Added global tables for VLD decoding. Also added instance initialization
// and termination. Several data structures are updated for H.263.
//
// Rev 1.1 17 Jul 1995 14:46:20 CZHU
//
//
// Rev 1.0 17 Jul 1995 14:14:40 CZHU
// Initial revision.
//////////////////////////////////////////////////////////////////////////////
#include "precomp.h"
static int iNumberOfGOBsBySourceFormat[2] = {
3, /* QCIF */
// 10,
12, /* CIF */
};
static int iNumberOfMBsInAGOBBySourceFormat[2] = {
33, /* QCIF */
33, /* CIF */
};
// rearch
//#ifndef LOSS_RECOVERY
#if 0
static LRESULT IAPass1ProcessFrame(
T_H263DecoderCatalog *DC,
T_BlkAction *fpBlockAction,
T_MBInfo *fpMBInfo,
BITSTREAM_STATE *fpbsState,
U8 *fpu8MaxPtr,
U32 *pN,
T_IQ_INDEX *pRUN_INVERSE_Q,
const I32 iNumberOfGOBs,
const I32 iNumberOfMBs,
const I32 iGOB_start,
const I32 iMB_start
);
#else
static LRESULT IAPass1ProcessFrameRTP(
T_H263DecoderCatalog *DC,
T_BlkAction *fpBlockAction,
T_MBInfo *fpMBInfo,
BITSTREAM_STATE *fpbsState,
U8 *fpu8MaxPtr,
U32 *pN,
T_IQ_INDEX *pRUN_INVERSE_Q,
const I32 iNumberOfGOBs,
const I32 iNumberOfMBs,
const I32 iGOB_start,
const I32 iMB_start
);
#endif
static void IAPass2ProcessFrame(
T_H263DecoderCatalog *DC,
T_BlkAction *fpBlockAction,
T_MBInfo *fpMBInfo,
U32 *pN,
T_IQ_INDEX *pRUN_INVERSE_Q,
const I32 iNumberOfGOBs,
const I32 iNumberOfMBs
);
// rearch
static long DibXY(ICDECOMPRESSEX FAR *lpicDecEx, LPINT lpiPitch, UINT yScale, BOOL bIsDCI);
static void GetDecoderOptions(T_H263DecoderCatalog *);
#define START_CODE 0xff18
static void ZeroFill(HPBYTE hpbY, HPBYTE hpbU, HPBYTE hpbV, int iPitch, U32 uWidth, U32 uHeight);
extern T_H263ColorConvertorCatalog ColorConvertorCatalog[];
extern void BlockCopy(
U32 uDstBlock,
U32 uSrcBlock);
extern void BlockEdgeFilter(U8 *YPlane, int Height, int Width, int Pitch, T_BlkAction *lpBlockAction);
LRESULT H263InitDecoderGlobal(void)
{ //For 32-bit decoder, this is empty for now, 7/29/95
//need to add code for 16 bit version.
return ICERR_OK;
}
/////////////////////////////////////////////////////////////////////////
//
// H263InitializeBlockActionStream
//
// Initialize the block action stream
//
static void H263InitializeBlockActionStream(
T_H263DecoderCatalog * DC)
{
U8 FAR * pu8;
U32 uFrameHeight = DC->uFrameHeight;
U32 uFrameWidth = DC->uFrameWidth;
U32 uCurBlock;
U32 uRefBlock;
U32 uBBlock;
U32 uYOffset;
U32 uUOffset;
U32 uVOffset;
U32 x;
U32 y;
U32 g;
U32 uPitch16;
U32 uPitch8;
U32 uYUpdate;
U32 uUVUpdate;
U32 uBlkNumber;
T_BlkAction FAR * fpBlockAction;
// Offsets for stepping thru GOBs for FCIF processing
static U32 uYGOBFCIFUpdate[12] =
{
(PITCH*3*16)-(FCIF_WIDTH>>1),
(FCIF_WIDTH>>1),
(PITCH*3*16)-(FCIF_WIDTH>>1),
(FCIF_WIDTH>>1),
(PITCH*3*16)-(FCIF_WIDTH>>1),
(FCIF_WIDTH>>1),
(PITCH*3*16)-(FCIF_WIDTH>>1),
(FCIF_WIDTH>>1),
(PITCH*3*16)-(FCIF_WIDTH>>1),
(FCIF_WIDTH>>1),
(PITCH*3*16)-(FCIF_WIDTH>>1),
(FCIF_WIDTH>>1),
};
static U32 uUVGOBFCIFUpdate[12] =
{
(PITCH*3*8)-(FCIF_WIDTH>>2),
(FCIF_WIDTH>>2),
(PITCH*3*8)-(FCIF_WIDTH>>2),
(FCIF_WIDTH>>2),
(PITCH*3*8)-(FCIF_WIDTH>>2),
(FCIF_WIDTH>>2),
(PITCH*3*8)-(FCIF_WIDTH>>2),
(FCIF_WIDTH>>2),
(PITCH*3*8)-(FCIF_WIDTH>>2),
(FCIF_WIDTH>>2),
(PITCH*3*8)-(FCIF_WIDTH>>2),
(FCIF_WIDTH>>2),
};
// assume that the width and height are multiples of 16
ASSERT((uFrameHeight & 0xF) == 0);
ASSERT((uFrameWidth & 0xF) == 0);
// Init uPitch16 and uPitch8
uPitch16 = PITCH*16;
uPitch8 = PITCH*8;
// Point to the allocated space
pu8 = (U8 FAR *) DC;
uCurBlock = (U32) (pu8 + DC->CurrFrame.X32_YPlane);
uRefBlock = (U32) (pu8 + DC->PrevFrame.X32_YPlane);
uBBlock = (U32) (pu8 + DC->PBFrame.X32_YPlane);
// skip the padding used for unconstrained motion vectors
uYOffset = Y_START;
uUOffset = DC->uSz_YPlane + UV_START;
uVOffset = uUOffset + (PITCH >> 1);
// start with block zero
uBlkNumber = 0;
if (uFrameWidth == QCIF_WIDTH)
{ /* if QCIF */
// calculate distance to the next row.
uYUpdate = (16 * PITCH) - uFrameWidth;
uUVUpdate = (8 * PITCH) - (uFrameWidth >> 1);
// Initialize the array
fpBlockAction = (T_BlkAction FAR *) (pu8 + DC->X16_BlkActionStream);
for (y = 0 ; y < uFrameHeight ; y += 16) {
for (x = 0 ; x < uFrameWidth ; x += 16) {
// Four Y Blocks
// Y0 Y1
// Y2 Y3
fpBlockAction->pCurBlock = uCurBlock + uYOffset;
fpBlockAction->pRefBlock = uRefBlock + uYOffset;
fpBlockAction->pBBlock = uBBlock + uYOffset;
fpBlockAction->u8BlkType = BT_EMPTY;
fpBlockAction->uBlkNumber = uBlkNumber++;
uYOffset += 8;
fpBlockAction++;
fpBlockAction->pCurBlock = uCurBlock + uYOffset;
fpBlockAction->pRefBlock = uRefBlock + uYOffset;
fpBlockAction->pBBlock = uBBlock + uYOffset;
fpBlockAction->u8BlkType = BT_EMPTY;
fpBlockAction->uBlkNumber = uBlkNumber++;
uYOffset = uYOffset - 8 + (8 * PITCH);
fpBlockAction++;
fpBlockAction->pCurBlock = uCurBlock + uYOffset;
fpBlockAction->pRefBlock = uRefBlock + uYOffset;
fpBlockAction->pBBlock = uBBlock + uYOffset;
fpBlockAction->u8BlkType = BT_EMPTY;
fpBlockAction->uBlkNumber = uBlkNumber++;
uYOffset += 8;
fpBlockAction++;
fpBlockAction->pCurBlock = uCurBlock + uYOffset;
fpBlockAction->pRefBlock = uRefBlock + uYOffset;
fpBlockAction->pBBlock = uBBlock + uYOffset;
fpBlockAction->u8BlkType = BT_EMPTY;
fpBlockAction->uBlkNumber = uBlkNumber++;
uYOffset = uYOffset + 8 - (8 * PITCH);
fpBlockAction++;
// One CR (V) Block
fpBlockAction->pCurBlock = uCurBlock + uVOffset;
fpBlockAction->pRefBlock = uRefBlock + uVOffset;
fpBlockAction->pBBlock = uBBlock + uVOffset;
fpBlockAction->u8BlkType = BT_EMPTY;
fpBlockAction->uBlkNumber = uBlkNumber++;
uVOffset += 8;
fpBlockAction++;
// One CB (U) Block
fpBlockAction->pCurBlock = uCurBlock + uUOffset;
fpBlockAction->pRefBlock = uRefBlock + uUOffset;
fpBlockAction->pBBlock = uBBlock + uUOffset;
fpBlockAction->u8BlkType = BT_EMPTY;
fpBlockAction->uBlkNumber = uBlkNumber++;
uUOffset += 8;
fpBlockAction++;
}
uYOffset += uYUpdate;
uUOffset += uUVUpdate;
uVOffset += uUVUpdate;
}
} /* end if QCIF */
if (uFrameWidth == FCIF_WIDTH)
{ /* if FCIF */
// calculate distance to the next row.
uYUpdate = (16 * PITCH) - (uFrameWidth >> 1);
uUVUpdate = (8 * PITCH) - (uFrameWidth >> 2);
// Initialize the array
fpBlockAction = (T_BlkAction FAR *) (pu8 + DC->X16_BlkActionStream);
for (g = 0; g < 12; g++) { /* for each GOB */
for (y = 0 ; y < 3 ; y++) { /* for each row in GOB */
for (x = 0 ; x < (uFrameWidth >> 1) ; x += 16) {
// Four Y Blocks
// Y0 Y1
// Y2 Y3
fpBlockAction->pCurBlock = uCurBlock + uYOffset;
fpBlockAction->pRefBlock = uRefBlock + uYOffset;
fpBlockAction->pBBlock = uBBlock + uYOffset;
fpBlockAction->u8BlkType = BT_EMPTY;
fpBlockAction->uBlkNumber = uBlkNumber++;
uYOffset += 8;
fpBlockAction++;
fpBlockAction->pCurBlock = uCurBlock + uYOffset;
fpBlockAction->pRefBlock = uRefBlock + uYOffset;
fpBlockAction->pBBlock = uBBlock + uYOffset;
fpBlockAction->u8BlkType = BT_EMPTY;
fpBlockAction->uBlkNumber = uBlkNumber++;
uYOffset = uYOffset - 8 + (8 * PITCH);
fpBlockAction++;
fpBlockAction->pCurBlock = uCurBlock + uYOffset;
fpBlockAction->pRefBlock = uRefBlock + uYOffset;
fpBlockAction->pBBlock = uBBlock + uYOffset;
fpBlockAction->u8BlkType = BT_EMPTY;
fpBlockAction->uBlkNumber = uBlkNumber++;
uYOffset += 8;
fpBlockAction++;
fpBlockAction->pCurBlock = uCurBlock + uYOffset;
fpBlockAction->pRefBlock = uRefBlock + uYOffset;
fpBlockAction->pBBlock = uBBlock + uYOffset;
fpBlockAction->u8BlkType = BT_EMPTY;
fpBlockAction->uBlkNumber = uBlkNumber++;
uYOffset = uYOffset + 8 - (8 * PITCH);
fpBlockAction++;
// One CR (V) Block
fpBlockAction->pCurBlock = uCurBlock + uVOffset;
fpBlockAction->pRefBlock = uRefBlock + uVOffset;
fpBlockAction->pBBlock = uBBlock + uVOffset;
fpBlockAction->u8BlkType = BT_EMPTY;
fpBlockAction->uBlkNumber = uBlkNumber++;
uVOffset += 8;
fpBlockAction++;
// One CB (U) Block
fpBlockAction->pCurBlock = uCurBlock + uUOffset;
fpBlockAction->pRefBlock = uRefBlock + uUOffset;
fpBlockAction->pBBlock = uBBlock + uUOffset;
fpBlockAction->u8BlkType = BT_EMPTY;
fpBlockAction->uBlkNumber = uBlkNumber++;
uUOffset += 8;
fpBlockAction++;
}
uYOffset += uPitch16 - (uFrameWidth >> 1);
uUOffset += uPitch8 - (uFrameWidth >> 2);
uVOffset += uPitch8 - (uFrameWidth >> 2);
}
uYOffset -= uYGOBFCIFUpdate[g];
uUOffset -= uUVGOBFCIFUpdate[g];
uVOffset -= uUVGOBFCIFUpdate[g];
}
} /* end if FCIF */
} // end H263InitializeBlockActionStream()
/////////////////////////////////////////////////////////////////////////
//
// H261ReInitializeBlockActionStream
//
// ReInitialize the block action stream
//
static void H261ReInitializeBlockActionStream(
T_H263DecoderCatalog * DC)
{
U8 FAR * pu8;
U32 uFrameHeight = DC->uFrameHeight;
U32 uFrameWidth = DC->uFrameWidth;
U32 utemp;
U32 x;
U32 y;
U32 g;
T_BlkAction FAR * fpBlockAction;
pu8 = (U8 FAR *) DC;
if (uFrameWidth == QCIF_WIDTH)
{ /* if QCIF */
// Initialize the array
fpBlockAction = (T_BlkAction FAR *) (pu8 + DC->X16_BlkActionStream);
for (y = 0 ; y < uFrameHeight ; y += 16) {
for (x = 0 ; x < uFrameWidth ; x += 16) {
// Four Y Blocks
// Y0 Y1
// Y2 Y3
utemp = fpBlockAction->pCurBlock;
fpBlockAction->pCurBlock = fpBlockAction->pRefBlock;
fpBlockAction->pRefBlock = utemp;
fpBlockAction->i8MVX=0;
fpBlockAction->i8MVY=0;
fpBlockAction->u8BlkType = BT_EMPTY;
fpBlockAction++;
utemp = fpBlockAction->pCurBlock;
fpBlockAction->pCurBlock = fpBlockAction->pRefBlock;
fpBlockAction->pRefBlock = utemp;
fpBlockAction->i8MVX=0;
fpBlockAction->i8MVY=0;
fpBlockAction->u8BlkType = BT_EMPTY;
fpBlockAction++;
utemp = fpBlockAction->pCurBlock;
fpBlockAction->pCurBlock = fpBlockAction->pRefBlock;
fpBlockAction->pRefBlock = utemp;
fpBlockAction->i8MVX=0;
fpBlockAction->i8MVY=0;
fpBlockAction->u8BlkType = BT_EMPTY;
fpBlockAction++;
utemp = fpBlockAction->pCurBlock;
fpBlockAction->pCurBlock = fpBlockAction->pRefBlock;
fpBlockAction->pRefBlock = utemp;
fpBlockAction->i8MVX=0;
fpBlockAction->i8MVY=0;
fpBlockAction->u8BlkType = BT_EMPTY;
fpBlockAction++;
// One CR (V) Block
utemp = fpBlockAction->pCurBlock;
fpBlockAction->pCurBlock = fpBlockAction->pRefBlock;
fpBlockAction->pRefBlock = utemp;
fpBlockAction->i8MVX=0;
fpBlockAction->i8MVY=0;
fpBlockAction->u8BlkType = BT_EMPTY;
fpBlockAction++;
// One CB (U) Block
utemp = fpBlockAction->pCurBlock;
fpBlockAction->pCurBlock = fpBlockAction->pRefBlock;
fpBlockAction->pRefBlock = utemp;
fpBlockAction->i8MVX=0;
fpBlockAction->i8MVY=0;
fpBlockAction->u8BlkType = BT_EMPTY;
fpBlockAction++;
}
}
} /* end if QCIF */
if (uFrameWidth == FCIF_WIDTH)
{ /* if FCIF */
// Initialize the array
fpBlockAction = (T_BlkAction FAR *) (pu8 + DC->X16_BlkActionStream);
for (g = 0; g < 12; g++) { /* for each GOB */
for (y = 0 ; y < 3 ; y++) { /* for each row in GOB */
for (x = 0 ; x < (uFrameWidth >> 1) ; x += 16) {
// Four Y Blocks
// Y0 Y1
// Y2 Y3
utemp = fpBlockAction->pCurBlock;
fpBlockAction->pCurBlock = fpBlockAction->pRefBlock;
fpBlockAction->pRefBlock = utemp;
fpBlockAction->i8MVX=0;
fpBlockAction->i8MVY=0;
fpBlockAction->u8BlkType = BT_EMPTY;
fpBlockAction++;
utemp = fpBlockAction->pCurBlock;
fpBlockAction->pCurBlock = fpBlockAction->pRefBlock;
fpBlockAction->pRefBlock = utemp;
fpBlockAction->i8MVX=0;
fpBlockAction->i8MVY=0;
fpBlockAction->u8BlkType = BT_EMPTY;
fpBlockAction++;
utemp = fpBlockAction->pCurBlock;
fpBlockAction->pCurBlock = fpBlockAction->pRefBlock;
fpBlockAction->pRefBlock = utemp;
fpBlockAction->i8MVX=0;
fpBlockAction->i8MVY=0;
fpBlockAction->u8BlkType = BT_EMPTY;
fpBlockAction++;
utemp = fpBlockAction->pCurBlock;
fpBlockAction->pCurBlock = fpBlockAction->pRefBlock;
fpBlockAction->pRefBlock = utemp;
fpBlockAction->i8MVX=0;
fpBlockAction->i8MVY=0;
fpBlockAction->u8BlkType = BT_EMPTY;
fpBlockAction++;
// One CR (V) Block
utemp = fpBlockAction->pCurBlock;
fpBlockAction->pCurBlock = fpBlockAction->pRefBlock;
fpBlockAction->pRefBlock = utemp;
fpBlockAction->i8MVX=0;
fpBlockAction->i8MVY=0;
fpBlockAction->u8BlkType = BT_EMPTY;
fpBlockAction++;
// One CB (U) Block
utemp = fpBlockAction->pCurBlock;
fpBlockAction->pCurBlock = fpBlockAction->pRefBlock;
fpBlockAction->pRefBlock = utemp;
fpBlockAction->i8MVX=0;
fpBlockAction->i8MVY=0;
fpBlockAction->u8BlkType = BT_EMPTY;
fpBlockAction++;
}
}
}
} /* end if FCIF */
} // end H261ReInitializeBlockActionStream()
//////////////////////////////////////////////////////////////////////////////
//
// H263InitDecoderInstance
//
// This function allocates and initializes the per-instance tables used by
// the H263 decoder. Note that in 16-bit Windows, the non-instance-specific
// global tables are copied to the per-instance data segment, so that they
// can be used without segment override prefixes.
//
LRESULT H263InitDecoderInstance(LPDECINST lpInst, int CodecID)
{
U32 u32YActiveHeight, u32YActiveWidth;
U32 u32UVActiveHeight, u32UVActiveWidth;
U32 u32YPlane, u32VUPlanes ,u32YVUPlanes,u32SizeBlkActionStream;
U32 uSizeBitStreamBuffer;
U32 uSizeDecTimingInfo;
U32 lOffset=0;
U32 u32TotalSize;
LRESULT iReturn= ICERR_OK;
U32 * pInitLimit;
U32 * pInitPtr;
// rearch
U32 u32SizeT_IQ_INDEXBuffer, u32SizepNBuffer, u32SizeMBInfoStream; // NEW
// rearch
T_H263DecoderCatalog * DC;
U8 * P32Inst;
SECURITY_ATTRIBUTES EventAttributes; // Used with Snapshot.
if(IsBadWritePtr((LPVOID)lpInst, sizeof(DECINSTINFO)))
{
DBOUT("ERROR :: H263InitDecoderInstance :: ICERR_BADPARAM");
iReturn = ICERR_BADPARAM;
goto done;
}
if ((CodecID == YUV12_CODEC && (lpInst->yres > 480 || lpInst->xres > 640)) ||
(CodecID == H263_CODEC && (lpInst->yres > 288 || lpInst->xres > 352)))
{
DBOUT("ERROR :: H263InitDecoderInstance :: ICERR_BADSIZE");
iReturn = ICERR_BADSIZE;
goto done;
}
if (CodecID == YUV12_CODEC)
{
/* The active height and width must be padded to a multiple of 8
* since the adjustpels routine relies on it.
*/
u32YActiveHeight = ((lpInst->yres + 0x7) & (~ 0x7));
u32YActiveWidth = ((lpInst->xres + 0x7) & (~ 0x7));
u32UVActiveHeight = ((lpInst->yres + 0xF) & (~ 0xF)) >> 1;
u32UVActiveWidth = ((lpInst->xres + 0xF) & (~ 0xF)) >> 1;
u32YPlane = u32YActiveWidth * u32YActiveHeight;
u32VUPlanes = u32UVActiveWidth * u32UVActiveHeight * 2;
u32YVUPlanes = u32YPlane + u32VUPlanes;
// added for I420 output support
// wasn't allocating enough memory for YUV12 output, no color convert case
// calculate the block action stream size. The Y portion has one block for
// every 8x8 region. The U and V portion has one block for every 16x16 region.
// We also want to make sure that the size is aligned to a cache line.
u32SizeBlkActionStream = (lpInst->xres >> 3) * (lpInst->yres >> 3);
u32SizeBlkActionStream += ((lpInst->xres >> 4) * (lpInst->yres >> 4)) * 2;
u32SizeBlkActionStream *= sizeof (T_BlkAction);
u32SizeBlkActionStream = (u32SizeBlkActionStream + 31) & ~0x1F;
// calculate the bitstream buffer size. We copy the input data to a buffer
// in our space because we read ahead up to 4 bytes beyond the end of the
// input data. The input data size changes for each frame. So the following
// is a very safe upper bound estimate.
// Add + 2 for extra zeros for start code emulation. AKK
uSizeBitStreamBuffer = lpInst->yres * lpInst->xres + 2;
#ifdef DECODE_STATS
uSizeDecTimingInfo = DEC_TIMING_INFO_FRAME_COUNT * sizeof (DEC_TIMING_INFO);
#else
uSizeDecTimingInfo = 0;
#endif
u32TotalSize = INSTANCE_DATA_FIXED_SIZE +
u32SizeBlkActionStream +
u32YVUPlanes + // current frame
u32YVUPlanes + // prev frame
BLOCK_BUFFER_SIZE +
FILTER_BLOCK_BUFFER_SIZE +
uSizeBitStreamBuffer + // input data
uSizeDecTimingInfo +
0x1F;
// u32TotalSize = 512L + 0x1FL; /* Just enough space for Decoder Catalog. */
}
else
{
ASSERT(CodecID == H263_CODEC);
u32YActiveHeight = lpInst->yres + UMV_EXPAND_Y + UMV_EXPAND_Y ;
u32YActiveWidth = lpInst->xres + UMV_EXPAND_Y + UMV_EXPAND_Y ;
u32UVActiveHeight = u32YActiveHeight/2;
u32UVActiveWidth = u32YActiveWidth /2;
u32YPlane = PITCH * u32YActiveHeight;
u32VUPlanes = PITCH * u32UVActiveHeight;
u32YVUPlanes = u32YPlane + u32VUPlanes;
// calculate the block action stream size. The Y portion has one block for
// every 8x8 region. The U and V portion has one block for every 16x16 region.
// We also want to make sure that the size is aligned to a cache line.
u32SizeBlkActionStream = (lpInst->xres >> 3) * (lpInst->yres >> 3);
u32SizeBlkActionStream += ((lpInst->xres >> 4) * (lpInst->yres >> 4)) * 2;
u32SizeBlkActionStream *= sizeof (T_BlkAction);
u32SizeBlkActionStream = (u32SizeBlkActionStream + 31) & ~0x1F;
// calculate the bitstream buffer size. We copy the input data to a buffer
// in our space because we read ahead up to 4 bytes beyond the end of the
// input data. The input data size changes for each frame. So the following
// is a very safe upper bound estimate.
// Add + 2 for extra zeros for start code emulation. AKK
// Add some additional to make sure stay dword align (rearch)
uSizeBitStreamBuffer = (lpInst->yres * lpInst->xres + 2 + 4) & ~0x3;
// rearch
// calculate sizes of NEW data structures
u32SizeT_IQ_INDEXBuffer = (lpInst->xres)*(lpInst->yres*2)*
sizeof(T_IQ_INDEX);
u32SizepNBuffer = (lpInst->xres>>4)*(lpInst->yres>>4)*sizeof(U32)*6;
u32SizeMBInfoStream = (lpInst->xres>>4)*(lpInst->yres>>4)*
sizeof(T_MBInfo);
// rearch
#ifdef DECODE_STATS
uSizeDecTimingInfo = DEC_TIMING_INFO_FRAME_COUNT * sizeof (DEC_TIMING_INFO);
#else
uSizeDecTimingInfo = 0;
#endif
u32TotalSize = INSTANCE_DATA_FIXED_SIZE +
u32SizeBlkActionStream +
u32YVUPlanes + // current frame
u32YVUPlanes + // prev frame
BLOCK_BUFFER_SIZE +
FILTER_BLOCK_BUFFER_SIZE +
uSizeBitStreamBuffer + // input data
u32SizeT_IQ_INDEXBuffer + // NEW
u32SizepNBuffer + // NEW
u32SizeMBInfoStream + // PB-NEW
uSizeDecTimingInfo +
0x1F;
}
/* If already initialized, terminate this instance before allocating
* another.
*/
if(lpInst->Initialized == TRUE)
{
H263TermDecoderInstance(lpInst);
}
// allocate the memory for the instance
lpInst->pDecoderInst = HeapAlloc(GetProcessHeap(), HEAP_ZERO_MEMORY,
u32TotalSize);
if (lpInst->pDecoderInst == NULL)
{
DBOUT("ERROR :: H263InitDecoderInstance :: ICERR_MEMORY");
iReturn = ICERR_MEMORY;
goto done;
}
//build the decoder catalog
P32Inst = (U8 *) ((((U32) lpInst->pDecoderInst) + 31) & ~0x1F);
//The catalog of per-instance data is at the start of the per-instance data.
DC = (T_H263DecoderCatalog *) P32Inst;
DC->DecoderType = CodecID;
DC->uFrameHeight = lpInst->yres;
DC->uFrameWidth = lpInst->xres;
DC->uYActiveHeight = u32YActiveHeight;
DC->uYActiveWidth = u32YActiveWidth;
DC->uUVActiveHeight = u32UVActiveHeight;
DC->uUVActiveWidth = u32UVActiveWidth;
DC->uSz_YPlane = u32YPlane;
DC->uSz_VUPlanes = u32VUPlanes;
DC->uSz_YVUPlanes = u32YVUPlanes;
DC->BrightnessSetting = H26X_DEFAULT_BRIGHTNESS;
DC->ContrastSetting = H26X_DEFAULT_CONTRAST;
DC->SaturationSetting = H26X_DEFAULT_SATURATION;
DC->iAPColorConvPrev = 0;
DC->pAPInstPrev = NULL; // assume no previous AP instance.
DC->p16InstPostProcess = NULL;
DC->a16InstPostProcess = NULL;
DC->bReadSrcFormat = 0;
EventAttributes.nLength = sizeof(SECURITY_ATTRIBUTES);
EventAttributes.lpSecurityDescriptor = NULL;
EventAttributes.bInheritHandle = FALSE;
DC->SnapshotEvent = CreateEvent(&EventAttributes, TRUE, FALSE, NULL);
/* Get the Options
*/
GetDecoderOptions(DC);
if (CodecID == H263_CODEC)
{
lOffset = INSTANCE_DATA_FIXED_SIZE;
DC->Ticker = 127;
//instance dependent table here
DC->X16_BlkActionStream = lOffset;
lOffset += u32SizeBlkActionStream;
DC-> CurrFrame.X32_YPlane = lOffset;
lOffset += DC->uSz_YPlane;
DC->CurrFrame.X32_VPlane = lOffset;
DC->CurrFrame.X32_UPlane = DC->CurrFrame.X32_VPlane + U_OFFSET;
lOffset += DC->uSz_VUPlanes;
//no padding is needed
DC->PrevFrame.X32_YPlane = lOffset;
lOffset += DC->uSz_YPlane;
DC->PrevFrame.X32_VPlane = lOffset;
DC->PrevFrame.X32_UPlane = DC->PrevFrame.X32_VPlane + U_OFFSET;
lOffset += DC->uSz_VUPlanes;
DC->uMBBuffer = lOffset;
lOffset += BLOCK_BUFFER_SIZE;
DC->uFilterBBuffer = lOffset;
lOffset += FILTER_BLOCK_BUFFER_SIZE;
// Bitstream
ASSERT((lOffset & 0x3) == 0); // DWORD alignment
DC->X32_BitStream = lOffset;
lOffset += uSizeBitStreamBuffer;
DC->uSizeBitStreamBuffer = uSizeBitStreamBuffer;
// rearch
ASSERT((lOffset & 0x3) == 0); // DWORD alignment
DC->X32_InverseQuant = lOffset;
lOffset += u32SizeT_IQ_INDEXBuffer;
ASSERT((lOffset & 0x3) == 0); // DWORD alignment
DC->X32_pN = lOffset;
lOffset += u32SizepNBuffer;
ASSERT((lOffset & 0x3) == 0); // DWORD alignment
DC->X32_uMBInfoStream = lOffset;
lOffset += u32SizeMBInfoStream;
// rearch
#ifdef DECODE_STATS
// Decode Timing Info
DC->X32_DecTimingInfo = lOffset;
lOffset += uSizeDecTimingInfo;
#endif
// init the data
ASSERT((U32)lOffset <= u32TotalSize);
pInitLimit = (U32 *) (P32Inst + lOffset);
pInitPtr = (U32 *) (P32Inst + DC->CurrFrame.X32_YPlane);
for (;pInitPtr < pInitLimit;pInitPtr++) *pInitPtr =0;
// Fill the Y,U,V Previous Frame space with black, this way
// even if we lost an I frame, the background will remain black
ZeroFill((HPBYTE)P32Inst + DC->PrevFrame.X32_YPlane + Y_START,
(HPBYTE)P32Inst + DC->PrevFrame.X32_UPlane + UV_START,
(HPBYTE)P32Inst + DC->PrevFrame.X32_VPlane + UV_START,
PITCH,
DC->uFrameWidth,
DC->uFrameHeight);
H263InitializeBlockActionStream(DC);
} // not YVU9
lpInst->Initialized = TRUE;
iReturn = ICERR_OK;
done:
return iReturn;
}
/***********************************************************************
* ZeroFill
* Fill the YVU data area with black.
***********************************************************************/
static void ZeroFill(HPBYTE hpbY, HPBYTE hpbU, HPBYTE hpbV, int iPitch, U32 uWidth, U32 uHeight)
{
U32 w,h;
int y,u,v;
U32 uNext;
HPBYTE pY, pU, pV;
y = 32;
uNext = iPitch - uWidth;
for (h = 0 ; h < uHeight ; h++) {
pY = hpbY;
for (w = 0; w < uWidth ; w++) {
*hpbY++ = (U8)16;
}
hpbY += uNext;
}
uWidth = uWidth / 2;
uHeight = uHeight / 2;
uNext = iPitch - uWidth;
for (h = 0 ; h < uHeight ; h++) {
pV = hpbV;
pU = hpbU;
for (w = 0; w < uWidth ; w++) {
*hpbV++ = (U8)128;
*hpbU++ = (U8)128;
}
hpbV += uNext;
hpbU += uNext;
}
}
//***********************************************************************
//
// TestFill
//
// Fill the YVU data area with a test pattern.
//
#if 0
static void
TestFill(
HPBYTE hpbY,
HPBYTE hpbU,
HPBYTE hpbV,
int iPitch,
U32 uWidth,
U32 uHeight)
{
U32 w,h;
int y,u,v;
U32 uNext;
HPBYTE pY, pU, pV;
y = 32;
uNext = iPitch - uWidth;
for (h = 0 ; h < uHeight ; h++) {
pY = hpbY;
for (w = 0; w < uWidth ; w++) {
*hpbY++ = (U8) (y + (w & ~0xF));
}
hpbY += uNext;
}
uWidth = uWidth / 2;
uHeight = uHeight / 2;
u = 0x4e * 2;
v = 44;
uNext = iPitch - uWidth;
for (h = 0 ; h < uHeight ; h++) {
pV = hpbV;
pU = hpbU;
for (w = 0; w < uWidth ; w++) {
*hpbV++ = (U8) v;
*hpbU++ = (U8) u;
}
hpbV += uNext;
hpbU += uNext;
}
} /* end TestFill */
static void
TestFillUV(
HPBYTE hpbU,
HPBYTE hpbV,
int iPitch,
U32 uWidth,
U32 uHeight)
{
U32 w,h;
int u,v;
U32 uNext;
HPBYTE pU, pV;
uWidth = uWidth / 2;
uHeight = uHeight / 2;
u = 128;
v = 128;
uNext = iPitch - uWidth;
for (h = 0 ; h < uHeight ; h++) {
pV = hpbV;
pU = hpbU;
for (w = 0; w < uWidth ; w++) {
*hpbV++ = (U8) v;
*hpbU++ = (U8) u;
}
hpbV += uNext;
hpbU += uNext;
}
} /* end TestFill */
#endif
//*********************************************************************
//H263Decompress -- This function drives the decompress
// and display of one frame
//*********************************************************************
LRESULT H263Decompress(
LPDECINST lpInst,
ICDECOMPRESSEX FAR * lpicDecEx,
BOOL bIsDCI)
{
LRESULT iReturn = ICERR_ERROR;
U8 FAR * fpSrc;
U8 FAR * P32Inst;
U8 FAR * fpu8MaxPtr;
T_H263DecoderCatalog * DC = NULL;
int iNumberOfGOBs;
int iNumberOfMBs;
T_BlkAction FAR * fpBlockAction;
LONG lOutput;
int intPitch;
U32 uNewOffsetToLine0;
U16 u16NewFrameHeight;
int bShapingFlag;
int uYPitch;
int uUVPitch;
U8 bMirror;
HPBYTE pSource, pDestination;
U32 utemp;
// rearch
T_IQ_INDEX * pRUN_INVERSE_Q;
U32 * pN;
T_MBInfo FAR * fpMBInfo;
I32 gob_start = 1, mb_start = 1;
// rearch
/* new variables added when change to color convertor/bef */
U32 uYPlane, uVPlane, uUPlane;
U8 *pFrame, *lpAligned;
T_H26X_RTP_BSINFO_TRAILER *pBsTrailer;
/* the following is for MB Checksum */
U32 uReadChecksum = 0;
#ifdef DECODE_STATS
U32 uStartLow;
U32 uStartHigh;
U32 uElapsed;
U32 uBefore;
U32 uDecodeFrameSum = 0;
U32 uHeadersSum = 0;
U32 uMemcpySum = 0;
U32 uFrameCopySum = 0;
U32 uOutputCCSum = 0;
U32 uInitBlkActStrSum = 0;
U32 uBEFSum = 0;
int bTimingThisFrame = 0;
DEC_TIMING_INFO * pDecTimingInfo = NULL;
#endif
#ifdef CHECKSUM_PICTURE
/* the following is for Picture Checksum */
YVUCheckSum pReadYVUCksum;
YVUCheckSum YVUChkSum;
U32 uCheckSumValid = 0; // flag to skip checksum check if
// encoder calling decoder before
// checksum valid
#endif
/* The following are used for reading bits */
U32 uWork;
U32 uBitsReady;
BITSTREAM_STATE bsState;
BITSTREAM_STATE FAR * fpbsState = &bsState;
#ifdef SKIP_DECODE
TBD("Skipping Decode");
iReturn = ICERR_OK;
goto done;
#endif
/* check the input pointers
*/
if (IsBadWritePtr((LPVOID)lpInst, sizeof(DECINSTINFO))||
IsBadReadPtr((LPVOID)lpicDecEx, sizeof(ICDECOMPRESSEX)))
{
DBOUT("ERROR :: H263Decompress :: ICERR_BADPARAM");
iReturn = ICERR_BADPARAM;
goto done;
}
/* Check for a bad length
*/
if (lpicDecEx->lpbiSrc->biSizeImage == 0) {
DBOUT("ERROR :: H263Decompress :: ICERR_BADIMAGESIZE");
iReturn = ICERR_BADIMAGESIZE;
goto done;
}
/* Lock the memory
*/
if (lpInst->pDecoderInst == NULL)
{
DBOUT("ERROR :: H263Decompress :: ICERR_MEMORY");
iReturn = ICERR_MEMORY;
goto done;
}
/* Set the frame mirroring flag
*/
bMirror = FALSE;
if (lpicDecEx->lpbiDst != 0)
{
if(lpicDecEx->lpbiSrc->biWidth * lpicDecEx->lpbiDst->biWidth < 0)
bMirror = TRUE;
}
/* for testing */
/* bMirror = TRUE; */
/* Build the decoder catalog pointer
*/
P32Inst = (U8 FAR *) ((((U32) lpInst->pDecoderInst) + 31) & ~0x1F);
DC = (T_H263DecoderCatalog FAR *) P32Inst;
if (DC->DecoderType == H263_CODEC)
{
#ifdef DECODE_STATS
if ((DC->uStatFrameCount < DEC_TIMING_INFO_FRAME_COUNT) &&
(DC->ColorConvertor != YUV12ForEnc))
{
ASSERT(DC->X32_DecTimingInfo > 0);
DC->pDecTimingInfo = (DEC_TIMING_INFO FAR *)( ((U8 FAR *)P32Inst) + DC->X32_DecTimingInfo );
TIMER_START(bTimingThisFrame,uStartLow,uStartHigh);
ASSERT(bTimingThisFrame);
DC->uStartLow = uStartLow;
DC->uStartHigh = uStartHigh;
}
else
{
DC->pDecTimingInfo = (DEC_TIMING_INFO FAR *) NULL;
ASSERT(!bTimingThisFrame);
}
DC->bTimingThisFrame = bTimingThisFrame;
#endif
/* Is there room to copy the bitstream? We could at most add 2 (zeros) and 3
padding bytes for DWORD alignment to the original bitstream */\
ASSERT(lpicDecEx->lpbiSrc->biSizeImage + 5 <= DC->uSizeBitStreamBuffer);
if ((lpicDecEx->lpbiSrc->biSizeImage + 5) > DC->uSizeBitStreamBuffer)
{
DBOUT("ERROR :: H263Decompress :: ICERR_ERROR: not enough room for bitstream");
iReturn = ICERR_ERROR;
goto done;
}
/* Copy the source data to the bitstream region.
* OPTIMIZE: Integrate MRV's BLKCOPY.ASM
*/
#ifdef DECODE_STATS
TIMER_BEFORE(bTimingThisFrame,uStartLow,uStartHigh,uBefore);
#endif
fpSrc = (U8 FAR *)(P32Inst + DC->X32_BitStream);
// New: we will first look for an EBS from the PPM. If there is one, then we will
// insert two bytes of zero between the H.261 bistream and the EBS part with
// DWORD alignment and update the total bitstream size. If no EBS is found,
// then we proceed as before.
DC->iVerifiedBsExt = FALSE;
DC->Sz_BitStream = lpicDecEx->lpbiSrc->biSizeImage ;
H26XRTP_VerifyBsInfoStream(DC,(U8 *) lpicDecEx->lpSrc,lpicDecEx->lpbiSrc->biSizeImage);
if (!DC->iValidBsExt)
{
memcpy((char FAR *)fpSrc, (const char FAR *) lpicDecEx->lpSrc, lpicDecEx->lpbiSrc->biSizeImage);
// also copy 16 bits of zero for end of frame detection
fpSrc[lpicDecEx->lpbiSrc->biSizeImage] = 0;
fpSrc[lpicDecEx->lpbiSrc->biSizeImage+1] = 0;
DC->Sz_BitStream += 2;
fpu8MaxPtr = fpSrc;
fpu8MaxPtr += (lpicDecEx->lpbiSrc->biSizeImage + 2 - 1);
}
else
{
// First the H.261 stream data - relying on PPM to fill the compressed size correctly
// in the trailer.
pBsTrailer = ( (T_H26X_RTP_BSINFO_TRAILER *)(DC->pBsTrailer) );
memcpy((char FAR *)fpSrc, (const char FAR *) lpicDecEx->lpSrc, pBsTrailer->uCompressedSize);
// Now write out two bytes of zeros at the end of the H.261 bitstream
fpSrc[pBsTrailer->uCompressedSize] = 0;
fpSrc[pBsTrailer->uCompressedSize + 1] = 0;
// Now tack on the EBS after DWORD alignment.
lpAligned = (U8 *) ( (U32) (fpSrc + (pBsTrailer->uCompressedSize + 2) + 3) &
0xfffffffc);
memcpy(lpAligned, DC->pBsInfo, DC->uNumOfPackets*sizeof(T_RTP_H261_BSINFO));
memcpy(lpAligned + DC->uNumOfPackets*sizeof(T_RTP_H261_BSINFO), DC->pBsTrailer,
sizeof(T_H26X_RTP_BSINFO_TRAILER));
// update lpicDecEx->lpbiSrc->biSizeImage
DC->Sz_BitStream = lpAligned + DC->uNumOfPackets*sizeof(T_RTP_H261_BSINFO) +
sizeof(T_H26X_RTP_BSINFO_TRAILER) - fpSrc;
fpu8MaxPtr = fpSrc;
fpu8MaxPtr += (pBsTrailer->uCompressedSize + 2 - 1);
}
#ifdef DECODE_STATS
TIMER_AFTER_P5(bTimingThisFrame,uStartLow,uStartHigh,uBefore,uElapsed,uMemcpySum)
#endif
/* Initialize the bit stream reader
*/
GET_BITS_INIT(uWork, uBitsReady);
// rearch
// Initialize pointers to data structures which carry info
// between passes
pRUN_INVERSE_Q = (T_IQ_INDEX *)(P32Inst + DC->X32_InverseQuant);
pN = (U32 *)(P32Inst + DC->X32_pN);
fpMBInfo = (T_MBInfo FAR *) (P32Inst + DC->X32_uMBInfoStream);
// rearch
// #ifdef LOSS_RECOVERY
#if 1
DC->iVerifiedBsExt = FALSE;
#endif
/* Decode the Picture Header */
#ifdef DECODE_STATS
TIMER_BEFORE(bTimingThisFrame,uStartLow,uStartHigh,uBefore);
#endif
#ifdef CHECKSUM_PICTURE
iReturn = H263DecodePictureHeader(DC, fpSrc, uBitsReady, uWork, fpbsState, &pReadYVUCksum, &uCheckSumValid);
#else
iReturn = H263DecodePictureHeader(DC, fpSrc, uBitsReady, uWork, fpbsState);
#endif
if (iReturn != ICERR_OK)
{
DBOUT("ERROR :: H263Decompress :: Error reading the picture header");
goto done;
}
#ifdef DECODE_STATS
TIMER_AFTER_P5(bTimingThisFrame,uStartLow,uStartHigh,uBefore,uElapsed,uHeadersSum)
#endif
/* Set a limit for testing for bitstream over-run
*/
/* For each GOB do... */
iNumberOfGOBs = iNumberOfGOBsBySourceFormat[DC->uSrcFormat];
iNumberOfMBs = iNumberOfMBsInAGOBBySourceFormat[DC->uSrcFormat];
/* In H263 a GOB is a single row of MB, and a MB is 16x16 */
/* In H261 a GOB is 33 MBs, and a MB is 16x16 */
/* Order of GOBs depends on source format */
if (DC->uSrcFormat == SRC_FORMAT_QCIF)
{
ASSERT(((U32)iNumberOfGOBs * 3 * 16) == DC->uFrameHeight);
if (((U32)iNumberOfGOBs * 3 * 16) != DC->uFrameHeight)
{
DBOUT("ERROR :: H263Decompress :: Error matching picture header SRC field and actual frame height");
iReturn = ICERR_ERROR;
goto done;
}
ASSERT(((U32)iNumberOfMBs / 3 * 16) == DC->uFrameWidth);
if (((U32)iNumberOfMBs / 3 * 16) != DC->uFrameWidth)
{
DBOUT("ERROR :: H263Decompress :: Error matching picture header SRC field and actual frame width");
iReturn = ICERR_ERROR;
goto done;
}
}
if (DC->uSrcFormat == SRC_FORMAT_CIF)
{
ASSERT(((U32)iNumberOfGOBs / 2 * 3 * 16) == DC->uFrameHeight);
if (((U32)iNumberOfGOBs / 2 * 3 * 16) != DC->uFrameHeight)
{
DBOUT("ERROR :: H263Decompress :: Error matching picture header SRC field and actual frame height");
iReturn = ICERR_ERROR;
goto done;
}
ASSERT(((U32)iNumberOfMBs / 3 * 2 * 16) == DC->uFrameWidth);
if (((U32)iNumberOfMBs / 3 * 2 * 16) != DC->uFrameWidth)
{
DBOUT("ERROR :: H263Decompress :: Error matching picture header SRC field and actual frame width");
iReturn = ICERR_ERROR;
goto done;
}
}
fpBlockAction = (T_BlkAction FAR *) (P32Inst + DC->X16_BlkActionStream);
// rearch
// H261, re initialize the block action stream for entire Frame
// at end of H263Decompress. High bit is set in BlockType to
// indicate if need to do BEF so can't re-init between GOBs.
// H261ReInitializeBlockActionStream(DC);
/*****************************************************************
FIRST PASS - bitream parsing and IDCT prep work
***************************************************************/
// #ifndef LOSS_RECOVERY
#if 0
iReturn = IAPass1ProcessFrame(DC,
fpBlockAction,
fpMBInfo,
fpbsState,
fpu8MaxPtr,
pN,
pRUN_INVERSE_Q,
iNumberOfGOBs,
iNumberOfMBs,
gob_start,
mb_start);
#else
iReturn = IAPass1ProcessFrameRTP(DC,
fpBlockAction,
fpMBInfo,
fpbsState,
fpu8MaxPtr,
pN,
pRUN_INVERSE_Q,
iNumberOfGOBs,
iNumberOfMBs,
gob_start,
mb_start);
#endif
if (iReturn != ICERR_OK) {
DBOUT("H261Decompress : Pass 1 error");
goto done;
}
/*****************************************************************
SECOND PASS - IDCT and motion compensation (MC)
*****************************************************************/
fpBlockAction = (T_BlkAction FAR *)(P32Inst + DC->X16_BlkActionStream);
pRUN_INVERSE_Q = (T_IQ_INDEX *)(P32Inst + DC->X32_InverseQuant);
pN = (U32 *)(P32Inst + DC->X32_pN);
fpMBInfo = (T_MBInfo FAR *)(P32Inst + DC->X32_uMBInfoStream);
IAPass2ProcessFrame(DC,
fpBlockAction,
fpMBInfo,
pN,
pRUN_INVERSE_Q,
iNumberOfGOBs,
iNumberOfMBs);
// rearch
//Prepare which frame to display for inter frames
DC->DispFrame.X32_YPlane = DC->CurrFrame.X32_YPlane;
DC->DispFrame.X32_VPlane = DC->CurrFrame.X32_VPlane;
DC->DispFrame.X32_UPlane = DC->CurrFrame.X32_UPlane;
utemp = DC->CurrFrame.X32_YPlane;
DC->CurrFrame.X32_YPlane = DC->PrevFrame.X32_YPlane;
DC->PrevFrame.X32_YPlane = utemp;
utemp = DC->CurrFrame.X32_VPlane ;
DC->CurrFrame.X32_VPlane = DC->PrevFrame.X32_VPlane;
DC->PrevFrame.X32_VPlane = utemp;
utemp = DC->CurrFrame.X32_UPlane ;
DC->CurrFrame.X32_UPlane = DC->PrevFrame.X32_UPlane;
DC->PrevFrame.X32_UPlane = utemp;
#ifdef CHECKSUM_PICTURE
if (uCheckSumValid)
{
/* compute and compare picture checksum data */
iReturn = H261ComputePictureCheckSum(P32Inst, &YVUChkSum);
iReturn = H261ComparePictureCheckSum(&YVUChkSum, &pReadYVUCksum);
}
#endif
} /* end if (DC->DecoderType == H263_CODEC) */
else
{
ASSERT(DC->DecoderType == YUV12_CODEC);
DC->DispFrame.X32_YPlane = DC->CurrFrame.X32_YPlane;
DC->DispFrame.X32_VPlane = DC->CurrFrame.X32_VPlane;
DC->DispFrame.X32_UPlane = DC->CurrFrame.X32_UPlane;
}
/* Return if there is no need to update screen yet.
*/
if ((lpicDecEx->dwFlags & ICDECOMPRESS_HURRYUP)
|| (lpicDecEx->dwFlags & ICDECOMPRESS_PREROLL))
{
DBOUT("H261Decompress : Display suppressed, HURRYUP or PREROLL");
iReturn = ICERR_DONTDRAW;
goto done;
}
#if 0
/* Fill the Y,U,V Current Frame space with a test pattern
*/
TestFill((HPBYTE)P32Inst + DC->CurrFrame.X32_YPlane + Y_START,
(HPBYTE)P32Inst + DC->CurrFrame.X32_UPlane + UV_START,
(HPBYTE)P32Inst + DC->CurrFrame.X32_VPlane + UV_START,
PITCH,
DC->uFrameWidth,
DC->uFrameHeight);
#endif
#if MAKE_GRAY
/* Fill the U,V Current Frame space with a test pattern
*/
TestFillUV((HPBYTE)P32Inst + DC->CurrFrame.X32_UPlane + UV_START,
(HPBYTE)P32Inst + DC->CurrFrame.X32_VPlane + UV_START,
PITCH,
DC->uFrameWidth,
DC->uFrameHeight);
#endif
/* Special case the YUV12 for the encoder because it should not include
* BEF, Shaping or aspect ratio correction...
*/
if (DC->ColorConvertor == YUV12ForEnc)
{
H26x_YUV12ForEnc ((HPBYTE)P32Inst,
DC->PrevFrame.X32_YPlane + Y_START,
DC->PrevFrame.X32_VPlane + UV_START,
DC->PrevFrame.X32_UPlane + UV_START,
DC->uFrameWidth,
DC->uFrameHeight,
PITCH,
(HPBYTE)lpicDecEx->lpDst,
(DWORD)Y_START,
(DWORD)(MAX_HEIGHT + 2L*UMV_EXPAND_Y) * PITCH + 8 + UV_START + PITCH / 2,
(DWORD)(MAX_HEIGHT + 2L*UMV_EXPAND_Y) * PITCH + 8 + UV_START);
iReturn = ICERR_OK;
goto done;
}
/* Copy Planes to Post Processing area if mirror and/or block edge filter.
*/
if (DC->DecoderType == H263_CODEC)
{
#ifdef DECODE_STATS
TIMER_BEFORE(bTimingThisFrame,uStartLow,uStartHigh,uBefore);
#endif
if(bMirror) { // copy with mirroring
pFrame = (U8 *)DC->p16InstPostProcess;
uYPlane = DC->PostFrame.X32_YPlane;
uUPlane = DC->PostFrame.X32_UPlane;
uVPlane = DC->PostFrame.X32_VPlane;
FrameMirror(((HPBYTE) P32Inst) + DC->DispFrame.X32_YPlane + Y_START,
((HPBYTE) DC->p16InstPostProcess) + DC->PostFrame.X32_YPlane,
DC->uFrameHeight,
DC->uFrameWidth,
PITCH);
FrameMirror(((HPBYTE) P32Inst)+ DC->DispFrame.X32_UPlane + UV_START,
((HPBYTE) DC->p16InstPostProcess) + DC->PostFrame.X32_UPlane,
DC->uFrameHeight/2,
DC->uFrameWidth/2,
PITCH);
FrameMirror(((HPBYTE) P32Inst)+ DC->DispFrame.X32_VPlane + UV_START,
((HPBYTE) DC->p16InstPostProcess) + DC->PostFrame.X32_VPlane,
DC->uFrameHeight/2,
DC->uFrameWidth/2,
PITCH);
}
else
{ /* no mirroring */
if ((DC->bUseBlockEdgeFilter) || (DC->bAdjustLuma) ||
(DC->bAdjustChroma))
{
/* copy for BEF */
pFrame = (U8 *)DC->p16InstPostProcess;
uYPlane = DC->PostFrame.X32_YPlane;
uUPlane = DC->PostFrame.X32_UPlane;
uVPlane = DC->PostFrame.X32_VPlane;
FrameCopy (((HPBYTE) P32Inst) +DC->DispFrame.X32_YPlane+Y_START,
((HPBYTE) DC->p16InstPostProcess) +DC->PostFrame.X32_YPlane,
DC->uFrameHeight,
DC->uFrameWidth,
PITCH);
FrameCopy (((HPBYTE) P32Inst)+DC->DispFrame.X32_UPlane+UV_START,
((HPBYTE) DC->p16InstPostProcess) +DC->PostFrame.X32_UPlane,
DC->uFrameHeight/2,
DC->uFrameWidth/2,
PITCH);
FrameCopy (((HPBYTE) P32Inst)+DC->DispFrame.X32_VPlane+UV_START,
((HPBYTE) DC->p16InstPostProcess) +DC->PostFrame.X32_VPlane,
DC->uFrameHeight/2,
DC->uFrameWidth/2,
PITCH);
} /* end if BEF on */
else
{
/* no BEF or mirror so don't need copy */
pFrame = (U8 *) DC;
uYPlane = DC->DispFrame.X32_YPlane + Y_START;
uUPlane = DC->DispFrame.X32_UPlane + UV_START;
uVPlane = DC->DispFrame.X32_VPlane + UV_START;
} /* end of else no BEF */
} /* end else no mirroring */
#ifdef DECODE_STATS
TIMER_AFTER_P5(bTimingThisFrame,uStartLow,uStartHigh,uBefore,uElapsed,uFrameCopySum)
#endif
uYPitch = PITCH;
uUVPitch = PITCH;
if (DC->bUseBlockEdgeFilter)
{
#ifdef DECODE_STATS
TIMER_BEFORE(bTimingThisFrame,uStartLow,uStartHigh,uBefore);
#endif
fpBlockAction=(T_BlkAction FAR *) (P32Inst+DC->X16_BlkActionStream);
BlockEdgeFilter(((HPBYTE) DC->p16InstPostProcess) + DC->PostFrame.X32_YPlane,
DC->uFrameHeight,
DC->uFrameWidth,
PITCH,
fpBlockAction);
#ifdef DECODE_STATS
TIMER_AFTER_P5(bTimingThisFrame,uStartLow,uStartHigh,uBefore,uElapsed,uBEFSum)
#endif
}
}
else
{ /* YUV12 */
const U32 uHeight = DC->uFrameHeight;
const U32 uWidth = DC->uFrameWidth;
const U32 uYPlaneSize = uHeight*uWidth;
if(bMirror) // mirroring and YUV12 need to do copy
{
pFrame = (U8 *)DC->p16InstPostProcess;
uYPlane = DC->PostFrame.X32_YPlane;
uUPlane = uYPlane + uYPlaneSize;
uVPlane = uUPlane + (uYPlaneSize>>2);
pSource = (HPBYTE)lpicDecEx->lpSrc;
pDestination = (HPBYTE)(DC->p16InstPostProcess + (DWORD)DC->PostFrame.X32_YPlane);
FrameMirror (pSource, pDestination, uHeight, uWidth, uWidth);
pSource += uYPlaneSize;
pDestination += uYPlaneSize;
FrameMirror (pSource, pDestination, (uHeight>>1), (uWidth>>1), (uWidth>>1));
pSource += (uYPlaneSize>>2);
pDestination += (uYPlaneSize>>2);
FrameMirror (pSource, pDestination, (uHeight>>1), (uWidth>>1), (uWidth>>1));
}
else // no mirroring
{
if ((DC->bAdjustLuma)||(DC->bAdjustChroma)) // copy when adjust pels
{
pFrame = (U8 *)DC->p16InstPostProcess;
//uYPlane = 0;
uYPlane = DC->PostFrame.X32_YPlane;
uUPlane = uYPlane + uYPlaneSize;
uVPlane = uUPlane + (uYPlaneSize>>2);
pSource = (HPBYTE)lpicDecEx->lpSrc;
pDestination = (HPBYTE)(DC->p16InstPostProcess + (DWORD)DC->PostFrame.X32_YPlane);
FrameCopy (pSource, pDestination, uHeight, uWidth, uWidth);
pSource += uYPlaneSize;
pDestination += uYPlaneSize;
FrameCopy (pSource, pDestination, (uHeight>>1), (uWidth>>1), (uWidth>>1));
pSource += (uYPlaneSize>>2);
pDestination += (uYPlaneSize>>2);
FrameCopy (pSource, pDestination, (uHeight>>1), (uWidth>>1), (uWidth>>1));
}
else
{
/* Do not have to do memcpy because color convertors don't
* destroy input planes.
*/
pFrame = (HPBYTE)lpicDecEx->lpSrc;
uYPlane = 0;
uUPlane = uYPlane + uYPlaneSize;
uVPlane = uUPlane + (uYPlaneSize>>2);
//memcpy(((char FAR *)(DC->p16InstPostProcess + (DWORD)DC->PostFrame.X32_YPlane)),
// (const char FAR *)lpicDecEx->lpSrc,
// lpicDecEx->lpbiSrc->biSizeImage);
}
} /* end else if no mirroring */
uYPitch = DC->uFrameWidth;
uUVPitch = DC->uFrameWidth >> 1;
} /* end else YUV12 */
if (DC->bForceOnAspectRatioCorrection || lpInst->bCorrectAspectRatio) {
bShapingFlag = 1;
u16NewFrameHeight = (U16) (DC->uFrameHeight * 11 / 12);
} else {
bShapingFlag = 0;
u16NewFrameHeight = (U16) DC->uFrameHeight;
}
/* Do the PEL color adjustments if necessary.
*/
if(DC->bAdjustLuma) {
/* width is rounded up to a multiple of 8
*/
AdjustPels(pFrame,
uYPlane,
DC->uFrameWidth,
uYPitch,
DC->uFrameHeight,
(U32) DC->X16_LumaAdjustment);
}
if(DC->bAdjustChroma) {
/* width = Y-Width / 4 and then rounded up to a multiple of 8
*/
AdjustPels(pFrame,
uUPlane,
(DC->uFrameWidth >> 1),
uUVPitch,
(DC->uFrameHeight >> 1),
(U32) DC->X16_ChromaAdjustment);
AdjustPels(pFrame,
uVPlane,
(DC->uFrameWidth >> 1),
uUVPitch,
(DC->uFrameHeight >> 1),
(U32) DC->X16_ChromaAdjustment);
}
/* Determine parameters (lOutput, intPitch, uNewOffsetToLine0)
* needed for color conversion.
*/
if (lpicDecEx->lpbiDst->biCompression == FOURCC_YUY2)
{
// We are assuming here a positive pitch for YUY2.
// This typically corresponds to a negative value for
// the destination bit map height.
// If we're ever asked to use YUY2 with a positive bit map
// height, we'll have to revisit these calculations.
intPitch = (lpicDecEx->lpbiDst->biBitCount >> 3)
* abs ((int)(lpicDecEx->lpbiDst->biWidth));
lOutput = 0;
uNewOffsetToLine0 = 0;
#if 0
// Aspect ratio correction is now supported for YUY2.
// This is necessary to enable direct draw under Active Movie 1.0.
bShapingFlag=FALSE;
#endif
DBOUT("Using YUY2 ........");
}
else if ((lpicDecEx->lpbiDst->biCompression == FOURCC_YUV12) || (lpicDecEx->lpbiDst->biCompression == FOURCC_IYUV))
{
intPitch = 0xdeadbeef; // should not be used
lOutput = 0;
uNewOffsetToLine0 = DC->CCOffsetToLine0;
bShapingFlag=FALSE;
DBOUT("Using YUV ........");
}
else if (lpicDecEx->lpbiDst->biCompression == FOURCC_IF09)
{
lOutput=0;
intPitch = abs((int)(lpicDecEx->lpbiDst->biWidth));
uNewOffsetToLine0 = DC->CCOffsetToLine0;
DBOUT("USing IF09........");
}
else
{
lOutput = DibXY(lpicDecEx, &intPitch, lpInst->YScale, bIsDCI);
uNewOffsetToLine0 = DC->CCOffsetToLine0;
if (!bIsDCI)
{
// DC->CCOffsetToLine0 was initialized without taking into
// account the sign of the destination bitmap height. Let's
// compensate for that here.
if (lpicDecEx->lpbiDst->biHeight < 0)
uNewOffsetToLine0 = 0;
// Adjust uNewOffsetToLine0 for aspect ratio correction.
if (uNewOffsetToLine0 > 0)
{
ASSERT(intPitch < 0);
if (lpInst->YScale == 2)
{
uNewOffsetToLine0 += 2 * (U32)intPitch *
((U32)DC->uFrameHeight - (U32)u16NewFrameHeight);
}
else
{
uNewOffsetToLine0 += (U32)intPitch *
((U32)DC->uFrameHeight - (U32)u16NewFrameHeight);
}
}
}
}
/* Call the color convertors
*/
/////////////////////////////////////////////////////////////////////////////
// Check to see if we need to copy a Snapshot into the output buffer.
// I added new fields to the Decoder Catalog to permit asynchronous
// transfer of data. These fields are:
// DC->SnapshotRequest
// DC->SnapshotBuffer
// DC->SnapshotEvent
// Ben - 09/25/96
/////////////////////////////////////////////////////////////////////////////
if(DC->SnapshotRequest == SNAPSHOT_REQUESTED)
{
UINT uiSZ_Snapshot;
DBOUT("D1DEC:DECOMPRESS::Snapshot requested");
uiSZ_Snapshot = (DC->uFrameWidth * DC->uFrameHeight * 12) >> 3;
if(!(IsBadWritePtr(DC->SnapshotBuffer, uiSZ_Snapshot)))
{
DC->SnapshotRequest = SNAPSHOT_COPY_STARTED;
DBOUT("D1DEC:DECOMPRESS::Snapshot copy started");
ColorConvertorCatalog[YUV12NOPITCH].ColorConvertor[0]
(
(LPSTR) pFrame + uYPlane,
(LPSTR) pFrame + uVPlane,
(LPSTR) pFrame + uUPlane,
(UN) DC->uFrameWidth,
(UN) DC->uFrameHeight,
(UN) uYPitch,
(UN) uUVPitch,
(UN) (bShapingFlag ? 12 : 9999),
(LPSTR) DC->SnapshotBuffer,
0,
0,
(int) DC->uFrameWidth,
YUV12NOPITCH
);
DC->SnapshotRequest = SNAPSHOT_COPY_FINISHED;
DBOUT("D1DEC:DECOMPRESS::Snapshot copy finished");
}
else
{
DC->SnapshotRequest = SNAPSHOT_COPY_REJECTED;
DBOUT("D1DEC:DECOMPRESS::Snapshot copy rejected");
}
SetEvent(DC->SnapshotEvent);
}
#ifndef RING0
#ifdef _DEBUG
{
char msg[180];
wsprintf(msg, "Decompress before CC: (%d,%d,%d,%d) (%d,%d,%d,%d) lOut %ld, NewOff %ld, DC->Off %ld, pitch %ld",
lpicDecEx->xSrc, lpicDecEx->ySrc, lpicDecEx->dxSrc, lpicDecEx->dySrc,
lpicDecEx->xDst, lpicDecEx->yDst, lpicDecEx->dxDst, lpicDecEx->dyDst,
lOutput, uNewOffsetToLine0, DC->CCOffsetToLine0, intPitch);
DBOUT(msg);
}
#endif
#endif
#ifdef DECODE_STATS
TIMER_BEFORE(bTimingThisFrame,uStartLow,uStartHigh,uBefore);
#endif
ColorConvertorCatalog[DC->ColorConvertor].ColorConvertor[PENTIUM_CC](
(LPSTR) pFrame + uYPlane,
(LPSTR) pFrame + uVPlane,
(LPSTR) pFrame + uUPlane,
(UN) DC->uFrameWidth,
(UN) DC->uFrameHeight,
(UN) uYPitch,
(UN) uUVPitch, // ??? BSE ??? //
(UN) (bShapingFlag ? 12 : 9999), // ??? BSE ??? //
(LPSTR) lpicDecEx->lpDst,
(U32) lOutput,
(U32) uNewOffsetToLine0,
(int) intPitch, // Color converter pitch
DC->ColorConvertor);
#ifdef DECODE_STATS
TIMER_AFTER_P5(bTimingThisFrame,uStartLow,uStartHigh,uBefore,uElapsed,uOutputCCSum);
#endif
iReturn = ICERR_OK;
done:
if (DC != NULL)
{
if (DC->DecoderType == H263_CODEC)
{
#ifdef DECODE_STATS
TIMER_BEFORE(bTimingThisFrame,uStartLow,uStartHigh,uBefore);
#endif
H261ReInitializeBlockActionStream(DC);
#ifdef DECODE_STATS
TIMER_AFTER_P5(bTimingThisFrame,uStartLow,uStartHigh,uBefore,uElapsed,uInitBlkActStrSum)
#endif
} /* end if (DC->DecoderType == H263_CODEC) */
#ifdef DECODE_STATS
TIMER_STOP(bTimingThisFrame,uStartLow,uStartHigh,uDecodeFrameSum);
if (bTimingThisFrame)
{
pDecTimingInfo = DC->pDecTimingInfo + DC->uStatFrameCount;
pDecTimingInfo->uDecodeFrame = uDecodeFrameSum;
pDecTimingInfo->uHeaders += uHeadersSum;
pDecTimingInfo->uMemcpy = uMemcpySum;
pDecTimingInfo->uFrameCopy = uFrameCopySum;
pDecTimingInfo->uOutputCC = uOutputCCSum;
pDecTimingInfo->uInitBlkActStr = uInitBlkActStrSum;
pDecTimingInfo->uBEF = uBEFSum;
DC->uStatFrameCount++;
/* Verify that we have time for all the required steps
*/
ASSERT(pDecTimingInfo->uDecodeFrame);
ASSERT(pDecTimingInfo->uHeaders);
ASSERT(pDecTimingInfo->uMemcpy);
ASSERT(pDecTimingInfo->uFrameCopy);
ASSERT(pDecTimingInfo->uOutputCC);
/* ASSERT(pDecTimingInfo->uDecodeBlock); 0 if all are empty */
ASSERT(pDecTimingInfo->uInitBlkActStr);
ASSERT(pDecTimingInfo->uBEF);
}
#endif
}
return iReturn;
}
//************************************************************************
//
//H263TermDecoderInstance -- This function frees the space allocated for an
// instance of the H263 decoder.
//
//************************************************************************
LRESULT H263TermDecoderInstance(LPDECINST lpInst)
{
LRESULT iReturn = ICERR_OK;
T_H263DecoderCatalog * DC;
if(IsBadWritePtr((LPVOID)lpInst, sizeof(DECINSTINFO)))
{
DBOUT("ERROR :: H263TermDecoderInstance :: ICERR_BADPARAM");
iReturn = ICERR_BADPARAM;
}
if(lpInst->Initialized == FALSE)
{
DBOUT("Warning: H263TermDecoderInstance(): Uninitialized instance")
return(ICERR_OK);
}
lpInst->Initialized = FALSE;
DC = (T_H263DecoderCatalog *) ((((U32) lpInst->pDecoderInst) + 31) & ~0x1F);
CloseHandle(DC->SnapshotEvent);
if (DC->a16InstPostProcess != NULL)
{
HeapFree(GetProcessHeap(), 0, DC->a16InstPostProcess);
// PhilF: Also freed in H263TerminateDecoderInstance! For now set to NULL to avoid second HeapFree.
// Investigate reason for 2nd call later...
DC->a16InstPostProcess = NULL;
}
HeapFree(GetProcessHeap(), 0, lpInst->pDecoderInst);
return iReturn;
}
//****************************************************************************
//DibXY -- This function is used to map color converted output to the screen.
//note: this function came from the H261 code base.
//****************************************************************************
static long DibXY(ICDECOMPRESSEX FAR *lpicDecEx, LPINT lpiPitch, UINT yScale, BOOL bIsDCI)
{
int iPitch; /* width of DIB */
long lOffset = 0;
LPBITMAPINFOHEADER lpbi = lpicDecEx->lpbiDst;
iPitch = (((abs((int)lpbi->biWidth) * (int)lpbi->biBitCount) >> 3) + 3) & ~3;
// The source and destination rectangles in lpicDecEx are only
// meaningful if bIsDCI is true (because throughout our codec, if bIsDCI
// is FALSE, we put zeroes in these rectangles). This may change, at
// some later point, if we decide (or are required) to make use of the
// rcSource and rcTarget rectangles that are associated with an Active
// Movie media sample.
if (!bIsDCI)
{
if (lpbi->biHeight >= 0)
{
// Typically for RGB, a positive bitmap height corresponds
// to a negative pitch.
iPitch = -iPitch;
}
}
else
{
if(lpicDecEx->xDst > 0) /* go to proper X position */
lOffset += ((long)lpicDecEx->xDst * (long)lpbi->biBitCount) >> 3;
if(lpbi->biHeight * lpicDecEx->dxSrc < 0)
{ /* DIB is bottom to top */
lOffset += (long) abs((int)lpbi->biWidth) *
(long) abs((int)lpbi->biHeight) *
((long) lpbi->biBitCount >> 3) -
(long) iPitch;
/***************************************************************************/
/***** This next line is used to subtract the amount that Brian added *****/
/***** to CCOffsetToLine0 in COLOR.C during initialization. This is *****/
/***** needed because for DCI, the pitch he used is incorrect. *****/
/***************************************************************************/
lOffset -= ((long) yScale * (long)lpicDecEx->dySrc - 1) *
(long) lpicDecEx->dxDst * ((long) lpbi->biBitCount >> 3);
iPitch = -iPitch;
}
if(lpicDecEx->yDst > 0) /* go to proper Y position */
lOffset += ((long)lpicDecEx->yDst * (long)iPitch);
if(lpicDecEx->dxSrc > 0) {
lOffset += ((long)lpicDecEx->dyDst * (long)iPitch) - (long)iPitch;
iPitch = -iPitch;
}
if((lpicDecEx->dxDst == 0) && (lpicDecEx->dyDst == 0))
iPitch = -iPitch;
}
*lpiPitch = iPitch;
return(lOffset);
}
/************************************************************************
*
* GetDecoderOptions
*
* Get the options, saving them in the catalog
*/
static void GetDecoderOptions(
T_H263DecoderCatalog * DC)
{
int bSetOptions = 1;
/* Default Options
*/
const int bDefaultForceOnAspectRatioCorrection = 0;
const int bDefaultUseBlockEdgeFilter = 1;
/* INI file variables
*/
#ifndef RING0
UN unResult;
#define SECTION_NAME "Decode"
#define INI_FILE_NAME "h261test.ini"
#ifdef _DEBUG
char buf132[132];
#endif
#endif
/* Read the options from the INI file
*/
#ifndef RING0
{
DBOUT("Getting decode options from the ini file h261test.ini");
/* BlockEdgeFilter
*/
unResult = GetPrivateProfileInt(SECTION_NAME, "BlockEdgeFilter", bDefaultUseBlockEdgeFilter, INI_FILE_NAME);
if (unResult != 0 && unResult != 1)
{
#ifdef _DEBUG
wsprintf(buf132,"BlockEdgeFilter ini value error (should be 0 or 1) - using default=%d",
(int) bDefaultUseBlockEdgeFilter);
DBOUT(buf132);
#endif
unResult = bDefaultUseBlockEdgeFilter;
}
DC->bUseBlockEdgeFilter = unResult;
/* Force on aspect ratio correction.
*/
unResult = GetPrivateProfileInt(SECTION_NAME, "ForceOnAspectRatioCorrection", bDefaultForceOnAspectRatioCorrection, INI_FILE_NAME);
if (unResult != 0 && unResult != 1)
{
#ifdef _DEBUG
wsprintf(buf132,"ForceOnAspectRatioCorrection ini value error (should be 0 or 1) - using default=%d",
(int) bDefaultForceOnAspectRatioCorrection);
DBOUT(buf132);
#endif
unResult = bDefaultForceOnAspectRatioCorrection;
}
DC->bForceOnAspectRatioCorrection = unResult;
bSetOptions = 0;
}
#endif
if (bSetOptions)
{
DC->bUseBlockEdgeFilter = bDefaultUseBlockEdgeFilter;
DC->bForceOnAspectRatioCorrection = bDefaultForceOnAspectRatioCorrection;
}
/* Can only use force aspect ratio correction on if SQCIF, QCIF, or CIF
*/
if (DC->bForceOnAspectRatioCorrection)
{
if (! ( ((DC->uFrameWidth == 128) && (DC->uFrameHeight == 96)) ||
((DC->uFrameWidth == 176) && (DC->uFrameHeight == 144)) ||
((DC->uFrameWidth == 352) && (DC->uFrameHeight == 288)) ) )
{
DBOUT("Aspect ratio correction can not be forced on unless the dimensions are SQCIF, QCIF, or CIF");
DC->bForceOnAspectRatioCorrection = 0;
}
}
/* Display the options
*/
if (DC->bUseBlockEdgeFilter)
{
DBOUT("Decoder option (BlockEdgeFilter) is ON");
}
else
{
DBOUT("Decoder option (BlockEdgeFilter) is OFF");
}
if (DC->bForceOnAspectRatioCorrection)
{
DBOUT("Decoder option (ForceOnAspectRatioCorrection) is ON");
}
else
{
DBOUT("Decoder option (ForceOnAspectRatioCorrection) is OFF");
}
DBOUT("Decoder option (MMX) is OFF: get a life, get MMX");
} /* end GetDecoderOptions() */
/***********************************************************************
* Description:
* This routine parses the bit-stream and initializes two major streams:
* 1) pN: no of coefficients in each of the block (biased by 65 for INTRA)
* 2) pRun_INVERSE_Q: de-quantized coefficient stream for the frame;
* MMX stream is scaled because we use scaled IDCT.
* Other information (e.g. MVs) is kept in decoder catalog, block action
* stream, and MB infor stream.
* Parameters:
* DC: Decoder catalog ptr
* fpBlockAction: block action stream ptr
* fpMBInfo: Macroblock info ptr
* fpbsState: bit-stream state pointer
* fpu8MaxPtr: sentinel value to check for bit-stream overruns
* pN: stream of no. of coeffs (biased by block type) for each block
* pRun_INVERSE_Q:stream of de-quantized (and scaled if using MMX) coefficients
* iNumberOfGOBs: no. of GOBs in the frame
* iNumberOfMBs: no. of MBs in a GOB in the frame
* iGOB_start:
* iMB_start:
* Note:
***********************************************************************/
#pragma code_seg("IACODE1")
// #ifndef LOSS_RECOVERY
#if 0
static LRESULT IAPass1ProcessFrame(
T_H263DecoderCatalog *DC,
T_BlkAction *fpBlockAction,
T_MBInfo *fpMBInfo,
BITSTREAM_STATE *fpbsState,
U8 *fpu8MaxPtr,
U32 *pN,
T_IQ_INDEX *pRUN_INVERSE_Q,
const I32 iNumberOfGOBs,
const I32 iNumberOfMBs,
const I32 iGOB_start,
const I32 iMB_start
)
{
I32 g, iReturn, iBlockNumber = 0 ;
I32 mb_start = iMB_start;
U32 *pNnew;
U32 uReadChecksum = 0;
I8 i;
I8 tmpcnt;
#ifdef DECODE_STATS
U32 uStartLow = DC->uStartLow;
U32 uStartHigh = DC->uStartHigh;
U32 uElapsed;
U32 uBefore;
U32 uHeadersSum = 0;
int bTimingThisFrame = DC->bTimingThisFrame;
DEC_TIMING_INFO *pDecTimingInfo = NULL;
#endif
#ifdef DECODE_STATS
TIMER_BEFORE(bTimingThisFrame,uStartLow,uStartHigh,uBefore);
#endif
/* move decode of GOB start code outside of GOB header processing */
/* because if processing skipped macroblocks, looking for the last MBA */
/* will find the next start code */
iReturn = H263DecodeGOBStartCode(DC, fpbsState);
if (iReturn != ICERR_OK)
{
DBOUT("ERROR :: H263Decompress :: Error reading the GOB StartCode");
goto done;
}
#ifdef DECODE_STATS
TIMER_AFTER_P5(bTimingThisFrame,uStartLow,uStartHigh,uBefore,uElapsed,uHeadersSum)
#endif
for (g = 1 ; g <= iNumberOfGOBs; g++)
{
#ifdef DECODE_STATS
TIMER_BEFORE(bTimingThisFrame,uStartLow,uStartHigh,uBefore);
#endif
iReturn = H263DecodeGOBHeader(DC, fpbsState, g);
if (iReturn != ICERR_OK)
{
DBOUT("ERROR :: H263Decompress :: Error reading the GOB header");
goto done;
}
#ifdef DECODE_STATS
TIMER_AFTER_P5(bTimingThisFrame,uStartLow,uStartHigh,uBefore,uElapsed,uHeadersSum)
#endif
DC->i16LastMBA = -1;
DC->i8MVDH = DC->i8MVDV = 0;
// re-sync iBlockNumber, fpBlockAction, fpMBInfo at this point
iBlockNumber = (g - 1) * iNumberOfMBs*6;
fpBlockAction = (T_BlkAction FAR *)((U8 *)DC + DC->X16_BlkActionStream);
fpMBInfo = (T_MBInfo FAR *) ((U8 *)DC + DC->X32_uMBInfoStream);
fpBlockAction += iBlockNumber;
fpMBInfo += iBlockNumber/6;
pNnew = (U32 *)((U8 *)DC + DC->X32_pN) + iBlockNumber;
while (pN < pNnew ) *pN++ = 0;
/* For each MB until START_CODE detected do ...
*/
for (; ; iBlockNumber += 6, fpBlockAction += 6, fpMBInfo++)
{
#ifdef DECODE_STATS
TIMER_BEFORE(bTimingThisFrame,uStartLow,uStartHigh,uBefore);
#endif
iReturn = H263DecodeMBHeader(DC, fpbsState, &uReadChecksum);
#ifdef DECODE_STATS
TIMER_AFTER_P5(bTimingThisFrame,uStartLow,uStartHigh,uBefore,uElapsed,uHeadersSum)
#endif
if (iReturn == START_CODE)
break;
/* If we didn't see a start code, then we either got an error,
* or we have another MBA delta in DC->uMBA.
*/
if (iReturn != ICERR_OK) {
DBOUT("ERROR :: H263Decompress (First Pass) :: Error reading MB header");
goto error;
}
/* Update MBA */
DC->i16LastMBA += (I16)DC->uMBA;
if (DC->i16LastMBA > 32)
{
DBOUT("ERROR :: H263Decompress :: Bad Macro Block Address");
goto done;
}
/* New for rearch */
/* adjust for empty macroblocks */
for ( tmpcnt = (I8)DC->uMBA; tmpcnt > 1; tmpcnt--)
{
for (i=0; i<6; i++)
{
*pN = 0;
pN++;
}
iBlockNumber += 6;
fpBlockAction += 6;
/* Default fpBlockAction values were already initialized
* in (Re)InitializeBlockActionStream.
*/
fpMBInfo->i8MBType = 2;
fpMBInfo++;
}
fpMBInfo->i8MBType = (I8)DC->uMBType; // New rearch
/* end of new rearch */
// decode and inverse quantize the transform coefficients
iReturn = H263DecodeMBData(DC,
fpBlockAction,
iBlockNumber,
fpbsState,
fpu8MaxPtr,
&uReadChecksum,
&pN,
&pRUN_INVERSE_Q);
if (iReturn != ICERR_OK) {
DBOUT("ERROR :: H263Decompress (First Pass) :: Error parsing MB data");
goto error;
}
} // end for each MB
/* Fill in arrays and advance Block Action stream when there
are skip MB at the end of each GOB
*/
while (iBlockNumber != (I32)g*198) {
for (i=0; i<6; i++)
{
*pN = 0;
pN++;
}
iBlockNumber += 6;
fpBlockAction+= 6;
/* Default fpBlockAction values were already initialized
* in (Re)InitializeBlockActionStream.
*/
fpMBInfo->i8MBType = 2;
fpMBInfo++;
}
/* allow the pointer to address up to four beyond the end - reading
* by DWORD using postincrement.
*/
// ASSERT(fpbsState->fpu8 <= fpu8MaxPtr+4);
if (fpbsState->fpu8 > fpu8MaxPtr+4)
goto error;
} // End for each GOB
#ifdef DECODE_STATS
if (bTimingThisFrame)
{
pDecTimingInfo = DC->pDecTimingInfo + DC->uStatFrameCount;
pDecTimingInfo->uHeaders += uHeadersSum;
}
#endif
done:
return ICERR_OK;
error:
return ICERR_ERROR;
}
#else
static LRESULT IAPass1ProcessFrameRTP(
T_H263DecoderCatalog *DC,
T_BlkAction *fpBlockAction,
T_MBInfo *fpMBInfo,
BITSTREAM_STATE *fpbsState,
U8 *fpu8MaxPtr,
U32 *pN,
T_IQ_INDEX *pRUN_INVERSE_Q,
const I32 iNumberOfGOBs,
const I32 iNumberOfMBs,
const I32 iGOB_start,
const I32 iMB_start
)
{
BITSTREAM_STATE fpbsStateSave;
I32 g, current_g, iReturn, iBlockNumber = 0 ;
I32 mb_start = iMB_start;
U32 *pNnew;
U32 uReadChecksum = 0;
I8 i;
I8 tmpcnt;
I32 g_skip, gtmp;
I32 uMaxGOBNumber, uGOBStep, uMaxBlockNumber;
#ifdef DECODE_STATS
U32 uStartLow = DC->uStartLow;
U32 uStartHigh = DC->uStartHigh;
U32 uElapsed;
U32 uBefore;
U32 uHeadersSum = 0;
int bTimingThisFrame = DC->bTimingThisFrame;
DEC_TIMING_INFO *pDecTimingInfo = NULL;
#endif
#ifdef DECODE_STATS
TIMER_BEFORE(bTimingThisFrame,uStartLow,uStartHigh,uBefore);
#endif
/* move decode of GOB start code outside of GOB header processing */
/* because if processing skipped macroblocks, looking for the last MBA */
/* will find the next start code */
iReturn = H263DecodeGOBStartCode(DC, fpbsState);
if (iReturn != ICERR_OK)
{
DBOUT("ERROR :: H261Decompress :: Error reading the GOB StartCode");
goto done;
}
#ifdef DECODE_STATS
TIMER_AFTER_P5(bTimingThisFrame,uStartLow,uStartHigh,uBefore,uElapsed,uHeadersSum)
#endif
if (iNumberOfGOBs == 3)
{
uMaxGOBNumber = 5;
uGOBStep = 2;
}
else
{
uMaxGOBNumber = 12;
uGOBStep = 1;
}
for (g = 1; g <= uMaxGOBNumber; g+=uGOBStep)
{
current_g = g;
#ifdef DECODE_STATS
TIMER_BEFORE(bTimingThisFrame,uStartLow,uStartHigh,uBefore);
#endif
iReturn = H263DecodeGOBHeader(DC, fpbsState, g);
// #ifndef LOSS_RECOVERY
#if 0
if (iReturn != ICERR_OK)
{
DBOUT("ERROR :: H261Decompress :: Error reading the GOB header");
goto done;
}
#else
if (iReturn == PACKET_FAULT_AT_MB_OR_GOB)
{
DBOUT("Packet fault at MBA or GBSC detected.");
current_g -= uGOBStep; // back up to previous GOB
iReturn = RtpH261FindNextPacket(DC, fpbsState, &pN,
(U32 *)&(DC->uPQuant), (int *)&mb_start, (int *) &g
);
switch (iReturn)
{
case NEXT_MODE_STARTS_GOB:
// Next packet is the start of a GOB; mark missing
// macroblocks as skipped, then read GOB start code,
// and continue in the GOB loop.
// Save bitstream state
DBOUT("Next packet is NEXT_MODE_STARTS_GOB");
fpbsStateSave.fpu8 = fpbsState->fpu8;
fpbsStateSave.uWork = fpbsState->uWork;
fpbsStateSave.uBitsReady = fpbsState->uBitsReady;
// Read GOB start code
iReturn = H263DecodeGOBStartCode(DC, fpbsState);
if (iReturn != ICERR_OK)
{
DBOUT("ERROR :: H261Decompress :: Error reading the GOB StartCode");
goto done;
}
// Read GOB Header
iReturn = H263DecodeGOBHeader(DC, fpbsState, g);
if (iReturn != ICERR_OK)
{
DBOUT("ERROR :: H261Decompress :: Error reading the GOB header");
goto done;
}
g = DC->uGroupNumber;
// Restore bitstream state
fpbsState->fpu8 = fpbsStateSave.fpu8;
fpbsState->uWork = fpbsStateSave.uWork;
fpbsState->uBitsReady = fpbsStateSave.uBitsReady;
// re-sync iBlockNumber, fpBlockAction, fpMBInfo at this point
if (DC->uSrcFormat == SRC_FORMAT_QCIF)
g_skip = (g - 1) >> 1;
else
g_skip = g - 1 ;
iBlockNumber = g_skip * iNumberOfMBs * 6;
fpBlockAction = (T_BlkAction FAR *)((U8 *)DC + DC->X16_BlkActionStream);
fpMBInfo = (T_MBInfo FAR *) ((U8 *)DC + DC->X32_uMBInfoStream);
fpBlockAction += iBlockNumber;
fpMBInfo += iBlockNumber/6;
pNnew = (U32 *)((U8 *)DC + DC->X32_pN) + iBlockNumber;
while (pN < pNnew )
*pN++ = 0;
// Now read the GOB start code and get ready to
// process the new GOB.
iReturn = H263DecodeGOBStartCode(DC, fpbsState);
if (iReturn != ICERR_OK)
{
DBOUT("ERROR :: H261Decompress :: Error reading the GOB StartCode");
goto done;
}
g -= uGOBStep;
continue;
break;
case NEXT_MODE_STARTS_MB :
// Next packet starts with a macroblock; check the
// GOB Number and mark all lost macroblocks as
// skipped; initialize MBA and motion vector
// predictors from the block action stream and
// jump to the macroblock loop
DBOUT("Next packet is NEXT_MODE_STARTS_MB");
if (DC->uSrcFormat == SRC_FORMAT_QCIF)
g_skip = (g - 1) >> 1;
else
g_skip = g - 1;
iBlockNumber = iNumberOfMBs * g_skip * 6 +
(mb_start+1) * 6;
fpBlockAction = (T_BlkAction FAR *)((U8 *)DC + DC->X16_BlkActionStream);
fpMBInfo = (T_MBInfo FAR *) ((U8 *)DC + DC->X32_uMBInfoStream);
fpBlockAction += iBlockNumber;
fpMBInfo += iBlockNumber/6;
DC->uMQuant = DC->uPQuant;
//DC->i16LastMBA = (U16) (mb_start - 1);
DC->i16LastMBA = (U16) (mb_start);
pNnew = (U32 *)((U8 *)DC + DC->X32_pN) + iBlockNumber;
while (pN < pNnew )
*pN++ = 0;
goto MB_LOOP;
break;
case NEXT_MODE_LAST: // all remaining packets in frame lost !!
DBOUT("Next packet is NEXT_MODE_LAST");
uMaxBlockNumber = iNumberOfMBs * iNumberOfGOBs * 6;
pNnew = (U32 *)((U8 *)DC + DC->X32_pN) + uMaxBlockNumber;
while (pN < pNnew )
*pN++ = 0;
iReturn = ICERR_OK;
goto done;
break;
default: // should never happen !!
iReturn = ICERR_ERROR;
goto done;
} // end switch
}
else
{
if (iReturn == PACKET_FAULT_AT_PSC) // can only happen for the PSC packet
{
DBOUT("PSC packet fault detected");
iReturn = RtpGetPicHeaderFromBsExt(DC);
if (iReturn != ICERR_OK)
{
DBOUT("ERROR:: cannot read Picture Header from RTP Trailer");
goto done;
}
iReturn = RtpH261FindNextPacket(DC, fpbsState, &pN,
(U32 *)&(DC->uPQuant), (int *)&mb_start, (int *) &g);
switch (iReturn)
{
case NEXT_MODE_STARTS_GOB:
// Next packet is the start of a GOB; mark missing
// macroblocks as skipped, then read GOB start code,
// and continue in the GOB loop.
// re-sync iBlockNumber, fpBlockAction, fpMBInfo at this point
// Save bitstream state
DBOUT("Next packet is NEXT_MODE_STARTS_GOB");
fpbsStateSave.fpu8 = fpbsState->fpu8;
fpbsStateSave.uWork = fpbsState->uWork;
fpbsStateSave.uBitsReady = fpbsState->uBitsReady;
// Read GOB start code
iReturn = H263DecodeGOBStartCode(DC, fpbsState);
if (iReturn != ICERR_OK)
{
DBOUT("ERROR :: H261Decompress :: Error reading the GOB StartCode");
goto done;
}
// Read GOB Header
iReturn = H263DecodeGOBHeader(DC, fpbsState, g);
if (iReturn != ICERR_OK)
{
DBOUT("ERROR :: H261Decompress :: Error reading the GOB header");
goto done;
}
g = DC->uGroupNumber;
// Restore bitstream state
fpbsState->fpu8 = fpbsStateSave.fpu8;
fpbsState->uWork = fpbsStateSave.uWork;
fpbsState->uBitsReady = fpbsStateSave.uBitsReady;
if (DC->uSrcFormat == SRC_FORMAT_QCIF)
g_skip = (g - 1) >> 1;
else
g_skip = g - 1;
iBlockNumber = g_skip * iNumberOfMBs * 6;
fpBlockAction = (T_BlkAction FAR *)((U8 *)DC + DC->X16_BlkActionStream);
fpMBInfo = (T_MBInfo FAR *) ((U8 *)DC + DC->X32_uMBInfoStream);
fpBlockAction += iBlockNumber;
fpMBInfo += iBlockNumber/6;
pNnew = (U32 *)((U8 *)DC + DC->X32_pN) + iBlockNumber;
while (pN < pNnew )
*pN++ = 0;
// Now read the GOB start code and get ready to
// process the new GOB.
iReturn = H263DecodeGOBStartCode(DC, fpbsState);
if (iReturn != ICERR_OK)
{
DBOUT("ERROR :: H261Decompress :: Error reading the GOB StartCode");
goto done;
}
g -= uGOBStep;
continue;
break;
case NEXT_MODE_STARTS_MB :
// Next packet starts with a macroblock; check the
// GOB Number and mark all lost macroblocks as
// skipped; initialize MBA and motion vector
// predictors from the block action stream and
// jump to the macroblock loop
DBOUT("Next packet is NEXT_MODE_STARTS_MB");
if (DC->uSrcFormat == SRC_FORMAT_QCIF)
g_skip = (g - 1) >> 1;
else
g_skip = g - 1;
iBlockNumber = iNumberOfMBs * g_skip * 6 +
(mb_start+1) * 6;
fpBlockAction = (T_BlkAction FAR *)((U8 *)DC + DC->X16_BlkActionStream);
fpMBInfo = (T_MBInfo FAR *) ((U8 *)DC + DC->X32_uMBInfoStream);
fpBlockAction += iBlockNumber;
fpMBInfo += iBlockNumber/6;
DC->uMQuant = DC->uPQuant;
//DC->i16LastMBA = (U16) (mb_start - 1);
DC->i16LastMBA = (U16) (mb_start);
pNnew = (U32 *)((U8 *)DC + DC->X32_pN) + iBlockNumber;
while (pN < pNnew )
*pN++ = 0;
goto MB_LOOP;
break;
case NEXT_MODE_LAST: // all remaining packets in frame lost !!
DBOUT("Next packet is NEXT_MODE_LAST");
uMaxBlockNumber = iNumberOfMBs * iNumberOfGOBs * 6;
pNnew = (U32 *)((U8 *)DC + DC->X32_pN) + uMaxBlockNumber;
while (pN < pNnew )
*pN++ = 0;
iReturn = ICERR_OK;
goto done;
break;
default: // should never happen !!
iReturn = ICERR_ERROR;
goto done;
} // end switch
} // if .. PACKET_FAULT_AT_PSC
else
{
if (iReturn == ICERR_ERROR)
{
DBOUT("ERROR :: H261Decompress :: Error reading GOB header");
DBOUT(" Packet fault not detected");
goto done;
}
// Outdated: Do the source format check here when it is known that
// the PSC was not the canned one from the PPM.
/* if (DC->bReadSrcFormat && DC->uPrevSrcFormat != DC->uSrcFormat)
{
DBOUT("ERROR::src format changed detected with no packet loss");
DBOUT(" not supported ... bailing out");
iReturn=ICERR_ERROR;
goto done;
}
DC->uPrevSrcFormat = DC->uSrcFormat;
DC->bReadSrcFormat = TRUE; */
}
}
#endif
#ifdef DECODE_STATS
TIMER_AFTER_P5(bTimingThisFrame,uStartLow,uStartHigh,uBefore,uElapsed,uHeadersSum)
#endif
DC->i16LastMBA = -1;
DC->i8MVDH = DC->i8MVDV = 0;
// re-sync iBlockNumber, fpBlockAction, fpMBInfo at this point
if (DC->uSrcFormat == SRC_FORMAT_QCIF)
iBlockNumber = ((g - 1)>>1) * iNumberOfMBs*6;
else
iBlockNumber = (g - 1)* iNumberOfMBs*6;
fpBlockAction = (T_BlkAction FAR *)((U8 *)DC + DC->X16_BlkActionStream);
fpMBInfo = (T_MBInfo FAR *) ((U8 *)DC + DC->X32_uMBInfoStream);
fpBlockAction += iBlockNumber;
fpMBInfo += iBlockNumber/6;
pNnew = (U32 *)((U8 *)DC + DC->X32_pN) + iBlockNumber;
while (pN < pNnew ) *pN++ = 0;
/* For each MB until START_CODE detected do ...
*/
MB_LOOP:
for (; ; iBlockNumber += 6, fpBlockAction += 6, fpMBInfo++)
{
#ifdef DECODE_STATS
TIMER_BEFORE(bTimingThisFrame,uStartLow,uStartHigh,uBefore);
#endif
iReturn = H263DecodeMBHeader(DC, fpbsState, &uReadChecksum);
#ifdef DECODE_STATS
TIMER_AFTER_P5(bTimingThisFrame,uStartLow,uStartHigh,uBefore,uElapsed,uHeadersSum)
#endif
if (iReturn == START_CODE)
break;
/* If we didn't see a start code, then we either got an error,
* or we have another MBA delta in DC->uMBA.
*/
if (iReturn != ICERR_OK) {
DBOUT("ERROR :: H263Decompress (First Pass) :: Error reading MB header");
goto error;
}
/* Update MBA */
DC->i16LastMBA += (I16)DC->uMBA;
if (DC->i16LastMBA > 32)
{
DBOUT("ERROR :: H263Decompress :: Bad Macro Block Address");
goto done;
}
/* New for rearch */
/* adjust for empty macroblocks */
for ( tmpcnt = (I8)DC->uMBA; tmpcnt > 1; tmpcnt--)
{
for (i=0; i<6; i++)
{
*pN = 0;
pN++;
}
iBlockNumber += 6;
fpBlockAction += 6;
/* Default fpBlockAction values were already initialized
* in (Re)InitializeBlockActionStream.
*/
fpMBInfo->i8MBType = 2;
fpMBInfo++;
}
fpMBInfo->i8MBType = (I8)DC->uMBType; // New rearch
/* end of new rearch */
// decode and inverse quantize the transform coefficients
iReturn = H263DecodeMBData(DC,
fpBlockAction,
iBlockNumber,
fpbsState,
fpu8MaxPtr,
&uReadChecksum,
&pN,
&pRUN_INVERSE_Q);
if (iReturn != ICERR_OK) {
DBOUT("ERROR :: H263Decompress (First Pass) :: Error parsing MB data");
goto error;
}
} // end for each MB
/* Fill in arrays and advance Block Action stream when there
are skip MB at the end of each GOB
*/
if (DC->uSrcFormat == SRC_FORMAT_QCIF)
{
switch (g)
{
case 1:
gtmp = 1;
break;
case 3:
gtmp = 2;
break;
case 5:
gtmp = 3;
break;
default:
DBOUT("Bad GOB Number");
iReturn = ICERR_ERROR;
goto error;
break;
}
}
else
gtmp = g;
while (iBlockNumber != (I32)gtmp*198) {
for (i=0; i<6; i++)
{
*pN = 0;
pN++;
}
iBlockNumber += 6;
fpBlockAction+= 6;
/* Default fpBlockAction values were already initialized
* in (Re)InitializeBlockActionStream.
*/
fpMBInfo->i8MBType = 2;
fpMBInfo++;
}
/* allow the pointer to address up to four beyond the end - reading
* by DWORD using postincrement.
*/
ASSERT(fpbsState->fpu8 <= fpu8MaxPtr+4);
} // End for each GOB
#ifdef DECODE_STATS
if (bTimingThisFrame)
{
pDecTimingInfo = DC->pDecTimingInfo + DC->uStatFrameCount;
pDecTimingInfo->uHeaders += uHeadersSum;
}
#endif
done:
return ICERR_OK;
error:
return ICERR_ERROR;
}
#endif
#pragma code_seg()
/***********************************************************************
* Description:
* This routines does IDCT and motion compensation.
* Parameters:
* DC: Decoder catalog ptr
* fpBlockAction: block action stream ptr
* fpMBInfo: Macroblock info ptr
* pN: stream of no. of coeffs (biased by block type) for each block
* pRun_INVERSE_Q:stream of de-quantized (and scaled if using MMX) coefficients
* iNumberOfGOBs: no. of GOBs in the frame
* iNumberOfMBs: no. of MBs in a GOB in the frame
* Note:
***********************************************************************/
#pragma code_seg("IACODE2")
static void IAPass2ProcessFrame(
T_H263DecoderCatalog *DC,
T_BlkAction *fpBlockAction,
T_MBInfo *fpMBInfo,
U32 *pN,
T_IQ_INDEX *pRUN_INVERSE_Q,
const I32 iNumberOfGOBs,
const I32 iNumberOfMBs
)
{
I32 g, m, b, iEdgeFlag=0;
// for each GOB do
for (g = 1 ; g <= iNumberOfGOBs; g++)
{
// for each MB do
for (m = 1; m <= iNumberOfMBs; m++, fpBlockAction+=6, fpMBInfo++)
{
// for each block do
for (b = 0; b < 6; b++) { // AP-NEW
// do inverse transform & motion compensation for the block
H263IDCTandMC(DC, fpBlockAction, b, m, g, pN, pRUN_INVERSE_Q,
fpMBInfo, iEdgeFlag); // AP-NEW
// Adjust pointers for next block
if ( *pN >= 65 )
pRUN_INVERSE_Q += *pN - 65;
else
pRUN_INVERSE_Q += *pN;
pN++;
} // end for each block
} // end for each MB
} // End for each GOB
}
#pragma code_seg()
// rearch