1 /* ------------------------------------------------------------------
2  * Copyright (C) 1998-2009 PacketVideo
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either
13  * express or implied.
14  * See the License for the specific language governing permissions
15  * and limitations under the License.
16  * -------------------------------------------------------------------
17  */
18 #include "mp4dec_lib.h"
19 #include "vlc_decode.h"
20 #include "bitstream.h"
21 #include "scaling.h"
22 #include "mbtype_mode.h"
23 #include "idct.h"
24 
25 #define OSCL_DISABLE_WARNING_CONDITIONAL_IS_CONSTANT
26 /* ======================================================================== */
27 /*  Function : DecodeFrameDataPartMode()                                    */
28 /*  Purpose  : Decode a frame of MPEG4 bitstream in datapartitioning mode.  */
29 /*  In/out   :                                                              */
30 /*  Return   :                                                              */
31 /*  Modified :                                                              */
32 /*                                                                          */
33 /*      04/25/2000 : Rewrite the data partitioning path completely  */
34 /*                           according to the pseudo codes in MPEG-4        */
35 /*                           standard.                                      */
36 /*  Modified : 09/18/2000 add fast VlcDecode+Dequant                    */
37 /*             04/17/2001 cleanup                                       */
38 /* ======================================================================== */
DecodeFrameDataPartMode(VideoDecData * video)39 PV_STATUS DecodeFrameDataPartMode(VideoDecData *video)
40 {
41     PV_STATUS status;
42     Vop *currVop = video->currVop;
43     BitstreamDecVideo *stream = video->bitstream;
44 
45     int nMBPerRow = video->nMBPerRow;
46 
47     int vopType = currVop->predictionType;
48     int mbnum;
49     int nTotalMB = video->nTotalMB;
50     int slice_counter;
51     int resync_marker_length;
52 
53     /* copy and pad to prev_Vop for INTER coding */
54     switch (vopType)
55     {
56         case I_VOP :
57 //      oscl_memset(Mode, MODE_INTRA, sizeof(uint8)*nTotalMB);
58             resync_marker_length = 17;
59             break;
60         case P_VOP :
61             oscl_memset(video->motX, 0, sizeof(MOT)*4*nTotalMB);
62             oscl_memset(video->motY, 0, sizeof(MOT)*4*nTotalMB);
63 //      oscl_memset(Mode, MODE_INTER, sizeof(uint8)*nTotalMB);
64             resync_marker_length = 16 + currVop->fcodeForward;
65             break;
66         default :
67             mp4dec_log("DecodeFrameDataPartMode(): Vop type not supported.\n");
68             return PV_FAIL;
69     }
70 
71     /** Initialize sliceNo ***/
72     mbnum = slice_counter = 0;
73 //  oscl_memset(video->sliceNo, 0, sizeof(uint8)*nTotalMB);
74 
75     do
76     {
77         /* This section is equivalent to motion_shape_texture() */
78         /* in the MPEG-4 standard.            04/13/2000      */
79         video->mbnum = mbnum;
80         video->mbnum_row = PV_GET_ROW(mbnum, nMBPerRow);   /*  This is needed if nbnum is read from the packet header */
81         video->mbnum_col = mbnum - video->mbnum_row * nMBPerRow;
82 
83         switch (vopType)
84         {
85             case I_VOP :
86                 status = DecodeDataPart_I_VideoPacket(video, slice_counter);
87                 break;
88 
89             case P_VOP :
90                 status = DecodeDataPart_P_VideoPacket(video, slice_counter);
91                 break;
92 
93             default :
94                 mp4dec_log("DecodeFrameDataPartMode(): Vop type not supported.\n");
95                 return PV_FAIL;
96         }
97 
98         while ((status = PV_ReadVideoPacketHeader(video, &mbnum)) == PV_FAIL)
99         {
100             if ((status = quickSearchVideoPacketHeader(stream, resync_marker_length)) != PV_SUCCESS)
101             {
102                 break;
103             }
104         }
105 
106         if (status == PV_END_OF_VOP)
107         {
108             mbnum = nTotalMB;
109         }
110 
111         if (mbnum > video->mbnum + 1)
112         {
113             ConcealPacket(video, video->mbnum, mbnum, slice_counter);
114         }
115         slice_counter++;
116         if (mbnum >= nTotalMB)
117         {
118             break;
119         }
120 
121 
122     }
123     while (TRUE);
124 
125     return PV_SUCCESS;
126 }
127 
128 
129 /* ======================================================================== */
130 /*  Function : DecodeDataPart_I_VideoPacket()                               */
131 /*  Date     : 04/25/2000                                                   */
132 /*  Purpose  : Decode Data Partitioned Mode Video Packet in I-VOP           */
133 /*  In/out   :                                                              */
134 /*  Return   : PV_SUCCESS if successed, PV_FAIL if failed.                  */
135 /*  Modified : 09/18/2000 add fast VlcDecode+Dequant                    */
136 /*             04/01/2001 fixed MB_stuffing, removed unnecessary code   */
137 /* ======================================================================== */
DecodeDataPart_I_VideoPacket(VideoDecData * video,int slice_counter)138 PV_STATUS DecodeDataPart_I_VideoPacket(VideoDecData *video, int slice_counter)
139 {
140     PV_STATUS status;
141     uint8 *Mode = video->headerInfo.Mode;
142     BitstreamDecVideo *stream = video->bitstream;
143     int  nTotalMB = video->nTotalMB;
144     int  mbnum, mb_start, mb_end;
145     int16 QP, *QPMB = video->QPMB;
146     int  MBtype, MCBPC, CBPY;
147     uint32 tmpvar;
148     uint code;
149     int nMBPerRow = video->nMBPerRow;
150     Bool valid_stuffing;
151     int32 startSecondPart, startFirstPart = getPointer(stream);
152 
153     /* decode the first partition */
154     QP = video->currVop->quantizer;
155     mb_start = mbnum = video->mbnum;
156     video->usePrevQP = 0;         /*  04/27/01 */
157 
158 
159     BitstreamShowBits16(stream, 9, &code);
160     while (code == 1)
161     {
162         PV_BitstreamFlushBits(stream, 9);
163         BitstreamShowBits16(stream, 9, &code);
164     }
165 
166     do
167     {
168         /* decode COD, MCBPC, ACpred_flag, CPBY and DQUANT */
169         MCBPC = PV_VlcDecMCBPC_com_intra(stream);
170 
171         if (!VLC_ERROR_DETECTED(MCBPC))
172         {
173             Mode[mbnum] = (uint8)(MBtype = MBtype_mode[MCBPC & 7]);
174             video->headerInfo.CBP[mbnum] = (uint8)((MCBPC >> 4) & 3);
175             status = GetMBheaderDataPart_DQUANT_DC(video, &QP);
176             video->usePrevQP = 1;        /* set it after the first coded MB      04/27/01 */
177         }
178         else
179         {
180             /* Report the error to the application.   06/20/2000 */
181             VideoDecoderErrorDetected(video);
182             video->mbnum = mb_start;
183             movePointerTo(stream, startFirstPart);
184             return PV_FAIL;
185         }
186 
187         video->sliceNo[mbnum] = (uint8) slice_counter;
188         QPMB[mbnum] = QP;
189         video->mbnum = ++mbnum;
190 
191         BitstreamShowBits16(stream, 9, &code);
192         while (code == 1)
193         {
194             PV_BitstreamFlushBits(stream, 9);
195             BitstreamShowBits16(stream, 9, &code);
196         }
197         /* have we reached the end of the video packet or vop? */
198         status = BitstreamShowBits32(stream, DC_MARKER_LENGTH, &tmpvar);
199 
200     }
201     while (tmpvar != DC_MARKER && video->mbnum < nTotalMB);
202 
203     if (tmpvar == DC_MARKER)
204     {
205         PV_BitstreamFlushBits(stream, DC_MARKER_LENGTH);
206     }
207     else
208     {
209         status = quickSearchDCM(stream);
210         if (status == PV_SUCCESS)
211         {
212             /* only way you can end up being here is in the last packet,and there is stuffing at
213             the end of the first partition */
214             PV_BitstreamFlushBits(stream, DC_MARKER_LENGTH);
215         }
216         else
217         {
218             /* Report the error to the application.   06/20/2000 */
219             VideoDecoderErrorDetected(video);
220             movePointerTo(stream, startFirstPart);
221             video->mbnum = mb_start;
222             /* concealment will be taken care of in the upper layer */
223             return PV_FAIL;
224         }
225     }
226 
227     /* decode the second partition */
228     startSecondPart = getPointer(stream);
229 
230     mb_end = video->mbnum;
231 
232     for (mbnum = mb_start; mbnum < mb_end; mbnum++)
233     {
234         MBtype = Mode[mbnum];
235         /* No skipped mode in I-packets  3/1/2001    */
236         video->mbnum = mbnum;
237 
238         video->mbnum_row = PV_GET_ROW(mbnum, nMBPerRow);   /*  This is needed if nbnum is read from the packet header */
239         video->mbnum_col = mbnum - video->mbnum_row * nMBPerRow;
240         /* there is always acdcpred in DataPart mode  04/10/01 */
241         video->acPredFlag[mbnum] = (uint8) BitstreamRead1Bits(stream);
242 
243         CBPY = PV_VlcDecCBPY(stream, MBtype & INTRA_MASK); /* MODE_INTRA || MODE_INTRA_Q */
244         if (CBPY < 0)
245         {
246             /* Report the error to the application.   06/20/2000 */
247             VideoDecoderErrorDetected(video);
248             movePointerTo(stream, startSecondPart); /*  */
249             /* Conceal packet,  05/15/2000 */
250             ConcealTexture_I(video, startFirstPart, mb_start, mb_end, slice_counter);
251             return PV_FAIL;
252         }
253 
254         video->headerInfo.CBP[mbnum] |= (uint8)(CBPY << 2);
255     }
256 
257     video->usePrevQP = 0;
258 
259     for (mbnum = mb_start; mbnum < mb_end; mbnum++)
260     {
261         video->mbnum = mbnum;
262 
263         video->mbnum_row = PV_GET_ROW(mbnum , nMBPerRow);  /*  This is needed if nbnum is read from the packet header */
264         video->mbnum_col = mbnum - video->mbnum_row * nMBPerRow;
265         /* No skipped mode in I-packets  3/1/2001    */
266         /* decode the DCT coeficients for the MB */
267         status = GetMBData_DataPart(video);
268         if (status != PV_SUCCESS)
269         {
270             /* Report the error to the application.   06/20/2000 */
271             VideoDecoderErrorDetected(video);
272             movePointerTo(stream, startSecondPart); /*  */
273             /* Conceal packet,  05/15/2000 */
274             ConcealTexture_I(video, startFirstPart, mb_start, mb_end, slice_counter);
275             return status;
276         }
277         video->usePrevQP = 1;           /*  04/27/01 should be set after decoding first MB */
278     }
279 
280     valid_stuffing = validStuffing(stream);
281     if (!valid_stuffing)
282     {
283         VideoDecoderErrorDetected(video);
284         movePointerTo(stream, startSecondPart);
285         ConcealTexture_I(video, startFirstPart, mb_start, mb_end, slice_counter);
286         return PV_FAIL;
287     }
288     return PV_SUCCESS;
289 }
290 
291 
292 /* ======================================================================== */
293 /*  Function : DecodeDataPart_P_VideoPacket()                               */
294 /*  Date     : 04/25/2000                                                   */
295 /*  Purpose  : Decode Data Partitioned Mode Video Packet in P-VOP           */
296 /*  In/out   :                                                              */
297 /*  Return   : PV_SUCCESS if successed, PV_FAIL if failed.                  */
298 /*  Modified :   09/18/2000,  fast VlcDecode+Dequant                        */
299 /*              04/13/2001,  fixed MB_stuffing, new ACDC pred structure,  */
300 /*                              cleanup                                     */
301 /*              08/07/2001,  remove MBzero                              */
302 /* ======================================================================== */
DecodeDataPart_P_VideoPacket(VideoDecData * video,int slice_counter)303 PV_STATUS DecodeDataPart_P_VideoPacket(VideoDecData *video, int slice_counter)
304 {
305     PV_STATUS status;
306     uint8 *Mode = video->headerInfo.Mode;
307     BitstreamDecVideo *stream = video->bitstream;
308     int nTotalMB = video->nTotalMB;
309     int mbnum, mb_start, mb_end;
310     int16 QP, *QPMB = video->QPMB;
311     int MBtype, CBPY;
312     Bool valid_stuffing;
313     int intra_MB;
314     uint32 tmpvar;
315     uint code;
316     int32  startFirstPart, startSecondPart;
317     int nMBPerRow = video->nMBPerRow;
318     uint8 *pbyte;
319     /* decode the first partition */
320     startFirstPart = getPointer(stream);
321     mb_start = video->mbnum;
322     video->usePrevQP = 0;            /*  04/27/01 */
323 
324     BitstreamShowBits16(stream, 10, &code);
325     while (code == 1)
326     {
327         PV_BitstreamFlushBits(stream, 10);
328         BitstreamShowBits16(stream, 10, &code);
329     }
330 
331     do
332     {
333         /* decode COD, MCBPC, ACpred_flag, CPBY and DQUANT */
334         /* We have to discard stuffed MB header */
335 
336         status = GetMBheaderDataPart_P(video);
337 
338         if (status != PV_SUCCESS)
339         {
340             /* Report the error to the application.   06/20/2000 */
341             VideoDecoderErrorDetected(video);
342             movePointerTo(stream, startFirstPart);
343             video->mbnum = mb_start;
344             return PV_FAIL;
345         }
346 
347         /* we must update slice_counter before motion vector decoding.   */
348         video->sliceNo[video->mbnum] = (uint8) slice_counter;
349 
350         if (Mode[video->mbnum] & INTER_MASK) /* INTER || INTER_Q || INTER_4V */
351         {
352             /* decode the motion vector (if there are any) */
353             status = PV_GetMBvectors(video, Mode[video->mbnum]);
354             if (status != PV_SUCCESS)
355             {
356                 /* Report the error to the application.   06/20/2000 */
357                 VideoDecoderErrorDetected(video);
358                 movePointerTo(stream, startFirstPart);
359                 video->mbnum = mb_start;
360                 return PV_FAIL;
361             }
362         }
363         video->mbnum++;
364 
365         video->mbnum_row = PV_GET_ROW(video->mbnum, nMBPerRow);   /*  This is needed if mbnum is read from the packet header */
366         video->mbnum_col = video->mbnum - video->mbnum_row * nMBPerRow;
367 
368         BitstreamShowBits16(stream, 10, &code);
369         while (code == 1)
370         {
371             PV_BitstreamFlushBits(stream, 10);
372             BitstreamShowBits16(stream, 10, &code);
373         }
374         /* have we reached the end of the video packet or vop? */
375         status = BitstreamShowBits32(stream, MOTION_MARKER_COMB_LENGTH, &tmpvar);
376         /*      if (status != PV_SUCCESS && status != PV_END_OF_BUFFER) return status;  */
377     }
378     while (tmpvar != MOTION_MARKER_COMB && video->mbnum < nTotalMB);
379 
380     if (tmpvar == MOTION_MARKER_COMB)
381     {
382         PV_BitstreamFlushBits(stream, MOTION_MARKER_COMB_LENGTH);
383     }
384     else
385     {
386         status = quickSearchMotionMarker(stream);
387         if (status == PV_SUCCESS)
388         {
389             /* only way you can end up being here is in the last packet,and there is stuffing at
390             the end of the first partition */
391             PV_BitstreamFlushBits(stream, MOTION_MARKER_COMB_LENGTH);
392         }
393         else
394         {
395             /* Report the error to the application.   06/20/2000 */
396             VideoDecoderErrorDetected(video);
397             movePointerTo(stream, startFirstPart);
398             video->mbnum = mb_start;
399             /* concealment will be taken care of in the upper layer  */
400             return PV_FAIL;
401         }
402     }
403 
404     /* decode the second partition */
405     startSecondPart = getPointer(stream);
406     QP = video->currVop->quantizer;
407 
408     mb_end = video->mbnum;
409 
410     for (mbnum = mb_start; mbnum < mb_end; mbnum++)
411     {
412         MBtype = Mode[mbnum];
413 
414         if (MBtype == MODE_SKIPPED)
415         {
416             QPMB[mbnum] = QP; /*  03/01/01 */
417             continue;
418         }
419         intra_MB = (MBtype & INTRA_MASK); /* (MBtype == MODE_INTRA || MBtype == MODE_INTRA_Q) */
420         video->mbnum = mbnum;
421         video->mbnum_row = PV_GET_ROW(mbnum, nMBPerRow);   /*  This is needed if nbnum is read from the packet header */
422         video->mbnum_col = mbnum - video->mbnum_row * nMBPerRow;
423 
424         /* there is always acdcprediction in DataPart mode    04/10/01 */
425         if (intra_MB)
426         {
427             video->acPredFlag[mbnum] = (uint8) BitstreamRead1Bits_INLINE(stream);
428         }
429 
430         CBPY = PV_VlcDecCBPY(stream, intra_MB);
431         if (CBPY < 0)
432         {
433             /* Report the error to the application.   06/20/2000 */
434             VideoDecoderErrorDetected(video);
435             /* Conceal second partition,  5/15/2000 */
436             movePointerTo(stream, startSecondPart);
437             ConcealTexture_P(video, mb_start, mb_end, slice_counter);
438             return PV_FAIL;
439         }
440 
441         video->headerInfo.CBP[mbnum] |= (uint8)(CBPY << 2);
442         if (intra_MB || MBtype == MODE_INTER_Q)                     /*  04/26/01 */
443         {
444             status = GetMBheaderDataPart_DQUANT_DC(video, &QP);
445             if (status != PV_SUCCESS) return status;
446         }
447         video->usePrevQP = 1;        /*  04/27/01 */
448         QPMB[mbnum] = QP;
449     }
450 
451     video->usePrevQP = 0;  /*  04/27/01 */
452 
453     for (mbnum = mb_start; mbnum < mb_end; mbnum++)
454     {
455         video->mbnum = mbnum;
456         video->mbnum_row = PV_GET_ROW(mbnum, nMBPerRow);  /*  This is needed if nbnum is read from the packet header */
457         video->mbnum_col = mbnum - video->mbnum_row * nMBPerRow;
458 
459 
460         if (Mode[mbnum] != MODE_SKIPPED)
461         {
462             /* decode the DCT coeficients for the MB */
463             status = GetMBData_DataPart(video);
464             if (status != PV_SUCCESS)
465             {
466                 /* Report the error to the application.   06/20/2000 */
467                 VideoDecoderErrorDetected(video);
468 
469                 /* Conceal second partition,  5/15/2000 */
470                 movePointerTo(stream, startSecondPart);
471                 ConcealTexture_P(video, mb_start, mb_end, slice_counter);
472                 return status;
473             }
474             video->usePrevQP = 1;  /*  04/27/01 */
475         }
476         else
477         {   // SKIPPED
478 
479             /* Motion compensation and put it to video->mblock->pred_block */
480             SkippedMBMotionComp(video);
481 
482             //oscl_memset(video->predDCAC_row + video->mbnum_col, 0, sizeof(typeDCACStore)); /*  SKIPPED_ACDC */
483             //oscl_memset(video->predDCAC_col, 0, sizeof(typeDCACStore));
484             /*  08/08/2005 */
485             pbyte = (uint8*)(video->predDCAC_row + video->mbnum_col);
486             ZERO_OUT_64BYTES(pbyte);
487             pbyte = (uint8*)(video->predDCAC_col);
488             ZERO_OUT_64BYTES(pbyte);
489 
490         }
491     }
492 
493     valid_stuffing = validStuffing(stream);   /*  */
494     if (!valid_stuffing)
495     {
496         VideoDecoderErrorDetected(video);
497         movePointerTo(stream, startSecondPart); /*  */
498         ConcealTexture_P(video, mb_start, mb_end, slice_counter);
499 
500         return PV_FAIL;
501     }
502     return PV_SUCCESS;
503 }
504 
505 
506 /* ======================================================================== */
507 /*  Function : GetMBheaderDataPart_DQUANT_DC()                              */
508 /*  Date     : 04/26/2000                                                   */
509 /*  Purpose  : Decode DQUANT and DC in Data Partitioned Mode for both       */
510 /*             I-VOP and P-VOP.                                             */
511 /*  In/out   :                                                              */
512 /*  Return   : PV_SUCCESS if successed, PV_FAIL if failed.                  */
513 /*  Modified : 02/13/2001 new ACDC prediction structure,        */
514 /*                                       cleanup                            */
515 /* ======================================================================== */
GetMBheaderDataPart_DQUANT_DC(VideoDecData * video,int16 * QP)516 PV_STATUS GetMBheaderDataPart_DQUANT_DC(VideoDecData *video, int16 *QP)
517 {
518     PV_STATUS status = PV_SUCCESS;
519     BitstreamDecVideo *stream = video->bitstream;
520     int mbnum = video->mbnum;
521     int intra_dc_vlc_thr = video->currVop->intraDCVlcThr;
522     uint8 *Mode = video->headerInfo.Mode;
523     int  MBtype = Mode[mbnum];
524     typeDCStore *DC = video->predDC + mbnum;
525     int  comp;
526     Bool switched;
527     uint  DQUANT;
528     int16 QP_tmp;
529 
530     const static int  DQ_tab[4] = { -1, -2, 1, 2};
531 
532     if (MBtype & Q_MASK)             /* INTRA_Q || INTER_Q */
533     {
534         DQUANT = BitstreamReadBits16(stream, 2);
535         *QP += DQ_tab[DQUANT];
536 
537         if (*QP < 1) *QP = 1;
538         else if (*QP > 31) *QP = 31;
539     }
540     if (MBtype & INTRA_MASK)  /* INTRA || INTRA_Q */ /* no switch, code DC separately */
541     {
542         QP_tmp = *QP;                      /* running QP  04/26/01*/
543         switched = 0;
544         if (intra_dc_vlc_thr)                 /*  04/27/01 */
545         {
546             if (video->usePrevQP)
547                 QP_tmp = video->QPMB[mbnum-1];
548             switched = (intra_dc_vlc_thr == 7 || QP_tmp >= intra_dc_vlc_thr * 2 + 11);
549         }
550         if (!switched)
551         {
552             for (comp = 0; comp < 6; comp++)
553             {
554                 status = PV_DecodePredictedIntraDC(comp, stream, (*DC + comp));   /*  03/01/01 */
555                 if (status != PV_SUCCESS) return PV_FAIL;
556             }
557         }
558         else
559         {
560             for (comp = 0; comp < 6; comp++)
561             {
562                 (*DC)[comp] = 0;   /*  04/26/01 needed for switched case*/
563             }
564         }
565     }
566     return status;
567 }
568 
569 
570 /***********************************************************CommentBegin******
571 *       04/25/2000 : Initial modification to the new PV Lib format.
572 *       04/17/2001 : new ACDC pred structure
573 ***********************************************************CommentEnd********/
GetMBheaderDataPart_P(VideoDecData * video)574 PV_STATUS GetMBheaderDataPart_P(VideoDecData *video)
575 {
576     BitstreamDecVideo *stream = video->bitstream;
577     int mbnum = video->mbnum;
578     uint8 *Mode = video->headerInfo.Mode;
579     typeDCStore *DC = video->predDC + mbnum;
580     uint no_dct_flag;
581     int comp;
582     int MCBPC;
583 
584     no_dct_flag = BitstreamRead1Bits_INLINE(stream);
585 
586     if (no_dct_flag)
587     {
588         /* skipped macroblock */
589         Mode[mbnum] = MODE_SKIPPED;
590 
591         for (comp = 0; comp < 6; comp++)
592         {
593             (*DC)[comp] = mid_gray;
594             /*  ACDC REMOVE AC coefs are set in DecodeDataPart_P */
595         }
596     }
597     else
598     {
599         /* coded macroblock */
600         MCBPC = PV_VlcDecMCBPC_com_inter(stream);
601 
602         if (VLC_ERROR_DETECTED(MCBPC))
603         {
604             return PV_FAIL;
605         }
606 
607         Mode[mbnum] = (uint8)MBtype_mode[MCBPC & 7];
608         video->headerInfo.CBP[mbnum] = (uint8)((MCBPC >> 4) & 3);
609     }
610 
611     return PV_SUCCESS;
612 }
613 
614 
615 /***********************************************************CommentBegin******
616 *       04/17/01  new ACDC pred structure, reorganized code, cleanup
617 ***********************************************************CommentEnd********/
GetMBData_DataPart(VideoDecData * video)618 PV_STATUS GetMBData_DataPart(VideoDecData *video)
619 {
620     int mbnum = video->mbnum;
621     int16 *dataBlock;
622     MacroBlock *mblock = video->mblock;
623     int QP = video->QPMB[mbnum];
624     int32 offset;
625     PIXEL *c_comp;
626     int width = video->width;
627     int intra_dc_vlc_thr = video->currVop->intraDCVlcThr;
628     uint CBP = video->headerInfo.CBP[mbnum];
629     uint8 mode = video->headerInfo.Mode[mbnum];
630     int x_pos = video->mbnum_col;
631     typeDCStore *DC = video->predDC + mbnum;
632     int  ncoeffs[6], *no_coeff = mblock->no_coeff;
633     int  comp;
634     Bool  switched;
635     int QP_tmp = QP;
636 
637     int y_pos = video->mbnum_row;
638 #ifdef PV_POSTPROC_ON
639     uint8 *pp_mod[6];
640     int TotalMB = video->nTotalMB;
641     int MB_in_width = video->nMBPerRow;
642 #endif
643 
644 
645 
646     /*****
647     *     Decoding of the 6 blocks (depending on transparent pattern)
648     *****/
649 #ifdef PV_POSTPROC_ON
650     if (video->postFilterType != PV_NO_POST_PROC)
651     {
652         /** post-processing ***/
653         pp_mod[0] = video->pstprcTypCur + (y_pos << 1) * (MB_in_width << 1) + (x_pos << 1);
654         pp_mod[1] = pp_mod[0] + 1;
655         pp_mod[2] = pp_mod[0] + (MB_in_width << 1);
656         pp_mod[3] = pp_mod[2] + 1;
657         pp_mod[4] = video->pstprcTypCur + (TotalMB << 2) + mbnum;
658         pp_mod[5] = pp_mod[4] + TotalMB;
659     }
660 #endif
661 
662     /*  oscl_memset(mblock->block, 0, sizeof(typeMBStore));    Aug 9,2005 */
663 
664     if (mode & INTRA_MASK) /* MODE_INTRA || mode == MODE_INTRA_Q */
665     {
666         switched = 0;
667         if (intra_dc_vlc_thr)
668         {
669             if (video->usePrevQP)
670                 QP_tmp = video->QPMB[mbnum-1];   /* running QP  04/26/01 */
671 
672             switched = (intra_dc_vlc_thr == 7 || QP_tmp >= intra_dc_vlc_thr * 2 + 11);
673         }
674 
675         mblock->DCScalarLum = cal_dc_scaler(QP, LUMINANCE_DC_TYPE);     /*   ACDC 03/01/01 */
676         mblock->DCScalarChr = cal_dc_scaler(QP, CHROMINANCE_DC_TYPE);
677 
678         for (comp = 0; comp < 6; comp++)
679         {
680             dataBlock = mblock->block[comp];    /*, 10/20/2000 */
681 
682             dataBlock[0] = (*DC)[comp];
683 
684             ncoeffs[comp] = VlcDequantH263IntraBlock(video, comp,
685                             switched, mblock->bitmapcol[comp], &mblock->bitmaprow[comp]);
686 
687             if (VLC_ERROR_DETECTED(ncoeffs[comp]))         /*  */
688             {
689                 if (switched)
690                     return PV_FAIL;
691                 else
692                 {
693                     ncoeffs[comp] = 1;
694                     oscl_memset((dataBlock + 1), 0, sizeof(int16)*63);
695                 }
696             }
697             no_coeff[comp] = ncoeffs[comp];
698             /*  modified to new semaphore for post-proc */
699             // Future work:: can be combined in the dequant function
700             // @todo Deblocking Semaphore for INTRA block
701 #ifdef PV_POSTPROC_ON
702             if (video->postFilterType != PV_NO_POST_PROC)
703                 *pp_mod[comp] = (uint8) PostProcSemaphore(dataBlock);
704 #endif
705         }
706         MBlockIDCT(video);
707     }
708     else /* MODE INTER*/
709     {
710 
711 
712 
713 
714         MBMotionComp(video, CBP);
715         offset = (int32)(y_pos << 4) * width + (x_pos << 4);
716         c_comp  = video->currVop->yChan + offset;
717 
718 
719         for (comp = 0; comp < 4; comp++)
720         {
721             (*DC)[comp] = mid_gray;
722 
723             if (CBP & (1 << (5 - comp)))
724             {
725                 ncoeffs[comp] = VlcDequantH263InterBlock(video, comp,
726                                 mblock->bitmapcol[comp], &mblock->bitmaprow[comp]);
727                 if (VLC_ERROR_DETECTED(ncoeffs[comp]))
728                     return PV_FAIL;
729 
730 
731                 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],
732                           mblock->bitmapcol[comp], mblock->bitmaprow[comp]);
733 
734             }
735             else
736             {
737                 ncoeffs[comp] = 0;
738             }
739 
740             /*  @todo Deblocking Semaphore for INTRA block, for inter just test for ringing  */
741 #ifdef PV_POSTPROC_ON
742             if (video->postFilterType != PV_NO_POST_PROC)
743                 *pp_mod[comp] = (uint8)((ncoeffs[comp] > 3) ? 4 : 0);
744 #endif
745         }
746 
747         (*DC)[4] = mid_gray;
748         if (CBP & 2)
749         {
750             ncoeffs[4] = VlcDequantH263InterBlock(video, 4,
751                                                   mblock->bitmapcol[4], &mblock->bitmaprow[4]);
752             if (VLC_ERROR_DETECTED(ncoeffs[4]))
753                 return PV_FAIL;
754 
755             BlockIDCT(video->currVop->uChan + (offset >> 2) + (x_pos << 2), mblock->pred_block + 256, mblock->block[4], width >> 1, ncoeffs[4],
756                       mblock->bitmapcol[4], mblock->bitmaprow[4]);
757 
758         }
759         else
760         {
761             ncoeffs[4] = 0;
762         }
763 #ifdef PV_POSTPROC_ON
764         if (video->postFilterType != PV_NO_POST_PROC)
765             *pp_mod[4] = (uint8)((ncoeffs[4] > 3) ? 4 : 0);
766 #endif
767         (*DC)[5] = mid_gray;
768         if (CBP & 1)
769         {
770             ncoeffs[5] = VlcDequantH263InterBlock(video, 5,
771                                                   mblock->bitmapcol[5], &mblock->bitmaprow[5]);
772             if (VLC_ERROR_DETECTED(ncoeffs[5]))
773                 return PV_FAIL;
774 
775             BlockIDCT(video->currVop->vChan + (offset >> 2) + (x_pos << 2), mblock->pred_block + 264, mblock->block[5], width >> 1, ncoeffs[5],
776                       mblock->bitmapcol[5], mblock->bitmaprow[5]);
777 
778         }
779         else
780         {
781             ncoeffs[5] = 0;
782         }
783 #ifdef PV_POSTPROC_ON
784         if (video->postFilterType != PV_NO_POST_PROC)
785             *pp_mod[5] = (uint8)((ncoeffs[5] > 3) ? 4 : 0);
786 #endif
787 
788 
789 
790 
791         /* Motion compensation and put it to video->mblock->pred_block */
792     }
793     return PV_SUCCESS;
794 }
795