source: 3DVCSoftware/trunk/source/Lib/TLibEncoder/TEncEntropy.cpp @ 1087

Last change on this file since 1087 was 1084, checked in by tech, 10 years ago

Merged branches/HTM-12.1-dev0@1083.

  • Property svn:eol-style set to native
File size: 23.2 KB
RevLine 
[5]1/* The copyright in this software is being made available under the BSD
2 * License, included below. This software may be subject to other third party
3 * and contributor rights, including patent rights, and no such rights are
[56]4 * granted under this license. 
[5]5 *
[872]6* Copyright (c) 2010-2014, ITU/ISO/IEC
[5]7 * All rights reserved.
8 *
9 * Redistribution and use in source and binary forms, with or without
10 * modification, are permitted provided that the following conditions are met:
11 *
12 *  * Redistributions of source code must retain the above copyright notice,
13 *    this list of conditions and the following disclaimer.
14 *  * Redistributions in binary form must reproduce the above copyright notice,
15 *    this list of conditions and the following disclaimer in the documentation
16 *    and/or other materials provided with the distribution.
[56]17 *  * Neither the name of the ITU/ISO/IEC nor the names of its contributors may
[5]18 *    be used to endorse or promote products derived from this software without
19 *    specific prior written permission.
20 *
21 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
22 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
23 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
24 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS
25 * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
26 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
27 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
28 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
29 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
30 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
31 * THE POSSIBILITY OF SUCH DAMAGE.
32 */
[2]33
34/** \file     TEncEntropy.cpp
35    \brief    entropy encoder class
36*/
37
38#include "TEncEntropy.h"
[56]39#include "TLibCommon/TypeDef.h"
40#include "TLibCommon/TComSampleAdaptiveOffset.h"
[2]41
[56]42//! \ingroup TLibEncoder
43//! \{
44
[2]45Void TEncEntropy::setEntropyCoder ( TEncEntropyIf* e, TComSlice* pcSlice )
46{
47  m_pcEntropyCoderIf = e;
48  m_pcEntropyCoderIf->setSlice ( pcSlice );
49}
50
51Void TEncEntropy::encodeSliceHeader ( TComSlice* pcSlice )
52{
53  m_pcEntropyCoderIf->codeSliceHeader( pcSlice );
54  return;
55}
56
[56]57Void  TEncEntropy::encodeTilesWPPEntryPoint( TComSlice* pSlice )
58{
59  m_pcEntropyCoderIf->codeTilesWPPEntryPoint( pSlice );
60}
61
[2]62Void TEncEntropy::encodeTerminatingBit      ( UInt uiIsLast )
63{
64  m_pcEntropyCoderIf->codeTerminatingBit( uiIsLast );
65 
66  return;
67}
68
69Void TEncEntropy::encodeSliceFinish()
70{
71  m_pcEntropyCoderIf->codeSliceFinish();
72}
73
74Void TEncEntropy::encodePPS( TComPPS* pcPPS )
75{
76  m_pcEntropyCoderIf->codePPS( pcPPS );
77  return;
78}
79
[608]80#if H_3D
81Void TEncEntropy::encodeSPS( TComSPS* pcSPS, Int viewIndex, Bool depthFlag )
[77]82{
[608]83  m_pcEntropyCoderIf->codeSPS( pcSPS, viewIndex, depthFlag );
[77]84  return;
85}
[56]86#else
[2]87Void TEncEntropy::encodeSPS( TComSPS* pcSPS )
88{
89  m_pcEntropyCoderIf->codeSPS( pcSPS );
90  return;
91}
[56]92#endif
[2]93
[608]94Void TEncEntropy::encodeCUTransquantBypassFlag( TComDataCU* pcCU, UInt uiAbsPartIdx, Bool bRD )
[2]95{
96  if( bRD )
[56]97  {
[2]98    uiAbsPartIdx = 0;
[56]99  }
[608]100  m_pcEntropyCoderIf->codeCUTransquantBypassFlag( pcCU, uiAbsPartIdx );
[2]101}
102
[608]103Void TEncEntropy::encodeVPS( TComVPS* pcVPS )
[189]104{
[608]105  m_pcEntropyCoderIf->codeVPS( pcVPS );
106  return;
107}
108
109Void TEncEntropy::encodeSkipFlag( TComDataCU* pcCU, UInt uiAbsPartIdx, Bool bRD )
110{
111  if ( pcCU->getSlice()->isIntra() )
[189]112  {
113    return;
114  }
115  if( bRD )
116  {
117    uiAbsPartIdx = 0;
118  }
[608]119  m_pcEntropyCoderIf->codeSkipFlag( pcCU, uiAbsPartIdx );
[189]120}
[1084]121#if H_3D_SINGLE_DEPTH
[1039]122Void TEncEntropy::encodeSingleDepthMode( TComDataCU* pcCU, UInt uiAbsPartIdx, Bool bRD )
123{
124  if ( !pcCU->getSlice()->getIsDepth() )
125  {
126    return;
127  }
128  if(!pcCU->getSlice()->getApplySingleDepthMode())
129  {
130     return;
131  }
132 
133  if( bRD )
134  {
135    uiAbsPartIdx = 0;
136  }
137  m_pcEntropyCoderIf->codeSingleDepthMode( pcCU, uiAbsPartIdx );
138}
139#endif
[2]140/** encode merge flag
141 * \param pcCU
142 * \param uiAbsPartIdx
143 * \returns Void
144 */
[608]145Void TEncEntropy::encodeMergeFlag( TComDataCU* pcCU, UInt uiAbsPartIdx )
[2]146{ 
[56]147  // at least one merge candidate exists
148  m_pcEntropyCoderIf->codeMergeFlag( pcCU, uiAbsPartIdx );
[2]149}
150
151/** encode merge index
152 * \param pcCU
153 * \param uiAbsPartIdx
154 * \param uiPUIdx
155 * \param bRD
156 * \returns Void
157 */
[608]158Void TEncEntropy::encodeMergeIndex( TComDataCU* pcCU, UInt uiAbsPartIdx, Bool bRD )
[2]159{
160  if( bRD )
161  {
162    uiAbsPartIdx = 0;
163    assert( pcCU->getPartitionSize(uiAbsPartIdx) == SIZE_2Nx2N );
164  }
[608]165  m_pcEntropyCoderIf->codeMergeIndex( pcCU, uiAbsPartIdx );
[2]166}
167
[608]168#if H_3D_IC
169Void TEncEntropy::encodeICFlag( TComDataCU* pcCU, UInt uiAbsPartIdx, Bool bRD )
[443]170{
[833]171  if ( pcCU->isIntra( uiAbsPartIdx ) || ( pcCU->getSlice()->getViewIndex() == 0 ) || pcCU->getSlice()->getIsDepth() || pcCU->getARPW( uiAbsPartIdx ) > 0 )
[443]172  {
173    return;
174  }
[608]175
176  if( !pcCU->getSlice()->getApplyIC() )
[976]177  {
[608]178    return;
[976]179  }
[608]180
[443]181  if( bRD )
182  {
183    uiAbsPartIdx = 0;
184  }
[950]185  else
186  {
[1066]187    Int ICEnableCandidate = pcCU->getSlice()->getICEnableCandidate(pcCU->getSlice()->getDepth());
188    Int ICEnableNum = pcCU->getSlice()->getICEnableNum(pcCU->getSlice()->getDepth());
189    ICEnableCandidate++;
190    if(pcCU->getICFlag(uiAbsPartIdx))
191    {
192      ICEnableNum++;
193    }
194    pcCU->getSlice()->setICEnableCandidate(pcCU->getSlice()->getDepth(), ICEnableCandidate);
195    pcCU->getSlice()->setICEnableNum(pcCU->getSlice()->getDepth(), ICEnableNum);
[950]196  }
[608]197  if( pcCU->isICFlagRequired( uiAbsPartIdx ) )
[976]198  {
[608]199    m_pcEntropyCoderIf->codeICFlag( pcCU, uiAbsPartIdx );
[976]200  }
[443]201}
202#endif
[2]203
[608]204#if H_3D_ARP
205Void TEncEntropy::encodeARPW( TComDataCU* pcCU, UInt uiAbsPartIdx )
[56]206{
[608]207  if( !pcCU->getSlice()->getARPStepNum() || pcCU->isIntra( uiAbsPartIdx ) ) 
[56]208  {
[608]209    return;
[56]210  }
211
[608]212  if ( pcCU->getPartitionSize(uiAbsPartIdx)!=SIZE_2Nx2N )
[56]213  {
[608]214    assert(pcCU->getARPW (uiAbsPartIdx) == 0);
[56]215  }
[608]216  else
[56]217  {
[608]218    m_pcEntropyCoderIf->codeARPW( pcCU, uiAbsPartIdx );
[56]219  }
220}
[608]221#endif
[56]222
[2]223/** encode prediction mode
224 * \param pcCU
225 * \param uiAbsPartIdx
226 * \param bRD
227 * \returns Void
228 */
229Void TEncEntropy::encodePredMode( TComDataCU* pcCU, UInt uiAbsPartIdx, Bool bRD )
230{
231  if( bRD )
[56]232  {
[2]233    uiAbsPartIdx = 0;
[56]234  }
[2]235  if ( pcCU->getSlice()->isIntra() )
236  {
237    return;
238  }
[56]239
[2]240  m_pcEntropyCoderIf->codePredMode( pcCU, uiAbsPartIdx );
241}
242
243// Split mode
244Void TEncEntropy::encodeSplitFlag( TComDataCU* pcCU, UInt uiAbsPartIdx, UInt uiDepth, Bool bRD )
245{
246  if( bRD )
[56]247  {
[2]248    uiAbsPartIdx = 0;
[56]249  }
[2]250  m_pcEntropyCoderIf->codeSplitFlag( pcCU, uiAbsPartIdx, uiDepth );
251}
252
253/** encode partition size
254 * \param pcCU
255 * \param uiAbsPartIdx
256 * \param uiDepth
257 * \param bRD
258 * \returns Void
259 */
260Void TEncEntropy::encodePartSize( TComDataCU* pcCU, UInt uiAbsPartIdx, UInt uiDepth, Bool bRD )
261{
262  if( bRD )
[56]263  {
[2]264    uiAbsPartIdx = 0;
[56]265  }
[833]266 
[2]267  m_pcEntropyCoderIf->codePartSize( pcCU, uiAbsPartIdx, uiDepth );
[833]268 
269#if H_3D_DBBP
[1039]270  if( pcCU->getSlice()->getVPS()->getUseDBBP(pcCU->getSlice()->getLayerIdInVps()) && (pcCU->getPartitionSize(uiAbsPartIdx) == SIZE_2NxN || pcCU->getPartitionSize(uiAbsPartIdx) == SIZE_Nx2N) && pcCU->getWidth(uiAbsPartIdx) > 8 )
[833]271  {
272    encodeDBBPFlag(pcCU, uiAbsPartIdx, bRD);
273  }
274#endif
[2]275}
276
[56]277/** Encode I_PCM information.
278 * \param pcCU pointer to CU
279 * \param uiAbsPartIdx CU index
280 * \param bRD flag indicating estimation or encoding
281 * \returns Void
282 */
283Void TEncEntropy::encodeIPCMInfo( TComDataCU* pcCU, UInt uiAbsPartIdx, Bool bRD )
[2]284{
[56]285  if(!pcCU->getSlice()->getSPS()->getUsePCM()
286    || pcCU->getWidth(uiAbsPartIdx) > (1<<pcCU->getSlice()->getSPS()->getPCMLog2MaxSize())
287    || pcCU->getWidth(uiAbsPartIdx) < (1<<pcCU->getSlice()->getSPS()->getPCMLog2MinSize()))
288  {
289    return;
290  }
[189]291 
[56]292  if( bRD )
293  {
294    uiAbsPartIdx = 0;
295  }
296 
[608]297  m_pcEntropyCoderIf->codeIPCMInfo ( pcCU, uiAbsPartIdx );
[56]298}
299
[608]300Void TEncEntropy::xEncodeTransform( TComDataCU* pcCU,UInt offsetLuma, UInt offsetChroma, UInt uiAbsPartIdx, UInt uiDepth, UInt width, UInt height, UInt uiTrIdx, Bool& bCodeDQP )
[56]301{
[2]302  const UInt uiSubdiv = pcCU->getTransformIdx( uiAbsPartIdx ) + pcCU->getDepth( uiAbsPartIdx ) > uiDepth;
303  const UInt uiLog2TrafoSize = g_aucConvertToBit[pcCU->getSlice()->getSPS()->getMaxCUWidth()]+2 - uiDepth;
[56]304  UInt cbfY = pcCU->getCbf( uiAbsPartIdx, TEXT_LUMA    , uiTrIdx );
305  UInt cbfU = pcCU->getCbf( uiAbsPartIdx, TEXT_CHROMA_U, uiTrIdx );
306  UInt cbfV = pcCU->getCbf( uiAbsPartIdx, TEXT_CHROMA_V, uiTrIdx );
307
308  if(uiTrIdx==0)
[2]309  {
[56]310    m_bakAbsPartIdxCU = uiAbsPartIdx;
[2]311  }
[56]312  if( uiLog2TrafoSize == 2 )
[2]313  {
[56]314    UInt partNum = pcCU->getPic()->getNumPartInCU() >> ( ( uiDepth - 1 ) << 1 );
315    if( ( uiAbsPartIdx % partNum ) == 0 )
316    {
317      m_uiBakAbsPartIdx   = uiAbsPartIdx;
318      m_uiBakChromaOffset = offsetChroma;
319    }
320    else if( ( uiAbsPartIdx % partNum ) == (partNum - 1) )
321    {
322      cbfU = pcCU->getCbf( m_uiBakAbsPartIdx, TEXT_CHROMA_U, uiTrIdx );
323      cbfV = pcCU->getCbf( m_uiBakAbsPartIdx, TEXT_CHROMA_V, uiTrIdx );
324    }
[2]325  }
[608]326 
327  if( pcCU->getPredictionMode(uiAbsPartIdx) == MODE_INTRA && pcCU->getPartitionSize(uiAbsPartIdx) == SIZE_NxN && uiDepth == pcCU->getDepth(uiAbsPartIdx) )
328  {
329    assert( uiSubdiv );
330  }
331  else if( pcCU->getPredictionMode(uiAbsPartIdx) == MODE_INTER && (pcCU->getPartitionSize(uiAbsPartIdx) != SIZE_2Nx2N) && uiDepth == pcCU->getDepth(uiAbsPartIdx) &&  (pcCU->getSlice()->getSPS()->getQuadtreeTUMaxDepthInter() == 1) )
332  {
333    if ( uiLog2TrafoSize > pcCU->getQuadtreeTULog2MinSizeInCU(uiAbsPartIdx) )
[2]334    {
335      assert( uiSubdiv );
336    }
[608]337    else
[56]338    {
[608]339      assert(!uiSubdiv );
[56]340    }
[608]341  }
342  else if( uiLog2TrafoSize > pcCU->getSlice()->getSPS()->getQuadtreeTULog2MaxSize() )
343  {
344    assert( uiSubdiv );
345  }
346  else if( uiLog2TrafoSize == pcCU->getSlice()->getSPS()->getQuadtreeTULog2MinSize() )
347  {
348    assert( !uiSubdiv );
349  }
350  else if( uiLog2TrafoSize == pcCU->getQuadtreeTULog2MinSizeInCU(uiAbsPartIdx) )
351  {
352    assert( !uiSubdiv );
353  }
354  else
355  {
356    assert( uiLog2TrafoSize > pcCU->getQuadtreeTULog2MinSizeInCU(uiAbsPartIdx) );
357    m_pcEntropyCoderIf->codeTransformSubdivFlag( uiSubdiv, 5 - uiLog2TrafoSize );
358  }
359
360  const UInt uiTrDepthCurr = uiDepth - pcCU->getDepth( uiAbsPartIdx );
361  const Bool bFirstCbfOfCU = uiTrDepthCurr == 0;
362  if( bFirstCbfOfCU || uiLog2TrafoSize > 2 )
363  {
364    if( bFirstCbfOfCU || pcCU->getCbf( uiAbsPartIdx, TEXT_CHROMA_U, uiTrDepthCurr - 1 ) )
[56]365    {
[608]366      m_pcEntropyCoderIf->codeQtCbf( pcCU, uiAbsPartIdx, TEXT_CHROMA_U, uiTrDepthCurr );
[56]367    }
[608]368    if( bFirstCbfOfCU || pcCU->getCbf( uiAbsPartIdx, TEXT_CHROMA_V, uiTrDepthCurr - 1 ) )
[2]369    {
[608]370      m_pcEntropyCoderIf->codeQtCbf( pcCU, uiAbsPartIdx, TEXT_CHROMA_V, uiTrDepthCurr );
[2]371    }
[608]372  }
373  else if( uiLog2TrafoSize == 2 )
374  {
375    assert( pcCU->getCbf( uiAbsPartIdx, TEXT_CHROMA_U, uiTrDepthCurr ) == pcCU->getCbf( uiAbsPartIdx, TEXT_CHROMA_U, uiTrDepthCurr - 1 ) );
376    assert( pcCU->getCbf( uiAbsPartIdx, TEXT_CHROMA_V, uiTrDepthCurr ) == pcCU->getCbf( uiAbsPartIdx, TEXT_CHROMA_V, uiTrDepthCurr - 1 ) );
377  }
378 
379  if( uiSubdiv )
380  {
381    UInt size;
382    width  >>= 1;
383    height >>= 1;
384    size = width*height;
385    uiTrIdx++;
386    ++uiDepth;
387    const UInt partNum = pcCU->getPic()->getNumPartInCU() >> (uiDepth << 1);
388   
389    xEncodeTransform( pcCU, offsetLuma, offsetChroma, uiAbsPartIdx, uiDepth, width, height, uiTrIdx, bCodeDQP );
390
391    uiAbsPartIdx += partNum;  offsetLuma += size;  offsetChroma += (size>>2);
392    xEncodeTransform( pcCU, offsetLuma, offsetChroma, uiAbsPartIdx, uiDepth, width, height, uiTrIdx, bCodeDQP );
393
394    uiAbsPartIdx += partNum;  offsetLuma += size;  offsetChroma += (size>>2);
395    xEncodeTransform( pcCU, offsetLuma, offsetChroma, uiAbsPartIdx, uiDepth, width, height, uiTrIdx, bCodeDQP );
396
397    uiAbsPartIdx += partNum;  offsetLuma += size;  offsetChroma += (size>>2);
398    xEncodeTransform( pcCU, offsetLuma, offsetChroma, uiAbsPartIdx, uiDepth, width, height, uiTrIdx, bCodeDQP );
399  }
400  else
401  {
402#if !H_MV_ENC_DEC_TRAC
[2]403    {
[608]404      DTRACE_CABAC_VL( g_nSymbolCounter++ );
405      DTRACE_CABAC_T( "\tTrIdx: abspart=" );
406      DTRACE_CABAC_V( uiAbsPartIdx );
407      DTRACE_CABAC_T( "\tdepth=" );
408      DTRACE_CABAC_V( uiDepth );
409      DTRACE_CABAC_T( "\ttrdepth=" );
410      DTRACE_CABAC_V( pcCU->getTransformIdx( uiAbsPartIdx ) );
411      DTRACE_CABAC_T( "\n" );
[2]412    }
[608]413#endif
414   
415    if( pcCU->getPredictionMode(uiAbsPartIdx) != MODE_INTRA && uiDepth == pcCU->getDepth( uiAbsPartIdx ) && !pcCU->getCbf( uiAbsPartIdx, TEXT_CHROMA_U, 0 ) && !pcCU->getCbf( uiAbsPartIdx, TEXT_CHROMA_V, 0 ) )
416    {
417      assert( pcCU->getCbf( uiAbsPartIdx, TEXT_LUMA, 0 ) );
418      //      printf( "saved one bin! " );
419    }
[2]420    else
421    {
[608]422      m_pcEntropyCoderIf->codeQtCbf( pcCU, uiAbsPartIdx, TEXT_LUMA, pcCU->getTransformIdx( uiAbsPartIdx ) );
[2]423    }
[56]424
[608]425
426    if ( cbfY || cbfU || cbfV )
[2]427    {
[608]428      // dQP: only for LCU once
429      if ( pcCU->getSlice()->getPPS()->getUseDQP() )
[2]430      {
[608]431        if ( bCodeDQP )
[2]432        {
[608]433          encodeQP( pcCU, m_bakAbsPartIdxCU );
434          bCodeDQP = false;
[2]435        }
436      }
437    }
[608]438    if( cbfY )
[2]439    {
[608]440      Int trWidth = width;
441      Int trHeight = height;
442      m_pcEntropyCoderIf->codeCoeffNxN( pcCU, (pcCU->getCoeffY()+offsetLuma), uiAbsPartIdx, trWidth, trHeight, uiDepth, TEXT_LUMA );
[2]443    }
[608]444    if( uiLog2TrafoSize > 2 )
[2]445    {
[608]446      Int trWidth = width >> 1;
447      Int trHeight = height >> 1;
448      if( cbfU )
[2]449      {
[608]450        m_pcEntropyCoderIf->codeCoeffNxN( pcCU, (pcCU->getCoeffCb()+offsetChroma), uiAbsPartIdx, trWidth, trHeight, uiDepth, TEXT_CHROMA_U );
[2]451      }
[608]452      if( cbfV )
[56]453      {
[608]454        m_pcEntropyCoderIf->codeCoeffNxN( pcCU, (pcCU->getCoeffCr()+offsetChroma), uiAbsPartIdx, trWidth, trHeight, uiDepth, TEXT_CHROMA_V );
[56]455      }
[608]456    }
457    else
458    {
459      UInt partNum = pcCU->getPic()->getNumPartInCU() >> ( ( uiDepth - 1 ) << 1 );
460      if( ( uiAbsPartIdx % partNum ) == (partNum - 1) )
[2]461      {
[56]462        Int trWidth = width;
463        Int trHeight = height;
464        if( cbfU )
[2]465        {
[608]466          m_pcEntropyCoderIf->codeCoeffNxN( pcCU, (pcCU->getCoeffCb()+m_uiBakChromaOffset), m_uiBakAbsPartIdx, trWidth, trHeight, uiDepth, TEXT_CHROMA_U );
[2]467        }
[56]468        if( cbfV )
469        {
[608]470          m_pcEntropyCoderIf->codeCoeffNxN( pcCU, (pcCU->getCoeffCr()+m_uiBakChromaOffset), m_uiBakAbsPartIdx, trWidth, trHeight, uiDepth, TEXT_CHROMA_V );
[56]471        }
[2]472      }
473    }
474  }
475}
476
477// Intra direction for Luma
[608]478Void TEncEntropy::encodeIntraDirModeLuma  ( TComDataCU* pcCU, UInt absPartIdx, Bool isMultiplePU )
[2]479{
[608]480  m_pcEntropyCoderIf->codeIntraDirLumaAng( pcCU, absPartIdx , isMultiplePU);
[2]481}
482
483// Intra direction for Chroma
484Void TEncEntropy::encodeIntraDirModeChroma( TComDataCU* pcCU, UInt uiAbsPartIdx, Bool bRD )
485{
486  if( bRD )
[56]487  {
[2]488    uiAbsPartIdx = 0;
[56]489  }
[2]490 
491  m_pcEntropyCoderIf->codeIntraDirChroma( pcCU, uiAbsPartIdx );
492}
493
[608]494Void TEncEntropy::encodePredInfo( TComDataCU* pcCU, UInt uiAbsPartIdx, Bool bRD )
[2]495{
496  if( bRD )
[56]497  {
[2]498    uiAbsPartIdx = 0;
[56]499  }
[2]500  if( pcCU->isIntra( uiAbsPartIdx ) )                                 // If it is Intra mode, encode intra prediction mode.
501  {
[608]502    encodeIntraDirModeLuma  ( pcCU, uiAbsPartIdx,true );
503#if H_3D_DIM_SDC
504    if(!pcCU->getSDCFlag(uiAbsPartIdx))
[443]505#endif
[608]506    encodeIntraDirModeChroma( pcCU, uiAbsPartIdx, bRD );
[2]507  }
508  else                                                                // if it is Inter mode, encode motion vector and reference index
509  {
[56]510    encodePUWise( pcCU, uiAbsPartIdx, bRD );
[2]511  }
512}
513
514/** encode motion information for every PU block
515 * \param pcCU
516 * \param uiAbsPartIdx
517 * \param bRD
518 * \returns Void
519 */
520Void TEncEntropy::encodePUWise( TComDataCU* pcCU, UInt uiAbsPartIdx, Bool bRD )
521{
522  if ( bRD )
[56]523  {
[2]524    uiAbsPartIdx = 0;
[56]525  }
526 
[2]527  PartSize ePartSize = pcCU->getPartitionSize( uiAbsPartIdx );
528  UInt uiNumPU = ( ePartSize == SIZE_2Nx2N ? 1 : ( ePartSize == SIZE_NxN ? 4 : 2 ) );
529  UInt uiDepth = pcCU->getDepth( uiAbsPartIdx );
530  UInt uiPUOffset = ( g_auiPUOffset[UInt( ePartSize )] << ( ( pcCU->getSlice()->getSPS()->getMaxCUDepth() - uiDepth ) << 1 ) ) >> 4;
531
532  for ( UInt uiPartIdx = 0, uiSubPartIdx = uiAbsPartIdx; uiPartIdx < uiNumPU; uiPartIdx++, uiSubPartIdx += uiPUOffset )
533  {
[608]534#if H_MV_ENC_DEC_TRAC
535    DTRACE_PU_S("=========== prediction_unit ===========\n")
536       //Todo:
537      //DTRACE_PU("x0", uiLPelX)
538      //DTRACE_PU("x1", uiTPelY)
539#endif
540    encodeMergeFlag( pcCU, uiSubPartIdx );
[2]541    if ( pcCU->getMergeFlag( uiSubPartIdx ) )
542    {
[608]543      encodeMergeIndex( pcCU, uiSubPartIdx );
[2]544    }
545    else
546    {
547      encodeInterDirPU( pcCU, uiSubPartIdx );
548      for ( UInt uiRefListIdx = 0; uiRefListIdx < 2; uiRefListIdx++ )
549      {
550        if ( pcCU->getSlice()->getNumRefIdx( RefPicList( uiRefListIdx ) ) > 0 )
551        {
552          encodeRefFrmIdxPU ( pcCU, uiSubPartIdx, RefPicList( uiRefListIdx ) );
553          encodeMvdPU       ( pcCU, uiSubPartIdx, RefPicList( uiRefListIdx ) );
554          encodeMVPIdxPU    ( pcCU, uiSubPartIdx, RefPicList( uiRefListIdx ) );
555        }
556      }
557    }
558  }
559
560  return;
561}
562
563Void TEncEntropy::encodeInterDirPU( TComDataCU* pcCU, UInt uiAbsPartIdx )
564{
565  if ( !pcCU->getSlice()->isInterB() )
566  {
567    return;
568  }
569
570  m_pcEntropyCoderIf->codeInterDir( pcCU, uiAbsPartIdx );
571  return;
572}
573
574/** encode reference frame index for a PU block
575 * \param pcCU
576 * \param uiAbsPartIdx
577 * \param eRefList
578 * \returns Void
579 */
580Void TEncEntropy::encodeRefFrmIdxPU( TComDataCU* pcCU, UInt uiAbsPartIdx, RefPicList eRefList )
581{
582  assert( !pcCU->isIntra( uiAbsPartIdx ) );
583  {
[56]584    if ( ( pcCU->getSlice()->getNumRefIdx( eRefList ) == 1 ) )
585    {
586      return;
587    }
[2]588
[56]589    if ( pcCU->getInterDir( uiAbsPartIdx ) & ( 1 << eRefList ) )
590    {
591      m_pcEntropyCoderIf->codeRefFrmIdx( pcCU, uiAbsPartIdx, eRefList );
592    }
[2]593  }
594
595  return;
596}
597
598/** encode motion vector difference for a PU block
599 * \param pcCU
600 * \param uiAbsPartIdx
601 * \param eRefList
602 * \returns Void
603 */
604Void TEncEntropy::encodeMvdPU( TComDataCU* pcCU, UInt uiAbsPartIdx, RefPicList eRefList )
605{
606  assert( !pcCU->isIntra( uiAbsPartIdx ) );
607
608  if ( pcCU->getInterDir( uiAbsPartIdx ) & ( 1 << eRefList ) )
609  {
610    m_pcEntropyCoderIf->codeMvd( pcCU, uiAbsPartIdx, eRefList );
611  }
612  return;
613}
614
615Void TEncEntropy::encodeMVPIdxPU( TComDataCU* pcCU, UInt uiAbsPartIdx, RefPicList eRefList )
616{
[608]617  if ( (pcCU->getInterDir( uiAbsPartIdx ) & ( 1 << eRefList )) )
[2]618  {
619    m_pcEntropyCoderIf->codeMVPIdx( pcCU, uiAbsPartIdx, eRefList );
620  }
621
622  return;
623}
624
625Void TEncEntropy::encodeQtCbf( TComDataCU* pcCU, UInt uiAbsPartIdx, TextType eType, UInt uiTrDepth )
626{
627  m_pcEntropyCoderIf->codeQtCbf( pcCU, uiAbsPartIdx, eType, uiTrDepth );
628}
629
630Void TEncEntropy::encodeTransformSubdivFlag( UInt uiSymbol, UInt uiCtx )
631{
632  m_pcEntropyCoderIf->codeTransformSubdivFlag( uiSymbol, uiCtx );
633}
634
635Void TEncEntropy::encodeQtRootCbf( TComDataCU* pcCU, UInt uiAbsPartIdx )
636{
637  m_pcEntropyCoderIf->codeQtRootCbf( pcCU, uiAbsPartIdx );
638}
639
[608]640Void TEncEntropy::encodeQtCbfZero( TComDataCU* pcCU, TextType eType, UInt uiTrDepth )
641{
642  m_pcEntropyCoderIf->codeQtCbfZero( pcCU, eType, uiTrDepth );
643}
644Void TEncEntropy::encodeQtRootCbfZero( TComDataCU* pcCU )
645{
646  m_pcEntropyCoderIf->codeQtRootCbfZero( pcCU );
647}
648
[2]649// dQP
650Void TEncEntropy::encodeQP( TComDataCU* pcCU, UInt uiAbsPartIdx, Bool bRD )
651{
652  if( bRD )
[56]653  {
[2]654    uiAbsPartIdx = 0;
[56]655  }
[2]656 
[56]657  if ( pcCU->getSlice()->getPPS()->getUseDQP() )
[2]658  {
659    m_pcEntropyCoderIf->codeDeltaQP( pcCU, uiAbsPartIdx );
660  }
661}
662
663
664// texture
665/** encode coefficients
666 * \param pcCU
667 * \param uiAbsPartIdx
668 * \param uiDepth
669 * \param uiWidth
670 * \param uiHeight
671 */
[56]672Void TEncEntropy::encodeCoeff( TComDataCU* pcCU, UInt uiAbsPartIdx, UInt uiDepth, UInt uiWidth, UInt uiHeight, Bool& bCodeDQP )
[2]673{
674  UInt uiMinCoeffSize = pcCU->getPic()->getMinCUWidth()*pcCU->getPic()->getMinCUHeight();
675  UInt uiLumaOffset   = uiMinCoeffSize*uiAbsPartIdx;
676  UInt uiChromaOffset = uiLumaOffset>>2;
[833]677#if H_3D_DIM_SDC
678  if( pcCU->getSDCFlag( uiAbsPartIdx ) && pcCU->isIntra( uiAbsPartIdx ) )
679  {
680    assert( pcCU->getPartitionSize(uiAbsPartIdx) == SIZE_2Nx2N );
681    assert( pcCU->getTransformIdx(uiAbsPartIdx) == 0 );
682    assert( pcCU->getCbf(uiAbsPartIdx, TEXT_LUMA) == 1 );
683    assert( pcCU->getCbf(uiAbsPartIdx, TEXT_CHROMA_U) == 1 );
684    assert( pcCU->getCbf(uiAbsPartIdx, TEXT_CHROMA_V) == 1 );
685  }
686
687  if( pcCU->getSDCFlag( uiAbsPartIdx ) && !pcCU->isIntra( uiAbsPartIdx ) )
688  {
689    assert( !pcCU->isSkipped( uiAbsPartIdx ) );
690    assert( !pcCU->isIntra( uiAbsPartIdx) );
691    assert( pcCU->getSlice()->getIsDepth() );
692  }
[884]693
[833]694  if( pcCU->getSlice()->getIsDepth() && ( pcCU->getSDCFlag( uiAbsPartIdx ) || pcCU->isIntra( uiAbsPartIdx ) ) )
695  {
696    Int iPartNum = ( pcCU->isIntra( uiAbsPartIdx ) && pcCU->getPartitionSize( uiAbsPartIdx ) == SIZE_NxN ) ? 4 : 1;
697    UInt uiPartOffset = ( pcCU->getPic()->getNumPartInCU() >> ( pcCU->getDepth( uiAbsPartIdx ) << 1 ) ) >> 2;
[608]698   
[833]699    if( !pcCU->getSDCFlag( uiAbsPartIdx ) )
700    {
701      for( Int iPart = 0; iPart < iPartNum; iPart++ )
702      {
703        if( getDimType( pcCU->getLumaIntraDir( uiAbsPartIdx + uiPartOffset*iPart ) ) < DIM_NUM_TYPE ) 
704        {
705          m_pcEntropyCoderIf->codeDeltaDC( pcCU, uiAbsPartIdx + uiPartOffset*iPart );
706        }
707      }
708    }
709    else
710    {
711      m_pcEntropyCoderIf->codeDeltaDC( pcCU, uiAbsPartIdx );
712      return;
713    }
714  }
[443]715#endif
[608]716
[2]717  if( pcCU->isIntra(uiAbsPartIdx) )
718  {
[608]719#if !H_MV
[56]720    DTRACE_CABAC_VL( g_nSymbolCounter++ )
[2]721    DTRACE_CABAC_T( "\tdecodeTransformIdx()\tCUDepth=" )
722    DTRACE_CABAC_V( uiDepth )
723    DTRACE_CABAC_T( "\n" )
[608]724#endif
[2]725  }
726  else
727  {
[608]728    if( !(pcCU->getMergeFlag( uiAbsPartIdx ) && pcCU->getPartitionSize(uiAbsPartIdx) == SIZE_2Nx2N ) )
[2]729    {
[608]730      m_pcEntropyCoderIf->codeQtRootCbf( pcCU, uiAbsPartIdx );
[2]731    }
[608]732    if ( !pcCU->getQtRootCbf( uiAbsPartIdx ) )
[189]733    {
[608]734      return;
[189]735    }
736  }
[608]737 
738  xEncodeTransform( pcCU, uiLumaOffset, uiChromaOffset, uiAbsPartIdx, uiDepth, uiWidth, uiHeight, 0, bCodeDQP);
[2]739}
740
[56]741Void TEncEntropy::encodeCoeffNxN( TComDataCU* pcCU, TCoeff* pcCoeff, UInt uiAbsPartIdx, UInt uiTrWidth, UInt uiTrHeight, UInt uiDepth, TextType eType )
[608]742{
743  // This is for Transform unit processing. This may be used at mode selection stage for Inter.
[56]744  m_pcEntropyCoderIf->codeCoeffNxN( pcCU, pcCoeff, uiAbsPartIdx, uiTrWidth, uiTrHeight, uiDepth, eType );
[2]745}
746
[56]747Void TEncEntropy::estimateBit (estBitsSbacStruct* pcEstBitsSbac, Int width, Int height, TextType eTType)
748{ 
749  eTType = eTType == TEXT_LUMA ? TEXT_LUMA : TEXT_CHROMA;
750 
751  m_pcEntropyCoderIf->estBit ( pcEstBitsSbac, width, height, eTType );
752}
753
754Int TEncEntropy::countNonZeroCoeffs( TCoeff* pcCoef, UInt uiSize )
755{
756  Int count = 0;
757 
758  for ( Int i = 0; i < uiSize; i++ )
[2]759  {
[56]760    count += pcCoef[i] != 0;
[2]761  }
[56]762 
763  return count;
764}
[2]765
[56]766/** encode quantization matrix
767 * \param scalingList quantization matrix information
768 */
769Void TEncEntropy::encodeScalingList( TComScalingList* scalingList )
770{
771  m_pcEntropyCoderIf->codeScalingList( scalingList );
[2]772}
773
[655]774#if H_3D_INTER_SDC
[833]775Void TEncEntropy::encodeDeltaDC  ( TComDataCU* pcCU, UInt absPartIdx )
776{
777  m_pcEntropyCoderIf->codeDeltaDC( pcCU, absPartIdx );
778}
779
780Void TEncEntropy::encodeSDCFlag( TComDataCU* pcCU, UInt uiAbsPartIdx, Bool bRD )
781{
782  if( ( !pcCU->isIntra( uiAbsPartIdx ) && !pcCU->getSlice()->getVPS()->getInterSDCFlag( pcCU->getSlice()->getLayerIdInVps() ) ) || 
783    ( pcCU->isIntra( uiAbsPartIdx ) && !pcCU->getSlice()->getVPS()->getVpsDepthModesFlag( pcCU->getSlice()->getLayerIdInVps() ) ) )
784  {
785    return;
786  }
787
788  if( !pcCU->getSlice()->getIsDepth() || pcCU->getPartitionSize( uiAbsPartIdx ) != SIZE_2Nx2N || pcCU->isSkipped( uiAbsPartIdx ) )
789  {
790    return;
791  }
792
793  assert( pcCU->getPartitionSize( uiAbsPartIdx ) == SIZE_2Nx2N || ( !pcCU->isIntra( uiAbsPartIdx ) && !pcCU->isSkipped( uiAbsPartIdx ) ) );
794
795  if( bRD )
796  {
797    uiAbsPartIdx = 0;
798  }
799
800  m_pcEntropyCoderIf->codeSDCFlag( pcCU, uiAbsPartIdx );
801}
[2]802
[189]803#endif
[833]804#if H_3D_DBBP
805Void TEncEntropy::encodeDBBPFlag( TComDataCU* pcCU, UInt uiAbsPartIdx, Bool bRD )
806{
807  if( bRD )
808  {
809    uiAbsPartIdx = 0;
810  }
811  m_pcEntropyCoderIf->codeDBBPFlag( pcCU, uiAbsPartIdx );
812}
813#endif
[56]814//! \}
Note: See TracBrowser for help on using the repository browser.