source: SHVCSoftware/branches/SHM-dev/source/Lib/TLibCommon/TComDataCU.cpp @ 1309

Last change on this file since 1309 was 1307, checked in by seregin, 9 years ago

port rev 4363

  • Property svn:eol-style set to native
File size: 125.8 KB
Line 
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
4 * granted under this license.
5 *
6 * Copyright (c) 2010-2015, ITU/ISO/IEC
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.
17 *  * Neither the name of the ITU/ISO/IEC nor the names of its contributors may
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 */
33
34/** \file     TComDataCU.cpp
35    \brief    CU data structure
36    \todo     not all entities are documented
37*/
38
39#include "TComDataCU.h"
40#include "TComTU.h"
41#include "TComPic.h"
42
43//! \ingroup TLibCommon
44//! \{
45
46// ====================================================================================================================
47// Constructor / destructor / create / destroy
48// ====================================================================================================================
49
50TComDataCU::TComDataCU()
51{
52  m_pcPic              = NULL;
53  m_pcSlice            = NULL;
54  m_puhDepth           = NULL;
55
56  m_skipFlag           = NULL;
57
58  m_pePartSize         = NULL;
59  m_pePredMode         = NULL;
60  m_CUTransquantBypass = NULL;
61  m_puhWidth           = NULL;
62  m_puhHeight          = NULL;
63  m_phQP               = NULL;
64  m_ChromaQpAdj        = NULL;
65  m_pbMergeFlag        = NULL;
66  m_puhMergeIndex      = NULL;
67  for(UInt i=0; i<MAX_NUM_CHANNEL_TYPE; i++)
68  {
69    m_puhIntraDir[i]     = NULL;
70  }
71  m_puhInterDir        = NULL;
72  m_puhTrIdx           = NULL;
73
74  for (UInt comp=0; comp<MAX_NUM_COMPONENT; comp++)
75  {
76    m_puhCbf[comp]                        = NULL;
77    m_crossComponentPredictionAlpha[comp] = NULL;
78    m_puhTransformSkip[comp]              = NULL;
79    m_pcTrCoeff[comp]                     = NULL;
80#if ADAPTIVE_QP_SELECTION
81    m_pcArlCoeff[comp]                    = NULL;
82#endif
83    m_pcIPCMSample[comp]                  = NULL;
84    m_explicitRdpcmMode[comp]             = NULL;
85  }
86#if ADAPTIVE_QP_SELECTION
87  m_ArlCoeffIsAliasedAllocation = false;
88#endif
89  m_pbIPCMFlag         = NULL;
90
91  m_pCtuAboveLeft      = NULL;
92  m_pCtuAboveRight     = NULL;
93  m_pCtuAbove          = NULL;
94  m_pCtuLeft           = NULL;
95
96  for(UInt i=0; i<NUM_REF_PIC_LIST_01; i++)
97  {
98    m_apcCUColocated[i]  = NULL;
99    m_apiMVPIdx[i]       = NULL;
100    m_apiMVPNum[i]       = NULL;
101  }
102
103  m_bDecSubCu          = false;
104}
105
106TComDataCU::~TComDataCU()
107{
108}
109
110Void TComDataCU::create( ChromaFormat chromaFormatIDC, UInt uiNumPartition, UInt uiWidth, UInt uiHeight, Bool bDecSubCu, Int unitSize
111#if ADAPTIVE_QP_SELECTION
112                        , TCoeff *pParentARLBuffer
113#endif
114                        )
115{
116  m_bDecSubCu = bDecSubCu;
117
118  m_pcPic              = NULL;
119  m_pcSlice            = NULL;
120  m_uiNumPartition     = uiNumPartition;
121  m_unitSize = unitSize;
122
123  if ( !bDecSubCu )
124  {
125    m_phQP               = (Char*     )xMalloc(Char,     uiNumPartition);
126    m_puhDepth           = (UChar*    )xMalloc(UChar,    uiNumPartition);
127    m_puhWidth           = (UChar*    )xMalloc(UChar,    uiNumPartition);
128    m_puhHeight          = (UChar*    )xMalloc(UChar,    uiNumPartition);
129
130    m_ChromaQpAdj        = new UChar[ uiNumPartition ];
131    m_skipFlag           = new Bool[ uiNumPartition ];
132    m_pePartSize         = new Char[ uiNumPartition ];
133    memset( m_pePartSize, NUMBER_OF_PART_SIZES,uiNumPartition * sizeof( *m_pePartSize ) );
134    m_pePredMode         = new Char[ uiNumPartition ];
135    m_CUTransquantBypass = new Bool[ uiNumPartition ];
136
137    m_pbMergeFlag        = (Bool*  )xMalloc(Bool,   uiNumPartition);
138    m_puhMergeIndex      = (UChar* )xMalloc(UChar,  uiNumPartition);
139
140    for (UInt ch=0; ch<MAX_NUM_CHANNEL_TYPE; ch++)
141    {
142      m_puhIntraDir[ch] = (UChar* )xMalloc(UChar,  uiNumPartition);
143    }
144    m_puhInterDir        = (UChar* )xMalloc(UChar,  uiNumPartition);
145
146    m_puhTrIdx           = (UChar* )xMalloc(UChar,  uiNumPartition);
147
148    for(UInt i=0; i<NUM_REF_PIC_LIST_01; i++)
149    {
150      const RefPicList rpl=RefPicList(i);
151      m_apiMVPIdx[rpl]       = new Char[ uiNumPartition ];
152      m_apiMVPNum[rpl]       = new Char[ uiNumPartition ];
153      memset( m_apiMVPIdx[rpl], -1,uiNumPartition * sizeof( Char ) );
154    }
155
156    for (UInt comp=0; comp<MAX_NUM_COMPONENT; comp++)
157    {
158      const ComponentID compID = ComponentID(comp);
159      const UInt chromaShift = getComponentScaleX(compID, chromaFormatIDC) + getComponentScaleY(compID, chromaFormatIDC);
160      const UInt totalSize   = (uiWidth * uiHeight) >> chromaShift;
161
162      m_crossComponentPredictionAlpha[compID] = (Char*  )xMalloc(Char,   uiNumPartition);
163      m_puhTransformSkip[compID]              = (UChar* )xMalloc(UChar,  uiNumPartition);
164      m_explicitRdpcmMode[compID]             = (UChar* )xMalloc(UChar,  uiNumPartition);
165      m_puhCbf[compID]                        = (UChar* )xMalloc(UChar,  uiNumPartition);
166      m_pcTrCoeff[compID]                     = (TCoeff*)xMalloc(TCoeff, totalSize);
167      memset( m_pcTrCoeff[compID], 0, (totalSize * sizeof( TCoeff )) );
168
169#if ADAPTIVE_QP_SELECTION
170      if( pParentARLBuffer != 0 )
171      {
172        m_pcArlCoeff[compID] = pParentARLBuffer;
173        m_ArlCoeffIsAliasedAllocation = true;
174        pParentARLBuffer += totalSize;
175      }
176      else
177      {
178        m_pcArlCoeff[compID] = (TCoeff*)xMalloc(TCoeff, totalSize);
179        m_ArlCoeffIsAliasedAllocation = false;
180      }
181#endif
182      m_pcIPCMSample[compID] = (Pel*   )xMalloc(Pel , totalSize);
183    }
184
185    m_pbIPCMFlag         = (Bool*  )xMalloc(Bool, uiNumPartition);
186
187    for(UInt i=0; i<NUM_REF_PIC_LIST_01; i++)
188    {
189      m_acCUMvField[i].create( uiNumPartition );
190    }
191
192  }
193  else
194  {
195    for(UInt i=0; i<NUM_REF_PIC_LIST_01; i++)
196    {
197      m_acCUMvField[i].setNumPartition(uiNumPartition );
198    }
199  }
200
201  // create motion vector fields
202
203  m_pCtuAboveLeft      = NULL;
204  m_pCtuAboveRight     = NULL;
205  m_pCtuAbove          = NULL;
206  m_pCtuLeft           = NULL;
207
208  for(UInt i=0; i<NUM_REF_PIC_LIST_01; i++)
209  {
210    m_apcCUColocated[i]  = NULL;
211  }
212}
213
214Void TComDataCU::destroy()
215{
216  // encoder-side buffer free
217  if ( !m_bDecSubCu )
218  {
219    if ( m_phQP )
220    {
221      xFree(m_phQP);
222      m_phQP = NULL;
223    }
224    if ( m_puhDepth )
225    {
226      xFree(m_puhDepth);
227      m_puhDepth = NULL;
228    }
229    if ( m_puhWidth )
230    {
231      xFree(m_puhWidth);
232      m_puhWidth = NULL;
233    }
234    if ( m_puhHeight )
235    {
236      xFree(m_puhHeight);
237      m_puhHeight = NULL;
238    }
239
240    if ( m_skipFlag )
241    {
242      delete[] m_skipFlag;
243      m_skipFlag = NULL;
244    }
245
246    if ( m_pePartSize )
247    {
248      delete[] m_pePartSize;
249      m_pePartSize = NULL;
250    }
251    if ( m_pePredMode )
252    {
253      delete[] m_pePredMode;
254      m_pePredMode = NULL;
255    }
256    if ( m_ChromaQpAdj )
257    {
258      delete[] m_ChromaQpAdj;
259      m_ChromaQpAdj = NULL;
260    }
261    if ( m_CUTransquantBypass )
262    {
263      delete[] m_CUTransquantBypass;
264      m_CUTransquantBypass = NULL;
265    }
266    if ( m_puhInterDir )
267    {
268      xFree(m_puhInterDir);
269      m_puhInterDir = NULL;
270    }
271    if ( m_pbMergeFlag )
272    {
273      xFree(m_pbMergeFlag);
274      m_pbMergeFlag = NULL;
275    }
276    if ( m_puhMergeIndex )
277    {
278      xFree(m_puhMergeIndex);
279      m_puhMergeIndex  = NULL;
280    }
281
282    for (UInt ch=0; ch<MAX_NUM_CHANNEL_TYPE; ch++)
283    {
284      xFree(m_puhIntraDir[ch]);
285      m_puhIntraDir[ch] = NULL;
286    }
287
288    if ( m_puhTrIdx )
289    {
290      xFree(m_puhTrIdx);
291      m_puhTrIdx = NULL;
292    }
293
294    for (UInt comp=0; comp<MAX_NUM_COMPONENT; comp++)
295    {
296      if ( m_crossComponentPredictionAlpha[comp] )
297      {
298        xFree(m_crossComponentPredictionAlpha[comp]);
299        m_crossComponentPredictionAlpha[comp] = NULL;
300      }
301      if ( m_puhTransformSkip[comp] )
302      {
303        xFree(m_puhTransformSkip[comp]);
304        m_puhTransformSkip[comp] = NULL;
305      }
306      if ( m_puhCbf[comp] )
307      {
308        xFree(m_puhCbf[comp]);
309        m_puhCbf[comp] = NULL;
310      }
311      if ( m_pcTrCoeff[comp] )
312      {
313        xFree(m_pcTrCoeff[comp]);
314        m_pcTrCoeff[comp] = NULL;
315      }
316      if ( m_explicitRdpcmMode[comp] )
317      {
318        xFree(m_explicitRdpcmMode[comp]);
319        m_explicitRdpcmMode[comp] = NULL;
320      }
321
322#if ADAPTIVE_QP_SELECTION
323      if (!m_ArlCoeffIsAliasedAllocation)
324      {
325        if ( m_pcArlCoeff[comp] )
326        {
327          xFree(m_pcArlCoeff[comp]);
328          m_pcArlCoeff[comp] = NULL;
329        }
330      }
331#endif
332
333      if ( m_pcIPCMSample[comp] )
334      {
335        xFree(m_pcIPCMSample[comp]);
336        m_pcIPCMSample[comp] = NULL;
337      }
338    }
339    if ( m_pbIPCMFlag )
340    {
341      xFree(m_pbIPCMFlag );
342      m_pbIPCMFlag = NULL;
343    }
344
345    for(UInt i=0; i<NUM_REF_PIC_LIST_01; i++)
346    {
347      const RefPicList rpl=RefPicList(i);
348      if ( m_apiMVPIdx[rpl] )
349      {
350        delete[] m_apiMVPIdx[rpl];
351        m_apiMVPIdx[rpl] = NULL;
352      }
353      if ( m_apiMVPNum[rpl] )
354      {
355        delete[] m_apiMVPNum[rpl];
356        m_apiMVPNum[rpl] = NULL;
357      }
358    }
359
360    for(UInt i=0; i<NUM_REF_PIC_LIST_01; i++)
361    {
362      const RefPicList rpl=RefPicList(i);
363      m_acCUMvField[rpl].destroy();
364    }
365  }
366
367  m_pcPic              = NULL;
368  m_pcSlice            = NULL;
369
370  m_pCtuAboveLeft      = NULL;
371  m_pCtuAboveRight     = NULL;
372  m_pCtuAbove          = NULL;
373  m_pCtuLeft           = NULL;
374
375
376  for(UInt i=0; i<NUM_REF_PIC_LIST_01; i++)
377  {
378    m_apcCUColocated[i]  = NULL;
379  }
380
381}
382
383Bool TComDataCU::CUIsFromSameTile            ( const TComDataCU *pCU /* Can be NULL */) const
384{
385  return pCU!=NULL &&
386         pCU->getSlice() != NULL &&
387         m_pcPic->getPicSym()->getTileIdxMap( pCU->getCtuRsAddr() ) == m_pcPic->getPicSym()->getTileIdxMap(getCtuRsAddr());
388}
389
390Bool TComDataCU::CUIsFromSameSliceAndTile    ( const TComDataCU *pCU /* Can be NULL */) const
391{
392  return pCU!=NULL &&
393         pCU->getSlice() != NULL &&
394         pCU->getSlice()->getSliceCurStartCtuTsAddr() == getSlice()->getSliceCurStartCtuTsAddr() &&
395         m_pcPic->getPicSym()->getTileIdxMap( pCU->getCtuRsAddr() ) == m_pcPic->getPicSym()->getTileIdxMap(getCtuRsAddr())
396         ;
397}
398
399Bool TComDataCU::CUIsFromSameSliceTileAndWavefrontRow( const TComDataCU *pCU /* Can be NULL */) const
400{
401  return CUIsFromSameSliceAndTile(pCU)
402         && (!getSlice()->getPPS()->getEntropyCodingSyncEnabledFlag() || getPic()->getCtu(getCtuRsAddr())->getCUPelY() == getPic()->getCtu(pCU->getCtuRsAddr())->getCUPelY());
403}
404
405Bool TComDataCU::isLastSubCUOfCtu(const UInt absPartIdx)
406{
407  const TComSPS &sps=*(getSlice()->getSPS());
408
409#if SVC_EXTENSION
410  TComSlice * pcSlice = m_pcPic->getSlice(m_pcPic->getCurrSliceIdx());
411  const UInt picWidth = pcSlice->getPicWidthInLumaSamples();
412  const UInt picHeight = pcSlice->getPicHeightInLumaSamples();
413#else
414  const UInt picWidth = sps.getPicWidthInLumaSamples();
415  const UInt picHeight = sps.getPicHeightInLumaSamples();
416#endif
417  const UInt granularityWidth = sps.getMaxCUWidth();
418
419  const UInt cuPosX = getCUPelX() + g_auiRasterToPelX[ g_auiZscanToRaster[absPartIdx] ];
420  const UInt cuPosY = getCUPelY() + g_auiRasterToPelY[ g_auiZscanToRaster[absPartIdx] ];
421
422  return (((cuPosX+getWidth( absPartIdx))%granularityWidth==0||(cuPosX+getWidth( absPartIdx)==picWidth ))
423       && ((cuPosY+getHeight(absPartIdx))%granularityWidth==0||(cuPosY+getHeight(absPartIdx)==picHeight)));
424}
425
426// ====================================================================================================================
427// Public member functions
428// ====================================================================================================================
429
430// --------------------------------------------------------------------------------------------------------------------
431// Initialization
432// --------------------------------------------------------------------------------------------------------------------
433
434/**
435 Initialize top-level CU: create internal buffers and set initial values before encoding the CTU.
436 
437 \param  pcPic       picture (TComPic) class pointer
438 \param  ctuRsAddr   CTU address in raster scan order
439 */
440Void TComDataCU::initCtu( TComPic* pcPic, UInt ctuRsAddr )
441{
442
443  const UInt maxCUWidth = pcPic->getPicSym()->getSPS().getMaxCUWidth();
444  const UInt maxCUHeight= pcPic->getPicSym()->getSPS().getMaxCUHeight();
445  m_pcPic              = pcPic;
446  m_pcSlice            = pcPic->getSlice(pcPic->getCurrSliceIdx());
447  m_ctuRsAddr          = ctuRsAddr;
448  m_uiCUPelX           = ( ctuRsAddr % pcPic->getFrameWidthInCtus() ) * maxCUWidth;
449  m_uiCUPelY           = ( ctuRsAddr / pcPic->getFrameWidthInCtus() ) * maxCUHeight;
450  m_absZIdxInCtu       = 0;
451  m_dTotalCost         = MAX_DOUBLE;
452  m_uiTotalDistortion  = 0;
453  m_uiTotalBits        = 0;
454  m_uiTotalBins        = 0;
455  m_uiNumPartition     = pcPic->getNumPartitionsInCtu();
456
457  memset( m_skipFlag          , false,                      m_uiNumPartition * sizeof( *m_skipFlag ) );
458
459#if SVC_EXTENSION
460  m_layerId           = pcPic->getLayerId();
461#endif
462
463  memset( m_pePartSize        , NUMBER_OF_PART_SIZES,       m_uiNumPartition * sizeof( *m_pePartSize ) );
464  memset( m_pePredMode        , NUMBER_OF_PREDICTION_MODES, m_uiNumPartition * sizeof( *m_pePredMode ) );
465  memset( m_CUTransquantBypass, false,                      m_uiNumPartition * sizeof( *m_CUTransquantBypass) );
466  memset( m_puhDepth          , 0,                          m_uiNumPartition * sizeof( *m_puhDepth ) );
467  memset( m_puhTrIdx          , 0,                          m_uiNumPartition * sizeof( *m_puhTrIdx ) );
468  memset( m_puhWidth          , maxCUWidth,                 m_uiNumPartition * sizeof( *m_puhWidth ) );
469  memset( m_puhHeight         , maxCUHeight,                m_uiNumPartition * sizeof( *m_puhHeight ) );
470  for(UInt i=0; i<NUM_REF_PIC_LIST_01; i++)
471  {
472    const RefPicList rpl=RefPicList(i);
473    memset( m_apiMVPIdx[rpl]  , -1,                         m_uiNumPartition * sizeof( *m_apiMVPIdx[rpl] ) );
474    memset( m_apiMVPNum[rpl]  , -1,                         m_uiNumPartition * sizeof( *m_apiMVPNum[rpl] ) );
475  }
476  memset( m_phQP              , getSlice()->getSliceQp(),   m_uiNumPartition * sizeof( *m_phQP ) );
477  memset( m_ChromaQpAdj       , 0,                          m_uiNumPartition * sizeof( *m_ChromaQpAdj ) );
478  for(UInt comp=0; comp<MAX_NUM_COMPONENT; comp++)
479  {
480    memset( m_crossComponentPredictionAlpha[comp] , 0,                     m_uiNumPartition * sizeof( *m_crossComponentPredictionAlpha[comp] ) );
481    memset( m_puhTransformSkip[comp]              , 0,                     m_uiNumPartition * sizeof( *m_puhTransformSkip[comp]) );
482    memset( m_puhCbf[comp]                        , 0,                     m_uiNumPartition * sizeof( *m_puhCbf[comp] ) );
483    memset( m_explicitRdpcmMode[comp]             , NUMBER_OF_RDPCM_MODES, m_uiNumPartition * sizeof( *m_explicitRdpcmMode[comp] ) );
484  }
485  memset( m_pbMergeFlag       , false,                    m_uiNumPartition * sizeof( *m_pbMergeFlag ) );
486  memset( m_puhMergeIndex     , 0,                        m_uiNumPartition * sizeof( *m_puhMergeIndex ) );
487  for (UInt ch=0; ch<MAX_NUM_CHANNEL_TYPE; ch++)
488  {
489    memset( m_puhIntraDir[ch] , ((ch==0) ? DC_IDX : 0),   m_uiNumPartition * sizeof( *(m_puhIntraDir[ch]) ) );
490  }
491  memset( m_puhInterDir       , 0,                        m_uiNumPartition * sizeof( *m_puhInterDir ) );
492  memset( m_pbIPCMFlag        , false,                    m_uiNumPartition * sizeof( *m_pbIPCMFlag ) );
493
494  const UInt numCoeffY    = maxCUWidth*maxCUHeight;
495  for (UInt comp=0; comp<MAX_NUM_COMPONENT; comp++)
496  {
497    const UInt componentShift = m_pcPic->getComponentScaleX(ComponentID(comp)) + m_pcPic->getComponentScaleY(ComponentID(comp));
498    memset( m_pcTrCoeff[comp], 0, sizeof(TCoeff)* numCoeffY>>componentShift );
499#if ADAPTIVE_QP_SELECTION
500    memset( m_pcArlCoeff[comp], 0, sizeof(TCoeff)* numCoeffY>>componentShift );
501#endif
502  }
503
504  for(UInt i=0; i<NUM_REF_PIC_LIST_01; i++)
505  {
506    m_acCUMvField[i].clearMvField();
507  }
508
509  // Setting neighbor CU
510  m_pCtuLeft        = NULL;
511  m_pCtuAbove       = NULL;
512  m_pCtuAboveLeft   = NULL;
513  m_pCtuAboveRight  = NULL;
514
515
516  for(UInt i=0; i<NUM_REF_PIC_LIST_01; i++)
517  {
518    m_apcCUColocated[i]  = NULL;
519  }
520
521  UInt frameWidthInCtus = pcPic->getFrameWidthInCtus();
522  if ( m_ctuRsAddr % frameWidthInCtus )
523  {
524    m_pCtuLeft = pcPic->getCtu( m_ctuRsAddr - 1 );
525  }
526
527  if ( m_ctuRsAddr / frameWidthInCtus )
528  {
529    m_pCtuAbove = pcPic->getCtu( m_ctuRsAddr - frameWidthInCtus );
530  }
531
532  if ( m_pCtuLeft && m_pCtuAbove )
533  {
534    m_pCtuAboveLeft = pcPic->getCtu( m_ctuRsAddr - frameWidthInCtus - 1 );
535  }
536
537  if ( m_pCtuAbove && ( (m_ctuRsAddr%frameWidthInCtus) < (frameWidthInCtus-1) )  )
538  {
539    m_pCtuAboveRight = pcPic->getCtu( m_ctuRsAddr - frameWidthInCtus + 1 );
540  }
541
542  for(UInt i=0; i<NUM_REF_PIC_LIST_01; i++)
543  {
544    const RefPicList rpl=RefPicList(i);
545    if ( getSlice()->getNumRefIdx( rpl ) > 0 )
546    {
547      m_apcCUColocated[rpl] = getSlice()->getRefPic( rpl, 0)->getCtu( m_ctuRsAddr );
548    }
549  }
550}
551
552
553/** Initialize prediction data with enabling sub-CTU-level delta QP.
554*   - set CU width and CU height according to depth
555*   - set qp value according to input qp
556*   - set last-coded qp value according to input last-coded qp
557*
558* \param  uiDepth            depth of the current CU
559* \param  qp                 qp for the current CU
560* \param  bTransquantBypass  true for transquant bypass
561*/
562Void TComDataCU::initEstData( const UInt uiDepth, const Int qp, const Bool bTransquantBypass )
563{
564  m_dTotalCost         = MAX_DOUBLE;
565  m_uiTotalDistortion  = 0;
566  m_uiTotalBits        = 0;
567  m_uiTotalBins        = 0;
568
569  const UChar uhWidth  = getSlice()->getSPS()->getMaxCUWidth()  >> uiDepth;
570  const UChar uhHeight = getSlice()->getSPS()->getMaxCUHeight() >> uiDepth;
571
572  for (UInt ui = 0; ui < m_uiNumPartition; ui++)
573  {
574    for(UInt i=0; i<NUM_REF_PIC_LIST_01; i++)
575    {
576      const RefPicList rpl=RefPicList(i);
577      m_apiMVPIdx[rpl][ui]  = -1;
578      m_apiMVPNum[rpl][ui]  = -1;
579    }
580    m_puhDepth  [ui]    = uiDepth;
581    m_puhWidth  [ui]    = uhWidth;
582    m_puhHeight [ui]    = uhHeight;
583    m_puhTrIdx  [ui]    = 0;
584    for(UInt comp=0; comp<MAX_NUM_COMPONENT; comp++)
585    {
586      m_crossComponentPredictionAlpha[comp][ui] = 0;
587      m_puhTransformSkip             [comp][ui] = 0;
588      m_explicitRdpcmMode            [comp][ui] = NUMBER_OF_RDPCM_MODES;
589    }
590    m_skipFlag[ui]      = false;
591    m_pePartSize[ui]    = NUMBER_OF_PART_SIZES;
592    m_pePredMode[ui]    = NUMBER_OF_PREDICTION_MODES;
593    m_CUTransquantBypass[ui] = bTransquantBypass;
594    m_pbIPCMFlag[ui]    = 0;
595    m_phQP[ui]          = qp;
596    m_ChromaQpAdj[ui]   = 0;
597    m_pbMergeFlag[ui]   = 0;
598    m_puhMergeIndex[ui] = 0;
599
600    for (UInt ch=0; ch<MAX_NUM_CHANNEL_TYPE; ch++)
601    {
602      m_puhIntraDir[ch][ui] = ((ch==0) ? DC_IDX : 0);
603    }
604
605    m_puhInterDir[ui] = 0;
606    for (UInt comp=0; comp<MAX_NUM_COMPONENT; comp++)
607    {
608      m_puhCbf[comp][ui] = 0;
609    }
610  }
611
612  for(UInt i=0; i<NUM_REF_PIC_LIST_01; i++)
613  {
614    m_acCUMvField[i].clearMvField();
615  }
616
617  const UInt numCoeffY = uhWidth*uhHeight;
618
619  for (UInt comp=0; comp<MAX_NUM_COMPONENT; comp++)
620  {
621    const ComponentID component = ComponentID(comp);
622    const UInt numCoeff = numCoeffY >> (getPic()->getComponentScaleX(component) + getPic()->getComponentScaleY(component));
623    memset( m_pcTrCoeff[comp],    0, numCoeff * sizeof( TCoeff ) );
624#if ADAPTIVE_QP_SELECTION
625    memset( m_pcArlCoeff[comp],   0, numCoeff * sizeof( TCoeff ) );
626#endif
627    memset( m_pcIPCMSample[comp], 0, numCoeff * sizeof( Pel) );
628  }
629}
630
631
632// initialize Sub partition
633Void TComDataCU::initSubCU( TComDataCU* pcCU, UInt uiPartUnitIdx, UInt uiDepth, Int qp )
634{
635  assert( uiPartUnitIdx<4 );
636
637  UInt uiPartOffset = ( pcCU->getTotalNumPart()>>2 )*uiPartUnitIdx;
638
639  m_pcPic              = pcCU->getPic();
640  m_pcSlice            = pcCU->getSlice();
641  m_ctuRsAddr          = pcCU->getCtuRsAddr();
642  m_absZIdxInCtu       = pcCU->getZorderIdxInCtu() + uiPartOffset;
643
644  const UChar uhWidth  = getSlice()->getSPS()->getMaxCUWidth()  >> uiDepth;
645  const UChar uhHeight = getSlice()->getSPS()->getMaxCUHeight() >> uiDepth;
646
647  m_uiCUPelX           = pcCU->getCUPelX() + ( uhWidth )*( uiPartUnitIdx &  1 );
648  m_uiCUPelY           = pcCU->getCUPelY() + ( uhHeight)*( uiPartUnitIdx >> 1 );
649
650  m_dTotalCost         = MAX_DOUBLE;
651  m_uiTotalDistortion  = 0;
652  m_uiTotalBits        = 0;
653  m_uiTotalBins        = 0;
654  m_uiNumPartition     = pcCU->getTotalNumPart() >> 2;
655
656  Int iSizeInUchar = sizeof( UChar  ) * m_uiNumPartition;
657  Int iSizeInBool  = sizeof( Bool   ) * m_uiNumPartition;
658  Int sizeInChar = sizeof( Char  ) * m_uiNumPartition;
659
660  memset( m_phQP,              qp,  sizeInChar );
661  memset( m_pbMergeFlag,        0, iSizeInBool  );
662  memset( m_puhMergeIndex,      0, iSizeInUchar );
663  for (UInt ch=0; ch<MAX_NUM_CHANNEL_TYPE; ch++)
664  {
665    memset( m_puhIntraDir[ch],  ((ch==0) ? DC_IDX : 0), iSizeInUchar );
666  }
667
668  memset( m_puhInterDir,        0, iSizeInUchar );
669  memset( m_puhTrIdx,           0, iSizeInUchar );
670
671  for(UInt comp=0; comp<MAX_NUM_COMPONENT; comp++)
672  {
673    memset( m_crossComponentPredictionAlpha[comp], 0, iSizeInUchar );
674    memset( m_puhTransformSkip[comp],              0, iSizeInUchar );
675    memset( m_puhCbf[comp],                        0, iSizeInUchar );
676    memset( m_explicitRdpcmMode[comp],             NUMBER_OF_RDPCM_MODES, iSizeInUchar );
677  }
678
679  memset( m_puhDepth,     uiDepth, iSizeInUchar );
680  memset( m_puhWidth,          uhWidth,  iSizeInUchar );
681  memset( m_puhHeight,         uhHeight, iSizeInUchar );
682  memset( m_pbIPCMFlag,        0, iSizeInBool  );
683  for (UInt ui = 0; ui < m_uiNumPartition; ui++)
684  {
685    m_skipFlag[ui]   = false;
686    m_pePartSize[ui] = NUMBER_OF_PART_SIZES;
687    m_pePredMode[ui] = NUMBER_OF_PREDICTION_MODES;
688    m_CUTransquantBypass[ui] = false;
689    m_ChromaQpAdj[ui] = 0;
690
691    for(UInt i=0; i<NUM_REF_PIC_LIST_01; i++)
692    {
693      const RefPicList rpl=RefPicList(i);
694      m_apiMVPIdx[rpl][ui] = -1;
695      m_apiMVPNum[rpl][ui] = -1;
696    }
697  }
698
699  const UInt numCoeffY    = uhWidth*uhHeight;
700  for (UInt ch=0; ch<MAX_NUM_COMPONENT; ch++)
701  {
702    const UInt componentShift = m_pcPic->getComponentScaleX(ComponentID(ch)) + m_pcPic->getComponentScaleY(ComponentID(ch));
703    memset( m_pcTrCoeff[ch],  0, sizeof(TCoeff)*(numCoeffY>>componentShift) );
704#if ADAPTIVE_QP_SELECTION
705    memset( m_pcArlCoeff[ch], 0, sizeof(TCoeff)*(numCoeffY>>componentShift) );
706#endif
707    memset( m_pcIPCMSample[ch], 0, sizeof(Pel)* (numCoeffY>>componentShift) );
708  }
709
710  for(UInt i=0; i<NUM_REF_PIC_LIST_01; i++)
711  {
712    m_acCUMvField[i].clearMvField();
713  }
714
715  m_pCtuLeft        = pcCU->getCtuLeft();
716  m_pCtuAbove       = pcCU->getCtuAbove();
717  m_pCtuAboveLeft   = pcCU->getCtuAboveLeft();
718  m_pCtuAboveRight  = pcCU->getCtuAboveRight();
719
720  for(UInt i=0; i<NUM_REF_PIC_LIST_01; i++)
721  {
722    m_apcCUColocated[i] = pcCU->getCUColocated(RefPicList(i));
723  }
724}
725
726Void TComDataCU::setOutsideCUPart( UInt uiAbsPartIdx, UInt uiDepth )
727{
728  const UInt     uiNumPartition = m_uiNumPartition >> (uiDepth << 1);
729  const UInt     uiSizeInUchar  = sizeof( UChar  ) * uiNumPartition;
730  const TComSPS &sps            = *(getSlice()->getSPS());
731  const UChar    uhWidth        = sps.getMaxCUWidth()  >> uiDepth;
732  const UChar    uhHeight       = sps.getMaxCUHeight() >> uiDepth;
733  memset( m_puhDepth    + uiAbsPartIdx,     uiDepth,  uiSizeInUchar );
734  memset( m_puhWidth    + uiAbsPartIdx,     uhWidth,  uiSizeInUchar );
735  memset( m_puhHeight   + uiAbsPartIdx,     uhHeight, uiSizeInUchar );
736}
737
738// --------------------------------------------------------------------------------------------------------------------
739// Copy
740// --------------------------------------------------------------------------------------------------------------------
741
742Void TComDataCU::copySubCU( TComDataCU* pcCU, UInt uiAbsPartIdx )
743{
744  UInt uiPart = uiAbsPartIdx;
745
746  m_pcPic              = pcCU->getPic();
747  m_pcSlice            = pcCU->getSlice();
748  m_ctuRsAddr          = pcCU->getCtuRsAddr();
749  m_absZIdxInCtu       = uiAbsPartIdx;
750
751  m_uiCUPelX           = pcCU->getCUPelX() + g_auiRasterToPelX[ g_auiZscanToRaster[uiAbsPartIdx] ];
752  m_uiCUPelY           = pcCU->getCUPelY() + g_auiRasterToPelY[ g_auiZscanToRaster[uiAbsPartIdx] ];
753
754  m_skipFlag=pcCU->getSkipFlag()          + uiPart;
755
756  m_phQP=pcCU->getQP()                    + uiPart;
757  m_ChromaQpAdj = pcCU->getChromaQpAdj()  + uiPart;
758  m_pePartSize = pcCU->getPartitionSize() + uiPart;
759  m_pePredMode=pcCU->getPredictionMode()  + uiPart;
760  m_CUTransquantBypass  = pcCU->getCUTransquantBypass()+uiPart;
761
762  m_pbMergeFlag         = pcCU->getMergeFlag()        + uiPart;
763  m_puhMergeIndex       = pcCU->getMergeIndex()       + uiPart;
764
765  for (UInt ch=0; ch<MAX_NUM_CHANNEL_TYPE; ch++)
766  {
767    m_puhIntraDir[ch]   = pcCU->getIntraDir(ChannelType(ch)) + uiPart;
768  }
769
770  m_puhInterDir         = pcCU->getInterDir()         + uiPart;
771  m_puhTrIdx            = pcCU->getTransformIdx()     + uiPart;
772
773  for(UInt comp=0; comp<MAX_NUM_COMPONENT; comp++)
774  {
775    m_crossComponentPredictionAlpha[comp] = pcCU->getCrossComponentPredictionAlpha(ComponentID(comp)) + uiPart;
776    m_puhTransformSkip[comp]              = pcCU->getTransformSkip(ComponentID(comp))                 + uiPart;
777    m_puhCbf[comp]                        = pcCU->getCbf(ComponentID(comp))                           + uiPart;
778    m_explicitRdpcmMode[comp]             = pcCU->getExplicitRdpcmMode(ComponentID(comp))             + uiPart;
779  }
780
781  m_puhDepth=pcCU->getDepth()                     + uiPart;
782  m_puhWidth=pcCU->getWidth()                     + uiPart;
783  m_puhHeight=pcCU->getHeight()                   + uiPart;
784
785  m_pbIPCMFlag         = pcCU->getIPCMFlag()        + uiPart;
786
787  m_pCtuAboveLeft      = pcCU->getCtuAboveLeft();
788  m_pCtuAboveRight     = pcCU->getCtuAboveRight();
789  m_pCtuAbove          = pcCU->getCtuAbove();
790  m_pCtuLeft           = pcCU->getCtuLeft();
791
792  for(UInt i=0; i<NUM_REF_PIC_LIST_01; i++)
793  {
794    const RefPicList rpl=RefPicList(i);
795    m_apcCUColocated[rpl] = pcCU->getCUColocated(rpl);
796    m_apiMVPIdx[rpl]=pcCU->getMVPIdx(rpl)  + uiPart;
797    m_apiMVPNum[rpl]=pcCU->getMVPNum(rpl)  + uiPart;
798  }
799
800  for(UInt i=0; i<NUM_REF_PIC_LIST_01; i++)
801  {
802    const RefPicList rpl=RefPicList(i);
803    m_acCUMvField[rpl].linkToWithOffset( pcCU->getCUMvField(rpl), uiPart );
804  }
805
806  UInt uiMaxCuWidth=pcCU->getSlice()->getSPS()->getMaxCUWidth();
807  UInt uiMaxCuHeight=pcCU->getSlice()->getSPS()->getMaxCUHeight();
808
809  UInt uiCoffOffset = uiMaxCuWidth*uiMaxCuHeight*uiAbsPartIdx/pcCU->getPic()->getNumPartitionsInCtu();
810
811  for (UInt ch=0; ch<MAX_NUM_COMPONENT; ch++)
812  {
813    const ComponentID component = ComponentID(ch);
814    const UInt componentShift   = m_pcPic->getComponentScaleX(component) + m_pcPic->getComponentScaleY(component);
815    const UInt offset           = uiCoffOffset >> componentShift;
816    m_pcTrCoeff[ch] = pcCU->getCoeff(component) + offset;
817#if ADAPTIVE_QP_SELECTION
818    m_pcArlCoeff[ch] = pcCU->getArlCoeff(component) + offset;
819#endif
820    m_pcIPCMSample[ch] = pcCU->getPCMSample(component) + offset;
821  }
822}
823
824// Copy inter prediction info from the biggest CU
825Void TComDataCU::copyInterPredInfoFrom    ( TComDataCU* pcCU, UInt uiAbsPartIdx, RefPicList eRefPicList )
826{
827  m_pcPic              = pcCU->getPic();
828  m_pcSlice            = pcCU->getSlice();
829  m_ctuRsAddr          = pcCU->getCtuRsAddr();
830  m_absZIdxInCtu       = uiAbsPartIdx;
831
832  Int iRastPartIdx     = g_auiZscanToRaster[uiAbsPartIdx];
833  m_uiCUPelX           = pcCU->getCUPelX() + m_pcPic->getMinCUWidth ()*( iRastPartIdx % m_pcPic->getNumPartInCtuWidth() );
834  m_uiCUPelY           = pcCU->getCUPelY() + m_pcPic->getMinCUHeight()*( iRastPartIdx / m_pcPic->getNumPartInCtuWidth() );
835
836  m_pCtuAboveLeft      = pcCU->getCtuAboveLeft();
837  m_pCtuAboveRight     = pcCU->getCtuAboveRight();
838  m_pCtuAbove          = pcCU->getCtuAbove();
839  m_pCtuLeft           = pcCU->getCtuLeft();
840
841  for(UInt i=0; i<NUM_REF_PIC_LIST_01; i++)
842  {
843    m_apcCUColocated[i]  = pcCU->getCUColocated(RefPicList(i));
844  }
845
846  m_skipFlag           = pcCU->getSkipFlag ()             + uiAbsPartIdx;
847
848  m_pePartSize         = pcCU->getPartitionSize ()        + uiAbsPartIdx;
849  m_pePredMode         = pcCU->getPredictionMode()        + uiAbsPartIdx;
850  m_ChromaQpAdj        = pcCU->getChromaQpAdj()           + uiAbsPartIdx;
851  m_CUTransquantBypass = pcCU->getCUTransquantBypass()    + uiAbsPartIdx;
852  m_puhInterDir        = pcCU->getInterDir      ()        + uiAbsPartIdx;
853
854  m_puhDepth           = pcCU->getDepth ()                + uiAbsPartIdx;
855  m_puhWidth           = pcCU->getWidth ()                + uiAbsPartIdx;
856  m_puhHeight          = pcCU->getHeight()                + uiAbsPartIdx;
857
858  m_pbMergeFlag        = pcCU->getMergeFlag()             + uiAbsPartIdx;
859  m_puhMergeIndex      = pcCU->getMergeIndex()            + uiAbsPartIdx;
860
861  m_apiMVPIdx[eRefPicList] = pcCU->getMVPIdx(eRefPicList) + uiAbsPartIdx;
862  m_apiMVPNum[eRefPicList] = pcCU->getMVPNum(eRefPicList) + uiAbsPartIdx;
863
864  m_acCUMvField[ eRefPicList ].linkToWithOffset( pcCU->getCUMvField(eRefPicList), uiAbsPartIdx );
865}
866
867// Copy small CU to bigger CU.
868// One of quarter parts overwritten by predicted sub part.
869Void TComDataCU::copyPartFrom( TComDataCU* pcCU, UInt uiPartUnitIdx, UInt uiDepth )
870{
871  assert( uiPartUnitIdx<4 );
872
873  m_dTotalCost         += pcCU->getTotalCost();
874  m_uiTotalDistortion  += pcCU->getTotalDistortion();
875  m_uiTotalBits        += pcCU->getTotalBits();
876
877  UInt uiOffset         = pcCU->getTotalNumPart()*uiPartUnitIdx;
878  const UInt numValidComp=pcCU->getPic()->getNumberValidComponents();
879  const UInt numValidChan=pcCU->getPic()->getChromaFormat()==CHROMA_400 ? 1:2;
880
881  UInt uiNumPartition = pcCU->getTotalNumPart();
882  Int iSizeInUchar  = sizeof( UChar ) * uiNumPartition;
883  Int iSizeInBool   = sizeof( Bool  ) * uiNumPartition;
884
885  Int sizeInChar  = sizeof( Char ) * uiNumPartition;
886  memcpy( m_skipFlag   + uiOffset, pcCU->getSkipFlag(),       sizeof( *m_skipFlag )   * uiNumPartition );
887  memcpy( m_phQP       + uiOffset, pcCU->getQP(),             sizeInChar                        );
888  memcpy( m_pePartSize + uiOffset, pcCU->getPartitionSize(),  sizeof( *m_pePartSize ) * uiNumPartition );
889  memcpy( m_pePredMode + uiOffset, pcCU->getPredictionMode(), sizeof( *m_pePredMode ) * uiNumPartition );
890  memcpy( m_ChromaQpAdj + uiOffset, pcCU->getChromaQpAdj(),   sizeof( *m_ChromaQpAdj ) * uiNumPartition );
891  memcpy( m_CUTransquantBypass + uiOffset, pcCU->getCUTransquantBypass(), sizeof( *m_CUTransquantBypass ) * uiNumPartition );
892  memcpy( m_pbMergeFlag         + uiOffset, pcCU->getMergeFlag(),         iSizeInBool  );
893  memcpy( m_puhMergeIndex       + uiOffset, pcCU->getMergeIndex(),        iSizeInUchar );
894
895  for (UInt ch=0; ch<numValidChan; ch++)
896  {
897    memcpy( m_puhIntraDir[ch]   + uiOffset, pcCU->getIntraDir(ChannelType(ch)), iSizeInUchar );
898  }
899
900  memcpy( m_puhInterDir         + uiOffset, pcCU->getInterDir(),          iSizeInUchar );
901  memcpy( m_puhTrIdx            + uiOffset, pcCU->getTransformIdx(),      iSizeInUchar );
902
903  for(UInt comp=0; comp<numValidComp; comp++)
904  {
905    memcpy( m_crossComponentPredictionAlpha[comp] + uiOffset, pcCU->getCrossComponentPredictionAlpha(ComponentID(comp)), iSizeInUchar );
906    memcpy( m_puhTransformSkip[comp]              + uiOffset, pcCU->getTransformSkip(ComponentID(comp))                , iSizeInUchar );
907    memcpy( m_puhCbf[comp]                        + uiOffset, pcCU->getCbf(ComponentID(comp))                          , iSizeInUchar );
908    memcpy( m_explicitRdpcmMode[comp]             + uiOffset, pcCU->getExplicitRdpcmMode(ComponentID(comp))            , iSizeInUchar );
909  }
910
911  memcpy( m_puhDepth  + uiOffset, pcCU->getDepth(),  iSizeInUchar );
912  memcpy( m_puhWidth  + uiOffset, pcCU->getWidth(),  iSizeInUchar );
913  memcpy( m_puhHeight + uiOffset, pcCU->getHeight(), iSizeInUchar );
914
915  memcpy( m_pbIPCMFlag + uiOffset, pcCU->getIPCMFlag(), iSizeInBool );
916
917  m_pCtuAboveLeft      = pcCU->getCtuAboveLeft();
918  m_pCtuAboveRight     = pcCU->getCtuAboveRight();
919  m_pCtuAbove          = pcCU->getCtuAbove();
920  m_pCtuLeft           = pcCU->getCtuLeft();
921
922  for(UInt i=0; i<NUM_REF_PIC_LIST_01; i++)
923  {
924    const RefPicList rpl=RefPicList(i);
925    memcpy( m_apiMVPIdx[rpl] + uiOffset, pcCU->getMVPIdx(rpl), iSizeInUchar );
926    memcpy( m_apiMVPNum[rpl] + uiOffset, pcCU->getMVPNum(rpl), iSizeInUchar );
927    m_apcCUColocated[rpl] = pcCU->getCUColocated(rpl);
928  }
929
930  for(UInt i=0; i<NUM_REF_PIC_LIST_01; i++)
931  {
932    const RefPicList rpl=RefPicList(i);
933    m_acCUMvField[rpl].copyFrom( pcCU->getCUMvField( rpl ), pcCU->getTotalNumPart(), uiOffset );
934  }
935
936  const UInt numCoeffY = (pcCU->getSlice()->getSPS()->getMaxCUWidth()*pcCU->getSlice()->getSPS()->getMaxCUHeight()) >> (uiDepth<<1);
937  const UInt offsetY   = uiPartUnitIdx*numCoeffY;
938  for (UInt ch=0; ch<numValidComp; ch++)
939  {
940    const ComponentID component = ComponentID(ch);
941    const UInt componentShift   = m_pcPic->getComponentScaleX(component) + m_pcPic->getComponentScaleY(component);
942    const UInt offset           = offsetY>>componentShift;
943    memcpy( m_pcTrCoeff [ch] + offset, pcCU->getCoeff(component),    sizeof(TCoeff)*(numCoeffY>>componentShift) );
944#if ADAPTIVE_QP_SELECTION
945    memcpy( m_pcArlCoeff[ch] + offset, pcCU->getArlCoeff(component), sizeof(TCoeff)*(numCoeffY>>componentShift) );
946#endif
947    memcpy( m_pcIPCMSample[ch] + offset, pcCU->getPCMSample(component), sizeof(Pel)*(numCoeffY>>componentShift) );
948  }
949
950  m_uiTotalBins += pcCU->getTotalBins();
951}
952
953// Copy current predicted part to a CU in picture.
954// It is used to predict for next part
955Void TComDataCU::copyToPic( UChar uhDepth )
956{
957  TComDataCU* pCtu = m_pcPic->getCtu( m_ctuRsAddr );
958  const UInt numValidComp=pCtu->getPic()->getNumberValidComponents();
959  const UInt numValidChan=pCtu->getPic()->getChromaFormat()==CHROMA_400 ? 1:2;
960
961  pCtu->getTotalCost()       = m_dTotalCost;
962  pCtu->getTotalDistortion() = m_uiTotalDistortion;
963  pCtu->getTotalBits()       = m_uiTotalBits;
964
965  Int iSizeInUchar  = sizeof( UChar ) * m_uiNumPartition;
966  Int iSizeInBool   = sizeof( Bool  ) * m_uiNumPartition;
967  Int sizeInChar  = sizeof( Char ) * m_uiNumPartition;
968
969  memcpy( pCtu->getSkipFlag() + m_absZIdxInCtu, m_skipFlag, sizeof( *m_skipFlag ) * m_uiNumPartition );
970
971  memcpy( pCtu->getQP() + m_absZIdxInCtu, m_phQP, sizeInChar  );
972
973  memcpy( pCtu->getPartitionSize()  + m_absZIdxInCtu, m_pePartSize, sizeof( *m_pePartSize ) * m_uiNumPartition );
974  memcpy( pCtu->getPredictionMode() + m_absZIdxInCtu, m_pePredMode, sizeof( *m_pePredMode ) * m_uiNumPartition );
975  memcpy( pCtu->getChromaQpAdj() + m_absZIdxInCtu, m_ChromaQpAdj, sizeof( *m_ChromaQpAdj ) * m_uiNumPartition );
976  memcpy( pCtu->getCUTransquantBypass()+ m_absZIdxInCtu, m_CUTransquantBypass, sizeof( *m_CUTransquantBypass ) * m_uiNumPartition );
977  memcpy( pCtu->getMergeFlag()         + m_absZIdxInCtu, m_pbMergeFlag,         iSizeInBool  );
978  memcpy( pCtu->getMergeIndex()        + m_absZIdxInCtu, m_puhMergeIndex,       iSizeInUchar );
979  for (UInt ch=0; ch<numValidChan; ch++)
980  {
981    memcpy( pCtu->getIntraDir(ChannelType(ch)) + m_absZIdxInCtu, m_puhIntraDir[ch], iSizeInUchar);
982  }
983
984  memcpy( pCtu->getInterDir()          + m_absZIdxInCtu, m_puhInterDir,         iSizeInUchar );
985  memcpy( pCtu->getTransformIdx()      + m_absZIdxInCtu, m_puhTrIdx,            iSizeInUchar );
986
987  for(UInt comp=0; comp<numValidComp; comp++)
988  {
989    memcpy( pCtu->getCrossComponentPredictionAlpha(ComponentID(comp)) + m_absZIdxInCtu, m_crossComponentPredictionAlpha[comp], iSizeInUchar );
990    memcpy( pCtu->getTransformSkip(ComponentID(comp))                 + m_absZIdxInCtu, m_puhTransformSkip[comp],              iSizeInUchar );
991    memcpy( pCtu->getCbf(ComponentID(comp))                           + m_absZIdxInCtu, m_puhCbf[comp],                        iSizeInUchar );
992    memcpy( pCtu->getExplicitRdpcmMode(ComponentID(comp))             + m_absZIdxInCtu, m_explicitRdpcmMode[comp],             iSizeInUchar );
993  }
994
995  memcpy( pCtu->getDepth()  + m_absZIdxInCtu, m_puhDepth,  iSizeInUchar );
996  memcpy( pCtu->getWidth()  + m_absZIdxInCtu, m_puhWidth,  iSizeInUchar );
997  memcpy( pCtu->getHeight() + m_absZIdxInCtu, m_puhHeight, iSizeInUchar );
998
999  for(UInt i=0; i<NUM_REF_PIC_LIST_01; i++)
1000  {
1001    const RefPicList rpl=RefPicList(i);
1002    memcpy( pCtu->getMVPIdx(rpl) + m_absZIdxInCtu, m_apiMVPIdx[rpl], iSizeInUchar );
1003    memcpy( pCtu->getMVPNum(rpl) + m_absZIdxInCtu, m_apiMVPNum[rpl], iSizeInUchar );
1004  }
1005
1006  for(UInt i=0; i<NUM_REF_PIC_LIST_01; i++)
1007  {
1008    const RefPicList rpl=RefPicList(i);
1009    m_acCUMvField[rpl].copyTo( pCtu->getCUMvField( rpl ), m_absZIdxInCtu );
1010  }
1011
1012  memcpy( pCtu->getIPCMFlag() + m_absZIdxInCtu, m_pbIPCMFlag,         iSizeInBool  );
1013
1014  const UInt numCoeffY    = (pCtu->getSlice()->getSPS()->getMaxCUWidth()*pCtu->getSlice()->getSPS()->getMaxCUHeight())>>(uhDepth<<1);
1015  const UInt offsetY      = m_absZIdxInCtu*m_pcPic->getMinCUWidth()*m_pcPic->getMinCUHeight();
1016  for (UInt comp=0; comp<numValidComp; comp++)
1017  {
1018    const ComponentID component = ComponentID(comp);
1019    const UInt componentShift   = m_pcPic->getComponentScaleX(component) + m_pcPic->getComponentScaleY(component);
1020    memcpy( pCtu->getCoeff(component)   + (offsetY>>componentShift), m_pcTrCoeff[component], sizeof(TCoeff)*(numCoeffY>>componentShift) );
1021#if ADAPTIVE_QP_SELECTION
1022    memcpy( pCtu->getArlCoeff(component) + (offsetY>>componentShift), m_pcArlCoeff[component], sizeof(TCoeff)*(numCoeffY>>componentShift) );
1023#endif
1024    memcpy( pCtu->getPCMSample(component) + (offsetY>>componentShift), m_pcIPCMSample[component], sizeof(Pel)*(numCoeffY>>componentShift) );
1025  }
1026
1027  pCtu->getTotalBins() = m_uiTotalBins;
1028}
1029
1030// --------------------------------------------------------------------------------------------------------------------
1031// Other public functions
1032// --------------------------------------------------------------------------------------------------------------------
1033
1034TComDataCU* TComDataCU::getPULeft( UInt& uiLPartUnitIdx,
1035                                   UInt uiCurrPartUnitIdx,
1036                                   Bool bEnforceSliceRestriction,
1037                                   Bool bEnforceTileRestriction )
1038{
1039  UInt uiAbsPartIdx       = g_auiZscanToRaster[uiCurrPartUnitIdx];
1040  UInt uiAbsZorderCUIdx   = g_auiZscanToRaster[m_absZIdxInCtu];
1041  const UInt numPartInCtuWidth = m_pcPic->getNumPartInCtuWidth();
1042
1043  if ( !RasterAddress::isZeroCol( uiAbsPartIdx, numPartInCtuWidth ) )
1044  {
1045    uiLPartUnitIdx = g_auiRasterToZscan[ uiAbsPartIdx - 1 ];
1046    if ( RasterAddress::isEqualCol( uiAbsPartIdx, uiAbsZorderCUIdx, numPartInCtuWidth ) )
1047    {
1048      return m_pcPic->getCtu( getCtuRsAddr() );
1049    }
1050    else
1051    {
1052      uiLPartUnitIdx -= m_absZIdxInCtu;
1053      return this;
1054    }
1055  }
1056
1057  uiLPartUnitIdx = g_auiRasterToZscan[ uiAbsPartIdx + numPartInCtuWidth - 1 ];
1058  if ( (bEnforceSliceRestriction && !CUIsFromSameSlice(m_pCtuLeft)) || (bEnforceTileRestriction && !CUIsFromSameTile(m_pCtuLeft)) )
1059  {
1060    return NULL;
1061  }
1062  return m_pCtuLeft;
1063}
1064
1065
1066TComDataCU* TComDataCU::getPUAbove( UInt& uiAPartUnitIdx,
1067                                    UInt uiCurrPartUnitIdx,
1068                                    Bool bEnforceSliceRestriction,
1069                                    Bool planarAtCtuBoundary,
1070                                    Bool bEnforceTileRestriction )
1071{
1072  UInt uiAbsPartIdx       = g_auiZscanToRaster[uiCurrPartUnitIdx];
1073  UInt uiAbsZorderCUIdx   = g_auiZscanToRaster[m_absZIdxInCtu];
1074  const UInt numPartInCtuWidth = m_pcPic->getNumPartInCtuWidth();
1075
1076  if ( !RasterAddress::isZeroRow( uiAbsPartIdx, numPartInCtuWidth ) )
1077  {
1078    uiAPartUnitIdx = g_auiRasterToZscan[ uiAbsPartIdx - numPartInCtuWidth ];
1079    if ( RasterAddress::isEqualRow( uiAbsPartIdx, uiAbsZorderCUIdx, numPartInCtuWidth ) )
1080    {
1081      return m_pcPic->getCtu( getCtuRsAddr() );
1082    }
1083    else
1084    {
1085      uiAPartUnitIdx -= m_absZIdxInCtu;
1086      return this;
1087    }
1088  }
1089
1090  if(planarAtCtuBoundary)
1091  {
1092    return NULL;
1093  }
1094
1095  uiAPartUnitIdx = g_auiRasterToZscan[ uiAbsPartIdx + m_pcPic->getNumPartitionsInCtu() - numPartInCtuWidth ];
1096
1097  if ( (bEnforceSliceRestriction && !CUIsFromSameSlice(m_pCtuAbove)) || (bEnforceTileRestriction && !CUIsFromSameTile(m_pCtuAbove)) )
1098  {
1099    return NULL;
1100  }
1101  return m_pCtuAbove;
1102}
1103
1104TComDataCU* TComDataCU::getPUAboveLeft( UInt& uiALPartUnitIdx, UInt uiCurrPartUnitIdx, Bool bEnforceSliceRestriction )
1105{
1106  UInt uiAbsPartIdx       = g_auiZscanToRaster[uiCurrPartUnitIdx];
1107  UInt uiAbsZorderCUIdx   = g_auiZscanToRaster[m_absZIdxInCtu];
1108  const UInt numPartInCtuWidth = m_pcPic->getNumPartInCtuWidth();
1109
1110  if ( !RasterAddress::isZeroCol( uiAbsPartIdx, numPartInCtuWidth ) )
1111  {
1112    if ( !RasterAddress::isZeroRow( uiAbsPartIdx, numPartInCtuWidth ) )
1113    {
1114      uiALPartUnitIdx = g_auiRasterToZscan[ uiAbsPartIdx - numPartInCtuWidth - 1 ];
1115      if ( RasterAddress::isEqualRowOrCol( uiAbsPartIdx, uiAbsZorderCUIdx, numPartInCtuWidth ) )
1116      {
1117        return m_pcPic->getCtu( getCtuRsAddr() );
1118      }
1119      else
1120      {
1121        uiALPartUnitIdx -= m_absZIdxInCtu;
1122        return this;
1123      }
1124    }
1125    uiALPartUnitIdx = g_auiRasterToZscan[ uiAbsPartIdx + getPic()->getNumPartitionsInCtu() - numPartInCtuWidth - 1 ];
1126    if ( bEnforceSliceRestriction && !CUIsFromSameSliceAndTile(m_pCtuAbove) )
1127    {
1128      return NULL;
1129    }
1130    return m_pCtuAbove;
1131  }
1132
1133  if ( !RasterAddress::isZeroRow( uiAbsPartIdx, numPartInCtuWidth ) )
1134  {
1135    uiALPartUnitIdx = g_auiRasterToZscan[ uiAbsPartIdx - 1 ];
1136    if ( bEnforceSliceRestriction && !CUIsFromSameSliceAndTile(m_pCtuLeft) )
1137    {
1138      return NULL;
1139    }
1140    return m_pCtuLeft;
1141  }
1142
1143  uiALPartUnitIdx = g_auiRasterToZscan[ m_pcPic->getNumPartitionsInCtu() - 1 ];
1144  if ( bEnforceSliceRestriction && !CUIsFromSameSliceAndTile(m_pCtuAboveLeft) )
1145  {
1146    return NULL;
1147  }
1148  return m_pCtuAboveLeft;
1149}
1150
1151TComDataCU* TComDataCU::getPUAboveRight( UInt& uiARPartUnitIdx, UInt uiCurrPartUnitIdx, Bool bEnforceSliceRestriction )
1152{
1153  UInt uiAbsPartIdxRT     = g_auiZscanToRaster[uiCurrPartUnitIdx];
1154  UInt uiAbsZorderCUIdx   = g_auiZscanToRaster[ m_absZIdxInCtu ] + m_puhWidth[0] / m_pcPic->getMinCUWidth() - 1;
1155  const UInt numPartInCtuWidth = m_pcPic->getNumPartInCtuWidth();
1156 
1157#if SVC_EXTENSION
1158  if( ( m_pcPic->getCtu(m_ctuRsAddr)->getCUPelX() + g_auiRasterToPelX[uiAbsPartIdxRT] + m_pcPic->getMinCUWidth() ) >= m_pcSlice->getPicWidthInLumaSamples() )
1159#else
1160  if( ( m_pcPic->getCtu(m_ctuRsAddr)->getCUPelX() + g_auiRasterToPelX[uiAbsPartIdxRT] + m_pcPic->getMinCUWidth() ) >= m_pcSlice->getSPS()->getPicWidthInLumaSamples() )
1161#endif
1162  {
1163    uiARPartUnitIdx = MAX_UINT;
1164    return NULL;
1165  }
1166
1167  if ( RasterAddress::lessThanCol( uiAbsPartIdxRT, numPartInCtuWidth - 1, numPartInCtuWidth ) )
1168  {
1169    if ( !RasterAddress::isZeroRow( uiAbsPartIdxRT, numPartInCtuWidth ) )
1170    {
1171      if ( uiCurrPartUnitIdx > g_auiRasterToZscan[ uiAbsPartIdxRT - numPartInCtuWidth + 1 ] )
1172      {
1173        uiARPartUnitIdx = g_auiRasterToZscan[ uiAbsPartIdxRT - numPartInCtuWidth + 1 ];
1174        if ( RasterAddress::isEqualRowOrCol( uiAbsPartIdxRT, uiAbsZorderCUIdx, numPartInCtuWidth ) )
1175        {
1176          return m_pcPic->getCtu( getCtuRsAddr() );
1177        }
1178        else
1179        {
1180          uiARPartUnitIdx -= m_absZIdxInCtu;
1181          return this;
1182        }
1183      }
1184      uiARPartUnitIdx = MAX_UINT;
1185      return NULL;
1186    }
1187    uiARPartUnitIdx = g_auiRasterToZscan[ uiAbsPartIdxRT + m_pcPic->getNumPartitionsInCtu() - numPartInCtuWidth + 1 ];
1188
1189    if ( bEnforceSliceRestriction && !CUIsFromSameSliceAndTile(m_pCtuAbove) )
1190    {
1191      return NULL;
1192    }
1193    return m_pCtuAbove;
1194  }
1195
1196  if ( !RasterAddress::isZeroRow( uiAbsPartIdxRT, numPartInCtuWidth ) )
1197  {
1198    uiARPartUnitIdx = MAX_UINT;
1199    return NULL;
1200  }
1201
1202  uiARPartUnitIdx = g_auiRasterToZscan[ m_pcPic->getNumPartitionsInCtu() - numPartInCtuWidth ];
1203
1204  if ( bEnforceSliceRestriction && !CUIsFromSameSliceAndTile(m_pCtuAboveRight) )
1205  {
1206    return NULL;
1207  }
1208  return m_pCtuAboveRight;
1209}
1210
1211TComDataCU* TComDataCU::getPUBelowLeft( UInt& uiBLPartUnitIdx, UInt uiCurrPartUnitIdx, Bool bEnforceSliceRestriction )
1212{
1213  UInt uiAbsPartIdxLB     = g_auiZscanToRaster[uiCurrPartUnitIdx];
1214  const UInt numPartInCtuWidth = m_pcPic->getNumPartInCtuWidth();
1215  UInt uiAbsZorderCUIdxLB = g_auiZscanToRaster[ m_absZIdxInCtu ] + (m_puhHeight[0] / m_pcPic->getMinCUHeight() - 1)*numPartInCtuWidth;
1216
1217#if SVC_EXTENSION
1218  if( ( m_pcPic->getCtu(m_ctuRsAddr)->getCUPelY() + g_auiRasterToPelY[uiAbsPartIdxLB] + m_pcPic->getMinCUHeight() ) >= m_pcSlice->getPicHeightInLumaSamples() )
1219#else
1220  if( ( m_pcPic->getCtu(m_ctuRsAddr)->getCUPelY() + g_auiRasterToPelY[uiAbsPartIdxLB] + m_pcPic->getMinCUHeight() ) >= m_pcSlice->getSPS()->getPicHeightInLumaSamples() )
1221#endif
1222  {
1223    uiBLPartUnitIdx = MAX_UINT;
1224    return NULL;
1225  }
1226
1227  if ( RasterAddress::lessThanRow( uiAbsPartIdxLB, m_pcPic->getNumPartInCtuHeight() - 1, numPartInCtuWidth ) )
1228  {
1229    if ( !RasterAddress::isZeroCol( uiAbsPartIdxLB, numPartInCtuWidth ) )
1230    {
1231      if ( uiCurrPartUnitIdx > g_auiRasterToZscan[ uiAbsPartIdxLB + numPartInCtuWidth - 1 ] )
1232      {
1233        uiBLPartUnitIdx = g_auiRasterToZscan[ uiAbsPartIdxLB + numPartInCtuWidth - 1 ];
1234        if ( RasterAddress::isEqualRowOrCol( uiAbsPartIdxLB, uiAbsZorderCUIdxLB, numPartInCtuWidth ) )
1235        {
1236          return m_pcPic->getCtu( getCtuRsAddr() );
1237        }
1238        else
1239        {
1240          uiBLPartUnitIdx -= m_absZIdxInCtu;
1241          return this;
1242        }
1243      }
1244      uiBLPartUnitIdx = MAX_UINT;
1245      return NULL;
1246    }
1247    uiBLPartUnitIdx = g_auiRasterToZscan[ uiAbsPartIdxLB + numPartInCtuWidth*2 - 1 ];
1248    if ( bEnforceSliceRestriction && !CUIsFromSameSliceAndTile(m_pCtuLeft) )
1249    {
1250      return NULL;
1251    }
1252    return m_pCtuLeft;
1253  }
1254
1255  uiBLPartUnitIdx = MAX_UINT;
1256  return NULL;
1257}
1258
1259TComDataCU* TComDataCU::getPUBelowLeftAdi(UInt& uiBLPartUnitIdx,  UInt uiCurrPartUnitIdx, UInt uiPartUnitOffset, Bool bEnforceSliceRestriction)
1260{
1261  UInt uiAbsPartIdxLB     = g_auiZscanToRaster[uiCurrPartUnitIdx];
1262  const UInt numPartInCtuWidth = m_pcPic->getNumPartInCtuWidth();
1263  UInt uiAbsZorderCUIdxLB = g_auiZscanToRaster[ m_absZIdxInCtu ] + ((m_puhHeight[0] / m_pcPic->getMinCUHeight()) - 1)*numPartInCtuWidth;
1264 
1265#if SVC_EXTENSION
1266  if( ( m_pcPic->getCtu(m_ctuRsAddr)->getCUPelY() + g_auiRasterToPelY[uiAbsPartIdxLB] + (m_pcPic->getPicSym()->getMinCUHeight() * uiPartUnitOffset)) >= m_pcSlice->getPicHeightInLumaSamples())
1267#else
1268  if( ( m_pcPic->getCtu(m_ctuRsAddr)->getCUPelY() + g_auiRasterToPelY[uiAbsPartIdxLB] + (m_pcPic->getPicSym()->getMinCUHeight() * uiPartUnitOffset)) >= m_pcSlice->getSPS()->getPicHeightInLumaSamples())
1269#endif
1270  {
1271    uiBLPartUnitIdx = MAX_UINT;
1272    return NULL;
1273  }
1274
1275  if ( RasterAddress::lessThanRow( uiAbsPartIdxLB, m_pcPic->getNumPartInCtuHeight() - uiPartUnitOffset, numPartInCtuWidth ) )
1276  {
1277    if ( !RasterAddress::isZeroCol( uiAbsPartIdxLB, numPartInCtuWidth ) )
1278    {
1279      if ( uiCurrPartUnitIdx > g_auiRasterToZscan[ uiAbsPartIdxLB + uiPartUnitOffset * numPartInCtuWidth - 1 ] )
1280      {
1281        uiBLPartUnitIdx = g_auiRasterToZscan[ uiAbsPartIdxLB + uiPartUnitOffset * numPartInCtuWidth - 1 ];
1282        if ( RasterAddress::isEqualRowOrCol( uiAbsPartIdxLB, uiAbsZorderCUIdxLB, numPartInCtuWidth ) )
1283        {
1284          return m_pcPic->getCtu( getCtuRsAddr() );
1285        }
1286        else
1287        {
1288          uiBLPartUnitIdx -= m_absZIdxInCtu;
1289          return this;
1290        }
1291      }
1292      uiBLPartUnitIdx = MAX_UINT;
1293      return NULL;
1294    }
1295    uiBLPartUnitIdx = g_auiRasterToZscan[ uiAbsPartIdxLB + (1+uiPartUnitOffset) * numPartInCtuWidth - 1 ];
1296    if ( bEnforceSliceRestriction && !CUIsFromSameSliceAndTile(m_pCtuLeft) )
1297    {
1298      return NULL;
1299    }
1300    return m_pCtuLeft;
1301  }
1302
1303  uiBLPartUnitIdx = MAX_UINT;
1304  return NULL;
1305}
1306
1307TComDataCU* TComDataCU::getPUAboveRightAdi(UInt&  uiARPartUnitIdx, UInt uiCurrPartUnitIdx, UInt uiPartUnitOffset, Bool bEnforceSliceRestriction)
1308{
1309  UInt uiAbsPartIdxRT     = g_auiZscanToRaster[uiCurrPartUnitIdx];
1310  UInt uiAbsZorderCUIdx   = g_auiZscanToRaster[ m_absZIdxInCtu ] + (m_puhWidth[0] / m_pcPic->getMinCUWidth()) - 1;
1311  const UInt numPartInCtuWidth = m_pcPic->getNumPartInCtuWidth();
1312 
1313#if SVC_EXTENSION
1314  if( ( m_pcPic->getCtu(m_ctuRsAddr)->getCUPelX() + g_auiRasterToPelX[uiAbsPartIdxRT] + (m_pcPic->getPicSym()->getMinCUHeight() * uiPartUnitOffset)) >= m_pcSlice->getPicWidthInLumaSamples() )
1315#else
1316  if( ( m_pcPic->getCtu(m_ctuRsAddr)->getCUPelX() + g_auiRasterToPelX[uiAbsPartIdxRT] + (m_pcPic->getPicSym()->getMinCUHeight() * uiPartUnitOffset)) >= m_pcSlice->getSPS()->getPicWidthInLumaSamples() )
1317#endif
1318  {
1319    uiARPartUnitIdx = MAX_UINT;
1320    return NULL;
1321  }
1322
1323  if ( RasterAddress::lessThanCol( uiAbsPartIdxRT, numPartInCtuWidth - uiPartUnitOffset, numPartInCtuWidth ) )
1324  {
1325    if ( !RasterAddress::isZeroRow( uiAbsPartIdxRT, numPartInCtuWidth ) )
1326    {
1327      if ( uiCurrPartUnitIdx > g_auiRasterToZscan[ uiAbsPartIdxRT - numPartInCtuWidth + uiPartUnitOffset ] )
1328      {
1329        uiARPartUnitIdx = g_auiRasterToZscan[ uiAbsPartIdxRT - numPartInCtuWidth + uiPartUnitOffset ];
1330        if ( RasterAddress::isEqualRowOrCol( uiAbsPartIdxRT, uiAbsZorderCUIdx, numPartInCtuWidth ) )
1331        {
1332          return m_pcPic->getCtu( getCtuRsAddr() );
1333        }
1334        else
1335        {
1336          uiARPartUnitIdx -= m_absZIdxInCtu;
1337          return this;
1338        }
1339      }
1340      uiARPartUnitIdx = MAX_UINT;
1341      return NULL;
1342    }
1343
1344    uiARPartUnitIdx = g_auiRasterToZscan[ uiAbsPartIdxRT + m_pcPic->getNumPartitionsInCtu() - numPartInCtuWidth + uiPartUnitOffset ];
1345    if ( bEnforceSliceRestriction && !CUIsFromSameSliceAndTile(m_pCtuAbove) )
1346    {
1347      return NULL;
1348    }
1349    return m_pCtuAbove;
1350  }
1351
1352  if ( !RasterAddress::isZeroRow( uiAbsPartIdxRT, numPartInCtuWidth ) )
1353  {
1354    uiARPartUnitIdx = MAX_UINT;
1355    return NULL;
1356  }
1357
1358  uiARPartUnitIdx = g_auiRasterToZscan[ m_pcPic->getNumPartitionsInCtu() - numPartInCtuWidth + uiPartUnitOffset-1 ];
1359  if ( bEnforceSliceRestriction && !CUIsFromSameSliceAndTile(m_pCtuAboveRight) )
1360  {
1361    return NULL;
1362  }
1363  return m_pCtuAboveRight;
1364}
1365
1366/** Get left QpMinCu
1367*\param   uiLPartUnitIdx
1368*\param   uiCurrAbsIdxInCtu
1369*\returns TComDataCU*   point of TComDataCU of left QpMinCu
1370*/
1371TComDataCU* TComDataCU::getQpMinCuLeft( UInt& uiLPartUnitIdx, UInt uiCurrAbsIdxInCtu )
1372{
1373  const UInt numPartInCtuWidth = m_pcPic->getNumPartInCtuWidth();
1374  const UInt maxCUDepth        = getSlice()->getSPS()->getMaxTotalCUDepth();
1375  const UInt maxCuDQPDepth     = getSlice()->getPPS()->getMaxCuDQPDepth();
1376  const UInt doubleDepthDifference = ((maxCUDepth - maxCuDQPDepth)<<1);
1377  UInt absZorderQpMinCUIdx = (uiCurrAbsIdxInCtu>>doubleDepthDifference)<<doubleDepthDifference;
1378  UInt absRorderQpMinCUIdx = g_auiZscanToRaster[absZorderQpMinCUIdx];
1379
1380  // check for left CTU boundary
1381  if ( RasterAddress::isZeroCol(absRorderQpMinCUIdx, numPartInCtuWidth) )
1382  {
1383    return NULL;
1384  }
1385
1386  // get index of left-CU relative to top-left corner of current quantization group
1387  uiLPartUnitIdx = g_auiRasterToZscan[absRorderQpMinCUIdx - 1];
1388
1389  // return pointer to current CTU
1390  return m_pcPic->getCtu( getCtuRsAddr() );
1391}
1392
1393/** Get Above QpMinCu
1394*\param   uiAPartUnitIdx
1395*\param   uiCurrAbsIdxInCtu
1396*\returns TComDataCU*   point of TComDataCU of above QpMinCu
1397*/
1398TComDataCU* TComDataCU::getQpMinCuAbove( UInt& uiAPartUnitIdx, UInt uiCurrAbsIdxInCtu )
1399{
1400  const UInt numPartInCtuWidth = m_pcPic->getNumPartInCtuWidth();
1401  const UInt maxCUDepth        = getSlice()->getSPS()->getMaxTotalCUDepth();
1402  const UInt maxCuDQPDepth     = getSlice()->getPPS()->getMaxCuDQPDepth();
1403  const UInt doubleDepthDifference = ((maxCUDepth - maxCuDQPDepth)<<1);
1404  UInt absZorderQpMinCUIdx = (uiCurrAbsIdxInCtu>>doubleDepthDifference)<<doubleDepthDifference;
1405  UInt absRorderQpMinCUIdx = g_auiZscanToRaster[absZorderQpMinCUIdx];
1406
1407  // check for top CTU boundary
1408  if ( RasterAddress::isZeroRow( absRorderQpMinCUIdx, numPartInCtuWidth) )
1409  {
1410    return NULL;
1411  }
1412
1413  // get index of top-CU relative to top-left corner of current quantization group
1414  uiAPartUnitIdx = g_auiRasterToZscan[absRorderQpMinCUIdx - numPartInCtuWidth];
1415
1416  // return pointer to current CTU
1417  return m_pcPic->getCtu( getCtuRsAddr() );
1418}
1419
1420
1421
1422/** Get reference QP from left QpMinCu or latest coded QP
1423*\param   uiCurrAbsIdxInCtu
1424*\returns Char   reference QP value
1425*/
1426Char TComDataCU::getRefQP( UInt uiCurrAbsIdxInCtu )
1427{
1428  UInt lPartIdx = MAX_UINT;
1429  UInt aPartIdx = MAX_UINT;
1430  TComDataCU* cULeft  = getQpMinCuLeft ( lPartIdx, m_absZIdxInCtu + uiCurrAbsIdxInCtu );
1431  TComDataCU* cUAbove = getQpMinCuAbove( aPartIdx, m_absZIdxInCtu + uiCurrAbsIdxInCtu );
1432  return (((cULeft? cULeft->getQP( lPartIdx ): getLastCodedQP( uiCurrAbsIdxInCtu )) + (cUAbove? cUAbove->getQP( aPartIdx ): getLastCodedQP( uiCurrAbsIdxInCtu )) + 1) >> 1);
1433}
1434
1435Int TComDataCU::getLastValidPartIdx( Int iAbsPartIdx )
1436{
1437  Int iLastValidPartIdx = iAbsPartIdx-1;
1438  while ( iLastValidPartIdx >= 0
1439       && getPredictionMode( iLastValidPartIdx ) == NUMBER_OF_PREDICTION_MODES )
1440  {
1441    UInt uiDepth = getDepth( iLastValidPartIdx );
1442    iLastValidPartIdx -= m_uiNumPartition>>(uiDepth<<1);
1443  }
1444  return iLastValidPartIdx;
1445}
1446
1447Char TComDataCU::getLastCodedQP( UInt uiAbsPartIdx )
1448{
1449  UInt uiQUPartIdxMask = ~((1<<((getSlice()->getSPS()->getMaxTotalCUDepth() - getSlice()->getPPS()->getMaxCuDQPDepth())<<1))-1);
1450  Int iLastValidPartIdx = getLastValidPartIdx( uiAbsPartIdx&uiQUPartIdxMask ); // A idx will be invalid if it is off the right or bottom edge of the picture.
1451  // If this CU is in the first CTU of the slice and there is no valid part before this one, use slice QP
1452  if ( getPic()->getPicSym()->getCtuTsToRsAddrMap(getSlice()->getSliceCurStartCtuTsAddr()) == getCtuRsAddr() && Int(getZorderIdxInCtu())+iLastValidPartIdx<0)
1453  {
1454    return getSlice()->getSliceQp();
1455  }
1456  else if ( iLastValidPartIdx >= 0 )
1457  {
1458    // If there is a valid part within the current Sub-CU, use it
1459    return getQP( iLastValidPartIdx );
1460  }
1461  else
1462  {
1463    if ( getZorderIdxInCtu() > 0 )
1464    {
1465      // If this wasn't the first sub-cu within the Ctu, explore the CTU itself.
1466      return getPic()->getCtu( getCtuRsAddr() )->getLastCodedQP( getZorderIdxInCtu() ); // TODO - remove this recursion
1467    }
1468    else if ( getPic()->getPicSym()->getCtuRsToTsAddrMap(getCtuRsAddr()) > 0
1469      && CUIsFromSameSliceTileAndWavefrontRow(getPic()->getCtu(getPic()->getPicSym()->getCtuTsToRsAddrMap(getPic()->getPicSym()->getCtuRsToTsAddrMap(getCtuRsAddr())-1))) )
1470    {
1471      // If this isn't the first Ctu (how can it be due to the first 'if'?), and the previous Ctu is from the same tile, examine the previous Ctu.
1472      return getPic()->getCtu( getPic()->getPicSym()->getCtuTsToRsAddrMap(getPic()->getPicSym()->getCtuRsToTsAddrMap(getCtuRsAddr())-1) )->getLastCodedQP( getPic()->getNumPartitionsInCtu() );  // TODO - remove this recursion
1473    }
1474    else
1475    {
1476      // No other options available - use the slice-level QP.
1477      return getSlice()->getSliceQp();
1478    }
1479  }
1480}
1481
1482
1483/** Check whether the CU is coded in lossless coding mode.
1484 * \param   absPartIdx
1485 * \returns true if the CU is coded in lossless coding mode; false if otherwise
1486 */
1487Bool TComDataCU::isLosslessCoded(UInt absPartIdx)
1488{
1489  return (getSlice()->getPPS()->getTransquantBypassEnableFlag() && getCUTransquantBypass (absPartIdx));
1490}
1491
1492
1493/** Get allowed chroma intra modes
1494*   - fills uiModeList with chroma intra modes
1495*
1496*\param   [in]  uiAbsPartIdx
1497*\param   [out] uiModeList pointer to chroma intra modes array
1498*/
1499Void TComDataCU::getAllowedChromaDir( UInt uiAbsPartIdx, UInt uiModeList[NUM_CHROMA_MODE] )
1500{
1501  uiModeList[0] = PLANAR_IDX;
1502  uiModeList[1] = VER_IDX;
1503  uiModeList[2] = HOR_IDX;
1504  uiModeList[3] = DC_IDX;
1505  uiModeList[4] = DM_CHROMA_IDX;
1506  assert(4<NUM_CHROMA_MODE);
1507
1508  UInt uiLumaMode = getIntraDir( CHANNEL_TYPE_LUMA, uiAbsPartIdx );
1509
1510  for( Int i = 0; i < NUM_CHROMA_MODE - 1; i++ )
1511  {
1512    if( uiLumaMode == uiModeList[i] )
1513    {
1514      uiModeList[i] = 34; // VER+8 mode
1515      break;
1516    }
1517  }
1518}
1519
1520/** Get most probable intra modes
1521*\param   uiAbsPartIdx    partition index
1522*\param   uiIntraDirPred  pointer to the array for MPM storage
1523*\param   compID          colour component ID
1524*\param   piMode          it is set with MPM mode in case both MPM are equal. It is used to restrict RD search at encode side.
1525*\returns Number of MPM
1526*/
1527Void TComDataCU::getIntraDirPredictor( UInt uiAbsPartIdx, Int uiIntraDirPred[NUM_MOST_PROBABLE_MODES], const ComponentID compID, Int* piMode  )
1528{
1529  TComDataCU* pcCULeft, *pcCUAbove;
1530  UInt        LeftPartIdx  = MAX_UINT;
1531  UInt        AbovePartIdx = MAX_UINT;
1532  Int         iLeftIntraDir, iAboveIntraDir;
1533  const TComSPS *sps=getSlice()->getSPS();
1534  const UInt partsPerMinCU = 1<<(2*(sps->getMaxTotalCUDepth() - sps->getLog2DiffMaxMinCodingBlockSize()));
1535
1536  const ChannelType chType = toChannelType(compID);
1537  const ChromaFormat chForm = getPic()->getChromaFormat();
1538  // Get intra direction of left PU
1539  pcCULeft = getPULeft( LeftPartIdx, m_absZIdxInCtu + uiAbsPartIdx );
1540
1541  if (isChroma(compID))
1542  {
1543    LeftPartIdx = getChromasCorrespondingPULumaIdx(LeftPartIdx, chForm, partsPerMinCU);
1544  }
1545  iLeftIntraDir  = pcCULeft ? ( pcCULeft->isIntra( LeftPartIdx ) ? pcCULeft->getIntraDir( chType, LeftPartIdx ) : DC_IDX ) : DC_IDX;
1546
1547  // Get intra direction of above PU
1548  pcCUAbove = getPUAbove( AbovePartIdx, m_absZIdxInCtu + uiAbsPartIdx, true, true );
1549
1550  if (isChroma(compID))
1551  {
1552    AbovePartIdx = getChromasCorrespondingPULumaIdx(AbovePartIdx, chForm, partsPerMinCU);
1553  }
1554  iAboveIntraDir = pcCUAbove ? ( pcCUAbove->isIntra( AbovePartIdx ) ? pcCUAbove->getIntraDir( chType, AbovePartIdx ) : DC_IDX ) : DC_IDX;
1555
1556  if (isChroma(chType))
1557  {
1558    if (iLeftIntraDir  == DM_CHROMA_IDX)
1559    {
1560      iLeftIntraDir  = pcCULeft-> getIntraDir( CHANNEL_TYPE_LUMA, LeftPartIdx  );
1561    }
1562    if (iAboveIntraDir == DM_CHROMA_IDX)
1563    {
1564      iAboveIntraDir = pcCUAbove->getIntraDir( CHANNEL_TYPE_LUMA, AbovePartIdx );
1565    }
1566  }
1567
1568  assert (2<NUM_MOST_PROBABLE_MODES);
1569  if(iLeftIntraDir == iAboveIntraDir)
1570  {
1571    if( piMode )
1572    {
1573      *piMode = 1;
1574    }
1575
1576    if (iLeftIntraDir > 1) // angular modes
1577    {
1578      uiIntraDirPred[0] = iLeftIntraDir;
1579      uiIntraDirPred[1] = ((iLeftIntraDir + 29) % 32) + 2;
1580      uiIntraDirPred[2] = ((iLeftIntraDir - 1 ) % 32) + 2;
1581    }
1582    else //non-angular
1583    {
1584      uiIntraDirPred[0] = PLANAR_IDX;
1585      uiIntraDirPred[1] = DC_IDX;
1586      uiIntraDirPred[2] = VER_IDX;
1587    }
1588  }
1589  else
1590  {
1591    if( piMode )
1592    {
1593      *piMode = 2;
1594    }
1595    uiIntraDirPred[0] = iLeftIntraDir;
1596    uiIntraDirPred[1] = iAboveIntraDir;
1597
1598    if (iLeftIntraDir && iAboveIntraDir ) //both modes are non-planar
1599    {
1600      uiIntraDirPred[2] = PLANAR_IDX;
1601    }
1602    else
1603    {
1604      uiIntraDirPred[2] =  (iLeftIntraDir+iAboveIntraDir)<2? VER_IDX : DC_IDX;
1605    }
1606  }
1607  for (UInt i=0; i<NUM_MOST_PROBABLE_MODES; i++)
1608  {
1609    assert(uiIntraDirPred[i] < 35);
1610  }
1611}
1612
1613UInt TComDataCU::getCtxSplitFlag( UInt uiAbsPartIdx, UInt uiDepth )
1614{
1615  TComDataCU* pcTempCU;
1616  UInt        uiTempPartIdx;
1617  UInt        uiCtx;
1618  // Get left split flag
1619  pcTempCU = getPULeft( uiTempPartIdx, m_absZIdxInCtu + uiAbsPartIdx );
1620  uiCtx  = ( pcTempCU ) ? ( ( pcTempCU->getDepth( uiTempPartIdx ) > uiDepth ) ? 1 : 0 ) : 0;
1621
1622  // Get above split flag
1623  pcTempCU = getPUAbove( uiTempPartIdx, m_absZIdxInCtu + uiAbsPartIdx );
1624  uiCtx += ( pcTempCU ) ? ( ( pcTempCU->getDepth( uiTempPartIdx ) > uiDepth ) ? 1 : 0 ) : 0;
1625
1626  return uiCtx;
1627}
1628
1629UInt TComDataCU::getCtxQtCbf( TComTU &rTu, const ChannelType chType )
1630{
1631  const UInt transformDepth = rTu.GetTransformDepthRel();
1632
1633  if (isChroma(chType))
1634  {
1635    return transformDepth;
1636  }
1637  else
1638  {
1639    const UInt uiCtx = ( transformDepth == 0 ? 1 : 0 );
1640    return uiCtx;
1641  }
1642}
1643
1644UInt TComDataCU::getQuadtreeTULog2MinSizeInCU( UInt absPartIdx )
1645{
1646  UInt log2CbSize = g_aucConvertToBit[getWidth( absPartIdx )] + 2;
1647  PartSize  partSize  = getPartitionSize( absPartIdx );
1648  UInt quadtreeTUMaxDepth = isIntra( absPartIdx ) ? m_pcSlice->getSPS()->getQuadtreeTUMaxDepthIntra() : m_pcSlice->getSPS()->getQuadtreeTUMaxDepthInter();
1649  Int intraSplitFlag = ( isIntra( absPartIdx ) && partSize == SIZE_NxN ) ? 1 : 0;
1650  Int interSplitFlag = ((quadtreeTUMaxDepth == 1) && isInter( absPartIdx ) && (partSize != SIZE_2Nx2N) );
1651
1652  UInt log2MinTUSizeInCU = 0;
1653  if (log2CbSize < (m_pcSlice->getSPS()->getQuadtreeTULog2MinSize() + quadtreeTUMaxDepth - 1 + interSplitFlag + intraSplitFlag) )
1654  {
1655    // when fully making use of signaled TUMaxDepth + inter/intraSplitFlag, resulting luma TB size is < QuadtreeTULog2MinSize
1656    log2MinTUSizeInCU = m_pcSlice->getSPS()->getQuadtreeTULog2MinSize();
1657  }
1658  else
1659  {
1660    // when fully making use of signaled TUMaxDepth + inter/intraSplitFlag, resulting luma TB size is still >= QuadtreeTULog2MinSize
1661    log2MinTUSizeInCU = log2CbSize - ( quadtreeTUMaxDepth - 1 + interSplitFlag + intraSplitFlag); // stop when trafoDepth == hierarchy_depth = splitFlag
1662    if ( log2MinTUSizeInCU > m_pcSlice->getSPS()->getQuadtreeTULog2MaxSize())
1663    {
1664      // when fully making use of signaled TUMaxDepth + inter/intraSplitFlag, resulting luma TB size is still > QuadtreeTULog2MaxSize
1665      log2MinTUSizeInCU = m_pcSlice->getSPS()->getQuadtreeTULog2MaxSize();
1666    }
1667  }
1668  return log2MinTUSizeInCU;
1669}
1670
1671UInt TComDataCU::getCtxSkipFlag( UInt uiAbsPartIdx )
1672{
1673  TComDataCU* pcTempCU;
1674  UInt        uiTempPartIdx;
1675  UInt        uiCtx = 0;
1676
1677  // Get BCBP of left PU
1678  pcTempCU = getPULeft( uiTempPartIdx, m_absZIdxInCtu + uiAbsPartIdx );
1679  uiCtx    = ( pcTempCU ) ? pcTempCU->isSkipped( uiTempPartIdx ) : 0;
1680
1681  // Get BCBP of above PU
1682  pcTempCU = getPUAbove( uiTempPartIdx, m_absZIdxInCtu + uiAbsPartIdx );
1683  uiCtx   += ( pcTempCU ) ? pcTempCU->isSkipped( uiTempPartIdx ) : 0;
1684
1685  return uiCtx;
1686}
1687
1688UInt TComDataCU::getCtxInterDir( UInt uiAbsPartIdx )
1689{
1690  return getDepth( uiAbsPartIdx );
1691}
1692
1693
1694UChar TComDataCU::getQtRootCbf( UInt uiIdx )
1695{
1696  const UInt numberValidComponents = getPic()->getNumberValidComponents();
1697  return getCbf( uiIdx, COMPONENT_Y, 0 )
1698          || ((numberValidComponents > COMPONENT_Cb) && getCbf( uiIdx, COMPONENT_Cb, 0 ))
1699          || ((numberValidComponents > COMPONENT_Cr) && getCbf( uiIdx, COMPONENT_Cr, 0 ));
1700}
1701
1702Void TComDataCU::setCbfSubParts( const UInt uiCbf[MAX_NUM_COMPONENT], UInt uiAbsPartIdx, UInt uiDepth )
1703{
1704  UInt uiCurrPartNumb = m_pcPic->getNumPartitionsInCtu() >> (uiDepth << 1);
1705  for(UInt comp=0; comp<MAX_NUM_COMPONENT; comp++)
1706  {
1707    memset( m_puhCbf[comp] + uiAbsPartIdx, uiCbf[comp], sizeof( UChar ) * uiCurrPartNumb );
1708  }
1709}
1710
1711Void TComDataCU::setCbfSubParts( UInt uiCbf, ComponentID compID, UInt uiAbsPartIdx, UInt uiDepth )
1712{
1713  UInt uiCurrPartNumb = m_pcPic->getNumPartitionsInCtu() >> (uiDepth << 1);
1714  memset( m_puhCbf[compID] + uiAbsPartIdx, uiCbf, sizeof( UChar ) * uiCurrPartNumb );
1715}
1716
1717/** Sets a coded block flag for all sub-partitions of a partition
1718 * \param uiCbf          The value of the coded block flag to be set
1719 * \param compID
1720 * \param uiAbsPartIdx
1721 * \param uiPartIdx
1722 * \param uiDepth
1723 */
1724Void TComDataCU::setCbfSubParts ( UInt uiCbf, ComponentID compID, UInt uiAbsPartIdx, UInt uiPartIdx, UInt uiDepth )
1725{
1726  setSubPart<UChar>( uiCbf, m_puhCbf[compID], uiAbsPartIdx, uiDepth, uiPartIdx );
1727}
1728
1729Void TComDataCU::setCbfPartRange ( UInt uiCbf, ComponentID compID, UInt uiAbsPartIdx, UInt uiCoveredPartIdxes )
1730{
1731  memset((m_puhCbf[compID] + uiAbsPartIdx), uiCbf, (sizeof(UChar) * uiCoveredPartIdxes));
1732}
1733
1734Void TComDataCU::bitwiseOrCbfPartRange( UInt uiCbf, ComponentID compID, UInt uiAbsPartIdx, UInt uiCoveredPartIdxes )
1735{
1736  const UInt stopAbsPartIdx = uiAbsPartIdx + uiCoveredPartIdxes;
1737
1738  for (UInt subPartIdx = uiAbsPartIdx; subPartIdx < stopAbsPartIdx; subPartIdx++)
1739  {
1740    m_puhCbf[compID][subPartIdx] |= uiCbf;
1741  }
1742}
1743
1744Void TComDataCU::setDepthSubParts( UInt uiDepth, UInt uiAbsPartIdx )
1745{
1746  UInt uiCurrPartNumb = m_pcPic->getNumPartitionsInCtu() >> (uiDepth << 1);
1747  memset( m_puhDepth + uiAbsPartIdx, uiDepth, sizeof(UChar)*uiCurrPartNumb );
1748}
1749
1750Bool TComDataCU::isFirstAbsZorderIdxInDepth (UInt uiAbsPartIdx, UInt uiDepth)
1751{
1752  UInt uiPartNumb = m_pcPic->getNumPartitionsInCtu() >> (uiDepth << 1);
1753  return (((m_absZIdxInCtu + uiAbsPartIdx)% uiPartNumb) == 0);
1754}
1755
1756Void TComDataCU::setPartSizeSubParts( PartSize eMode, UInt uiAbsPartIdx, UInt uiDepth )
1757{
1758  assert( sizeof( *m_pePartSize) == 1 );
1759  memset( m_pePartSize + uiAbsPartIdx, eMode, m_pcPic->getNumPartitionsInCtu() >> ( 2 * uiDepth ) );
1760}
1761
1762Void TComDataCU::setCUTransquantBypassSubParts( Bool flag, UInt uiAbsPartIdx, UInt uiDepth )
1763{
1764  memset( m_CUTransquantBypass + uiAbsPartIdx, flag, m_pcPic->getNumPartitionsInCtu() >> ( 2 * uiDepth ) );
1765}
1766
1767Void TComDataCU::setSkipFlagSubParts( Bool skip, UInt absPartIdx, UInt depth )
1768{
1769  assert( sizeof( *m_skipFlag) == 1 );
1770  memset( m_skipFlag + absPartIdx, skip, m_pcPic->getNumPartitionsInCtu() >> ( 2 * depth ) );
1771}
1772
1773Void TComDataCU::setPredModeSubParts( PredMode eMode, UInt uiAbsPartIdx, UInt uiDepth )
1774{
1775  assert( sizeof( *m_pePredMode) == 1 );
1776  memset( m_pePredMode + uiAbsPartIdx, eMode, m_pcPic->getNumPartitionsInCtu() >> ( 2 * uiDepth ) );
1777}
1778
1779Void TComDataCU::setChromaQpAdjSubParts( UChar val, Int absPartIdx, Int depth )
1780{
1781  assert( sizeof(*m_ChromaQpAdj) == 1 );
1782  memset( m_ChromaQpAdj + absPartIdx, val, m_pcPic->getNumPartitionsInCtu() >> ( 2 * depth ) );
1783}
1784
1785Void TComDataCU::setQPSubCUs( Int qp, UInt absPartIdx, UInt depth, Bool &foundNonZeroCbf )
1786{
1787  UInt currPartNumb = m_pcPic->getNumPartitionsInCtu() >> (depth << 1);
1788  UInt currPartNumQ = currPartNumb >> 2;
1789  const UInt numValidComp = m_pcPic->getNumberValidComponents();
1790
1791  if(!foundNonZeroCbf)
1792  {
1793    if(getDepth(absPartIdx) > depth)
1794    {
1795      for ( UInt partUnitIdx = 0; partUnitIdx < 4; partUnitIdx++ )
1796      {
1797        setQPSubCUs( qp, absPartIdx+partUnitIdx*currPartNumQ, depth+1, foundNonZeroCbf );
1798      }
1799    }
1800    else
1801    {
1802      if(getCbf( absPartIdx, COMPONENT_Y ) || (numValidComp>COMPONENT_Cb && getCbf( absPartIdx, COMPONENT_Cb )) || (numValidComp>COMPONENT_Cr && getCbf( absPartIdx, COMPONENT_Cr) ) )
1803      {
1804        foundNonZeroCbf = true;
1805      }
1806      else
1807      {
1808        setQPSubParts(qp, absPartIdx, depth);
1809      }
1810    }
1811  }
1812}
1813
1814Void TComDataCU::setQPSubParts( Int qp, UInt uiAbsPartIdx, UInt uiDepth )
1815{
1816  const UInt numPart = m_pcPic->getNumPartitionsInCtu() >> (uiDepth << 1);
1817  memset(m_phQP+uiAbsPartIdx, qp, numPart);
1818}
1819
1820Void TComDataCU::setIntraDirSubParts( const ChannelType channelType, const UInt dir, const UInt absPartIdx, const UInt depth )
1821{
1822  UInt numPart = m_pcPic->getNumPartitionsInCtu() >> (depth << 1);
1823  memset( m_puhIntraDir[channelType] + absPartIdx, dir,sizeof(UChar)*numPart );
1824}
1825
1826template<typename T>
1827Void TComDataCU::setSubPart( T uiParameter, T* puhBaseCtu, UInt uiCUAddr, UInt uiCUDepth, UInt uiPUIdx )
1828{
1829  assert( sizeof(T) == 1 ); // Using memset() works only for types of size 1
1830
1831  UInt uiCurrPartNumQ = (m_pcPic->getNumPartitionsInCtu() >> (2 * uiCUDepth)) >> 2;
1832  switch ( m_pePartSize[ uiCUAddr ] )
1833  {
1834    case SIZE_2Nx2N:
1835      memset( puhBaseCtu + uiCUAddr, uiParameter, 4 * uiCurrPartNumQ );
1836      break;
1837    case SIZE_2NxN:
1838      memset( puhBaseCtu + uiCUAddr, uiParameter, 2 * uiCurrPartNumQ );
1839      break;
1840    case SIZE_Nx2N:
1841      memset( puhBaseCtu + uiCUAddr, uiParameter, uiCurrPartNumQ );
1842      memset( puhBaseCtu + uiCUAddr + 2 * uiCurrPartNumQ, uiParameter, uiCurrPartNumQ );
1843      break;
1844    case SIZE_NxN:
1845      memset( puhBaseCtu + uiCUAddr, uiParameter, uiCurrPartNumQ );
1846      break;
1847    case SIZE_2NxnU:
1848      if ( uiPUIdx == 0 )
1849      {
1850        memset( puhBaseCtu + uiCUAddr, uiParameter, (uiCurrPartNumQ >> 1) );
1851        memset( puhBaseCtu + uiCUAddr + uiCurrPartNumQ, uiParameter, (uiCurrPartNumQ >> 1) );
1852      }
1853      else if ( uiPUIdx == 1 )
1854      {
1855        memset( puhBaseCtu + uiCUAddr, uiParameter, (uiCurrPartNumQ >> 1) );
1856        memset( puhBaseCtu + uiCUAddr + uiCurrPartNumQ, uiParameter, ((uiCurrPartNumQ >> 1) + (uiCurrPartNumQ << 1)) );
1857      }
1858      else
1859      {
1860        assert(0);
1861      }
1862      break;
1863    case SIZE_2NxnD:
1864      if ( uiPUIdx == 0 )
1865      {
1866        memset( puhBaseCtu + uiCUAddr, uiParameter, ((uiCurrPartNumQ << 1) + (uiCurrPartNumQ >> 1)) );
1867        memset( puhBaseCtu + uiCUAddr + (uiCurrPartNumQ << 1) + uiCurrPartNumQ, uiParameter, (uiCurrPartNumQ >> 1) );
1868      }
1869      else if ( uiPUIdx == 1 )
1870      {
1871        memset( puhBaseCtu + uiCUAddr, uiParameter, (uiCurrPartNumQ >> 1) );
1872        memset( puhBaseCtu + uiCUAddr + uiCurrPartNumQ, uiParameter, (uiCurrPartNumQ >> 1) );
1873      }
1874      else
1875      {
1876        assert(0);
1877      }
1878      break;
1879    case SIZE_nLx2N:
1880      if ( uiPUIdx == 0 )
1881      {
1882        memset( puhBaseCtu + uiCUAddr, uiParameter, (uiCurrPartNumQ >> 2) );
1883        memset( puhBaseCtu + uiCUAddr + (uiCurrPartNumQ >> 1), uiParameter, (uiCurrPartNumQ >> 2) );
1884        memset( puhBaseCtu + uiCUAddr + (uiCurrPartNumQ << 1), uiParameter, (uiCurrPartNumQ >> 2) );
1885        memset( puhBaseCtu + uiCUAddr + (uiCurrPartNumQ << 1) + (uiCurrPartNumQ >> 1), uiParameter, (uiCurrPartNumQ >> 2) );
1886      }
1887      else if ( uiPUIdx == 1 )
1888      {
1889        memset( puhBaseCtu + uiCUAddr, uiParameter, (uiCurrPartNumQ >> 2) );
1890        memset( puhBaseCtu + uiCUAddr + (uiCurrPartNumQ >> 1), uiParameter, (uiCurrPartNumQ + (uiCurrPartNumQ >> 2)) );
1891        memset( puhBaseCtu + uiCUAddr + (uiCurrPartNumQ << 1), uiParameter, (uiCurrPartNumQ >> 2) );
1892        memset( puhBaseCtu + uiCUAddr + (uiCurrPartNumQ << 1) + (uiCurrPartNumQ >> 1), uiParameter, (uiCurrPartNumQ + (uiCurrPartNumQ >> 2)) );
1893      }
1894      else
1895      {
1896        assert(0);
1897      }
1898      break;
1899    case SIZE_nRx2N:
1900      if ( uiPUIdx == 0 )
1901      {
1902        memset( puhBaseCtu + uiCUAddr, uiParameter, (uiCurrPartNumQ + (uiCurrPartNumQ >> 2)) );
1903        memset( puhBaseCtu + uiCUAddr + uiCurrPartNumQ + (uiCurrPartNumQ >> 1), uiParameter, (uiCurrPartNumQ >> 2) );
1904        memset( puhBaseCtu + uiCUAddr + (uiCurrPartNumQ << 1), uiParameter, (uiCurrPartNumQ + (uiCurrPartNumQ >> 2)) );
1905        memset( puhBaseCtu + uiCUAddr + (uiCurrPartNumQ << 1) + uiCurrPartNumQ + (uiCurrPartNumQ >> 1), uiParameter, (uiCurrPartNumQ >> 2) );
1906      }
1907      else if ( uiPUIdx == 1 )
1908      {
1909        memset( puhBaseCtu + uiCUAddr, uiParameter, (uiCurrPartNumQ >> 2) );
1910        memset( puhBaseCtu + uiCUAddr + (uiCurrPartNumQ >> 1), uiParameter, (uiCurrPartNumQ >> 2) );
1911        memset( puhBaseCtu + uiCUAddr + (uiCurrPartNumQ << 1), uiParameter, (uiCurrPartNumQ >> 2) );
1912        memset( puhBaseCtu + uiCUAddr + (uiCurrPartNumQ << 1) + (uiCurrPartNumQ >> 1), uiParameter, (uiCurrPartNumQ >> 2) );
1913      }
1914      else
1915      {
1916        assert(0);
1917      }
1918      break;
1919    default:
1920      assert( 0 );
1921      break;
1922  }
1923}
1924
1925Void TComDataCU::setMergeFlagSubParts ( Bool bMergeFlag, UInt uiAbsPartIdx, UInt uiPartIdx, UInt uiDepth )
1926{
1927  setSubPart( bMergeFlag, m_pbMergeFlag, uiAbsPartIdx, uiDepth, uiPartIdx );
1928}
1929
1930Void TComDataCU::setMergeIndexSubParts ( UInt uiMergeIndex, UInt uiAbsPartIdx, UInt uiPartIdx, UInt uiDepth )
1931{
1932  setSubPart<UChar>( uiMergeIndex, m_puhMergeIndex, uiAbsPartIdx, uiDepth, uiPartIdx );
1933}
1934
1935Void TComDataCU::setInterDirSubParts( UInt uiDir, UInt uiAbsPartIdx, UInt uiPartIdx, UInt uiDepth )
1936{
1937  setSubPart<UChar>( uiDir, m_puhInterDir, uiAbsPartIdx, uiDepth, uiPartIdx );
1938}
1939
1940Void TComDataCU::setMVPIdxSubParts( Int iMVPIdx, RefPicList eRefPicList, UInt uiAbsPartIdx, UInt uiPartIdx, UInt uiDepth )
1941{
1942  setSubPart<Char>( iMVPIdx, m_apiMVPIdx[eRefPicList], uiAbsPartIdx, uiDepth, uiPartIdx );
1943}
1944
1945Void TComDataCU::setMVPNumSubParts( Int iMVPNum, RefPicList eRefPicList, UInt uiAbsPartIdx, UInt uiPartIdx, UInt uiDepth )
1946{
1947  setSubPart<Char>( iMVPNum, m_apiMVPNum[eRefPicList], uiAbsPartIdx, uiDepth, uiPartIdx );
1948}
1949
1950
1951Void TComDataCU::setTrIdxSubParts( UInt uiTrIdx, UInt uiAbsPartIdx, UInt uiDepth )
1952{
1953  UInt uiCurrPartNumb = m_pcPic->getNumPartitionsInCtu() >> (uiDepth << 1);
1954
1955  memset( m_puhTrIdx + uiAbsPartIdx, uiTrIdx, sizeof(UChar)*uiCurrPartNumb );
1956}
1957
1958Void TComDataCU::setTransformSkipSubParts( const UInt useTransformSkip[MAX_NUM_COMPONENT], UInt uiAbsPartIdx, UInt uiDepth )
1959{
1960  UInt uiCurrPartNumb = m_pcPic->getNumPartitionsInCtu() >> (uiDepth << 1);
1961
1962  for(UInt i=0; i<MAX_NUM_COMPONENT; i++)
1963  {
1964    memset( m_puhTransformSkip[i] + uiAbsPartIdx, useTransformSkip[i], sizeof( UChar ) * uiCurrPartNumb );
1965  }
1966}
1967
1968Void TComDataCU::setTransformSkipSubParts( UInt useTransformSkip, ComponentID compID, UInt uiAbsPartIdx, UInt uiDepth)
1969{
1970  UInt uiCurrPartNumb = m_pcPic->getNumPartitionsInCtu() >> (uiDepth << 1);
1971
1972  memset( m_puhTransformSkip[compID] + uiAbsPartIdx, useTransformSkip, sizeof( UChar ) * uiCurrPartNumb );
1973}
1974
1975Void TComDataCU::setTransformSkipPartRange ( UInt useTransformSkip, ComponentID compID, UInt uiAbsPartIdx, UInt uiCoveredPartIdxes )
1976{
1977  memset((m_puhTransformSkip[compID] + uiAbsPartIdx), useTransformSkip, (sizeof(UChar) * uiCoveredPartIdxes));
1978}
1979
1980Void TComDataCU::setCrossComponentPredictionAlphaPartRange( Char alphaValue, ComponentID compID, UInt uiAbsPartIdx, UInt uiCoveredPartIdxes )
1981{
1982  memset((m_crossComponentPredictionAlpha[compID] + uiAbsPartIdx), alphaValue, (sizeof(Char) * uiCoveredPartIdxes));
1983}
1984
1985Void TComDataCU::setExplicitRdpcmModePartRange ( UInt rdpcmMode, ComponentID compID, UInt uiAbsPartIdx, UInt uiCoveredPartIdxes )
1986{
1987  memset((m_explicitRdpcmMode[compID] + uiAbsPartIdx), rdpcmMode, (sizeof(UChar) * uiCoveredPartIdxes));
1988}
1989
1990Void TComDataCU::setSizeSubParts( UInt uiWidth, UInt uiHeight, UInt uiAbsPartIdx, UInt uiDepth )
1991{
1992  UInt uiCurrPartNumb = m_pcPic->getNumPartitionsInCtu() >> (uiDepth << 1);
1993
1994  memset( m_puhWidth  + uiAbsPartIdx, uiWidth,  sizeof(UChar)*uiCurrPartNumb );
1995  memset( m_puhHeight + uiAbsPartIdx, uiHeight, sizeof(UChar)*uiCurrPartNumb );
1996}
1997
1998UChar TComDataCU::getNumPartitions(const UInt uiAbsPartIdx)
1999{
2000  UChar iNumPart = 0;
2001
2002  switch ( m_pePartSize[uiAbsPartIdx] )
2003  {
2004    case SIZE_2Nx2N:    iNumPart = 1; break;
2005    case SIZE_2NxN:     iNumPart = 2; break;
2006    case SIZE_Nx2N:     iNumPart = 2; break;
2007    case SIZE_NxN:      iNumPart = 4; break;
2008    case SIZE_2NxnU:    iNumPart = 2; break;
2009    case SIZE_2NxnD:    iNumPart = 2; break;
2010    case SIZE_nLx2N:    iNumPart = 2; break;
2011    case SIZE_nRx2N:    iNumPart = 2; break;
2012    default:            assert (0);   break;
2013  }
2014
2015  return  iNumPart;
2016}
2017
2018// This is for use by a leaf/sub CU object only, with no additional AbsPartIdx
2019Void TComDataCU::getPartIndexAndSize( UInt uiPartIdx, UInt& ruiPartAddr, Int& riWidth, Int& riHeight )
2020{
2021  switch ( m_pePartSize[0] )
2022  {
2023    case SIZE_2NxN:
2024      riWidth = getWidth(0);      riHeight = getHeight(0) >> 1; ruiPartAddr = ( uiPartIdx == 0 )? 0 : m_uiNumPartition >> 1;
2025      break;
2026    case SIZE_Nx2N:
2027      riWidth = getWidth(0) >> 1; riHeight = getHeight(0);      ruiPartAddr = ( uiPartIdx == 0 )? 0 : m_uiNumPartition >> 2;
2028      break;
2029    case SIZE_NxN:
2030      riWidth = getWidth(0) >> 1; riHeight = getHeight(0) >> 1; ruiPartAddr = ( m_uiNumPartition >> 2 ) * uiPartIdx;
2031      break;
2032    case SIZE_2NxnU:
2033      riWidth     = getWidth(0);
2034      riHeight    = ( uiPartIdx == 0 ) ?  getHeight(0) >> 2 : ( getHeight(0) >> 2 ) + ( getHeight(0) >> 1 );
2035      ruiPartAddr = ( uiPartIdx == 0 ) ? 0 : m_uiNumPartition >> 3;
2036      break;
2037    case SIZE_2NxnD:
2038      riWidth     = getWidth(0);
2039      riHeight    = ( uiPartIdx == 0 ) ?  ( getHeight(0) >> 2 ) + ( getHeight(0) >> 1 ) : getHeight(0) >> 2;
2040      ruiPartAddr = ( uiPartIdx == 0 ) ? 0 : (m_uiNumPartition >> 1) + (m_uiNumPartition >> 3);
2041      break;
2042    case SIZE_nLx2N:
2043      riWidth     = ( uiPartIdx == 0 ) ? getWidth(0) >> 2 : ( getWidth(0) >> 2 ) + ( getWidth(0) >> 1 );
2044      riHeight    = getHeight(0);
2045      ruiPartAddr = ( uiPartIdx == 0 ) ? 0 : m_uiNumPartition >> 4;
2046      break;
2047    case SIZE_nRx2N:
2048      riWidth     = ( uiPartIdx == 0 ) ? ( getWidth(0) >> 2 ) + ( getWidth(0) >> 1 ) : getWidth(0) >> 2;
2049      riHeight    = getHeight(0);
2050      ruiPartAddr = ( uiPartIdx == 0 ) ? 0 : (m_uiNumPartition >> 2) + (m_uiNumPartition >> 4);
2051      break;
2052    default:
2053      assert ( m_pePartSize[0] == SIZE_2Nx2N );
2054      riWidth = getWidth(0);      riHeight = getHeight(0);      ruiPartAddr = 0;
2055      break;
2056  }
2057}
2058
2059
2060Void TComDataCU::getMvField ( TComDataCU* pcCU, UInt uiAbsPartIdx, RefPicList eRefPicList, TComMvField& rcMvField )
2061{
2062  if ( pcCU == NULL )  // OUT OF BOUNDARY
2063  {
2064    TComMv  cZeroMv;
2065    rcMvField.setMvField( cZeroMv, NOT_VALID );
2066    return;
2067  }
2068
2069  TComCUMvField*  pcCUMvField = pcCU->getCUMvField( eRefPicList );
2070  rcMvField.setMvField( pcCUMvField->getMv( uiAbsPartIdx ), pcCUMvField->getRefIdx( uiAbsPartIdx ) );
2071}
2072
2073Void TComDataCU::deriveLeftRightTopIdxGeneral ( UInt uiAbsPartIdx, UInt uiPartIdx, UInt& ruiPartIdxLT, UInt& ruiPartIdxRT )
2074{
2075  ruiPartIdxLT = m_absZIdxInCtu + uiAbsPartIdx;
2076  UInt uiPUWidth = 0;
2077
2078  switch ( m_pePartSize[uiAbsPartIdx] )
2079  {
2080    case SIZE_2Nx2N: uiPUWidth = m_puhWidth[uiAbsPartIdx];  break;
2081    case SIZE_2NxN:  uiPUWidth = m_puhWidth[uiAbsPartIdx];   break;
2082    case SIZE_Nx2N:  uiPUWidth = m_puhWidth[uiAbsPartIdx]  >> 1;  break;
2083    case SIZE_NxN:   uiPUWidth = m_puhWidth[uiAbsPartIdx]  >> 1; break;
2084    case SIZE_2NxnU:   uiPUWidth = m_puhWidth[uiAbsPartIdx]; break;
2085    case SIZE_2NxnD:   uiPUWidth = m_puhWidth[uiAbsPartIdx]; break;
2086    case SIZE_nLx2N:
2087      if ( uiPartIdx == 0 )
2088      {
2089        uiPUWidth = m_puhWidth[uiAbsPartIdx]  >> 2;
2090      }
2091      else if ( uiPartIdx == 1 )
2092      {
2093        uiPUWidth = (m_puhWidth[uiAbsPartIdx]  >> 1) + (m_puhWidth[uiAbsPartIdx]  >> 2);
2094      }
2095      else
2096      {
2097        assert(0);
2098      }
2099      break;
2100    case SIZE_nRx2N:
2101      if ( uiPartIdx == 0 )
2102      {
2103        uiPUWidth = (m_puhWidth[uiAbsPartIdx]  >> 1) + (m_puhWidth[uiAbsPartIdx]  >> 2);
2104      }
2105      else if ( uiPartIdx == 1 )
2106      {
2107        uiPUWidth = m_puhWidth[uiAbsPartIdx]  >> 2;
2108      }
2109      else
2110      {
2111        assert(0);
2112      }
2113      break;
2114    default:
2115      assert (0);
2116      break;
2117  }
2118
2119  ruiPartIdxRT = g_auiRasterToZscan [g_auiZscanToRaster[ ruiPartIdxLT ] + uiPUWidth / m_pcPic->getMinCUWidth() - 1 ];
2120}
2121
2122Void TComDataCU::deriveLeftBottomIdxGeneral( UInt uiAbsPartIdx, UInt uiPartIdx, UInt& ruiPartIdxLB )
2123{
2124  UInt uiPUHeight = 0;
2125  switch ( m_pePartSize[uiAbsPartIdx] )
2126  {
2127    case SIZE_2Nx2N: uiPUHeight = m_puhHeight[uiAbsPartIdx];    break;
2128    case SIZE_2NxN:  uiPUHeight = m_puhHeight[uiAbsPartIdx] >> 1;    break;
2129    case SIZE_Nx2N:  uiPUHeight = m_puhHeight[uiAbsPartIdx];  break;
2130    case SIZE_NxN:   uiPUHeight = m_puhHeight[uiAbsPartIdx] >> 1;    break;
2131    case SIZE_2NxnU:
2132      if ( uiPartIdx == 0 )
2133      {
2134        uiPUHeight = m_puhHeight[uiAbsPartIdx] >> 2;
2135      }
2136      else if ( uiPartIdx == 1 )
2137      {
2138        uiPUHeight = (m_puhHeight[uiAbsPartIdx] >> 1) + (m_puhHeight[uiAbsPartIdx] >> 2);
2139      }
2140      else
2141      {
2142        assert(0);
2143      }
2144      break;
2145    case SIZE_2NxnD:
2146      if ( uiPartIdx == 0 )
2147      {
2148        uiPUHeight = (m_puhHeight[uiAbsPartIdx] >> 1) + (m_puhHeight[uiAbsPartIdx] >> 2);
2149      }
2150      else if ( uiPartIdx == 1 )
2151      {
2152        uiPUHeight = m_puhHeight[uiAbsPartIdx] >> 2;
2153      }
2154      else
2155      {
2156        assert(0);
2157      }
2158      break;
2159    case SIZE_nLx2N: uiPUHeight = m_puhHeight[uiAbsPartIdx];  break;
2160    case SIZE_nRx2N: uiPUHeight = m_puhHeight[uiAbsPartIdx];  break;
2161    default:
2162      assert (0);
2163      break;
2164  }
2165
2166  ruiPartIdxLB      = g_auiRasterToZscan [g_auiZscanToRaster[ m_absZIdxInCtu + uiAbsPartIdx ] + ((uiPUHeight / m_pcPic->getMinCUHeight()) - 1)*m_pcPic->getNumPartInCtuWidth()];
2167}
2168
2169Void TComDataCU::deriveLeftRightTopIdx ( UInt uiPartIdx, UInt& ruiPartIdxLT, UInt& ruiPartIdxRT )
2170{
2171  ruiPartIdxLT = m_absZIdxInCtu;
2172  ruiPartIdxRT = g_auiRasterToZscan [g_auiZscanToRaster[ ruiPartIdxLT ] + m_puhWidth[0] / m_pcPic->getMinCUWidth() - 1 ];
2173
2174  switch ( m_pePartSize[0] )
2175  {
2176    case SIZE_2Nx2N:                                                                                                                                break;
2177    case SIZE_2NxN:
2178      ruiPartIdxLT += ( uiPartIdx == 0 )? 0 : m_uiNumPartition >> 1; ruiPartIdxRT += ( uiPartIdx == 0 )? 0 : m_uiNumPartition >> 1;
2179      break;
2180    case SIZE_Nx2N:
2181      ruiPartIdxLT += ( uiPartIdx == 0 )? 0 : m_uiNumPartition >> 2; ruiPartIdxRT -= ( uiPartIdx == 1 )? 0 : m_uiNumPartition >> 2;
2182      break;
2183    case SIZE_NxN:
2184      ruiPartIdxLT += ( m_uiNumPartition >> 2 ) * uiPartIdx;         ruiPartIdxRT +=  ( m_uiNumPartition >> 2 ) * ( uiPartIdx - 1 );
2185      break;
2186    case SIZE_2NxnU:
2187      ruiPartIdxLT += ( uiPartIdx == 0 )? 0 : m_uiNumPartition >> 3;
2188      ruiPartIdxRT += ( uiPartIdx == 0 )? 0 : m_uiNumPartition >> 3;
2189      break;
2190    case SIZE_2NxnD:
2191      ruiPartIdxLT += ( uiPartIdx == 0 )? 0 : ( m_uiNumPartition >> 1 ) + ( m_uiNumPartition >> 3 );
2192      ruiPartIdxRT += ( uiPartIdx == 0 )? 0 : ( m_uiNumPartition >> 1 ) + ( m_uiNumPartition >> 3 );
2193      break;
2194    case SIZE_nLx2N:
2195      ruiPartIdxLT += ( uiPartIdx == 0 )? 0 : m_uiNumPartition >> 4;
2196      ruiPartIdxRT -= ( uiPartIdx == 1 )? 0 : ( m_uiNumPartition >> 2 ) + ( m_uiNumPartition >> 4 );
2197      break;
2198    case SIZE_nRx2N:
2199      ruiPartIdxLT += ( uiPartIdx == 0 )? 0 : ( m_uiNumPartition >> 2 ) + ( m_uiNumPartition >> 4 );
2200      ruiPartIdxRT -= ( uiPartIdx == 1 )? 0 : m_uiNumPartition >> 4;
2201      break;
2202    default:
2203      assert (0);
2204      break;
2205  }
2206
2207}
2208
2209Void TComDataCU::deriveLeftBottomIdx( UInt  uiPartIdx,      UInt&      ruiPartIdxLB )
2210{
2211  ruiPartIdxLB      = g_auiRasterToZscan [g_auiZscanToRaster[ m_absZIdxInCtu ] + ( ((m_puhHeight[0] / m_pcPic->getMinCUHeight())>>1) - 1)*m_pcPic->getNumPartInCtuWidth()];
2212
2213  switch ( m_pePartSize[0] )
2214  {
2215    case SIZE_2Nx2N:
2216      ruiPartIdxLB += m_uiNumPartition >> 1;
2217      break;
2218    case SIZE_2NxN:
2219      ruiPartIdxLB += ( uiPartIdx == 0 )? 0 : m_uiNumPartition >> 1;
2220      break;
2221    case SIZE_Nx2N:
2222      ruiPartIdxLB += ( uiPartIdx == 0 )? m_uiNumPartition >> 1 : (m_uiNumPartition >> 2)*3;
2223      break;
2224    case SIZE_NxN:
2225      ruiPartIdxLB += ( m_uiNumPartition >> 2 ) * uiPartIdx;
2226      break;
2227    case SIZE_2NxnU:
2228      ruiPartIdxLB += ( uiPartIdx == 0 ) ? -((Int)m_uiNumPartition >> 3) : m_uiNumPartition >> 1;
2229      break;
2230    case SIZE_2NxnD:
2231      ruiPartIdxLB += ( uiPartIdx == 0 ) ? (m_uiNumPartition >> 2) + (m_uiNumPartition >> 3): m_uiNumPartition >> 1;
2232      break;
2233    case SIZE_nLx2N:
2234      ruiPartIdxLB += ( uiPartIdx == 0 ) ? m_uiNumPartition >> 1 : (m_uiNumPartition >> 1) + (m_uiNumPartition >> 4);
2235      break;
2236    case SIZE_nRx2N:
2237      ruiPartIdxLB += ( uiPartIdx == 0 ) ? m_uiNumPartition >> 1 : (m_uiNumPartition >> 1) + (m_uiNumPartition >> 2) + (m_uiNumPartition >> 4);
2238      break;
2239    default:
2240      assert (0);
2241      break;
2242  }
2243}
2244
2245/** Derive the partition index of neighbouring bottom right block
2246 * \param [in]  uiPartIdx     current partition index
2247 * \param [out] ruiPartIdxRB  partition index of neighbouring bottom right block
2248 */
2249Void TComDataCU::deriveRightBottomIdx( UInt uiPartIdx, UInt &ruiPartIdxRB )
2250{
2251  ruiPartIdxRB      = g_auiRasterToZscan [g_auiZscanToRaster[ m_absZIdxInCtu ] + ( ((m_puhHeight[0] / m_pcPic->getMinCUHeight())>>1) - 1)*m_pcPic->getNumPartInCtuWidth() +  m_puhWidth[0] / m_pcPic->getMinCUWidth() - 1];
2252
2253  switch ( m_pePartSize[0] )
2254  {
2255    case SIZE_2Nx2N:
2256      ruiPartIdxRB += m_uiNumPartition >> 1;
2257      break;
2258    case SIZE_2NxN:
2259      ruiPartIdxRB += ( uiPartIdx == 0 )? 0 : m_uiNumPartition >> 1;
2260      break;
2261    case SIZE_Nx2N:
2262      ruiPartIdxRB += ( uiPartIdx == 0 )? m_uiNumPartition >> 2 : (m_uiNumPartition >> 1);
2263      break;
2264    case SIZE_NxN:
2265      ruiPartIdxRB += ( m_uiNumPartition >> 2 ) * ( uiPartIdx - 1 );
2266      break;
2267    case SIZE_2NxnU:
2268      ruiPartIdxRB += ( uiPartIdx == 0 ) ? -((Int)m_uiNumPartition >> 3) : m_uiNumPartition >> 1;
2269      break;
2270    case SIZE_2NxnD:
2271      ruiPartIdxRB += ( uiPartIdx == 0 ) ? (m_uiNumPartition >> 2) + (m_uiNumPartition >> 3): m_uiNumPartition >> 1;
2272      break;
2273    case SIZE_nLx2N:
2274      ruiPartIdxRB += ( uiPartIdx == 0 ) ? (m_uiNumPartition >> 3) + (m_uiNumPartition >> 4): m_uiNumPartition >> 1;
2275      break;
2276    case SIZE_nRx2N:
2277      ruiPartIdxRB += ( uiPartIdx == 0 ) ? (m_uiNumPartition >> 2) + (m_uiNumPartition >> 3) + (m_uiNumPartition >> 4) : m_uiNumPartition >> 1;
2278      break;
2279    default:
2280      assert (0);
2281      break;
2282  }
2283}
2284
2285Void TComDataCU::deriveLeftRightTopIdxAdi ( UInt& ruiPartIdxLT, UInt& ruiPartIdxRT, UInt uiPartOffset, UInt uiPartDepth )
2286{
2287  UInt uiNumPartInWidth = (m_puhWidth[0]/m_pcPic->getMinCUWidth())>>uiPartDepth;
2288  ruiPartIdxLT = m_absZIdxInCtu + uiPartOffset;
2289  ruiPartIdxRT = g_auiRasterToZscan[ g_auiZscanToRaster[ ruiPartIdxLT ] + uiNumPartInWidth - 1 ];
2290}
2291
2292Void TComDataCU::deriveLeftBottomIdxAdi( UInt& ruiPartIdxLB, UInt uiPartOffset, UInt uiPartDepth )
2293{
2294  UInt uiAbsIdx;
2295  UInt uiMinCuWidth, uiWidthInMinCus;
2296
2297  uiMinCuWidth    = getPic()->getMinCUWidth();
2298  uiWidthInMinCus = (getWidth(0)/uiMinCuWidth)>>uiPartDepth;
2299  uiAbsIdx        = getZorderIdxInCtu()+uiPartOffset+(m_uiNumPartition>>(uiPartDepth<<1))-1;
2300  uiAbsIdx        = g_auiZscanToRaster[uiAbsIdx]-(uiWidthInMinCus-1);
2301  ruiPartIdxLB    = g_auiRasterToZscan[uiAbsIdx];
2302}
2303
2304Bool TComDataCU::hasEqualMotion( UInt uiAbsPartIdx, TComDataCU* pcCandCU, UInt uiCandAbsPartIdx )
2305{
2306  if ( getInterDir( uiAbsPartIdx ) != pcCandCU->getInterDir( uiCandAbsPartIdx ) )
2307  {
2308    return false;
2309  }
2310
2311  for ( UInt uiRefListIdx = 0; uiRefListIdx < 2; uiRefListIdx++ )
2312  {
2313    if ( getInterDir( uiAbsPartIdx ) & ( 1 << uiRefListIdx ) )
2314    {
2315      if ( getCUMvField( RefPicList( uiRefListIdx ) )->getMv( uiAbsPartIdx )     != pcCandCU->getCUMvField( RefPicList( uiRefListIdx ) )->getMv( uiCandAbsPartIdx ) ||
2316        getCUMvField( RefPicList( uiRefListIdx ) )->getRefIdx( uiAbsPartIdx ) != pcCandCU->getCUMvField( RefPicList( uiRefListIdx ) )->getRefIdx( uiCandAbsPartIdx ) )
2317      {
2318        return false;
2319      }
2320    }
2321  }
2322
2323  return true;
2324}
2325
2326//! Construct a list of merging candidates
2327Void TComDataCU::getInterMergeCandidates( UInt uiAbsPartIdx, UInt uiPUIdx, TComMvField* pcMvFieldNeighbours, UChar* puhInterDirNeighbours, Int& numValidMergeCand, Int mrgCandIdx )
2328{
2329  UInt uiAbsPartAddr = m_absZIdxInCtu + uiAbsPartIdx;
2330  Bool abCandIsInter[ MRG_MAX_NUM_CANDS ];
2331  for( UInt ui = 0; ui < getSlice()->getMaxNumMergeCand(); ++ui )
2332  {
2333    abCandIsInter[ui] = false;
2334    pcMvFieldNeighbours[ ( ui << 1 )     ].setRefIdx(NOT_VALID);
2335    pcMvFieldNeighbours[ ( ui << 1 ) + 1 ].setRefIdx(NOT_VALID);
2336  }
2337  numValidMergeCand = getSlice()->getMaxNumMergeCand();
2338  // compute the location of the current PU
2339  Int xP, yP, nPSW, nPSH;
2340  this->getPartPosition(uiPUIdx, xP, yP, nPSW, nPSH);
2341
2342  Int iCount = 0;
2343
2344  UInt uiPartIdxLT, uiPartIdxRT, uiPartIdxLB;
2345  PartSize cCurPS = getPartitionSize( uiAbsPartIdx );
2346  deriveLeftRightTopIdxGeneral( uiAbsPartIdx, uiPUIdx, uiPartIdxLT, uiPartIdxRT );
2347  deriveLeftBottomIdxGeneral( uiAbsPartIdx, uiPUIdx, uiPartIdxLB );
2348
2349  //left
2350  UInt uiLeftPartIdx = 0;
2351  TComDataCU* pcCULeft = 0;
2352  pcCULeft = getPULeft( uiLeftPartIdx, uiPartIdxLB );
2353
2354  Bool isAvailableA1 = pcCULeft &&
2355                       pcCULeft->isDiffMER(xP -1, yP+nPSH-1, xP, yP) &&
2356                       !( uiPUIdx == 1 && (cCurPS == SIZE_Nx2N || cCurPS == SIZE_nLx2N || cCurPS == SIZE_nRx2N) ) &&
2357                       pcCULeft->isInter( uiLeftPartIdx ) ;
2358
2359  if ( isAvailableA1 )
2360  {
2361    abCandIsInter[iCount] = true;
2362    // get Inter Dir
2363    puhInterDirNeighbours[iCount] = pcCULeft->getInterDir( uiLeftPartIdx );
2364    // get Mv from Left
2365    pcCULeft->getMvField( pcCULeft, uiLeftPartIdx, REF_PIC_LIST_0, pcMvFieldNeighbours[iCount<<1] );
2366    if ( getSlice()->isInterB() )
2367    {
2368      pcCULeft->getMvField( pcCULeft, uiLeftPartIdx, REF_PIC_LIST_1, pcMvFieldNeighbours[(iCount<<1)+1] );
2369    }
2370    if ( mrgCandIdx == iCount )
2371    {
2372      return;
2373    }
2374    iCount ++;
2375  }
2376
2377  // early termination
2378  if (iCount == getSlice()->getMaxNumMergeCand())
2379  {
2380    return;
2381  }
2382  // above
2383  UInt uiAbovePartIdx = 0;
2384  TComDataCU* pcCUAbove = 0;
2385  pcCUAbove = getPUAbove( uiAbovePartIdx, uiPartIdxRT );
2386
2387  Bool isAvailableB1 = pcCUAbove &&
2388                       pcCUAbove->isDiffMER(xP+nPSW-1, yP-1, xP, yP) &&
2389                       !( uiPUIdx == 1 && (cCurPS == SIZE_2NxN || cCurPS == SIZE_2NxnU || cCurPS == SIZE_2NxnD) ) &&
2390                       pcCUAbove->isInter( uiAbovePartIdx );
2391
2392  if ( isAvailableB1 && (!isAvailableA1 || !pcCULeft->hasEqualMotion( uiLeftPartIdx, pcCUAbove, uiAbovePartIdx ) ) )
2393  {
2394    abCandIsInter[iCount] = true;
2395    // get Inter Dir
2396    puhInterDirNeighbours[iCount] = pcCUAbove->getInterDir( uiAbovePartIdx );
2397    // get Mv from Left
2398    pcCUAbove->getMvField( pcCUAbove, uiAbovePartIdx, REF_PIC_LIST_0, pcMvFieldNeighbours[iCount<<1] );
2399    if ( getSlice()->isInterB() )
2400    {
2401      pcCUAbove->getMvField( pcCUAbove, uiAbovePartIdx, REF_PIC_LIST_1, pcMvFieldNeighbours[(iCount<<1)+1] );
2402    }
2403    if ( mrgCandIdx == iCount )
2404    {
2405      return;
2406    }
2407    iCount ++;
2408  }
2409  // early termination
2410  if (iCount == getSlice()->getMaxNumMergeCand())
2411  {
2412    return;
2413  }
2414
2415  // above right
2416  UInt uiAboveRightPartIdx = 0;
2417  TComDataCU* pcCUAboveRight = 0;
2418  pcCUAboveRight = getPUAboveRight( uiAboveRightPartIdx, uiPartIdxRT );
2419
2420  Bool isAvailableB0 = pcCUAboveRight &&
2421                       pcCUAboveRight->isDiffMER(xP+nPSW, yP-1, xP, yP) &&
2422                       pcCUAboveRight->isInter( uiAboveRightPartIdx );
2423
2424  if ( isAvailableB0 && ( !isAvailableB1 || !pcCUAbove->hasEqualMotion( uiAbovePartIdx, pcCUAboveRight, uiAboveRightPartIdx ) ) )
2425  {
2426    abCandIsInter[iCount] = true;
2427    // get Inter Dir
2428    puhInterDirNeighbours[iCount] = pcCUAboveRight->getInterDir( uiAboveRightPartIdx );
2429    // get Mv from Left
2430    pcCUAboveRight->getMvField( pcCUAboveRight, uiAboveRightPartIdx, REF_PIC_LIST_0, pcMvFieldNeighbours[iCount<<1] );
2431    if ( getSlice()->isInterB() )
2432    {
2433      pcCUAboveRight->getMvField( pcCUAboveRight, uiAboveRightPartIdx, REF_PIC_LIST_1, pcMvFieldNeighbours[(iCount<<1)+1] );
2434    }
2435    if ( mrgCandIdx == iCount )
2436    {
2437      return;
2438    }
2439    iCount ++;
2440  }
2441  // early termination
2442  if (iCount == getSlice()->getMaxNumMergeCand())
2443  {
2444    return;
2445  }
2446
2447  //left bottom
2448  UInt uiLeftBottomPartIdx = 0;
2449  TComDataCU* pcCULeftBottom = 0;
2450  pcCULeftBottom = this->getPUBelowLeft( uiLeftBottomPartIdx, uiPartIdxLB );
2451
2452  Bool isAvailableA0 = pcCULeftBottom &&
2453                       pcCULeftBottom->isDiffMER(xP-1, yP+nPSH, xP, yP) &&
2454                       pcCULeftBottom->isInter( uiLeftBottomPartIdx ) ;
2455
2456  if ( isAvailableA0 && ( !isAvailableA1 || !pcCULeft->hasEqualMotion( uiLeftPartIdx, pcCULeftBottom, uiLeftBottomPartIdx ) ) )
2457  {
2458    abCandIsInter[iCount] = true;
2459    // get Inter Dir
2460    puhInterDirNeighbours[iCount] = pcCULeftBottom->getInterDir( uiLeftBottomPartIdx );
2461    // get Mv from Left
2462    pcCULeftBottom->getMvField( pcCULeftBottom, uiLeftBottomPartIdx, REF_PIC_LIST_0, pcMvFieldNeighbours[iCount<<1] );
2463    if ( getSlice()->isInterB() )
2464    {
2465      pcCULeftBottom->getMvField( pcCULeftBottom, uiLeftBottomPartIdx, REF_PIC_LIST_1, pcMvFieldNeighbours[(iCount<<1)+1] );
2466    }
2467    if ( mrgCandIdx == iCount )
2468    {
2469      return;
2470    }
2471    iCount ++;
2472  }
2473  // early termination
2474  if (iCount == getSlice()->getMaxNumMergeCand())
2475  {
2476    return;
2477  }
2478
2479  // above left
2480  if( iCount < 4 )
2481  {
2482    UInt uiAboveLeftPartIdx = 0;
2483    TComDataCU* pcCUAboveLeft = 0;
2484    pcCUAboveLeft = getPUAboveLeft( uiAboveLeftPartIdx, uiAbsPartAddr );
2485
2486    Bool isAvailableB2 = pcCUAboveLeft &&
2487                         pcCUAboveLeft->isDiffMER(xP-1, yP-1, xP, yP) &&
2488                         pcCUAboveLeft->isInter( uiAboveLeftPartIdx );
2489
2490    if ( isAvailableB2 && ( !isAvailableA1 || !pcCULeft->hasEqualMotion( uiLeftPartIdx, pcCUAboveLeft, uiAboveLeftPartIdx ) )
2491        && ( !isAvailableB1 || !pcCUAbove->hasEqualMotion( uiAbovePartIdx, pcCUAboveLeft, uiAboveLeftPartIdx ) ) )
2492    {
2493      abCandIsInter[iCount] = true;
2494      // get Inter Dir
2495      puhInterDirNeighbours[iCount] = pcCUAboveLeft->getInterDir( uiAboveLeftPartIdx );
2496      // get Mv from Left
2497      pcCUAboveLeft->getMvField( pcCUAboveLeft, uiAboveLeftPartIdx, REF_PIC_LIST_0, pcMvFieldNeighbours[iCount<<1] );
2498      if ( getSlice()->isInterB() )
2499      {
2500        pcCUAboveLeft->getMvField( pcCUAboveLeft, uiAboveLeftPartIdx, REF_PIC_LIST_1, pcMvFieldNeighbours[(iCount<<1)+1] );
2501      }
2502      if ( mrgCandIdx == iCount )
2503      {
2504        return;
2505      }
2506      iCount ++;
2507    }
2508  }
2509  // early termination
2510  if (iCount == getSlice()->getMaxNumMergeCand())
2511  {
2512    return;
2513  }
2514
2515  if ( getSlice()->getEnableTMVPFlag() )
2516  {
2517    //>> MTK colocated-RightBottom
2518    UInt uiPartIdxRB;
2519
2520    deriveRightBottomIdx( uiPUIdx, uiPartIdxRB );
2521
2522    UInt uiAbsPartIdxTmp = g_auiZscanToRaster[uiPartIdxRB];
2523    const UInt numPartInCtuWidth  = m_pcPic->getNumPartInCtuWidth();
2524    const UInt numPartInCtuHeight = m_pcPic->getNumPartInCtuHeight();
2525
2526    TComMv cColMv;
2527    Int iRefIdx;
2528    Int ctuRsAddr = -1;
2529
2530#if SVC_EXTENSION
2531    if (   ( ( m_pcPic->getCtu(m_ctuRsAddr)->getCUPelX() + g_auiRasterToPelX[uiAbsPartIdxTmp] + m_pcPic->getMinCUWidth () ) < m_pcSlice->getPicWidthInLumaSamples() )  // image boundary check
2532        && ( ( m_pcPic->getCtu(m_ctuRsAddr)->getCUPelY() + g_auiRasterToPelY[uiAbsPartIdxTmp] + m_pcPic->getMinCUHeight() ) < m_pcSlice->getPicHeightInLumaSamples() ) )
2533#else
2534    if (   ( ( m_pcPic->getCtu(m_ctuRsAddr)->getCUPelX() + g_auiRasterToPelX[uiAbsPartIdxTmp] + m_pcPic->getMinCUWidth () ) < m_pcSlice->getSPS()->getPicWidthInLumaSamples () )  // image boundary check
2535        && ( ( m_pcPic->getCtu(m_ctuRsAddr)->getCUPelY() + g_auiRasterToPelY[uiAbsPartIdxTmp] + m_pcPic->getMinCUHeight() ) < m_pcSlice->getSPS()->getPicHeightInLumaSamples() ) )
2536#endif   
2537    {
2538      if ( ( uiAbsPartIdxTmp % numPartInCtuWidth < numPartInCtuWidth - 1 ) &&           // is not at the last column of CTU
2539        ( uiAbsPartIdxTmp / numPartInCtuWidth < numPartInCtuHeight - 1 ) )              // is not at the last row    of CTU
2540      {
2541        uiAbsPartAddr = g_auiRasterToZscan[ uiAbsPartIdxTmp + numPartInCtuWidth + 1 ];
2542        ctuRsAddr = getCtuRsAddr();
2543      }
2544      else if ( uiAbsPartIdxTmp % numPartInCtuWidth < numPartInCtuWidth - 1 )           // is not at the last column of CTU But is last row of CTU
2545      {
2546        uiAbsPartAddr = g_auiRasterToZscan[ (uiAbsPartIdxTmp + numPartInCtuWidth + 1) % m_pcPic->getNumPartitionsInCtu() ];
2547      }
2548      else if ( uiAbsPartIdxTmp / numPartInCtuWidth < numPartInCtuHeight - 1 )          // is not at the last row of CTU But is last column of CTU
2549      {
2550        uiAbsPartAddr = g_auiRasterToZscan[ uiAbsPartIdxTmp + 1 ];
2551        ctuRsAddr = getCtuRsAddr() + 1;
2552      }
2553      else //is the right bottom corner of CTU
2554      {
2555        uiAbsPartAddr = 0;
2556      }
2557    }
2558
2559    iRefIdx = 0;
2560
2561    Bool bExistMV = false;
2562    UInt uiPartIdxCenter;
2563    Int dir = 0;
2564    UInt uiArrayAddr = iCount;
2565    xDeriveCenterIdx( uiPUIdx, uiPartIdxCenter );
2566    bExistMV = ctuRsAddr >= 0 && xGetColMVP( REF_PIC_LIST_0, ctuRsAddr, uiAbsPartAddr, cColMv, iRefIdx );
2567    if( bExistMV == false )
2568    {
2569      bExistMV = xGetColMVP( REF_PIC_LIST_0, getCtuRsAddr(), uiPartIdxCenter,  cColMv, iRefIdx );
2570    }
2571    if( bExistMV )
2572    {
2573      dir |= 1;
2574      pcMvFieldNeighbours[ 2 * uiArrayAddr ].setMvField( cColMv, iRefIdx );
2575    }
2576
2577    if ( getSlice()->isInterB() )
2578    {
2579      bExistMV = ctuRsAddr >= 0 && xGetColMVP( REF_PIC_LIST_1, ctuRsAddr, uiAbsPartAddr, cColMv, iRefIdx);
2580      if( bExistMV == false )
2581      {
2582        bExistMV = xGetColMVP( REF_PIC_LIST_1, getCtuRsAddr(), uiPartIdxCenter, cColMv, iRefIdx );
2583      }
2584      if( bExistMV )
2585      {
2586        dir |= 2;
2587        pcMvFieldNeighbours[ 2 * uiArrayAddr + 1 ].setMvField( cColMv, iRefIdx );
2588      }
2589    }
2590
2591    if (dir != 0)
2592    {
2593      puhInterDirNeighbours[uiArrayAddr] = dir;
2594      abCandIsInter[uiArrayAddr] = true;
2595
2596      if ( mrgCandIdx == iCount )
2597      {
2598        return;
2599      }
2600      iCount++;
2601    }
2602  }
2603  // early termination
2604  if (iCount == getSlice()->getMaxNumMergeCand())
2605  {
2606    return;
2607  }
2608
2609  UInt uiArrayAddr = iCount;
2610  UInt uiCutoff = uiArrayAddr;
2611
2612  if ( getSlice()->isInterB() )
2613  {
2614    static const UInt NUM_PRIORITY_LIST=12;
2615    static const UInt uiPriorityList0[NUM_PRIORITY_LIST] = {0 , 1, 0, 2, 1, 2, 0, 3, 1, 3, 2, 3};
2616    static const UInt uiPriorityList1[NUM_PRIORITY_LIST] = {1 , 0, 2, 0, 2, 1, 3, 0, 3, 1, 3, 2};
2617
2618    for (Int idx=0; idx<uiCutoff*(uiCutoff-1) && uiArrayAddr!= getSlice()->getMaxNumMergeCand(); idx++)
2619    {
2620      assert(idx<NUM_PRIORITY_LIST);
2621      Int i = uiPriorityList0[idx];
2622      Int j = uiPriorityList1[idx];
2623      if (abCandIsInter[i] && abCandIsInter[j]&& (puhInterDirNeighbours[i]&0x1)&&(puhInterDirNeighbours[j]&0x2))
2624      {
2625        abCandIsInter[uiArrayAddr] = true;
2626        puhInterDirNeighbours[uiArrayAddr] = 3;
2627
2628        // get Mv from cand[i] and cand[j]
2629        pcMvFieldNeighbours[uiArrayAddr << 1].setMvField(pcMvFieldNeighbours[i<<1].getMv(), pcMvFieldNeighbours[i<<1].getRefIdx());
2630        pcMvFieldNeighbours[( uiArrayAddr << 1 ) + 1].setMvField(pcMvFieldNeighbours[(j<<1)+1].getMv(), pcMvFieldNeighbours[(j<<1)+1].getRefIdx());
2631
2632        Int iRefPOCL0 = m_pcSlice->getRefPOC( REF_PIC_LIST_0, pcMvFieldNeighbours[(uiArrayAddr<<1)].getRefIdx() );
2633        Int iRefPOCL1 = m_pcSlice->getRefPOC( REF_PIC_LIST_1, pcMvFieldNeighbours[(uiArrayAddr<<1)+1].getRefIdx() );
2634        if (iRefPOCL0 == iRefPOCL1 && pcMvFieldNeighbours[(uiArrayAddr<<1)].getMv() == pcMvFieldNeighbours[(uiArrayAddr<<1)+1].getMv())
2635        {
2636          abCandIsInter[uiArrayAddr] = false;
2637        }
2638        else
2639        {
2640          uiArrayAddr++;
2641        }
2642      }
2643    }
2644  }
2645  // early termination
2646  if (uiArrayAddr == getSlice()->getMaxNumMergeCand())
2647  {
2648    return;
2649  }
2650
2651  Int iNumRefIdx = (getSlice()->isInterB()) ? min(m_pcSlice->getNumRefIdx(REF_PIC_LIST_0), m_pcSlice->getNumRefIdx(REF_PIC_LIST_1)) : m_pcSlice->getNumRefIdx(REF_PIC_LIST_0);
2652
2653  Int r = 0;
2654  Int refcnt = 0;
2655  while (uiArrayAddr < getSlice()->getMaxNumMergeCand())
2656  {
2657    abCandIsInter[uiArrayAddr] = true;
2658    puhInterDirNeighbours[uiArrayAddr] = 1;
2659    pcMvFieldNeighbours[uiArrayAddr << 1].setMvField( TComMv(0, 0), r);
2660
2661    if ( getSlice()->isInterB() )
2662    {
2663      puhInterDirNeighbours[uiArrayAddr] = 3;
2664      pcMvFieldNeighbours[(uiArrayAddr << 1) + 1].setMvField(TComMv(0, 0), r);
2665    }
2666    uiArrayAddr++;
2667
2668    if ( refcnt == iNumRefIdx - 1 )
2669    {
2670      r = 0;
2671    }
2672    else
2673    {
2674      ++r;
2675      ++refcnt;
2676    }
2677  }
2678  numValidMergeCand = uiArrayAddr;
2679}
2680
2681/** Check whether the current PU and a spatial neighboring PU are in a same ME region.
2682 * \param xN, yN   location of the upper-left corner pixel of a neighboring PU
2683 * \param xP, yP   location of the upper-left corner pixel of the current PU
2684 */
2685Bool TComDataCU::isDiffMER(Int xN, Int yN, Int xP, Int yP)
2686{
2687
2688  UInt plevel = this->getSlice()->getPPS()->getLog2ParallelMergeLevelMinus2() + 2;
2689  if ((xN>>plevel)!= (xP>>plevel))
2690  {
2691    return true;
2692  }
2693  if ((yN>>plevel)!= (yP>>plevel))
2694  {
2695    return true;
2696  }
2697  return false;
2698}
2699
2700/** Calculate the location of upper-left corner pixel and size of the current PU.
2701 * \param partIdx       PU index within a CU
2702 * \param xP, yP        location of the upper-left corner pixel of the current PU
2703 * \param nPSW, nPSH    size of the current PU
2704 */
2705Void TComDataCU::getPartPosition( UInt partIdx, Int& xP, Int& yP, Int& nPSW, Int& nPSH)
2706{
2707  UInt col = m_uiCUPelX;
2708  UInt row = m_uiCUPelY;
2709
2710  switch ( m_pePartSize[0] )
2711  {
2712  case SIZE_2NxN:
2713    nPSW = getWidth(0);
2714    nPSH = getHeight(0) >> 1;
2715    xP   = col;
2716    yP   = (partIdx ==0)? row: row + nPSH;
2717    break;
2718  case SIZE_Nx2N:
2719    nPSW = getWidth(0) >> 1;
2720    nPSH = getHeight(0);
2721    xP   = (partIdx ==0)? col: col + nPSW;
2722    yP   = row;
2723    break;
2724  case SIZE_NxN:
2725    nPSW = getWidth(0) >> 1;
2726    nPSH = getHeight(0) >> 1;
2727    xP   = col + (partIdx&0x1)*nPSW;
2728    yP   = row + (partIdx>>1)*nPSH;
2729    break;
2730  case SIZE_2NxnU:
2731    nPSW = getWidth(0);
2732    nPSH = ( partIdx == 0 ) ?  getHeight(0) >> 2 : ( getHeight(0) >> 2 ) + ( getHeight(0) >> 1 );
2733    xP   = col;
2734    yP   = (partIdx ==0)? row: row + getHeight(0) - nPSH;
2735
2736    break;
2737  case SIZE_2NxnD:
2738    nPSW = getWidth(0);
2739    nPSH = ( partIdx == 0 ) ?  ( getHeight(0) >> 2 ) + ( getHeight(0) >> 1 ) : getHeight(0) >> 2;
2740    xP   = col;
2741    yP   = (partIdx ==0)? row: row + getHeight(0) - nPSH;
2742    break;
2743  case SIZE_nLx2N:
2744    nPSW = ( partIdx == 0 ) ? getWidth(0) >> 2 : ( getWidth(0) >> 2 ) + ( getWidth(0) >> 1 );
2745    nPSH = getHeight(0);
2746    xP   = (partIdx ==0)? col: col + getWidth(0) - nPSW;
2747    yP   = row;
2748    break;
2749  case SIZE_nRx2N:
2750    nPSW = ( partIdx == 0 ) ? ( getWidth(0) >> 2 ) + ( getWidth(0) >> 1 ) : getWidth(0) >> 2;
2751    nPSH = getHeight(0);
2752    xP   = (partIdx ==0)? col: col + getWidth(0) - nPSW;
2753    yP   = row;
2754    break;
2755  default:
2756    assert ( m_pePartSize[0] == SIZE_2Nx2N );
2757    nPSW = getWidth(0);
2758    nPSH = getHeight(0);
2759    xP   = col ;
2760    yP   = row ;
2761
2762    break;
2763  }
2764}
2765
2766/** Constructs a list of candidates for AMVP
2767 * \param uiPartIdx
2768 * \param uiPartAddr
2769 * \param eRefPicList
2770 * \param iRefIdx
2771 * \param pInfo
2772 */
2773Void TComDataCU::fillMvpCand ( UInt uiPartIdx, UInt uiPartAddr, RefPicList eRefPicList, Int iRefIdx, AMVPInfo* pInfo )
2774{
2775  TComMv cMvPred;
2776  Bool bAddedSmvp = false;
2777
2778  pInfo->iN = 0;
2779  if (iRefIdx < 0)
2780  {
2781    return;
2782  }
2783
2784  //-- Get Spatial MV
2785  UInt uiPartIdxLT, uiPartIdxRT, uiPartIdxLB;
2786  const UInt numPartInCtuWidth  = m_pcPic->getNumPartInCtuWidth();
2787  const UInt numPartInCtuHeight = m_pcPic->getNumPartInCtuHeight();
2788  Bool bAdded = false;
2789
2790  deriveLeftRightTopIdx( uiPartIdx, uiPartIdxLT, uiPartIdxRT );
2791  deriveLeftBottomIdx( uiPartIdx, uiPartIdxLB );
2792
2793  TComDataCU* tmpCU = NULL;
2794  UInt idx;
2795  tmpCU = getPUBelowLeft(idx, uiPartIdxLB);
2796  bAddedSmvp = (tmpCU != NULL) && (tmpCU->isInter(idx));
2797
2798  if (!bAddedSmvp)
2799  {
2800    tmpCU = getPULeft(idx, uiPartIdxLB);
2801    bAddedSmvp = (tmpCU != NULL) && (tmpCU->isInter(idx));
2802  }
2803
2804  // Left predictor search
2805  bAdded = xAddMVPCand( pInfo, eRefPicList, iRefIdx, uiPartIdxLB, MD_BELOW_LEFT);
2806  if (!bAdded)
2807  {
2808    bAdded = xAddMVPCand( pInfo, eRefPicList, iRefIdx, uiPartIdxLB, MD_LEFT );
2809  }
2810
2811  if(!bAdded)
2812  {
2813    bAdded = xAddMVPCandOrder( pInfo, eRefPicList, iRefIdx, uiPartIdxLB, MD_BELOW_LEFT);
2814    if (!bAdded)
2815    {
2816      xAddMVPCandOrder( pInfo, eRefPicList, iRefIdx, uiPartIdxLB, MD_LEFT );
2817    }
2818  }
2819
2820  // Above predictor search
2821  bAdded = xAddMVPCand( pInfo, eRefPicList, iRefIdx, uiPartIdxRT, MD_ABOVE_RIGHT);
2822
2823  if (!bAdded)
2824  {
2825    bAdded = xAddMVPCand( pInfo, eRefPicList, iRefIdx, uiPartIdxRT, MD_ABOVE);
2826  }
2827
2828  if(!bAdded)
2829  {
2830    xAddMVPCand( pInfo, eRefPicList, iRefIdx, uiPartIdxLT, MD_ABOVE_LEFT);
2831  }
2832
2833  if(!bAddedSmvp)
2834  {
2835    bAdded = xAddMVPCandOrder( pInfo, eRefPicList, iRefIdx, uiPartIdxRT, MD_ABOVE_RIGHT);
2836    if (!bAdded)
2837    {
2838      bAdded = xAddMVPCandOrder( pInfo, eRefPicList, iRefIdx, uiPartIdxRT, MD_ABOVE);
2839    }
2840
2841    if(!bAdded)
2842    {
2843      xAddMVPCandOrder( pInfo, eRefPicList, iRefIdx, uiPartIdxLT, MD_ABOVE_LEFT);
2844    }
2845  }
2846
2847  if ( pInfo->iN == 2 )
2848  {
2849    if ( pInfo->m_acMvCand[ 0 ] == pInfo->m_acMvCand[ 1 ] )
2850    {
2851      pInfo->iN = 1;
2852    }
2853  }
2854
2855  if ( getSlice()->getEnableTMVPFlag() )
2856  {
2857    // Get Temporal Motion Predictor
2858    Int iRefIdx_Col = iRefIdx;
2859    TComMv cColMv;
2860    UInt uiPartIdxRB;
2861    UInt uiAbsPartIdx;
2862    UInt uiAbsPartAddr;
2863
2864    deriveRightBottomIdx( uiPartIdx, uiPartIdxRB );
2865    uiAbsPartAddr = m_absZIdxInCtu + uiPartAddr;
2866
2867    //----  co-located RightBottom Temporal Predictor (H) ---//
2868    uiAbsPartIdx = g_auiZscanToRaster[uiPartIdxRB];
2869    Int ctuRsAddr = -1;
2870
2871#if SVC_EXTENSION
2872    if (  ( ( m_pcPic->getCtu(m_ctuRsAddr)->getCUPelX() + g_auiRasterToPelX[uiAbsPartIdx] + m_pcPic->getMinCUWidth () ) < m_pcSlice->getPicWidthInLumaSamples())   // image boundary check
2873       && ( ( m_pcPic->getCtu(m_ctuRsAddr)->getCUPelY() + g_auiRasterToPelY[uiAbsPartIdx] + m_pcPic->getMinCUHeight() ) < m_pcSlice->getPicHeightInLumaSamples() ) )   
2874#else
2875    if (  ( ( m_pcPic->getCtu(m_ctuRsAddr)->getCUPelX() + g_auiRasterToPelX[uiAbsPartIdx] + m_pcPic->getMinCUWidth () ) < m_pcSlice->getSPS()->getPicWidthInLumaSamples () )  // image boundary check
2876       && ( ( m_pcPic->getCtu(m_ctuRsAddr)->getCUPelY() + g_auiRasterToPelY[uiAbsPartIdx] + m_pcPic->getMinCUHeight() ) < m_pcSlice->getSPS()->getPicHeightInLumaSamples() ) )
2877#endif   
2878    {
2879      if ( ( uiAbsPartIdx % numPartInCtuWidth < numPartInCtuWidth - 1 ) &&  // is not at the last column of CTU
2880           ( uiAbsPartIdx / numPartInCtuWidth < numPartInCtuHeight - 1 ) )  // is not at the last row    of CTU
2881      {
2882        uiAbsPartAddr = g_auiRasterToZscan[ uiAbsPartIdx + numPartInCtuWidth + 1 ];
2883        ctuRsAddr = getCtuRsAddr();
2884      }
2885      else if ( uiAbsPartIdx % numPartInCtuWidth < numPartInCtuWidth - 1 )  // is not at the last column of CTU But is last row of CTU
2886      {
2887        uiAbsPartAddr = g_auiRasterToZscan[ (uiAbsPartIdx + numPartInCtuWidth + 1) % m_pcPic->getNumPartitionsInCtu() ];
2888      }
2889      else if ( uiAbsPartIdx / numPartInCtuWidth < numPartInCtuHeight - 1 ) // is not at the last row of CTU But is last column of CTU
2890      {
2891        uiAbsPartAddr = g_auiRasterToZscan[ uiAbsPartIdx + 1 ];
2892        ctuRsAddr = getCtuRsAddr() + 1;
2893      }
2894      else //is the right bottom corner of CTU
2895      {
2896        uiAbsPartAddr = 0;
2897      }
2898    }
2899    if ( ctuRsAddr >= 0 && xGetColMVP( eRefPicList, ctuRsAddr, uiAbsPartAddr, cColMv, iRefIdx_Col ) )
2900    {
2901      pInfo->m_acMvCand[pInfo->iN++] = cColMv;
2902    }
2903    else
2904    {
2905      UInt uiPartIdxCenter;
2906      xDeriveCenterIdx( uiPartIdx, uiPartIdxCenter );
2907      if (xGetColMVP( eRefPicList, getCtuRsAddr(), uiPartIdxCenter,  cColMv, iRefIdx_Col ))
2908      {
2909        pInfo->m_acMvCand[pInfo->iN++] = cColMv;
2910      }
2911    }
2912    //----  co-located RightBottom Temporal Predictor  ---//
2913  }
2914
2915  if (pInfo->iN > AMVP_MAX_NUM_CANDS)
2916  {
2917    pInfo->iN = AMVP_MAX_NUM_CANDS;
2918  }
2919
2920  while (pInfo->iN < AMVP_MAX_NUM_CANDS)
2921  {
2922    pInfo->m_acMvCand[pInfo->iN].set(0,0);
2923    pInfo->iN++;
2924  }
2925  return ;
2926}
2927
2928
2929Bool TComDataCU::isBipredRestriction(UInt puIdx)
2930{
2931  Int width = 0;
2932  Int height = 0;
2933  UInt partAddr;
2934
2935  getPartIndexAndSize( puIdx, partAddr, width, height );
2936  if ( getWidth(0) == 8 && (width < 8 || height < 8) )
2937  {
2938    return true;
2939  }
2940  return false;
2941}
2942
2943
2944Void TComDataCU::clipMv    (TComMv&  rcMv)
2945{
2946  const TComSPS &sps=*(m_pcSlice->getSPS());
2947  Int  iMvShift = 2;
2948  Int iOffset = 8;
2949#if SVC_EXTENSION
2950  Int iHorMax = ( m_pcSlice->getPicWidthInLumaSamples() + iOffset - (Int)m_uiCUPelX - 1 ) << iMvShift;
2951#else
2952  Int iHorMax = ( sps.getPicWidthInLumaSamples() + iOffset - (Int)m_uiCUPelX - 1 ) << iMvShift;
2953#endif
2954  Int iHorMin = (      -(Int)sps.getMaxCUWidth() - iOffset - (Int)m_uiCUPelX + 1 ) << iMvShift;
2955 
2956#if SVC_EXTENSION
2957  Int iVerMax = ( m_pcSlice->getPicHeightInLumaSamples() + iOffset - (Int)m_uiCUPelY - 1 ) << iMvShift;
2958#else
2959  Int iVerMax = ( sps.getPicHeightInLumaSamples() + iOffset - (Int)m_uiCUPelY - 1 ) << iMvShift;
2960#endif
2961  Int iVerMin = (      -(Int)sps.getMaxCUHeight() - iOffset - (Int)m_uiCUPelY + 1 ) << iMvShift;
2962
2963  rcMv.setHor( min (iHorMax, max (iHorMin, rcMv.getHor())) );
2964  rcMv.setVer( min (iVerMax, max (iVerMin, rcMv.getVer())) );
2965}
2966
2967
2968UInt TComDataCU::getIntraSizeIdx(UInt uiAbsPartIdx)
2969{
2970  UInt uiShift = ( m_pePartSize[uiAbsPartIdx]==SIZE_NxN ? 1 : 0 );
2971
2972  UChar uiWidth = m_puhWidth[uiAbsPartIdx]>>uiShift;
2973  UInt  uiCnt = 0;
2974  while( uiWidth )
2975  {
2976    uiCnt++;
2977    uiWidth>>=1;
2978  }
2979  uiCnt-=2;
2980  return uiCnt > 6 ? 6 : uiCnt;
2981}
2982
2983Void TComDataCU::clearCbf( UInt uiIdx, ComponentID compID, UInt uiNumParts )
2984{
2985  memset( &m_puhCbf[compID][uiIdx], 0, sizeof(UChar)*uiNumParts);
2986}
2987
2988/** Set a I_PCM flag for all sub-partitions of a partition.
2989 * \param bIpcmFlag I_PCM flag
2990 * \param uiAbsPartIdx patition index
2991 * \param uiDepth CU depth
2992 * \returns Void
2993 */
2994Void TComDataCU::setIPCMFlagSubParts  (Bool bIpcmFlag, UInt uiAbsPartIdx, UInt uiDepth)
2995{
2996  UInt uiCurrPartNumb = m_pcPic->getNumPartitionsInCtu() >> (uiDepth << 1);
2997
2998  memset(m_pbIPCMFlag + uiAbsPartIdx, bIpcmFlag, sizeof(Bool)*uiCurrPartNumb );
2999}
3000
3001/** Test whether the block at uiPartIdx is skipped.
3002 * \param uiPartIdx Partition index
3003 * \returns true if the current the block is skipped
3004 */
3005Bool TComDataCU::isSkipped( UInt uiPartIdx )
3006{
3007  return ( getSkipFlag( uiPartIdx ) );
3008}
3009
3010// ====================================================================================================================
3011// Protected member functions
3012// ====================================================================================================================
3013
3014Bool TComDataCU::xAddMVPCand( AMVPInfo* pInfo, RefPicList eRefPicList, Int iRefIdx, UInt uiPartUnitIdx, MVP_DIR eDir )
3015{
3016  TComDataCU* pcTmpCU = NULL;
3017  UInt uiIdx;
3018  switch( eDir )
3019  {
3020    case MD_LEFT:
3021    {
3022      pcTmpCU = getPULeft(uiIdx, uiPartUnitIdx);
3023      break;
3024    }
3025    case MD_ABOVE:
3026    {
3027      pcTmpCU = getPUAbove(uiIdx, uiPartUnitIdx);
3028      break;
3029    }
3030    case MD_ABOVE_RIGHT:
3031    {
3032      pcTmpCU = getPUAboveRight(uiIdx, uiPartUnitIdx);
3033      break;
3034    }
3035    case MD_BELOW_LEFT:
3036    {
3037      pcTmpCU = getPUBelowLeft(uiIdx, uiPartUnitIdx);
3038      break;
3039    }
3040    case MD_ABOVE_LEFT:
3041    {
3042      pcTmpCU = getPUAboveLeft(uiIdx, uiPartUnitIdx);
3043      break;
3044    }
3045    default:
3046    {
3047      break;
3048    }
3049  }
3050
3051  if ( pcTmpCU == NULL )
3052  {
3053    return false;
3054  }
3055
3056  if ( pcTmpCU->getCUMvField(eRefPicList)->getRefIdx(uiIdx) >= 0 && m_pcSlice->getRefPic( eRefPicList, iRefIdx)->getPOC() == pcTmpCU->getSlice()->getRefPOC( eRefPicList, pcTmpCU->getCUMvField(eRefPicList)->getRefIdx(uiIdx) ))
3057  {
3058    TComMv cMvPred = pcTmpCU->getCUMvField(eRefPicList)->getMv(uiIdx);
3059
3060    pInfo->m_acMvCand[ pInfo->iN++] = cMvPred;
3061    return true;
3062  }
3063
3064  RefPicList eRefPicList2nd = REF_PIC_LIST_0;
3065  if(       eRefPicList == REF_PIC_LIST_0 )
3066  {
3067    eRefPicList2nd = REF_PIC_LIST_1;
3068  }
3069  else if ( eRefPicList == REF_PIC_LIST_1)
3070  {
3071    eRefPicList2nd = REF_PIC_LIST_0;
3072  }
3073
3074
3075  Int iCurrRefPOC = m_pcSlice->getRefPic( eRefPicList, iRefIdx)->getPOC();
3076  Int iNeibRefPOC;
3077
3078
3079  if( pcTmpCU->getCUMvField(eRefPicList2nd)->getRefIdx(uiIdx) >= 0 )
3080  {
3081    iNeibRefPOC = pcTmpCU->getSlice()->getRefPOC( eRefPicList2nd, pcTmpCU->getCUMvField(eRefPicList2nd)->getRefIdx(uiIdx) );
3082    if( iNeibRefPOC == iCurrRefPOC ) // Same Reference Frame But Diff List//
3083    {
3084      TComMv cMvPred = pcTmpCU->getCUMvField(eRefPicList2nd)->getMv(uiIdx);
3085      pInfo->m_acMvCand[ pInfo->iN++] = cMvPred;
3086      return true;
3087    }
3088  }
3089  return false;
3090}
3091
3092/**
3093 * \param pInfo
3094 * \param eRefPicList
3095 * \param iRefIdx
3096 * \param uiPartUnitIdx
3097 * \param eDir
3098 * \returns Bool
3099 */
3100Bool TComDataCU::xAddMVPCandOrder( AMVPInfo* pInfo, RefPicList eRefPicList, Int iRefIdx, UInt uiPartUnitIdx, MVP_DIR eDir )
3101{
3102  TComDataCU* pcTmpCU = NULL;
3103  UInt uiIdx;
3104  switch( eDir )
3105  {
3106  case MD_LEFT:
3107    {
3108      pcTmpCU = getPULeft(uiIdx, uiPartUnitIdx);
3109      break;
3110    }
3111  case MD_ABOVE:
3112    {
3113      pcTmpCU = getPUAbove(uiIdx, uiPartUnitIdx);
3114      break;
3115    }
3116  case MD_ABOVE_RIGHT:
3117    {
3118      pcTmpCU = getPUAboveRight(uiIdx, uiPartUnitIdx);
3119      break;
3120    }
3121  case MD_BELOW_LEFT:
3122    {
3123      pcTmpCU = getPUBelowLeft(uiIdx, uiPartUnitIdx);
3124      break;
3125    }
3126  case MD_ABOVE_LEFT:
3127    {
3128      pcTmpCU = getPUAboveLeft(uiIdx, uiPartUnitIdx);
3129      break;
3130    }
3131  default:
3132    {
3133      break;
3134    }
3135  }
3136
3137  if ( pcTmpCU == NULL )
3138  {
3139    return false;
3140  }
3141
3142  RefPicList eRefPicList2nd = REF_PIC_LIST_0;
3143  if(       eRefPicList == REF_PIC_LIST_0 )
3144  {
3145    eRefPicList2nd = REF_PIC_LIST_1;
3146  }
3147  else if ( eRefPicList == REF_PIC_LIST_1)
3148  {
3149    eRefPicList2nd = REF_PIC_LIST_0;
3150  }
3151
3152  Int iCurrPOC = m_pcSlice->getPOC();
3153  Int iCurrRefPOC = m_pcSlice->getRefPic( eRefPicList, iRefIdx)->getPOC();
3154  Int iNeibPOC = iCurrPOC;
3155  Int iNeibRefPOC;
3156  Bool bIsCurrRefLongTerm = m_pcSlice->getRefPic( eRefPicList, iRefIdx)->getIsLongTerm();
3157  Bool bIsNeibRefLongTerm = false;
3158
3159  //---------------  V1 (END) ------------------//
3160  if( pcTmpCU->getCUMvField(eRefPicList)->getRefIdx(uiIdx) >= 0)
3161  {
3162    iNeibRefPOC = pcTmpCU->getSlice()->getRefPOC( eRefPicList, pcTmpCU->getCUMvField(eRefPicList)->getRefIdx(uiIdx) );
3163    TComMv cMvPred = pcTmpCU->getCUMvField(eRefPicList)->getMv(uiIdx);
3164    TComMv rcMv;
3165
3166    bIsNeibRefLongTerm = pcTmpCU->getSlice()->getRefPic( eRefPicList, pcTmpCU->getCUMvField(eRefPicList)->getRefIdx(uiIdx) )->getIsLongTerm();
3167    if ( bIsCurrRefLongTerm == bIsNeibRefLongTerm )
3168    {
3169      if ( bIsCurrRefLongTerm || bIsNeibRefLongTerm )
3170      {
3171        rcMv = cMvPred;
3172      }
3173      else
3174      {
3175        Int iScale = xGetDistScaleFactor( iCurrPOC, iCurrRefPOC, iNeibPOC, iNeibRefPOC );
3176        if ( iScale == 4096 )
3177        {
3178          rcMv = cMvPred;
3179        }
3180        else
3181        {
3182          rcMv = cMvPred.scaleMv( iScale );
3183        }
3184      }
3185
3186      pInfo->m_acMvCand[ pInfo->iN++] = rcMv;
3187      return true;
3188    }
3189  }
3190  //---------------------- V2(END) --------------------//
3191  if( pcTmpCU->getCUMvField(eRefPicList2nd)->getRefIdx(uiIdx) >= 0)
3192  {
3193    iNeibRefPOC = pcTmpCU->getSlice()->getRefPOC( eRefPicList2nd, pcTmpCU->getCUMvField(eRefPicList2nd)->getRefIdx(uiIdx) );
3194    TComMv cMvPred = pcTmpCU->getCUMvField(eRefPicList2nd)->getMv(uiIdx);
3195    TComMv rcMv;
3196
3197    bIsNeibRefLongTerm = pcTmpCU->getSlice()->getRefPic( eRefPicList2nd, pcTmpCU->getCUMvField(eRefPicList2nd)->getRefIdx(uiIdx) )->getIsLongTerm();
3198    if ( bIsCurrRefLongTerm == bIsNeibRefLongTerm )
3199    {
3200      if ( bIsCurrRefLongTerm || bIsNeibRefLongTerm )
3201      {
3202        rcMv = cMvPred;
3203      }
3204      else
3205      {
3206        Int iScale = xGetDistScaleFactor( iCurrPOC, iCurrRefPOC, iNeibPOC, iNeibRefPOC );
3207        if ( iScale == 4096 )
3208        {
3209          rcMv = cMvPred;
3210        }
3211        else
3212        {
3213          rcMv = cMvPred.scaleMv( iScale );
3214        }
3215      }
3216
3217      pInfo->m_acMvCand[ pInfo->iN++] = rcMv;
3218      return true;
3219    }
3220  }
3221  //---------------------- V3(END) --------------------//
3222  return false;
3223}
3224
3225Bool TComDataCU::xGetColMVP( RefPicList eRefPicList, Int ctuRsAddr, Int uiPartUnitIdx, TComMv& rcMv, Int& riRefIdx )
3226{
3227  UInt uiAbsPartAddr = uiPartUnitIdx;
3228
3229  RefPicList  eColRefPicList;
3230  Int iColPOC, iColRefPOC, iCurrPOC, iCurrRefPOC, iScale;
3231  TComMv cColMv;
3232
3233  // use coldir.
3234  TComPic *pColPic = getSlice()->getRefPic( RefPicList(getSlice()->isInterB() ? 1-getSlice()->getColFromL0Flag() : 0), getSlice()->getColRefIdx());
3235  TComDataCU *pColCtu = pColPic->getCtu( ctuRsAddr );
3236  if(pColCtu->getPic()==0||pColCtu->getPartitionSize(uiPartUnitIdx)==NUMBER_OF_PART_SIZES)
3237  {
3238    return false;
3239  }
3240  iCurrPOC = m_pcSlice->getPOC();
3241  iColPOC = pColCtu->getSlice()->getPOC();
3242
3243  if (!pColCtu->isInter(uiAbsPartAddr))
3244  {
3245    return false;
3246  }
3247
3248  eColRefPicList = getSlice()->getCheckLDC() ? eRefPicList : RefPicList(getSlice()->getColFromL0Flag());
3249
3250  Int iColRefIdx = pColCtu->getCUMvField(RefPicList(eColRefPicList))->getRefIdx(uiAbsPartAddr);
3251
3252  if (iColRefIdx < 0 )
3253  {
3254    eColRefPicList = RefPicList(1 - eColRefPicList);
3255    iColRefIdx = pColCtu->getCUMvField(RefPicList(eColRefPicList))->getRefIdx(uiAbsPartAddr);
3256
3257    if (iColRefIdx < 0 )
3258    {
3259      return false;
3260    }
3261  }
3262
3263  // Scale the vector.
3264  iColRefPOC = pColCtu->getSlice()->getRefPOC(eColRefPicList, iColRefIdx);
3265  cColMv = pColCtu->getCUMvField(eColRefPicList)->getMv(uiAbsPartAddr);
3266
3267  iCurrRefPOC = m_pcSlice->getRefPic(eRefPicList, riRefIdx)->getPOC();
3268
3269  Bool bIsCurrRefLongTerm = m_pcSlice->getRefPic(eRefPicList, riRefIdx)->getIsLongTerm();
3270  Bool bIsColRefLongTerm = pColCtu->getSlice()->getIsUsedAsLongTerm(eColRefPicList, iColRefIdx);
3271
3272  if ( bIsCurrRefLongTerm != bIsColRefLongTerm )
3273  {
3274    return false;
3275  }
3276
3277  if ( bIsCurrRefLongTerm || bIsColRefLongTerm )
3278  {
3279    rcMv = cColMv;
3280  }
3281  else
3282  {
3283    iScale = xGetDistScaleFactor(iCurrPOC, iCurrRefPOC, iColPOC, iColRefPOC);
3284    if ( iScale == 4096 )
3285    {
3286      rcMv = cColMv;
3287    }
3288    else
3289    {
3290      rcMv = cColMv.scaleMv( iScale );
3291    }
3292  }
3293
3294  return true;
3295}
3296
3297UInt TComDataCU::xGetMvdBits(TComMv cMvd)
3298{
3299  return ( xGetComponentBits(cMvd.getHor()) + xGetComponentBits(cMvd.getVer()) );
3300}
3301
3302UInt TComDataCU::xGetComponentBits(Int iVal)
3303{
3304  UInt uiLength = 1;
3305  UInt uiTemp   = ( iVal <= 0) ? (-iVal<<1)+1: (iVal<<1);
3306
3307  assert ( uiTemp );
3308
3309  while ( 1 != uiTemp )
3310  {
3311    uiTemp >>= 1;
3312    uiLength += 2;
3313  }
3314
3315  return uiLength;
3316}
3317
3318
3319Int TComDataCU::xGetDistScaleFactor(Int iCurrPOC, Int iCurrRefPOC, Int iColPOC, Int iColRefPOC)
3320{
3321  Int iDiffPocD = iColPOC - iColRefPOC;
3322  Int iDiffPocB = iCurrPOC - iCurrRefPOC;
3323
3324  if( iDiffPocD == iDiffPocB )
3325  {
3326    return 4096;
3327  }
3328  else
3329  {
3330    Int iTDB      = Clip3( -128, 127, iDiffPocB );
3331    Int iTDD      = Clip3( -128, 127, iDiffPocD );
3332    Int iX        = (0x4000 + abs(iTDD/2)) / iTDD;
3333    Int iScale    = Clip3( -4096, 4095, (iTDB * iX + 32) >> 6 );
3334    return iScale;
3335  }
3336}
3337
3338Void TComDataCU::xDeriveCenterIdx( UInt uiPartIdx, UInt& ruiPartIdxCenter )
3339{
3340  UInt uiPartAddr;
3341  Int  iPartWidth;
3342  Int  iPartHeight;
3343  getPartIndexAndSize( uiPartIdx, uiPartAddr, iPartWidth, iPartHeight);
3344
3345  ruiPartIdxCenter = m_absZIdxInCtu+uiPartAddr; // partition origin.
3346  ruiPartIdxCenter = g_auiRasterToZscan[ g_auiZscanToRaster[ ruiPartIdxCenter ]
3347                                        + ( iPartHeight/m_pcPic->getMinCUHeight()  )/2*m_pcPic->getNumPartInCtuWidth()
3348                                        + ( iPartWidth/m_pcPic->getMinCUWidth()  )/2];
3349}
3350
3351Void TComDataCU::compressMV()
3352{
3353  Int scaleFactor = 4 * AMVP_DECIMATION_FACTOR / m_unitSize;
3354  if (scaleFactor > 0)
3355  {
3356    for(UInt i=0; i<NUM_REF_PIC_LIST_01; i++)
3357    {
3358      m_acCUMvField[i].compress(m_pePredMode, scaleFactor);
3359    }
3360  }
3361}
3362
3363UInt TComDataCU::getCoefScanIdx(const UInt uiAbsPartIdx, const UInt uiWidth, const UInt uiHeight, const ComponentID compID) const
3364{
3365  //------------------------------------------------
3366
3367  //this mechanism is available for intra only
3368
3369  if (!isIntra(uiAbsPartIdx))
3370  {
3371    return SCAN_DIAG;
3372  }
3373
3374  //------------------------------------------------
3375
3376  //check that MDCS can be used for this TU
3377
3378  const ChromaFormat format = getPic()->getChromaFormat();
3379
3380  const UInt maximumWidth  = MDCS_MAXIMUM_WIDTH  >> getComponentScaleX(compID, format);
3381  const UInt maximumHeight = MDCS_MAXIMUM_HEIGHT >> getComponentScaleY(compID, format);
3382
3383  if ((uiWidth > maximumWidth) || (uiHeight > maximumHeight))
3384  {
3385    return SCAN_DIAG;
3386  }
3387
3388  //------------------------------------------------
3389
3390  //otherwise, select the appropriate mode
3391
3392  UInt uiDirMode  = getIntraDir(toChannelType(compID), uiAbsPartIdx);
3393
3394  if (uiDirMode==DM_CHROMA_IDX)
3395  {
3396    const TComSPS *sps=getSlice()->getSPS();
3397    const UInt partsPerMinCU = 1<<(2*(sps->getMaxTotalCUDepth() - sps->getLog2DiffMaxMinCodingBlockSize()));
3398    uiDirMode = getIntraDir(CHANNEL_TYPE_LUMA, getChromasCorrespondingPULumaIdx(uiAbsPartIdx, getPic()->getChromaFormat(), partsPerMinCU));
3399  }
3400
3401  if (isChroma(compID) && (format == CHROMA_422))
3402  {
3403    uiDirMode = g_chroma422IntraAngleMappingTable[uiDirMode];
3404  }
3405
3406  //------------------
3407
3408  if      (abs((Int)uiDirMode - VER_IDX) <= MDCS_ANGLE_LIMIT)
3409  {
3410    return SCAN_HOR;
3411  }
3412  else if (abs((Int)uiDirMode - HOR_IDX) <= MDCS_ANGLE_LIMIT)
3413  {
3414    return SCAN_VER;
3415  }
3416  else
3417  {
3418    return SCAN_DIAG;
3419  }
3420}
3421
3422#if SVC_EXTENSION
3423TComDataCU* TComDataCU::getBaseColCU( UInt refLayerIdc, UInt uiCuAbsPartIdx, UInt &uiCUAddrBase, UInt &uiAbsPartIdxBase, Bool motionMapping )
3424{
3425  UInt uiPelX = getCUPelX() + g_auiRasterToPelX[ g_auiZscanToRaster[uiCuAbsPartIdx] ];
3426  UInt uiPelY = getCUPelY() + g_auiRasterToPelY[ g_auiZscanToRaster[uiCuAbsPartIdx] ];
3427
3428  return getBaseColCU( refLayerIdc, uiPelX, uiPelY, uiCUAddrBase, uiAbsPartIdxBase, motionMapping );
3429}
3430
3431TComDataCU* TComDataCU::getBaseColCU( UInt refLayerIdc, UInt pelX, UInt pelY, UInt &uiCUAddrBase, UInt &uiAbsPartIdxBase, Bool motionMapping )
3432{
3433  TComPic* baseColPic = m_pcSlice->getBaseColPic(refLayerIdc);
3434
3435  Int iPelX = Clip3<Int>(0, m_pcPic->getPicYuvRec()->getWidth(COMPONENT_Y)  - 1, pelX);
3436  Int iPelY = Clip3<Int>(0, m_pcPic->getPicYuvRec()->getHeight(COMPONENT_Y) - 1, pelY);
3437
3438  // centre of the collocated 16x16 block for motion mapping
3439  if( motionMapping )
3440  {
3441    iPelX = pelX + 8;
3442    iPelY = pelY + 8;
3443  }
3444
3445  Int leftStartL = m_pcSlice->getPPS()->getScaledRefLayerWindowForLayer(baseColPic->getSlice(0)->getVPS()->getRefLayerId(getSlice()->getLayerId(), refLayerIdc)).getWindowLeftOffset();
3446  Int topStartL  = m_pcSlice->getPPS()->getScaledRefLayerWindowForLayer(baseColPic->getSlice(0)->getVPS()->getRefLayerId(getSlice()->getLayerId(), refLayerIdc)).getWindowTopOffset();
3447
3448  const Window &windowRL = m_pcSlice->getPPS()->getRefLayerWindowForLayer(baseColPic->getSlice(0)->getVPS()->getRefLayerId(getSlice()->getLayerId(), refLayerIdc));
3449  Int iBX = (((iPelX - leftStartL)*g_posScalingFactor[refLayerIdc][0] + (1<<15)) >> 16) + windowRL.getWindowLeftOffset();
3450  Int iBY = (((iPelY - topStartL )*g_posScalingFactor[refLayerIdc][1] + (1<<15)) >> 16) + windowRL.getWindowTopOffset();
3451
3452  // offset for collocated block in the motion mapping
3453  if( motionMapping )
3454  {
3455    if( m_pcPic->equalPictureSizeAndOffsetFlag(refLayerIdc) )
3456    {
3457      // copy motion field from the same sample position for the case of 1x scaling ratio and same offset value between the current and reference layers
3458      iBX = pelX;
3459      iBY = pelY;
3460    }
3461    else
3462    {
3463      // actually, motion field compression is performed in the Void TComPic::compressMotion() function, but with (+4) the rounding may have effect on the picture boundary check.
3464      iBX = ( ( iBX + 4 ) >> 4 ) << 4;
3465      iBY = ( ( iBY + 4 ) >> 4 ) << 4;
3466    }
3467  }
3468
3469  if ( iBX < 0 || iBX >= baseColPic->getPicYuvRec()->getWidth(COMPONENT_Y) || iBY < 0 || iBY >= baseColPic->getPicYuvRec()->getHeight(COMPONENT_Y) )
3470  {
3471    return NULL;
3472  }
3473
3474  UInt baseMaxCUHeight = baseColPic->getSlice(0)->getSPS()->getMaxCUHeight();
3475  UInt baseMaxCUWidth  = baseColPic->getSlice(0)->getSPS()->getMaxCUWidth();
3476  UInt baseMinUnitSize = baseColPic->getMinCUWidth();
3477 
3478  uiCUAddrBase = ( iBY / baseMaxCUHeight ) * baseColPic->getFrameWidthInCtus() + ( iBX / baseMaxCUWidth );
3479
3480  assert(uiCUAddrBase < baseColPic->getNumberOfCtusInFrame());
3481
3482  UInt uiRasterAddrBase = ( iBY - (iBY/baseMaxCUHeight)*baseMaxCUHeight ) / baseMinUnitSize * baseColPic->getNumPartInCtuWidth() + ( iBX - (iBX/baseMaxCUWidth)*baseMaxCUWidth ) / baseMinUnitSize; 
3483
3484#if LAYER_CTB
3485  uiAbsPartIdxBase = g_auiLayerRasterToZscan[baseColPic->getLayerId()][uiRasterAddrBase];
3486#else
3487  uiAbsPartIdxBase = g_auiRasterToZscan[uiRasterAddrBase];
3488#endif
3489
3490  return baseColPic->getCtu(uiCUAddrBase);
3491}
3492
3493Void TComDataCU::scaleBaseMV( UInt refLayerIdc, TComMvField& rcMvFieldEnhance, TComMvField& rcMvFieldBase )
3494{
3495  TComMvField cMvFieldBase;
3496  TComMv cMv;
3497
3498  cMv = rcMvFieldBase.getMv().scaleMv( g_mvScalingFactor[refLayerIdc][0], g_mvScalingFactor[refLayerIdc][1] );
3499
3500  rcMvFieldEnhance.setMvField( cMv, rcMvFieldBase.getRefIdx() );
3501}
3502
3503#if FAST_INTRA_SHVC
3504/** generate limited set of remaining modes
3505*\param   uiAbsPartIdx
3506*\param   uiIntraDirPred  pointer to the array for MPM storage
3507*\returns Number of intra coding modes (nb of remaining modes + 3 MPMs)
3508*/
3509Int TComDataCU::reduceSetOfIntraModes( UInt uiAbsPartIdx, Int* uiIntraDirPred, Int &fullSetOfModes )
3510{
3511  // check BL mode
3512  UInt uiCUAddrBase = 0, uiAbsPartAddrBase = 0;
3513  // the right reference layerIdc should be specified, currently it is set to m_layerId-1
3514  TComDataCU* pcTempCU = getBaseColCU(m_layerId - 1, uiAbsPartIdx, uiCUAddrBase, uiAbsPartAddrBase, false );
3515
3516  if( pcTempCU->getPredictionMode( uiAbsPartAddrBase ) != MODE_INTRA )
3517  {
3518    return( NUM_INTRA_MODE-1 );
3519  }
3520
3521  // compute set of enabled modes g_reducedSetIntraModes[...]
3522  Int authorizedMode[NUM_INTRA_MODE-1]={1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1,1};
3523  Int nbModes;
3524  for (nbModes=0; nbModes<3; nbModes++)  // add 3 MPMs 1st
3525  {
3526    g_reducedSetIntraModes[nbModes] = uiIntraDirPred[nbModes];
3527    authorizedMode[ uiIntraDirPred[nbModes] ] = 0;
3528  }
3529
3530  Int iColBaseDir = pcTempCU->getIntraDir( CHANNEL_TYPE_LUMA, uiAbsPartAddrBase );
3531  if ( authorizedMode[iColBaseDir] )  //possibly add BL mode
3532  {
3533    g_reducedSetIntraModes[nbModes++] = iColBaseDir;
3534    authorizedMode[ iColBaseDir ] = 0;
3535  }
3536
3537  Int iRefMode = ( iColBaseDir > 1 ) ? iColBaseDir : uiIntraDirPred[0];
3538  if ( iRefMode > 1 )    //add neighboring modes of refMode
3539  {
3540    UInt Left  = iRefMode;
3541    UInt Right = iRefMode;
3542    while ( nbModes < NB_REMAIN_MODES+3 )
3543    {
3544      Left = ((Left + 29) % 32) + 2;
3545      Right = ((Right - 1 ) % 32) + 2;
3546      if ( authorizedMode[Left] )   g_reducedSetIntraModes[nbModes++] = Left;
3547      if ( authorizedMode[Right] )  g_reducedSetIntraModes[nbModes++] = Right;
3548    }
3549  }
3550  else      //add pre-defined modes
3551  {
3552    Int  idx = 0;
3553    while ( nbModes < NB_REMAIN_MODES+3 )
3554    {
3555      UInt mode = g_predefSetIntraModes[idx++];
3556      if ( authorizedMode[mode] )   g_reducedSetIntraModes[nbModes++] = mode;
3557    }
3558  }
3559
3560  fullSetOfModes = 0;
3561  return ( nbModes );
3562}
3563#endif
3564
3565#if REF_IDX_ME_ZEROMV
3566Bool TComDataCU::xCheckZeroMVILRMerge(UChar uhInterDir, TComMvField& cMvFieldL0, TComMvField& cMvFieldL1)
3567{
3568  Bool checkZeroMVILR = true;
3569
3570  if(uhInterDir&0x1)  //list0
3571  {
3572    Int refIdxL0 = cMvFieldL0.getRefIdx();
3573    TComPic* refPic = m_pcSlice->getRefPic(REF_PIC_LIST_0, refIdxL0);
3574
3575    if(refPic->isILR(m_layerId))
3576    {
3577      checkZeroMVILR &= (cMvFieldL0.getHor() == 0 && cMvFieldL0.getVer() == 0);
3578
3579      // It is a requirement of bitstream conformance that when the reference picture represented by the variable refIdxLX is an inter-layer reference picture,
3580      // VpsInterLayerSamplePredictionEnabled[ LayerIdxInVps[ currLayerId ] ][ LayerIdxInVps[ rLId ] ] shall be equal to 1, where rLId is set equal to nuh_layer_id of the inter-layer picture
3581      checkZeroMVILR &= m_pcSlice->getVPS()->isSamplePredictionType( getLayerIdx(), refPic->getLayerIdx() );
3582    }
3583  }
3584  if(uhInterDir&0x2)  //list1
3585  {
3586    Int refIdxL1  = cMvFieldL1.getRefIdx();
3587    TComPic* refPic = m_pcSlice->getRefPic(REF_PIC_LIST_1, refIdxL1);
3588
3589    if(refPic->isILR(m_layerId))
3590    {
3591      checkZeroMVILR &= (cMvFieldL1.getHor() == 0 && cMvFieldL1.getVer() == 0);
3592
3593      // It is a requirement of bitstream conformance that when the reference picture represented by the variable refIdxLX is an inter-layer reference picture,
3594      // VpsInterLayerSamplePredictionEnabled[ LayerIdxInVps[ currLayerId ] ][ LayerIdxInVps[ rLId ] ] shall be equal to 1, where rLId is set equal to nuh_layer_id of the inter-layer picture
3595      checkZeroMVILR &= m_pcSlice->getVPS()->isSamplePredictionType( getLayerIdx(), refPic->getLayerIdx() );
3596    }
3597  }
3598
3599  return checkZeroMVILR;
3600}
3601
3602Bool TComDataCU::xCheckZeroMVILRMvdL1Zero(Int iRefList, Int iRefIdx, Int MvpIdx)
3603{
3604  RefPicList eRefPicList = iRefList > 0? REF_PIC_LIST_1: REF_PIC_LIST_0;
3605  assert(eRefPicList == REF_PIC_LIST_1);
3606
3607  Bool checkZeroMVILR = true;
3608
3609  if(getSlice()->getRefPic(eRefPicList, iRefIdx)->isILR(m_layerId))
3610  {
3611    AMVPInfo* pcAMVPInfo = getCUMvField(eRefPicList)->getAMVPInfo();
3612    TComMv    cMv        = pcAMVPInfo->m_acMvCand[MvpIdx];
3613    checkZeroMVILR &= (cMv.getHor() == 0 && cMv.getVer() == 0);
3614  }
3615
3616  return checkZeroMVILR;
3617}
3618#endif
3619
3620#if N0383_IL_CONSTRAINED_TILE_SETS_SEI
3621Bool TComDataCU::isInterLayerReference(UChar uhInterDir, TComMvField& cMvFieldL0, TComMvField& cMvFieldL1)
3622{
3623  Bool checkILR = false;
3624
3625  if(uhInterDir&0x1)  //list0
3626  {
3627    Int refIdxL0 = cMvFieldL0.getRefIdx();
3628    checkILR = getSlice()->getRefPic(REF_PIC_LIST_0, refIdxL0)->isILR(m_layerId);
3629  }
3630  if(uhInterDir&0x2)  //list1
3631  {
3632    Int refIdxL1  = cMvFieldL1.getRefIdx();
3633    checkILR = checkILR || getSlice()->getRefPic(REF_PIC_LIST_1, refIdxL1)->isILR(m_layerId);
3634  }
3635
3636  return checkILR;
3637}
3638#endif
3639
3640#endif //SVC_EXTENSION
3641//! \}
Note: See TracBrowser for help on using the repository browser.