10c1bc742181ded4930842b46e9507372f0b1b963James Dong/*
20c1bc742181ded4930842b46e9507372f0b1b963James Dong * Copyright (C) 2009 The Android Open Source Project
30c1bc742181ded4930842b46e9507372f0b1b963James Dong *
40c1bc742181ded4930842b46e9507372f0b1b963James Dong * Licensed under the Apache License, Version 2.0 (the "License");
50c1bc742181ded4930842b46e9507372f0b1b963James Dong * you may not use this file except in compliance with the License.
60c1bc742181ded4930842b46e9507372f0b1b963James Dong * You may obtain a copy of the License at
70c1bc742181ded4930842b46e9507372f0b1b963James Dong *
80c1bc742181ded4930842b46e9507372f0b1b963James Dong *      http://www.apache.org/licenses/LICENSE-2.0
90c1bc742181ded4930842b46e9507372f0b1b963James Dong *
100c1bc742181ded4930842b46e9507372f0b1b963James Dong * Unless required by applicable law or agreed to in writing, software
110c1bc742181ded4930842b46e9507372f0b1b963James Dong * distributed under the License is distributed on an "AS IS" BASIS,
120c1bc742181ded4930842b46e9507372f0b1b963James Dong * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
130c1bc742181ded4930842b46e9507372f0b1b963James Dong * See the License for the specific language governing permissions and
140c1bc742181ded4930842b46e9507372f0b1b963James Dong * limitations under the License.
150c1bc742181ded4930842b46e9507372f0b1b963James Dong */
160c1bc742181ded4930842b46e9507372f0b1b963James Dong
170c1bc742181ded4930842b46e9507372f0b1b963James Dong/*------------------------------------------------------------------------------
180c1bc742181ded4930842b46e9507372f0b1b963James Dong
190c1bc742181ded4930842b46e9507372f0b1b963James Dong    Table of contents
200c1bc742181ded4930842b46e9507372f0b1b963James Dong
210c1bc742181ded4930842b46e9507372f0b1b963James Dong     1. Include headers
220c1bc742181ded4930842b46e9507372f0b1b963James Dong     2. External compiler flags
230c1bc742181ded4930842b46e9507372f0b1b963James Dong     3. Module defines
240c1bc742181ded4930842b46e9507372f0b1b963James Dong     4. Local function prototypes
250c1bc742181ded4930842b46e9507372f0b1b963James Dong     5. Functions
260c1bc742181ded4930842b46e9507372f0b1b963James Dong          h264bsdDecodePicOrderCnt
270c1bc742181ded4930842b46e9507372f0b1b963James Dong
280c1bc742181ded4930842b46e9507372f0b1b963James Dong------------------------------------------------------------------------------*/
290c1bc742181ded4930842b46e9507372f0b1b963James Dong
300c1bc742181ded4930842b46e9507372f0b1b963James Dong/*------------------------------------------------------------------------------
310c1bc742181ded4930842b46e9507372f0b1b963James Dong    1. Include headers
320c1bc742181ded4930842b46e9507372f0b1b963James Dong------------------------------------------------------------------------------*/
330c1bc742181ded4930842b46e9507372f0b1b963James Dong
340c1bc742181ded4930842b46e9507372f0b1b963James Dong#include "h264bsd_util.h"
350c1bc742181ded4930842b46e9507372f0b1b963James Dong#include "h264bsd_pic_order_cnt.h"
360c1bc742181ded4930842b46e9507372f0b1b963James Dong
370c1bc742181ded4930842b46e9507372f0b1b963James Dong/*------------------------------------------------------------------------------
380c1bc742181ded4930842b46e9507372f0b1b963James Dong    2. External compiler flags
390c1bc742181ded4930842b46e9507372f0b1b963James Dong--------------------------------------------------------------------------------
400c1bc742181ded4930842b46e9507372f0b1b963James Dong
410c1bc742181ded4930842b46e9507372f0b1b963James Dong--------------------------------------------------------------------------------
420c1bc742181ded4930842b46e9507372f0b1b963James Dong    3. Module defines
430c1bc742181ded4930842b46e9507372f0b1b963James Dong------------------------------------------------------------------------------*/
440c1bc742181ded4930842b46e9507372f0b1b963James Dong
450c1bc742181ded4930842b46e9507372f0b1b963James Dong/*------------------------------------------------------------------------------
460c1bc742181ded4930842b46e9507372f0b1b963James Dong    4. Local function prototypes
470c1bc742181ded4930842b46e9507372f0b1b963James Dong------------------------------------------------------------------------------*/
480c1bc742181ded4930842b46e9507372f0b1b963James Dong
490c1bc742181ded4930842b46e9507372f0b1b963James Dong/*------------------------------------------------------------------------------
500c1bc742181ded4930842b46e9507372f0b1b963James Dong
510c1bc742181ded4930842b46e9507372f0b1b963James Dong    Function: h264bsdDecodePicOrderCnt
520c1bc742181ded4930842b46e9507372f0b1b963James Dong
530c1bc742181ded4930842b46e9507372f0b1b963James Dong        Functional description:
540c1bc742181ded4930842b46e9507372f0b1b963James Dong            Compute picture order count for a picture. Function implements
550c1bc742181ded4930842b46e9507372f0b1b963James Dong            computation of all POC types (0, 1 and 2), type is obtained from
560c1bc742181ded4930842b46e9507372f0b1b963James Dong            sps. See standard for description of the POC types and how POC is
570c1bc742181ded4930842b46e9507372f0b1b963James Dong            computed for each type.
580c1bc742181ded4930842b46e9507372f0b1b963James Dong
590c1bc742181ded4930842b46e9507372f0b1b963James Dong            Function returns the minimum of top field and bottom field pic
600c1bc742181ded4930842b46e9507372f0b1b963James Dong            order counts.
610c1bc742181ded4930842b46e9507372f0b1b963James Dong
620c1bc742181ded4930842b46e9507372f0b1b963James Dong        Inputs:
630c1bc742181ded4930842b46e9507372f0b1b963James Dong            poc         pointer to previous results
640c1bc742181ded4930842b46e9507372f0b1b963James Dong            sps         pointer to sequence parameter set
650c1bc742181ded4930842b46e9507372f0b1b963James Dong            slicHeader  pointer to current slice header, frame number and
660c1bc742181ded4930842b46e9507372f0b1b963James Dong                        other params needed for POC computation
670c1bc742181ded4930842b46e9507372f0b1b963James Dong            pNalUnit    pointer to current NAL unit structrue, function needs
680c1bc742181ded4930842b46e9507372f0b1b963James Dong                        to know if this is an IDR picture and also if this is
690c1bc742181ded4930842b46e9507372f0b1b963James Dong                        a reference picture
700c1bc742181ded4930842b46e9507372f0b1b963James Dong
710c1bc742181ded4930842b46e9507372f0b1b963James Dong        Outputs:
720c1bc742181ded4930842b46e9507372f0b1b963James Dong            poc         results stored here for computation of next POC
730c1bc742181ded4930842b46e9507372f0b1b963James Dong
740c1bc742181ded4930842b46e9507372f0b1b963James Dong        Returns:
750c1bc742181ded4930842b46e9507372f0b1b963James Dong            picture order count
760c1bc742181ded4930842b46e9507372f0b1b963James Dong
770c1bc742181ded4930842b46e9507372f0b1b963James Dong------------------------------------------------------------------------------*/
780c1bc742181ded4930842b46e9507372f0b1b963James Dong
790c1bc742181ded4930842b46e9507372f0b1b963James Dongi32 h264bsdDecodePicOrderCnt(pocStorage_t *poc, seqParamSet_t *sps,
800c1bc742181ded4930842b46e9507372f0b1b963James Dong    sliceHeader_t *pSliceHeader, nalUnit_t *pNalUnit)
810c1bc742181ded4930842b46e9507372f0b1b963James Dong{
820c1bc742181ded4930842b46e9507372f0b1b963James Dong
830c1bc742181ded4930842b46e9507372f0b1b963James Dong/* Variables */
840c1bc742181ded4930842b46e9507372f0b1b963James Dong
850c1bc742181ded4930842b46e9507372f0b1b963James Dong    u32 i;
860c1bc742181ded4930842b46e9507372f0b1b963James Dong    i32 picOrderCnt;
870c1bc742181ded4930842b46e9507372f0b1b963James Dong    u32 frameNumOffset, absFrameNum, picOrderCntCycleCnt;
880c1bc742181ded4930842b46e9507372f0b1b963James Dong    u32 frameNumInPicOrderCntCycle;
890c1bc742181ded4930842b46e9507372f0b1b963James Dong    i32 expectedDeltaPicOrderCntCycle;
900c1bc742181ded4930842b46e9507372f0b1b963James Dong    u32 containsMmco5;
910c1bc742181ded4930842b46e9507372f0b1b963James Dong
920c1bc742181ded4930842b46e9507372f0b1b963James Dong/* Code */
930c1bc742181ded4930842b46e9507372f0b1b963James Dong
940c1bc742181ded4930842b46e9507372f0b1b963James Dong    ASSERT(poc);
950c1bc742181ded4930842b46e9507372f0b1b963James Dong    ASSERT(sps);
960c1bc742181ded4930842b46e9507372f0b1b963James Dong    ASSERT(pSliceHeader);
970c1bc742181ded4930842b46e9507372f0b1b963James Dong    ASSERT(pNalUnit);
980c1bc742181ded4930842b46e9507372f0b1b963James Dong    ASSERT(sps->picOrderCntType <= 2);
990c1bc742181ded4930842b46e9507372f0b1b963James Dong
1000c1bc742181ded4930842b46e9507372f0b1b963James Dong#if 0
1010c1bc742181ded4930842b46e9507372f0b1b963James Dong    /* JanSa: I don't think this is necessary, don't see any reason to
1020c1bc742181ded4930842b46e9507372f0b1b963James Dong     * increment prevFrameNum one by one instead of one big increment.
1030c1bc742181ded4930842b46e9507372f0b1b963James Dong     * However, standard specifies that this should be done -> if someone
1040c1bc742181ded4930842b46e9507372f0b1b963James Dong     * figures out any case when the outcome would be different for step by
1050c1bc742181ded4930842b46e9507372f0b1b963James Dong     * step increment, this part of the code should be enabled */
1060c1bc742181ded4930842b46e9507372f0b1b963James Dong
1070c1bc742181ded4930842b46e9507372f0b1b963James Dong    /* if there was a gap in frame numbering and picOrderCntType is 1 or 2 ->
1080c1bc742181ded4930842b46e9507372f0b1b963James Dong     * "compute" pic order counts for non-existing frames. These are not
1090c1bc742181ded4930842b46e9507372f0b1b963James Dong     * actually computed, but process needs to be done to update the
1100c1bc742181ded4930842b46e9507372f0b1b963James Dong     * prevFrameNum and prevFrameNumOffset */
1110c1bc742181ded4930842b46e9507372f0b1b963James Dong    if ( sps->picOrderCntType > 0 &&
1120c1bc742181ded4930842b46e9507372f0b1b963James Dong         pSliceHeader->frameNum != poc->prevFrameNum &&
1130c1bc742181ded4930842b46e9507372f0b1b963James Dong         pSliceHeader->frameNum != ((poc->prevFrameNum + 1) % sps->maxFrameNum))
1140c1bc742181ded4930842b46e9507372f0b1b963James Dong    {
1150c1bc742181ded4930842b46e9507372f0b1b963James Dong
1160c1bc742181ded4930842b46e9507372f0b1b963James Dong        /* use variable i for unUsedShortTermFrameNum */
1170c1bc742181ded4930842b46e9507372f0b1b963James Dong        i = (poc->prevFrameNum + 1) % sps->maxFrameNum;
1180c1bc742181ded4930842b46e9507372f0b1b963James Dong
1190c1bc742181ded4930842b46e9507372f0b1b963James Dong        do
1200c1bc742181ded4930842b46e9507372f0b1b963James Dong        {
1210c1bc742181ded4930842b46e9507372f0b1b963James Dong            if (poc->prevFrameNum > i)
1220c1bc742181ded4930842b46e9507372f0b1b963James Dong                frameNumOffset = poc->prevFrameNumOffset + sps->maxFrameNum;
1230c1bc742181ded4930842b46e9507372f0b1b963James Dong            else
1240c1bc742181ded4930842b46e9507372f0b1b963James Dong                frameNumOffset = poc->prevFrameNumOffset;
1250c1bc742181ded4930842b46e9507372f0b1b963James Dong
1260c1bc742181ded4930842b46e9507372f0b1b963James Dong            poc->prevFrameNumOffset = frameNumOffset;
1270c1bc742181ded4930842b46e9507372f0b1b963James Dong            poc->prevFrameNum = i;
1280c1bc742181ded4930842b46e9507372f0b1b963James Dong
1290c1bc742181ded4930842b46e9507372f0b1b963James Dong            i = (i + 1) % sps->maxFrameNum;
1300c1bc742181ded4930842b46e9507372f0b1b963James Dong
1310c1bc742181ded4930842b46e9507372f0b1b963James Dong        } while (i != pSliceHeader->frameNum);
1320c1bc742181ded4930842b46e9507372f0b1b963James Dong    }
1330c1bc742181ded4930842b46e9507372f0b1b963James Dong#endif
1340c1bc742181ded4930842b46e9507372f0b1b963James Dong
1350c1bc742181ded4930842b46e9507372f0b1b963James Dong    /* check if current slice includes mmco equal to 5 */
1360c1bc742181ded4930842b46e9507372f0b1b963James Dong    containsMmco5 = HANTRO_FALSE;
1370c1bc742181ded4930842b46e9507372f0b1b963James Dong    if (pSliceHeader->decRefPicMarking.adaptiveRefPicMarkingModeFlag)
1380c1bc742181ded4930842b46e9507372f0b1b963James Dong    {
1390c1bc742181ded4930842b46e9507372f0b1b963James Dong        i = 0;
1400c1bc742181ded4930842b46e9507372f0b1b963James Dong        while (pSliceHeader->decRefPicMarking.operation[i].
1410c1bc742181ded4930842b46e9507372f0b1b963James Dong            memoryManagementControlOperation)
1420c1bc742181ded4930842b46e9507372f0b1b963James Dong        {
1430c1bc742181ded4930842b46e9507372f0b1b963James Dong            if (pSliceHeader->decRefPicMarking.operation[i].
1440c1bc742181ded4930842b46e9507372f0b1b963James Dong                memoryManagementControlOperation == 5)
1450c1bc742181ded4930842b46e9507372f0b1b963James Dong            {
1460c1bc742181ded4930842b46e9507372f0b1b963James Dong                containsMmco5 = HANTRO_TRUE;
1470c1bc742181ded4930842b46e9507372f0b1b963James Dong                break;
1480c1bc742181ded4930842b46e9507372f0b1b963James Dong            }
1490c1bc742181ded4930842b46e9507372f0b1b963James Dong            i++;
1500c1bc742181ded4930842b46e9507372f0b1b963James Dong        }
1510c1bc742181ded4930842b46e9507372f0b1b963James Dong    }
1520c1bc742181ded4930842b46e9507372f0b1b963James Dong    switch (sps->picOrderCntType)
1530c1bc742181ded4930842b46e9507372f0b1b963James Dong    {
1540c1bc742181ded4930842b46e9507372f0b1b963James Dong
1550c1bc742181ded4930842b46e9507372f0b1b963James Dong        case 0:
1560c1bc742181ded4930842b46e9507372f0b1b963James Dong            /* set prevPicOrderCnt values for IDR frame */
1570c1bc742181ded4930842b46e9507372f0b1b963James Dong            if (IS_IDR_NAL_UNIT(pNalUnit))
1580c1bc742181ded4930842b46e9507372f0b1b963James Dong            {
1590c1bc742181ded4930842b46e9507372f0b1b963James Dong                poc->prevPicOrderCntMsb = 0;
1600c1bc742181ded4930842b46e9507372f0b1b963James Dong                poc->prevPicOrderCntLsb = 0;
1610c1bc742181ded4930842b46e9507372f0b1b963James Dong            }
1620c1bc742181ded4930842b46e9507372f0b1b963James Dong
1630c1bc742181ded4930842b46e9507372f0b1b963James Dong            /* compute picOrderCntMsb (stored in picOrderCnt variable) */
1640c1bc742181ded4930842b46e9507372f0b1b963James Dong            if ( (pSliceHeader->picOrderCntLsb < poc->prevPicOrderCntLsb) &&
1650c1bc742181ded4930842b46e9507372f0b1b963James Dong                ((poc->prevPicOrderCntLsb - pSliceHeader->picOrderCntLsb) >=
1660c1bc742181ded4930842b46e9507372f0b1b963James Dong                 sps->maxPicOrderCntLsb/2) )
1670c1bc742181ded4930842b46e9507372f0b1b963James Dong            {
1680c1bc742181ded4930842b46e9507372f0b1b963James Dong                picOrderCnt = poc->prevPicOrderCntMsb +
1690c1bc742181ded4930842b46e9507372f0b1b963James Dong                    (i32)sps->maxPicOrderCntLsb;
1700c1bc742181ded4930842b46e9507372f0b1b963James Dong            }
1710c1bc742181ded4930842b46e9507372f0b1b963James Dong            else if ((pSliceHeader->picOrderCntLsb > poc->prevPicOrderCntLsb) &&
1720c1bc742181ded4930842b46e9507372f0b1b963James Dong                ((pSliceHeader->picOrderCntLsb - poc->prevPicOrderCntLsb) >
1730c1bc742181ded4930842b46e9507372f0b1b963James Dong                 sps->maxPicOrderCntLsb/2) )
1740c1bc742181ded4930842b46e9507372f0b1b963James Dong            {
1750c1bc742181ded4930842b46e9507372f0b1b963James Dong                picOrderCnt = poc->prevPicOrderCntMsb -
1760c1bc742181ded4930842b46e9507372f0b1b963James Dong                    (i32)sps->maxPicOrderCntLsb;
1770c1bc742181ded4930842b46e9507372f0b1b963James Dong            }
1780c1bc742181ded4930842b46e9507372f0b1b963James Dong            else
1790c1bc742181ded4930842b46e9507372f0b1b963James Dong                picOrderCnt = poc->prevPicOrderCntMsb;
1800c1bc742181ded4930842b46e9507372f0b1b963James Dong
1810c1bc742181ded4930842b46e9507372f0b1b963James Dong            /* standard specifies that prevPicOrderCntMsb is from previous
1820c1bc742181ded4930842b46e9507372f0b1b963James Dong             * rererence frame -> replace old value only if current frame is
1830c1bc742181ded4930842b46e9507372f0b1b963James Dong             * rererence frame */
1840c1bc742181ded4930842b46e9507372f0b1b963James Dong            if (pNalUnit->nalRefIdc)
1850c1bc742181ded4930842b46e9507372f0b1b963James Dong                poc->prevPicOrderCntMsb = picOrderCnt;
1860c1bc742181ded4930842b46e9507372f0b1b963James Dong
1870c1bc742181ded4930842b46e9507372f0b1b963James Dong            /* compute top field order cnt (stored in picOrderCnt) */
1880c1bc742181ded4930842b46e9507372f0b1b963James Dong            picOrderCnt += (i32)pSliceHeader->picOrderCntLsb;
1890c1bc742181ded4930842b46e9507372f0b1b963James Dong
1900c1bc742181ded4930842b46e9507372f0b1b963James Dong            /* if delta for bottom field is negative -> bottom will be the
1910c1bc742181ded4930842b46e9507372f0b1b963James Dong             * minimum pic order count */
1920c1bc742181ded4930842b46e9507372f0b1b963James Dong            if (pSliceHeader->deltaPicOrderCntBottom < 0)
1930c1bc742181ded4930842b46e9507372f0b1b963James Dong                picOrderCnt += pSliceHeader->deltaPicOrderCntBottom;
1940c1bc742181ded4930842b46e9507372f0b1b963James Dong
1950c1bc742181ded4930842b46e9507372f0b1b963James Dong            /* standard specifies that prevPicOrderCntLsb is from previous
1960c1bc742181ded4930842b46e9507372f0b1b963James Dong             * rererence frame -> replace old value only if current frame is
1970c1bc742181ded4930842b46e9507372f0b1b963James Dong             * rererence frame */
1980c1bc742181ded4930842b46e9507372f0b1b963James Dong            if (pNalUnit->nalRefIdc)
1990c1bc742181ded4930842b46e9507372f0b1b963James Dong            {
2000c1bc742181ded4930842b46e9507372f0b1b963James Dong                /* if current frame contains mmco5 -> modify values to be
2010c1bc742181ded4930842b46e9507372f0b1b963James Dong                 * stored */
2020c1bc742181ded4930842b46e9507372f0b1b963James Dong                if (containsMmco5)
2030c1bc742181ded4930842b46e9507372f0b1b963James Dong                {
2040c1bc742181ded4930842b46e9507372f0b1b963James Dong                    poc->prevPicOrderCntMsb = 0;
2050c1bc742181ded4930842b46e9507372f0b1b963James Dong                    /* prevPicOrderCntLsb should be the top field picOrderCnt
2060c1bc742181ded4930842b46e9507372f0b1b963James Dong                     * if previous frame included mmco5. Top field picOrderCnt
2070c1bc742181ded4930842b46e9507372f0b1b963James Dong                     * for frames containing mmco5 is obtained by subtracting
2080c1bc742181ded4930842b46e9507372f0b1b963James Dong                     * the picOrderCnt from original top field order count ->
2090c1bc742181ded4930842b46e9507372f0b1b963James Dong                     * value is zero if top field was the minimum, i.e. delta
2100c1bc742181ded4930842b46e9507372f0b1b963James Dong                     * for bottom was positive, otherwise value is
2110c1bc742181ded4930842b46e9507372f0b1b963James Dong                     * -deltaPicOrderCntBottom */
2120c1bc742181ded4930842b46e9507372f0b1b963James Dong                    if (pSliceHeader->deltaPicOrderCntBottom < 0)
2130c1bc742181ded4930842b46e9507372f0b1b963James Dong                        poc->prevPicOrderCntLsb =
2140c1bc742181ded4930842b46e9507372f0b1b963James Dong                            (u32)(-pSliceHeader->deltaPicOrderCntBottom);
2150c1bc742181ded4930842b46e9507372f0b1b963James Dong                    else
2160c1bc742181ded4930842b46e9507372f0b1b963James Dong                        poc->prevPicOrderCntLsb = 0;
2170c1bc742181ded4930842b46e9507372f0b1b963James Dong                    picOrderCnt = 0;
2180c1bc742181ded4930842b46e9507372f0b1b963James Dong                }
2190c1bc742181ded4930842b46e9507372f0b1b963James Dong                else
2200c1bc742181ded4930842b46e9507372f0b1b963James Dong                {
2210c1bc742181ded4930842b46e9507372f0b1b963James Dong                    poc->prevPicOrderCntLsb = pSliceHeader->picOrderCntLsb;
2220c1bc742181ded4930842b46e9507372f0b1b963James Dong                }
2230c1bc742181ded4930842b46e9507372f0b1b963James Dong            }
2240c1bc742181ded4930842b46e9507372f0b1b963James Dong
2250c1bc742181ded4930842b46e9507372f0b1b963James Dong            break;
2260c1bc742181ded4930842b46e9507372f0b1b963James Dong
2270c1bc742181ded4930842b46e9507372f0b1b963James Dong        case 1:
2280c1bc742181ded4930842b46e9507372f0b1b963James Dong
2290c1bc742181ded4930842b46e9507372f0b1b963James Dong            /* step 1 (in the description in the standard) */
2300c1bc742181ded4930842b46e9507372f0b1b963James Dong            if (IS_IDR_NAL_UNIT(pNalUnit))
2310c1bc742181ded4930842b46e9507372f0b1b963James Dong                frameNumOffset = 0;
2320c1bc742181ded4930842b46e9507372f0b1b963James Dong            else if (poc->prevFrameNum > pSliceHeader->frameNum)
2330c1bc742181ded4930842b46e9507372f0b1b963James Dong                frameNumOffset = poc->prevFrameNumOffset + sps->maxFrameNum;
2340c1bc742181ded4930842b46e9507372f0b1b963James Dong            else
2350c1bc742181ded4930842b46e9507372f0b1b963James Dong                frameNumOffset = poc->prevFrameNumOffset;
2360c1bc742181ded4930842b46e9507372f0b1b963James Dong
2370c1bc742181ded4930842b46e9507372f0b1b963James Dong            /* step 2 */
2380c1bc742181ded4930842b46e9507372f0b1b963James Dong            if (sps->numRefFramesInPicOrderCntCycle)
2390c1bc742181ded4930842b46e9507372f0b1b963James Dong                absFrameNum = frameNumOffset + pSliceHeader->frameNum;
2400c1bc742181ded4930842b46e9507372f0b1b963James Dong            else
2410c1bc742181ded4930842b46e9507372f0b1b963James Dong                absFrameNum = 0;
2420c1bc742181ded4930842b46e9507372f0b1b963James Dong
2430c1bc742181ded4930842b46e9507372f0b1b963James Dong            if (pNalUnit->nalRefIdc == 0 && absFrameNum > 0)
2440c1bc742181ded4930842b46e9507372f0b1b963James Dong                absFrameNum -= 1;
2450c1bc742181ded4930842b46e9507372f0b1b963James Dong
2460c1bc742181ded4930842b46e9507372f0b1b963James Dong            /* step 3 */
2470c1bc742181ded4930842b46e9507372f0b1b963James Dong            if (absFrameNum > 0)
2480c1bc742181ded4930842b46e9507372f0b1b963James Dong            {
2490c1bc742181ded4930842b46e9507372f0b1b963James Dong                picOrderCntCycleCnt =
2500c1bc742181ded4930842b46e9507372f0b1b963James Dong                    (absFrameNum - 1)/sps->numRefFramesInPicOrderCntCycle;
2510c1bc742181ded4930842b46e9507372f0b1b963James Dong                frameNumInPicOrderCntCycle =
2520c1bc742181ded4930842b46e9507372f0b1b963James Dong                    (absFrameNum - 1)%sps->numRefFramesInPicOrderCntCycle;
2530c1bc742181ded4930842b46e9507372f0b1b963James Dong            }
2540c1bc742181ded4930842b46e9507372f0b1b963James Dong
2550c1bc742181ded4930842b46e9507372f0b1b963James Dong            /* step 4 */
2560c1bc742181ded4930842b46e9507372f0b1b963James Dong            expectedDeltaPicOrderCntCycle = 0;
2570c1bc742181ded4930842b46e9507372f0b1b963James Dong            for (i = 0; i < sps->numRefFramesInPicOrderCntCycle; i++)
2580c1bc742181ded4930842b46e9507372f0b1b963James Dong                expectedDeltaPicOrderCntCycle += sps->offsetForRefFrame[i];
2590c1bc742181ded4930842b46e9507372f0b1b963James Dong
2600c1bc742181ded4930842b46e9507372f0b1b963James Dong            /* step 5 (picOrderCnt used to store expectedPicOrderCnt) */
2610c1bc742181ded4930842b46e9507372f0b1b963James Dong            /*lint -esym(644,picOrderCntCycleCnt) always initialized */
2620c1bc742181ded4930842b46e9507372f0b1b963James Dong            /*lint -esym(644,frameNumInPicOrderCntCycle) always initialized */
2630c1bc742181ded4930842b46e9507372f0b1b963James Dong            if (absFrameNum > 0)
2640c1bc742181ded4930842b46e9507372f0b1b963James Dong            {
2650c1bc742181ded4930842b46e9507372f0b1b963James Dong                picOrderCnt =
2660c1bc742181ded4930842b46e9507372f0b1b963James Dong                    (i32)picOrderCntCycleCnt * expectedDeltaPicOrderCntCycle;
2670c1bc742181ded4930842b46e9507372f0b1b963James Dong                for (i = 0; i <= frameNumInPicOrderCntCycle; i++)
2680c1bc742181ded4930842b46e9507372f0b1b963James Dong                    picOrderCnt += sps->offsetForRefFrame[i];
2690c1bc742181ded4930842b46e9507372f0b1b963James Dong            }
2700c1bc742181ded4930842b46e9507372f0b1b963James Dong            else
2710c1bc742181ded4930842b46e9507372f0b1b963James Dong                picOrderCnt = 0;
2720c1bc742181ded4930842b46e9507372f0b1b963James Dong
2730c1bc742181ded4930842b46e9507372f0b1b963James Dong            if (pNalUnit->nalRefIdc == 0)
2740c1bc742181ded4930842b46e9507372f0b1b963James Dong                picOrderCnt += sps->offsetForNonRefPic;
2750c1bc742181ded4930842b46e9507372f0b1b963James Dong
2760c1bc742181ded4930842b46e9507372f0b1b963James Dong            /* step 6 (picOrderCnt is top field order cnt if delta for bottom
2770c1bc742181ded4930842b46e9507372f0b1b963James Dong             * is positive, otherwise it is bottom field order cnt) */
2780c1bc742181ded4930842b46e9507372f0b1b963James Dong            picOrderCnt += pSliceHeader->deltaPicOrderCnt[0];
2790c1bc742181ded4930842b46e9507372f0b1b963James Dong
2800c1bc742181ded4930842b46e9507372f0b1b963James Dong            if ( (sps->offsetForTopToBottomField +
2810c1bc742181ded4930842b46e9507372f0b1b963James Dong                    pSliceHeader->deltaPicOrderCnt[1]) < 0 )
2820c1bc742181ded4930842b46e9507372f0b1b963James Dong            {
2830c1bc742181ded4930842b46e9507372f0b1b963James Dong                picOrderCnt += sps->offsetForTopToBottomField +
2840c1bc742181ded4930842b46e9507372f0b1b963James Dong                    pSliceHeader->deltaPicOrderCnt[1];
2850c1bc742181ded4930842b46e9507372f0b1b963James Dong            }
2860c1bc742181ded4930842b46e9507372f0b1b963James Dong
2870c1bc742181ded4930842b46e9507372f0b1b963James Dong            /* if current picture contains mmco5 -> set prevFrameNumOffset and
2880c1bc742181ded4930842b46e9507372f0b1b963James Dong             * prevFrameNum to 0 for computation of picOrderCnt of next
2890c1bc742181ded4930842b46e9507372f0b1b963James Dong             * frame, otherwise store frameNum and frameNumOffset to poc
2900c1bc742181ded4930842b46e9507372f0b1b963James Dong             * structure */
2910c1bc742181ded4930842b46e9507372f0b1b963James Dong            if (!containsMmco5)
2920c1bc742181ded4930842b46e9507372f0b1b963James Dong            {
2930c1bc742181ded4930842b46e9507372f0b1b963James Dong                poc->prevFrameNumOffset = frameNumOffset;
2940c1bc742181ded4930842b46e9507372f0b1b963James Dong                poc->prevFrameNum = pSliceHeader->frameNum;
2950c1bc742181ded4930842b46e9507372f0b1b963James Dong            }
2960c1bc742181ded4930842b46e9507372f0b1b963James Dong            else
2970c1bc742181ded4930842b46e9507372f0b1b963James Dong            {
2980c1bc742181ded4930842b46e9507372f0b1b963James Dong                poc->prevFrameNumOffset = 0;
2990c1bc742181ded4930842b46e9507372f0b1b963James Dong                poc->prevFrameNum = 0;
3000c1bc742181ded4930842b46e9507372f0b1b963James Dong                picOrderCnt = 0;
3010c1bc742181ded4930842b46e9507372f0b1b963James Dong            }
3020c1bc742181ded4930842b46e9507372f0b1b963James Dong            break;
3030c1bc742181ded4930842b46e9507372f0b1b963James Dong
3040c1bc742181ded4930842b46e9507372f0b1b963James Dong        default: /* case 2 */
3050c1bc742181ded4930842b46e9507372f0b1b963James Dong            /* derive frameNumOffset */
3060c1bc742181ded4930842b46e9507372f0b1b963James Dong            if (IS_IDR_NAL_UNIT(pNalUnit))
3070c1bc742181ded4930842b46e9507372f0b1b963James Dong                frameNumOffset = 0;
3080c1bc742181ded4930842b46e9507372f0b1b963James Dong            else if (poc->prevFrameNum > pSliceHeader->frameNum)
3090c1bc742181ded4930842b46e9507372f0b1b963James Dong                frameNumOffset = poc->prevFrameNumOffset + sps->maxFrameNum;
3100c1bc742181ded4930842b46e9507372f0b1b963James Dong            else
3110c1bc742181ded4930842b46e9507372f0b1b963James Dong                frameNumOffset = poc->prevFrameNumOffset;
3120c1bc742181ded4930842b46e9507372f0b1b963James Dong
3130c1bc742181ded4930842b46e9507372f0b1b963James Dong            /* derive picOrderCnt (type 2 has same value for top and bottom
3140c1bc742181ded4930842b46e9507372f0b1b963James Dong             * field order cnts) */
3150c1bc742181ded4930842b46e9507372f0b1b963James Dong            if (IS_IDR_NAL_UNIT(pNalUnit))
3160c1bc742181ded4930842b46e9507372f0b1b963James Dong                picOrderCnt = 0;
3170c1bc742181ded4930842b46e9507372f0b1b963James Dong            else if (pNalUnit->nalRefIdc == 0)
3180c1bc742181ded4930842b46e9507372f0b1b963James Dong                picOrderCnt =
3190c1bc742181ded4930842b46e9507372f0b1b963James Dong                    2 * (i32)(frameNumOffset + pSliceHeader->frameNum) - 1;
3200c1bc742181ded4930842b46e9507372f0b1b963James Dong            else
3210c1bc742181ded4930842b46e9507372f0b1b963James Dong                picOrderCnt =
3220c1bc742181ded4930842b46e9507372f0b1b963James Dong                    2 * (i32)(frameNumOffset + pSliceHeader->frameNum);
3230c1bc742181ded4930842b46e9507372f0b1b963James Dong
3240c1bc742181ded4930842b46e9507372f0b1b963James Dong            /* if current picture contains mmco5 -> set prevFrameNumOffset and
3250c1bc742181ded4930842b46e9507372f0b1b963James Dong             * prevFrameNum to 0 for computation of picOrderCnt of next
3260c1bc742181ded4930842b46e9507372f0b1b963James Dong             * frame, otherwise store frameNum and frameNumOffset to poc
3270c1bc742181ded4930842b46e9507372f0b1b963James Dong             * structure */
3280c1bc742181ded4930842b46e9507372f0b1b963James Dong            if (!containsMmco5)
3290c1bc742181ded4930842b46e9507372f0b1b963James Dong            {
3300c1bc742181ded4930842b46e9507372f0b1b963James Dong                poc->prevFrameNumOffset = frameNumOffset;
3310c1bc742181ded4930842b46e9507372f0b1b963James Dong                poc->prevFrameNum = pSliceHeader->frameNum;
3320c1bc742181ded4930842b46e9507372f0b1b963James Dong            }
3330c1bc742181ded4930842b46e9507372f0b1b963James Dong            else
3340c1bc742181ded4930842b46e9507372f0b1b963James Dong            {
3350c1bc742181ded4930842b46e9507372f0b1b963James Dong                poc->prevFrameNumOffset = 0;
3360c1bc742181ded4930842b46e9507372f0b1b963James Dong                poc->prevFrameNum = 0;
3370c1bc742181ded4930842b46e9507372f0b1b963James Dong                picOrderCnt = 0;
3380c1bc742181ded4930842b46e9507372f0b1b963James Dong            }
3390c1bc742181ded4930842b46e9507372f0b1b963James Dong            break;
3400c1bc742181ded4930842b46e9507372f0b1b963James Dong
3410c1bc742181ded4930842b46e9507372f0b1b963James Dong    }
3420c1bc742181ded4930842b46e9507372f0b1b963James Dong
3430c1bc742181ded4930842b46e9507372f0b1b963James Dong    /*lint -esym(644,picOrderCnt) always initialized */
3440c1bc742181ded4930842b46e9507372f0b1b963James Dong    return(picOrderCnt);
3450c1bc742181ded4930842b46e9507372f0b1b963James Dong
3460c1bc742181ded4930842b46e9507372f0b1b963James Dong}
3470c1bc742181ded4930842b46e9507372f0b1b963James Dong
348