blob: 00db04b8276c1319815e351e67f2e0a286a34b85 [file] [log] [blame]
/* ------------------------------------------------------------------
* Copyright (C) 1998-2009 PacketVideo
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either
* express or implied.
* See the License for the specific language governing permissions
* and limitations under the License.
* -------------------------------------------------------------------
*/
#include "mp4dec_lib.h"
#include "vlc_decode.h"
#include "bitstream.h"
#include "scaling.h"
#include "mbtype_mode.h"
#include "idct.h"
#define OSCL_DISABLE_WARNING_CONDITIONAL_IS_CONSTANT
/* ======================================================================== */
/* Function : DecodeFrameDataPartMode() */
/* Purpose : Decode a frame of MPEG4 bitstream in datapartitioning mode. */
/* In/out : */
/* Return : */
/* Modified : */
/* */
/* 04/25/2000 : Rewrite the data partitioning path completely */
/* according to the pseudo codes in MPEG-4 */
/* standard. */
/* Modified : 09/18/2000 add fast VlcDecode+Dequant */
/* 04/17/2001 cleanup */
/* ======================================================================== */
PV_STATUS DecodeFrameDataPartMode(VideoDecData *video)
{
PV_STATUS status;
Vop *currVop = video->currVop;
BitstreamDecVideo *stream = video->bitstream;
int nMBPerRow = video->nMBPerRow;
int vopType = currVop->predictionType;
int mbnum;
int nTotalMB = video->nTotalMB;
int slice_counter;
int resync_marker_length;
/* copy and pad to prev_Vop for INTER coding */
switch (vopType)
{
case I_VOP :
// oscl_memset(Mode, MODE_INTRA, sizeof(uint8)*nTotalMB);
resync_marker_length = 17;
break;
case P_VOP :
oscl_memset(video->motX, 0, sizeof(MOT)*4*nTotalMB);
oscl_memset(video->motY, 0, sizeof(MOT)*4*nTotalMB);
// oscl_memset(Mode, MODE_INTER, sizeof(uint8)*nTotalMB);
resync_marker_length = 16 + currVop->fcodeForward;
break;
default :
mp4dec_log("DecodeFrameDataPartMode(): Vop type not supported.\n");
return PV_FAIL;
}
/** Initialize sliceNo ***/
mbnum = slice_counter = 0;
// oscl_memset(video->sliceNo, 0, sizeof(uint8)*nTotalMB);
do
{
/* This section is equivalent to motion_shape_texture() */
/* in the MPEG-4 standard. 04/13/2000 */
video->mbnum = mbnum;
video->mbnum_row = PV_GET_ROW(mbnum, nMBPerRow); /* This is needed if nbnum is read from the packet header */
video->mbnum_col = mbnum - video->mbnum_row * nMBPerRow;
switch (vopType)
{
case I_VOP :
status = DecodeDataPart_I_VideoPacket(video, slice_counter);
break;
case P_VOP :
status = DecodeDataPart_P_VideoPacket(video, slice_counter);
break;
default :
mp4dec_log("DecodeFrameDataPartMode(): Vop type not supported.\n");
return PV_FAIL;
}
while ((status = PV_ReadVideoPacketHeader(video, &mbnum)) == PV_FAIL)
{
if ((status = quickSearchVideoPacketHeader(stream, resync_marker_length)) != PV_SUCCESS)
{
break;
}
}
if (status == PV_END_OF_VOP)
{
mbnum = nTotalMB;
}
if (mbnum > video->mbnum + 1)
{
ConcealPacket(video, video->mbnum, mbnum, slice_counter);
}
slice_counter++;
if (mbnum >= nTotalMB)
{
break;
}
}
while (TRUE);
return PV_SUCCESS;
}
/* ======================================================================== */
/* Function : DecodeDataPart_I_VideoPacket() */
/* Date : 04/25/2000 */
/* Purpose : Decode Data Partitioned Mode Video Packet in I-VOP */
/* In/out : */
/* Return : PV_SUCCESS if successed, PV_FAIL if failed. */
/* Modified : 09/18/2000 add fast VlcDecode+Dequant */
/* 04/01/2001 fixed MB_stuffing, removed unnecessary code */
/* ======================================================================== */
PV_STATUS DecodeDataPart_I_VideoPacket(VideoDecData *video, int slice_counter)
{
PV_STATUS status;
uint8 *Mode = video->headerInfo.Mode;
BitstreamDecVideo *stream = video->bitstream;
int nTotalMB = video->nTotalMB;
int mbnum, mb_start, mb_end;
int16 QP, *QPMB = video->QPMB;
int MBtype, MCBPC, CBPY;
uint32 tmpvar;
uint code;
int nMBPerRow = video->nMBPerRow;
Bool valid_stuffing;
int32 startSecondPart, startFirstPart = getPointer(stream);
/* decode the first partition */
QP = video->currVop->quantizer;
mb_start = mbnum = video->mbnum;
video->usePrevQP = 0; /* 04/27/01 */
BitstreamShowBits16(stream, 9, &code);
while (code == 1)
{
PV_BitstreamFlushBits(stream, 9);
BitstreamShowBits16(stream, 9, &code);
}
do
{
/* decode COD, MCBPC, ACpred_flag, CPBY and DQUANT */
MCBPC = PV_VlcDecMCBPC_com_intra(stream);
if (!VLC_ERROR_DETECTED(MCBPC))
{
Mode[mbnum] = (uint8)(MBtype = MBtype_mode[MCBPC & 7]);
video->headerInfo.CBP[mbnum] = (uint8)((MCBPC >> 4) & 3);
status = GetMBheaderDataPart_DQUANT_DC(video, &QP);
video->usePrevQP = 1; /* set it after the first coded MB 04/27/01 */
}
else
{
/* Report the error to the application. 06/20/2000 */
VideoDecoderErrorDetected(video);
video->mbnum = mb_start;
movePointerTo(stream, startFirstPart);
return PV_FAIL;
}
video->sliceNo[mbnum] = (uint8) slice_counter;
QPMB[mbnum] = QP;
video->mbnum = ++mbnum;
BitstreamShowBits16(stream, 9, &code);
while (code == 1)
{
PV_BitstreamFlushBits(stream, 9);
BitstreamShowBits16(stream, 9, &code);
}
/* have we reached the end of the video packet or vop? */
status = BitstreamShowBits32(stream, DC_MARKER_LENGTH, &tmpvar);
}
while (tmpvar != DC_MARKER && video->mbnum < nTotalMB);
if (tmpvar == DC_MARKER)
{
PV_BitstreamFlushBits(stream, DC_MARKER_LENGTH);
}
else
{
status = quickSearchDCM(stream);
if (status == PV_SUCCESS)
{
/* only way you can end up being here is in the last packet,and there is stuffing at
the end of the first partition */
PV_BitstreamFlushBits(stream, DC_MARKER_LENGTH);
}
else
{
/* Report the error to the application. 06/20/2000 */
VideoDecoderErrorDetected(video);
movePointerTo(stream, startFirstPart);
video->mbnum = mb_start;
/* concealment will be taken care of in the upper layer */
return PV_FAIL;
}
}
/* decode the second partition */
startSecondPart = getPointer(stream);
mb_end = video->mbnum;
for (mbnum = mb_start; mbnum < mb_end; mbnum++)
{
MBtype = Mode[mbnum];
/* No skipped mode in I-packets 3/1/2001 */
video->mbnum = mbnum;
video->mbnum_row = PV_GET_ROW(mbnum, nMBPerRow); /* This is needed if nbnum is read from the packet header */
video->mbnum_col = mbnum - video->mbnum_row * nMBPerRow;
/* there is always acdcpred in DataPart mode 04/10/01 */
video->acPredFlag[mbnum] = (uint8) BitstreamRead1Bits(stream);
CBPY = PV_VlcDecCBPY(stream, MBtype & INTRA_MASK); /* MODE_INTRA || MODE_INTRA_Q */
if (CBPY < 0)
{
/* Report the error to the application. 06/20/2000 */
VideoDecoderErrorDetected(video);
movePointerTo(stream, startSecondPart); /* */
/* Conceal packet, 05/15/2000 */
ConcealTexture_I(video, startFirstPart, mb_start, mb_end, slice_counter);
return PV_FAIL;
}
video->headerInfo.CBP[mbnum] |= (uint8)(CBPY << 2);
}
video->usePrevQP = 0;
for (mbnum = mb_start; mbnum < mb_end; mbnum++)
{
video->mbnum = mbnum;
video->mbnum_row = PV_GET_ROW(mbnum , nMBPerRow); /* This is needed if nbnum is read from the packet header */
video->mbnum_col = mbnum - video->mbnum_row * nMBPerRow;
/* No skipped mode in I-packets 3/1/2001 */
/* decode the DCT coeficients for the MB */
status = GetMBData_DataPart(video);
if (status != PV_SUCCESS)
{
/* Report the error to the application. 06/20/2000 */
VideoDecoderErrorDetected(video);
movePointerTo(stream, startSecondPart); /* */
/* Conceal packet, 05/15/2000 */
ConcealTexture_I(video, startFirstPart, mb_start, mb_end, slice_counter);
return status;
}
video->usePrevQP = 1; /* 04/27/01 should be set after decoding first MB */
}
valid_stuffing = validStuffing(stream);
if (!valid_stuffing)
{
VideoDecoderErrorDetected(video);
movePointerTo(stream, startSecondPart);
ConcealTexture_I(video, startFirstPart, mb_start, mb_end, slice_counter);
return PV_FAIL;
}
return PV_SUCCESS;
}
/* ======================================================================== */
/* Function : DecodeDataPart_P_VideoPacket() */
/* Date : 04/25/2000 */
/* Purpose : Decode Data Partitioned Mode Video Packet in P-VOP */
/* In/out : */
/* Return : PV_SUCCESS if successed, PV_FAIL if failed. */
/* Modified : 09/18/2000, fast VlcDecode+Dequant */
/* 04/13/2001, fixed MB_stuffing, new ACDC pred structure, */
/* cleanup */
/* 08/07/2001, remove MBzero */
/* ======================================================================== */
PV_STATUS DecodeDataPart_P_VideoPacket(VideoDecData *video, int slice_counter)
{
PV_STATUS status;
uint8 *Mode = video->headerInfo.Mode;
BitstreamDecVideo *stream = video->bitstream;
int nTotalMB = video->nTotalMB;
int mbnum, mb_start, mb_end;
int16 QP, *QPMB = video->QPMB;
int MBtype, CBPY;
Bool valid_stuffing;
int intra_MB;
uint32 tmpvar;
uint code;
int32 startFirstPart, startSecondPart;
int nMBPerRow = video->nMBPerRow;
uint8 *pbyte;
/* decode the first partition */
startFirstPart = getPointer(stream);
mb_start = video->mbnum;
video->usePrevQP = 0; /* 04/27/01 */
BitstreamShowBits16(stream, 10, &code);
while (code == 1)
{
PV_BitstreamFlushBits(stream, 10);
BitstreamShowBits16(stream, 10, &code);
}
do
{
/* decode COD, MCBPC, ACpred_flag, CPBY and DQUANT */
/* We have to discard stuffed MB header */
status = GetMBheaderDataPart_P(video);
if (status != PV_SUCCESS)
{
/* Report the error to the application. 06/20/2000 */
VideoDecoderErrorDetected(video);
movePointerTo(stream, startFirstPart);
video->mbnum = mb_start;
return PV_FAIL;
}
/* we must update slice_counter before motion vector decoding. */
video->sliceNo[video->mbnum] = (uint8) slice_counter;
if (Mode[video->mbnum] & INTER_MASK) /* INTER || INTER_Q || INTER_4V */
{
/* decode the motion vector (if there are any) */
status = PV_GetMBvectors(video, Mode[video->mbnum]);
if (status != PV_SUCCESS)
{
/* Report the error to the application. 06/20/2000 */
VideoDecoderErrorDetected(video);
movePointerTo(stream, startFirstPart);
video->mbnum = mb_start;
return PV_FAIL;
}
}
video->mbnum++;
video->mbnum_row = PV_GET_ROW(video->mbnum, nMBPerRow); /* This is needed if mbnum is read from the packet header */
video->mbnum_col = video->mbnum - video->mbnum_row * nMBPerRow;
BitstreamShowBits16(stream, 10, &code);
while (code == 1)
{
PV_BitstreamFlushBits(stream, 10);
BitstreamShowBits16(stream, 10, &code);
}
/* have we reached the end of the video packet or vop? */
status = BitstreamShowBits32(stream, MOTION_MARKER_COMB_LENGTH, &tmpvar);
/* if (status != PV_SUCCESS && status != PV_END_OF_BUFFER) return status; */
}
while (tmpvar != MOTION_MARKER_COMB && video->mbnum < nTotalMB);
if (tmpvar == MOTION_MARKER_COMB)
{
PV_BitstreamFlushBits(stream, MOTION_MARKER_COMB_LENGTH);
}
else
{
status = quickSearchMotionMarker(stream);
if (status == PV_SUCCESS)
{
/* only way you can end up being here is in the last packet,and there is stuffing at
the end of the first partition */
PV_BitstreamFlushBits(stream, MOTION_MARKER_COMB_LENGTH);
}
else
{
/* Report the error to the application. 06/20/2000 */
VideoDecoderErrorDetected(video);
movePointerTo(stream, startFirstPart);
video->mbnum = mb_start;
/* concealment will be taken care of in the upper layer */
return PV_FAIL;
}
}
/* decode the second partition */
startSecondPart = getPointer(stream);
QP = video->currVop->quantizer;
mb_end = video->mbnum;
for (mbnum = mb_start; mbnum < mb_end; mbnum++)
{
MBtype = Mode[mbnum];
if (MBtype == MODE_SKIPPED)
{
QPMB[mbnum] = QP; /* 03/01/01 */
continue;
}
intra_MB = (MBtype & INTRA_MASK); /* (MBtype == MODE_INTRA || MBtype == MODE_INTRA_Q) */
video->mbnum = mbnum;
video->mbnum_row = PV_GET_ROW(mbnum, nMBPerRow); /* This is needed if nbnum is read from the packet header */
video->mbnum_col = mbnum - video->mbnum_row * nMBPerRow;
/* there is always acdcprediction in DataPart mode 04/10/01 */
if (intra_MB)
{
video->acPredFlag[mbnum] = (uint8) BitstreamRead1Bits_INLINE(stream);
}
CBPY = PV_VlcDecCBPY(stream, intra_MB);
if (CBPY < 0)
{
/* Report the error to the application. 06/20/2000 */
VideoDecoderErrorDetected(video);
/* Conceal second partition, 5/15/2000 */
movePointerTo(stream, startSecondPart);
ConcealTexture_P(video, mb_start, mb_end, slice_counter);
return PV_FAIL;
}
video->headerInfo.CBP[mbnum] |= (uint8)(CBPY << 2);
if (intra_MB || MBtype == MODE_INTER_Q) /* 04/26/01 */
{
status = GetMBheaderDataPart_DQUANT_DC(video, &QP);
if (status != PV_SUCCESS) return status;
}
video->usePrevQP = 1; /* 04/27/01 */
QPMB[mbnum] = QP;
}
video->usePrevQP = 0; /* 04/27/01 */
for (mbnum = mb_start; mbnum < mb_end; mbnum++)
{
video->mbnum = mbnum;
video->mbnum_row = PV_GET_ROW(mbnum, nMBPerRow); /* This is needed if nbnum is read from the packet header */
video->mbnum_col = mbnum - video->mbnum_row * nMBPerRow;
if (Mode[mbnum] != MODE_SKIPPED)
{
/* decode the DCT coeficients for the MB */
status = GetMBData_DataPart(video);
if (status != PV_SUCCESS)
{
/* Report the error to the application. 06/20/2000 */
VideoDecoderErrorDetected(video);
/* Conceal second partition, 5/15/2000 */
movePointerTo(stream, startSecondPart);
ConcealTexture_P(video, mb_start, mb_end, slice_counter);
return status;
}
video->usePrevQP = 1; /* 04/27/01 */
}
else
{ // SKIPPED
/* Motion compensation and put it to video->mblock->pred_block */
SkippedMBMotionComp(video);
//oscl_memset(video->predDCAC_row + video->mbnum_col, 0, sizeof(typeDCACStore)); /* SKIPPED_ACDC */
//oscl_memset(video->predDCAC_col, 0, sizeof(typeDCACStore));
/* 08/08/2005 */
pbyte = (uint8*)(video->predDCAC_row + video->mbnum_col);
ZERO_OUT_64BYTES(pbyte);
pbyte = (uint8*)(video->predDCAC_col);
ZERO_OUT_64BYTES(pbyte);
}
}
valid_stuffing = validStuffing(stream); /* */
if (!valid_stuffing)
{
VideoDecoderErrorDetected(video);
movePointerTo(stream, startSecondPart); /* */
ConcealTexture_P(video, mb_start, mb_end, slice_counter);
return PV_FAIL;
}
return PV_SUCCESS;
}
/* ======================================================================== */
/* Function : GetMBheaderDataPart_DQUANT_DC() */
/* Date : 04/26/2000 */
/* Purpose : Decode DQUANT and DC in Data Partitioned Mode for both */
/* I-VOP and P-VOP. */
/* In/out : */
/* Return : PV_SUCCESS if successed, PV_FAIL if failed. */
/* Modified : 02/13/2001 new ACDC prediction structure, */
/* cleanup */
/* ======================================================================== */
PV_STATUS GetMBheaderDataPart_DQUANT_DC(VideoDecData *video, int16 *QP)
{
PV_STATUS status = PV_SUCCESS;
BitstreamDecVideo *stream = video->bitstream;
int mbnum = video->mbnum;
int intra_dc_vlc_thr = video->currVop->intraDCVlcThr;
uint8 *Mode = video->headerInfo.Mode;
int MBtype = Mode[mbnum];
typeDCStore *DC = video->predDC + mbnum;
int comp;
Bool switched;
uint DQUANT;
int16 QP_tmp;
const static int DQ_tab[4] = { -1, -2, 1, 2};
if (MBtype & Q_MASK) /* INTRA_Q || INTER_Q */
{
DQUANT = BitstreamReadBits16(stream, 2);
*QP += DQ_tab[DQUANT];
if (*QP < 1) *QP = 1;
else if (*QP > 31) *QP = 31;
}
if (MBtype & INTRA_MASK) /* INTRA || INTRA_Q */ /* no switch, code DC separately */
{
QP_tmp = *QP; /* running QP 04/26/01*/
switched = 0;
if (intra_dc_vlc_thr) /* 04/27/01 */
{
if (video->usePrevQP)
QP_tmp = video->QPMB[mbnum-1];
switched = (intra_dc_vlc_thr == 7 || QP_tmp >= intra_dc_vlc_thr * 2 + 11);
}
if (!switched)
{
for (comp = 0; comp < 6; comp++)
{
status = PV_DecodePredictedIntraDC(comp, stream, (*DC + comp)); /* 03/01/01 */
if (status != PV_SUCCESS) return PV_FAIL;
}
}
else
{
for (comp = 0; comp < 6; comp++)
{
(*DC)[comp] = 0; /* 04/26/01 needed for switched case*/
}
}
}
return status;
}
/***********************************************************CommentBegin******
* 04/25/2000 : Initial modification to the new PV Lib format.
* 04/17/2001 : new ACDC pred structure
***********************************************************CommentEnd********/
PV_STATUS GetMBheaderDataPart_P(VideoDecData *video)
{
BitstreamDecVideo *stream = video->bitstream;
int mbnum = video->mbnum;
uint8 *Mode = video->headerInfo.Mode;
typeDCStore *DC = video->predDC + mbnum;
uint no_dct_flag;
int comp;
int MCBPC;
no_dct_flag = BitstreamRead1Bits_INLINE(stream);
if (no_dct_flag)
{
/* skipped macroblock */
Mode[mbnum] = MODE_SKIPPED;
for (comp = 0; comp < 6; comp++)
{
(*DC)[comp] = mid_gray;
/* ACDC REMOVE AC coefs are set in DecodeDataPart_P */
}
}
else
{
/* coded macroblock */
MCBPC = PV_VlcDecMCBPC_com_inter(stream);
if (VLC_ERROR_DETECTED(MCBPC))
{
return PV_FAIL;
}
Mode[mbnum] = (uint8)MBtype_mode[MCBPC & 7];
video->headerInfo.CBP[mbnum] = (uint8)((MCBPC >> 4) & 3);
}
return PV_SUCCESS;
}
/***********************************************************CommentBegin******
* 04/17/01 new ACDC pred structure, reorganized code, cleanup
***********************************************************CommentEnd********/
PV_STATUS GetMBData_DataPart(VideoDecData *video)
{
int mbnum = video->mbnum;
int16 *dataBlock;
MacroBlock *mblock = video->mblock;
int QP = video->QPMB[mbnum];
int32 offset;
PIXEL *c_comp;
int width = video->width;
int intra_dc_vlc_thr = video->currVop->intraDCVlcThr;
uint CBP = video->headerInfo.CBP[mbnum];
uint8 mode = video->headerInfo.Mode[mbnum];
int x_pos = video->mbnum_col;
typeDCStore *DC = video->predDC + mbnum;
int ncoeffs[6], *no_coeff = mblock->no_coeff;
int comp;
Bool switched;
int QP_tmp = QP;
int y_pos = video->mbnum_row;
#ifdef PV_POSTPROC_ON
uint8 *pp_mod[6];
int TotalMB = video->nTotalMB;
int MB_in_width = video->nMBPerRow;
#endif
/*****
* Decoding of the 6 blocks (depending on transparent pattern)
*****/
#ifdef PV_POSTPROC_ON
if (video->postFilterType != PV_NO_POST_PROC)
{
/** post-processing ***/
pp_mod[0] = video->pstprcTypCur + (y_pos << 1) * (MB_in_width << 1) + (x_pos << 1);
pp_mod[1] = pp_mod[0] + 1;
pp_mod[2] = pp_mod[0] + (MB_in_width << 1);
pp_mod[3] = pp_mod[2] + 1;
pp_mod[4] = video->pstprcTypCur + (TotalMB << 2) + mbnum;
pp_mod[5] = pp_mod[4] + TotalMB;
}
#endif
/* oscl_memset(mblock->block, 0, sizeof(typeMBStore)); Aug 9,2005 */
if (mode & INTRA_MASK) /* MODE_INTRA || mode == MODE_INTRA_Q */
{
switched = 0;
if (intra_dc_vlc_thr)
{
if (video->usePrevQP)
QP_tmp = video->QPMB[mbnum-1]; /* running QP 04/26/01 */
switched = (intra_dc_vlc_thr == 7 || QP_tmp >= intra_dc_vlc_thr * 2 + 11);
}
mblock->DCScalarLum = cal_dc_scaler(QP, LUMINANCE_DC_TYPE); /* ACDC 03/01/01 */
mblock->DCScalarChr = cal_dc_scaler(QP, CHROMINANCE_DC_TYPE);
for (comp = 0; comp < 6; comp++)
{
dataBlock = mblock->block[comp]; /*, 10/20/2000 */
dataBlock[0] = (*DC)[comp];
ncoeffs[comp] = VlcDequantH263IntraBlock(video, comp,
switched, mblock->bitmapcol[comp], &mblock->bitmaprow[comp]);
if (VLC_ERROR_DETECTED(ncoeffs[comp])) /* */
{
if (switched)
return PV_FAIL;
else
{
ncoeffs[comp] = 1;
oscl_memset((dataBlock + 1), 0, sizeof(int16)*63);
}
}
no_coeff[comp] = ncoeffs[comp];
/* modified to new semaphore for post-proc */
// Future work:: can be combined in the dequant function
// @todo Deblocking Semaphore for INTRA block
#ifdef PV_POSTPROC_ON
if (video->postFilterType != PV_NO_POST_PROC)
*pp_mod[comp] = (uint8) PostProcSemaphore(dataBlock);
#endif
}
MBlockIDCT(video);
}
else /* MODE INTER*/
{
MBMotionComp(video, CBP);
offset = (int32)(y_pos << 4) * width + (x_pos << 4);
c_comp = video->currVop->yChan + offset;
for (comp = 0; comp < 4; comp++)
{
(*DC)[comp] = mid_gray;
if (CBP & (1 << (5 - comp)))
{
ncoeffs[comp] = VlcDequantH263InterBlock(video, comp,
mblock->bitmapcol[comp], &mblock->bitmaprow[comp]);
if (VLC_ERROR_DETECTED(ncoeffs[comp]))
return PV_FAIL;
BlockIDCT(c_comp + (comp&2)*(width << 2) + 8*(comp&1), mblock->pred_block + (comp&2)*64 + 8*(comp&1), mblock->block[comp], width, ncoeffs[comp],
mblock->bitmapcol[comp], mblock->bitmaprow[comp]);
}
else
{
ncoeffs[comp] = 0;
}
/* @todo Deblocking Semaphore for INTRA block, for inter just test for ringing */
#ifdef PV_POSTPROC_ON
if (video->postFilterType != PV_NO_POST_PROC)
*pp_mod[comp] = (uint8)((ncoeffs[comp] > 3) ? 4 : 0);
#endif
}
(*DC)[4] = mid_gray;
if (CBP & 2)
{
ncoeffs[4] = VlcDequantH263InterBlock(video, 4,
mblock->bitmapcol[4], &mblock->bitmaprow[4]);
if (VLC_ERROR_DETECTED(ncoeffs[4]))
return PV_FAIL;
BlockIDCT(video->currVop->uChan + (offset >> 2) + (x_pos << 2), mblock->pred_block + 256, mblock->block[4], width >> 1, ncoeffs[4],
mblock->bitmapcol[4], mblock->bitmaprow[4]);
}
else
{
ncoeffs[4] = 0;
}
#ifdef PV_POSTPROC_ON
if (video->postFilterType != PV_NO_POST_PROC)
*pp_mod[4] = (uint8)((ncoeffs[4] > 3) ? 4 : 0);
#endif
(*DC)[5] = mid_gray;
if (CBP & 1)
{
ncoeffs[5] = VlcDequantH263InterBlock(video, 5,
mblock->bitmapcol[5], &mblock->bitmaprow[5]);
if (VLC_ERROR_DETECTED(ncoeffs[5]))
return PV_FAIL;
BlockIDCT(video->currVop->vChan + (offset >> 2) + (x_pos << 2), mblock->pred_block + 264, mblock->block[5], width >> 1, ncoeffs[5],
mblock->bitmapcol[5], mblock->bitmaprow[5]);
}
else
{
ncoeffs[5] = 0;
}
#ifdef PV_POSTPROC_ON
if (video->postFilterType != PV_NO_POST_PROC)
*pp_mod[5] = (uint8)((ncoeffs[5] > 3) ? 4 : 0);
#endif
/* Motion compensation and put it to video->mblock->pred_block */
}
return PV_SUCCESS;
}