source: SHVCSoftware/branches/SHM-dev/source/Lib/TLibEncoder/TEncSlice.cpp @ 1242

Last change on this file since 1242 was 1235, checked in by seregin, 9 years ago

port rev 4219 and rev 4246

  • Property svn:eol-style set to native
File size: 49.9 KB
RevLine 
[313]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
[494]4 * granted under this license.
[313]5 *
[595]6 * Copyright (c) 2010-2014, ITU/ISO/IEC
[313]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     TEncSlice.cpp
35    \brief    slice encoder class
36*/
37
38#include "TEncTop.h"
39#include "TEncSlice.h"
40#include <math.h>
41
42//! \ingroup TLibEncoder
43//! \{
44
45// ====================================================================================================================
46// Constructor / destructor / create / destroy
47// ====================================================================================================================
48
49TEncSlice::TEncSlice()
[1235]50: m_encCABACTableIdx(I_SLICE)
[313]51{
52  m_apcPicYuvPred = NULL;
53  m_apcPicYuvResi = NULL;
[494]54
[313]55  m_pdRdPicLambda = NULL;
56  m_pdRdPicQp     = NULL;
57  m_piRdPicQp     = NULL;
58}
59
60TEncSlice::~TEncSlice()
61{
62}
63
[494]64Void TEncSlice::create( Int iWidth, Int iHeight, ChromaFormat chromaFormat, UInt iMaxCUWidth, UInt iMaxCUHeight, UChar uhTotalDepth )
65{
66  // create prediction picture
67  if ( m_apcPicYuvPred == NULL )
68  {
69    m_apcPicYuvPred  = new TComPicYuv;
70    m_apcPicYuvPred->create( iWidth, iHeight, chromaFormat, iMaxCUWidth, iMaxCUHeight, uhTotalDepth );
71  }
72
73  // create residual picture
74  if( m_apcPicYuvResi == NULL )
75  {
76    m_apcPicYuvResi  = new TComPicYuv;
77    m_apcPicYuvResi->create( iWidth, iHeight, chromaFormat, iMaxCUWidth, iMaxCUHeight, uhTotalDepth );
78  }
79}
80
[313]81Void TEncSlice::destroy()
82{
83  // destroy prediction picture
84  if ( m_apcPicYuvPred )
85  {
86    m_apcPicYuvPred->destroy();
87    delete m_apcPicYuvPred;
88    m_apcPicYuvPred  = NULL;
89  }
[494]90
[313]91  // destroy residual picture
92  if ( m_apcPicYuvResi )
93  {
94    m_apcPicYuvResi->destroy();
95    delete m_apcPicYuvResi;
96    m_apcPicYuvResi  = NULL;
97  }
[494]98
[313]99  // free lambda and QP arrays
100  if ( m_pdRdPicLambda ) { xFree( m_pdRdPicLambda ); m_pdRdPicLambda = NULL; }
101  if ( m_pdRdPicQp     ) { xFree( m_pdRdPicQp     ); m_pdRdPicQp     = NULL; }
102  if ( m_piRdPicQp     ) { xFree( m_piRdPicQp     ); m_piRdPicQp     = NULL; }
103}
104
105Void TEncSlice::init( TEncTop* pcEncTop )
106{
107  m_pcCfg             = pcEncTop;
108  m_pcListPic         = pcEncTop->getListPic();
109#if SVC_EXTENSION
110  m_ppcTEncTop        = pcEncTop->getLayerEnc();
[494]111#endif
[313]112  m_pcGOPEncoder      = pcEncTop->getGOPEncoder();
113  m_pcCuEncoder       = pcEncTop->getCuEncoder();
114  m_pcPredSearch      = pcEncTop->getPredSearch();
[494]115
[313]116  m_pcEntropyCoder    = pcEncTop->getEntropyCoder();
117  m_pcSbacCoder       = pcEncTop->getSbacCoder();
118  m_pcBinCABAC        = pcEncTop->getBinCABAC();
119  m_pcTrQuant         = pcEncTop->getTrQuant();
[494]120
[313]121  m_pcRdCost          = pcEncTop->getRdCost();
122  m_pppcRDSbacCoder   = pcEncTop->getRDSbacCoder();
123  m_pcRDGoOnSbacCoder = pcEncTop->getRDGoOnSbacCoder();
[494]124
[313]125  // create lambda and QP arrays
126  m_pdRdPicLambda     = (Double*)xMalloc( Double, m_pcCfg->getDeltaQpRD() * 2 + 1 );
127  m_pdRdPicQp         = (Double*)xMalloc( Double, m_pcCfg->getDeltaQpRD() * 2 + 1 );
128  m_piRdPicQp         = (Int*   )xMalloc( Int,    m_pcCfg->getDeltaQpRD() * 2 + 1 );
129  m_pcRateCtrl        = pcEncTop->getRateCtrl();
130}
131
[1029]132
133
134Void
135#if JCTVC_M0259_LAMBDAREFINEMENT
136TEncSlice::setUpLambda(TComSlice* slice, Double dLambda, Int iQP, Int depth)
137#else
138TEncSlice::setUpLambda(TComSlice* slice, const Double dLambda, Int iQP)
139#endif
140{
141#if JCTVC_M0259_LAMBDAREFINEMENT
[1065]142  Int layerIdx = slice->getLayerIdx();
143
144  if( slice->getLayerId() > 0 && m_ppcTEncTop[layerIdx]->getNumActiveRefLayers() && depth >= 3 && m_pcCfg->getGOPSize() == ( 1 << depth ) )
[1029]145  {
[1062]146    UInt prevLayerIdx = 0; 
[1065]147    if( m_ppcTEncTop[layerIdx]->getNumActiveRefLayers() > 0 )
[1062]148    {
149      prevLayerIdx = m_ppcTEncTop[layerIdx]->getPredLayerIdx( m_ppcTEncTop[layerIdx]->getNumActiveRefLayers() - 1);
150    }
[1065]151
[1057]152    Double gamma = xCalEnhLambdaFactor( m_ppcTEncTop[prevLayerIdx]->getQP() - m_ppcTEncTop[layerIdx]->getQP() ,
[1065]153                                        1.0 * m_ppcTEncTop[layerIdx]->getSourceWidth() * m_ppcTEncTop[layerIdx]->getSourceHeight()
154                                        / m_ppcTEncTop[prevLayerIdx]->getSourceWidth() / m_ppcTEncTop[prevLayerIdx]->getSourceHeight() );
155
[1029]156    dLambda *= gamma;
157  }
158#endif
159
160  // store lambda
161  m_pcRdCost ->setLambda( dLambda );
162
163  // for RDO
164  // in RdCost there is only one lambda because the luma and chroma bits are not separated, instead we weight the distortion of chroma.
165  Double dLambdas[MAX_NUM_COMPONENT] = { dLambda };
166  for(UInt compIdx=1; compIdx<MAX_NUM_COMPONENT; compIdx++)
167  {
168    const ComponentID compID=ComponentID(compIdx);
169    Int chromaQPOffset = slice->getPPS()->getQpOffset(compID) + slice->getSliceChromaQpDelta(compID);
170    Int qpc=(iQP + chromaQPOffset < 0) ? iQP : getScaledChromaQP(iQP + chromaQPOffset, m_pcCfg->getChromaFormatIdc());
171    Double tmpWeight = pow( 2.0, (iQP-qpc)/3.0 );  // takes into account of the chroma qp mapping and chroma qp Offset
172
173#if JCTVC_M0259_LAMBDAREFINEMENT
[1057]174    if( slice->getLayerId() > 0 && m_ppcTEncTop[slice->getLayerIdx()]->getNumActiveRefLayers() && m_pcCfg->getGOPSize() >= 8 && slice->isIntra() == false && depth == 0 )
[1029]175    {
176      dLambdas[0] = dLambda * 1.1;
177      m_pcRdCost->setLambda( dLambdas[0] );
178
179      m_pcRdCost->setDistortionWeight(compID, tmpWeight * 1.15);
180      dLambdas[compIdx] = dLambdas[0] / tmpWeight / 1.15;
181    }
182    else
183    {
184#endif
185    m_pcRdCost->setDistortionWeight(compID, tmpWeight);
186    dLambdas[compIdx]=dLambda/tmpWeight;
187#if JCTVC_M0259_LAMBDAREFINEMENT
188    }
189#endif
190  }
191
192#if RDOQ_CHROMA_LAMBDA
193// for RDOQ
194  m_pcTrQuant->setLambdas( dLambdas );
195#else
196  m_pcTrQuant->setLambda( dLambda );
197#endif
198
199// For SAO
200  slice   ->setLambdas( dLambdas );
201}
202
203
204
[313]205/**
206 - non-referenced frame marking
207 - QP computation based on temporal structure
208 - lambda computation based on QP
209 - set temporal layer ID and the parameter sets
210 .
211 \param pcPic         picture class
[1029]212 \param pocLast      POC of last picture
213 \param pocCurr     current POC
[313]214 \param iNumPicRcvd   number of received pictures
215 \param iTimeOffset   POC offset for hierarchical structure
216 \param iDepth        temporal layer depth
217 \param rpcSlice      slice header class
218 \param pSPS          SPS associated with the slice
219 \param pPPS          PPS associated with the slice
220 */
221#if SVC_EXTENSION
222//\param vps          VPS associated with the slice
[1235]223Void TEncSlice::initEncSlice( TComPic* pcPic, Int pocLast, Int pocCurr, Int iNumPicRcvd, Int iGOPid, TComSlice*& rpcSlice, Bool isField )
[313]224#else
[1235]225Void TEncSlice::initEncSlice( TComPic* pcPic, Int pocLast, Int pocCurr, Int iNumPicRcvd, Int iGOPid, TComSlice*& rpcSlice, const TComSPS* pSPS, const TComPPS *pPPS, Bool isField )
[313]226#endif
227{
228  Double dQP;
229  Double dLambda;
[494]230
[313]231  rpcSlice = pcPic->getSlice(0);
232  rpcSlice->setSliceBits(0);
233  rpcSlice->setPic( pcPic );
234#if SVC_EXTENSION
[1235]235  const TComPPS* pPPS = &pcPic->getPicSym()->getPPS();
236
[313]237  UInt layerId = pcPic->getLayerId();
238  rpcSlice->initSlice( layerId );
239#else
240  rpcSlice->initSlice();
241#endif
242  rpcSlice->setPicOutputFlag( true );
243  rpcSlice->setPOC( pocCurr );
[494]244
[313]245  // depth computation based on GOP size
246  Int depth;
247  {
[713]248    Int poc = rpcSlice->getPOC();
249    if(isField)
250    {
[1029]251      poc = (poc/2) % (m_pcCfg->getGOPSize()/2);
[713]252    }
253    else
254    {
[1029]255      poc = poc % m_pcCfg->getGOPSize();   
[713]256    }
[1029]257
[313]258    if ( poc == 0 )
259    {
260      depth = 0;
261    }
262    else
263    {
264      Int step = m_pcCfg->getGOPSize();
265      depth    = 0;
266      for( Int i=step>>1; i>=1; i>>=1 )
267      {
268        for ( Int j=i; j<m_pcCfg->getGOPSize(); j+=step )
269        {
270          if ( j == poc )
271          {
272            i=0;
273            break;
274          }
275        }
276        step >>= 1;
277        depth++;
278      }
279    }
[1029]280
[713]281#if HARMONIZE_GOP_FIRST_FIELD_COUPLE
282    if(poc != 0)
283    {
284#endif
[1029]285      if (isField && ((rpcSlice->getPOC() % 2) == 1))
286      {
287        depth ++;
288      }
289#if HARMONIZE_GOP_FIRST_FIELD_COUPLE
[713]290    }
291#endif
292  }
[494]293
[313]294  // slice type
295  SliceType eSliceType;
[494]296
[313]297  eSliceType=B_SLICE;
[713]298#if EFFICIENT_FIELD_IRAP
299  if(!(isField && pocLast == 1))
300  {
301#endif // EFFICIENT_FIELD_IRAP
302#if ALLOW_RECOVERY_POINT_AS_RAP
[1029]303    if(m_pcCfg->getDecodingRefreshType() == 3)
304    {
305      eSliceType = (pocLast == 0 || pocCurr % m_pcCfg->getIntraPeriod() == 0             || m_pcGOPEncoder->getGOPSize() == 0) ? I_SLICE : eSliceType;
306    }
307    else
308    {
[713]309#endif
[1029]310      eSliceType = (pocLast == 0 || (pocCurr - (isField ? 1 : 0)) % m_pcCfg->getIntraPeriod() == 0 || m_pcGOPEncoder->getGOPSize() == 0) ? I_SLICE : eSliceType;
311#if ALLOW_RECOVERY_POINT_AS_RAP
312    }
313#endif
[713]314#if EFFICIENT_FIELD_IRAP
315  }
316#endif
[494]317
[313]318  rpcSlice->setSliceType    ( eSliceType );
[494]319
[313]320  // ------------------------------------------------------------------------------------------------------------------
321  // Non-referenced frame marking
322  // ------------------------------------------------------------------------------------------------------------------
[494]323
[313]324  if(pocLast == 0)
325  {
326    rpcSlice->setTemporalLayerNonReferenceFlag(false);
327  }
328  else
329  {
330    rpcSlice->setTemporalLayerNonReferenceFlag(!m_pcCfg->getGOPEntry(iGOPid).m_refPic);
331  }
332  rpcSlice->setReferenced(true);
[494]333
[313]334  // ------------------------------------------------------------------------------------------------------------------
335  // QP setting
336  // ------------------------------------------------------------------------------------------------------------------
[494]337
[313]338  dQP = m_pcCfg->getQP();
339  if(eSliceType!=I_SLICE)
340  {
[1203]341#if SVC_EXTENSION
[595]342    if (!(( m_pcCfg->getMaxDeltaQP() == 0 ) && (dQP == -rpcSlice->getQpBDOffsetY() ) && (rpcSlice->getPPS()->getTransquantBypassEnableFlag())))
[442]343#else
[1029]344    if (!(( m_pcCfg->getMaxDeltaQP() == 0 ) && (dQP == -rpcSlice->getSPS()->getQpBDOffset(CHANNEL_TYPE_LUMA) ) && (rpcSlice->getPPS()->getTransquantBypassEnableFlag())))
[442]345#endif
[313]346    {
347      dQP += m_pcCfg->getGOPEntry(iGOPid).m_QPOffset;
348    }
349  }
[494]350
[313]351  // modify QP
352  Int* pdQPs = m_pcCfg->getdQPs();
353  if ( pdQPs )
354  {
355    dQP += pdQPs[ rpcSlice->getPOC() ];
356  }
[1029]357
358  if (m_pcCfg->getCostMode()==COST_LOSSLESS_CODING)
359  {
360    dQP=LOSSLESS_AND_MIXED_LOSSLESS_RD_COST_TEST_QP;
361    m_pcCfg->setDeltaQpRD(0);
362  }
363
[313]364  // ------------------------------------------------------------------------------------------------------------------
365  // Lambda computation
366  // ------------------------------------------------------------------------------------------------------------------
[494]367
[313]368  Int iQP;
369  Double dOrigQP = dQP;
370
371  // pre-compute lambda and QP values for all possible QP candidates
372  for ( Int iDQpIdx = 0; iDQpIdx < 2 * m_pcCfg->getDeltaQpRD() + 1; iDQpIdx++ )
373  {
374    // compute QP value
375    dQP = dOrigQP + ((iDQpIdx+1)>>1)*(iDQpIdx%2 ? -1 : 1);
[494]376
[313]377    // compute lambda value
378    Int    NumberBFrames = ( m_pcCfg->getGOPSize() - 1 );
379    Int    SHIFT_QP = 12;
[442]380
381    Double dLambda_scale = 1.0 - Clip3( 0.0, 0.5, 0.05*(Double)(isField ? NumberBFrames/2 : NumberBFrames) );
382
[313]383#if FULL_NBIT
[1029]384    Int    bitdepth_luma_qp_scale = 6 * (g_bitDepth[CHANNEL_TYPE_LUMA] - 8);
[313]385#else
386    Int    bitdepth_luma_qp_scale = 0;
387#endif
388    Double qp_temp = (Double) dQP + bitdepth_luma_qp_scale - SHIFT_QP;
389#if FULL_NBIT
390    Double qp_temp_orig = (Double) dQP - SHIFT_QP;
391#endif
392    // Case #1: I or P-slices (key-frame)
393    Double dQPFactor = m_pcCfg->getGOPEntry(iGOPid).m_QPFactor;
394    if ( eSliceType==I_SLICE )
395    {
396      dQPFactor=0.57*dLambda_scale;
397    }
398    dLambda = dQPFactor*pow( 2.0, qp_temp/3.0 );
399
400    if ( depth>0 )
401    {
402#if FULL_NBIT
403        dLambda *= Clip3( 2.00, 4.00, (qp_temp_orig / 6.0) ); // (j == B_SLICE && p_cur_frm->layer != 0 )
404#else
405        dLambda *= Clip3( 2.00, 4.00, (qp_temp / 6.0) ); // (j == B_SLICE && p_cur_frm->layer != 0 )
406#endif
407    }
[494]408
[313]409    // if hadamard is used in ME process
410    if ( !m_pcCfg->getUseHADME() && rpcSlice->getSliceType( ) != I_SLICE )
411    {
412      dLambda *= 0.95;
413    }
[494]414
[1203]415#if SVC_EXTENSION
[442]416    iQP = max( -rpcSlice->getQpBDOffsetY(), min( MAX_QP, (Int) floor( dQP + 0.5 ) ) );
417#else
[1029]418    iQP = max( -pSPS->getQpBDOffset(CHANNEL_TYPE_LUMA), min( MAX_QP, (Int) floor( dQP + 0.5 ) ) );
[442]419#endif
[313]420
421    m_pdRdPicLambda[iDQpIdx] = dLambda;
422    m_pdRdPicQp    [iDQpIdx] = dQP;
423    m_piRdPicQp    [iDQpIdx] = iQP;
424  }
[494]425
[313]426  // obtain dQP = 0 case
427  dLambda = m_pdRdPicLambda[0];
428  dQP     = m_pdRdPicQp    [0];
429  iQP     = m_piRdPicQp    [0];
[494]430
[313]431  if( rpcSlice->getSliceType( ) != I_SLICE )
432  {
433    dLambda *= m_pcCfg->getLambdaModifier( m_pcCfg->getGOPEntry(iGOPid).m_temporalId );
434  }
435
436#if JCTVC_M0259_LAMBDAREFINEMENT
[1029]437  setUpLambda(rpcSlice, dLambda, iQP, depth);
[313]438#else
[1029]439  setUpLambda(rpcSlice, dLambda, iQP);
[313]440#endif
441#if HB_LAMBDA_FOR_LDC
442  // restore original slice type
[1029]443
[713]444#if EFFICIENT_FIELD_IRAP
445  if(!(isField && pocLast == 1))
446  {
447#endif // EFFICIENT_FIELD_IRAP
448#if ALLOW_RECOVERY_POINT_AS_RAP
[1029]449    if(m_pcCfg->getDecodingRefreshType() == 3)
450    {
451      eSliceType = (pocLast == 0 || (pocCurr)                     % m_pcCfg->getIntraPeriod() == 0 || m_pcGOPEncoder->getGOPSize() == 0) ? I_SLICE : eSliceType;
452    }
453    else
454    {
[713]455#endif
[1029]456      eSliceType = (pocLast == 0 || (pocCurr - (isField ? 1 : 0)) % m_pcCfg->getIntraPeriod() == 0 || m_pcGOPEncoder->getGOPSize() == 0) ? I_SLICE : eSliceType;
457#if ALLOW_RECOVERY_POINT_AS_RAP
458    }
459#endif
[713]460#if EFFICIENT_FIELD_IRAP
461  }
462#endif // EFFICIENT_FIELD_IRAP
[494]463
[442]464#if SVC_EXTENSION
[540]465  if( m_pcCfg->getLayerId() > 0 && m_pcCfg->getNumActiveRefLayers() > 0 )
[313]466  {
467    eSliceType=B_SLICE;
468  }
469#endif
470  rpcSlice->setSliceType        ( eSliceType );
471#endif
[494]472
[313]473  if (m_pcCfg->getUseRecalculateQPAccordingToLambda())
474  {
475    dQP = xGetQPValueAccordingToLambda( dLambda );
[1203]476#if SVC_EXTENSION
[494]477    iQP = max( -rpcSlice->getQpBDOffsetY(), min( MAX_QP, (Int) floor( dQP + 0.5 ) ) );
[442]478#else
[1029]479    iQP = max( -pSPS->getQpBDOffset(CHANNEL_TYPE_LUMA), min( MAX_QP, (Int) floor( dQP + 0.5 ) ) );
[442]480#endif
[313]481  }
482
[1029]483  rpcSlice->setSliceQp           ( iQP );
[313]484#if ADAPTIVE_QP_SELECTION
[1029]485  rpcSlice->setSliceQpBase       ( iQP );
[313]486#endif
[1029]487  rpcSlice->setSliceQpDelta      ( 0 );
488  rpcSlice->setSliceChromaQpDelta( COMPONENT_Cb, 0 );
489  rpcSlice->setSliceChromaQpDelta( COMPONENT_Cr, 0 );
490  rpcSlice->setUseChromaQpAdj( pPPS->getChromaQpAdjTableSize() > 0 );
[313]491  rpcSlice->setNumRefIdx(REF_PIC_LIST_0,m_pcCfg->getGOPEntry(iGOPid).m_numRefPicsActive);
492  rpcSlice->setNumRefIdx(REF_PIC_LIST_1,m_pcCfg->getGOPEntry(iGOPid).m_numRefPicsActive);
[494]493
[313]494  if ( m_pcCfg->getDeblockingFilterMetric() )
495  {
496    rpcSlice->setDeblockingFilterOverrideFlag(true);
497    rpcSlice->setDeblockingFilterDisable(false);
498    rpcSlice->setDeblockingFilterBetaOffsetDiv2( 0 );
499    rpcSlice->setDeblockingFilterTcOffsetDiv2( 0 );
[1029]500  }
501  else if (rpcSlice->getPPS()->getDeblockingFilterControlPresentFlag())
[313]502  {
[1235]503    rpcSlice->setDeblockingFilterOverrideFlag( rpcSlice->getPPS()->getDeblockingFilterOverrideEnabledFlag() );
504    rpcSlice->setDeblockingFilterDisable( rpcSlice->getPPS()->getPicDisableDeblockingFilterFlag() );
[313]505    if ( !rpcSlice->getDeblockingFilterDisable())
506    {
[1235]507      if ( rpcSlice->getDeblockingFilterOverrideFlag() && eSliceType!=I_SLICE)
[313]508      {
509        rpcSlice->setDeblockingFilterBetaOffsetDiv2( m_pcCfg->getGOPEntry(iGOPid).m_betaOffsetDiv2 + m_pcCfg->getLoopFilterBetaOffset()  );
510        rpcSlice->setDeblockingFilterTcOffsetDiv2( m_pcCfg->getGOPEntry(iGOPid).m_tcOffsetDiv2 + m_pcCfg->getLoopFilterTcOffset() );
511      }
512      else
513      {
[1029]514        rpcSlice->setDeblockingFilterBetaOffsetDiv2( m_pcCfg->getLoopFilterBetaOffset() );
515        rpcSlice->setDeblockingFilterTcOffsetDiv2( m_pcCfg->getLoopFilterTcOffset() );
[313]516      }
517    }
518  }
519  else
520  {
521    rpcSlice->setDeblockingFilterOverrideFlag( false );
522    rpcSlice->setDeblockingFilterDisable( false );
523    rpcSlice->setDeblockingFilterBetaOffsetDiv2( 0 );
524    rpcSlice->setDeblockingFilterTcOffsetDiv2( 0 );
525  }
526
527  rpcSlice->setDepth            ( depth );
[494]528
[313]529  pcPic->setTLayer( m_pcCfg->getGOPEntry(iGOPid).m_temporalId );
530  if(eSliceType==I_SLICE)
531  {
532    pcPic->setTLayer(0);
533  }
534  rpcSlice->setTLayer( pcPic->getTLayer() );
535
536  assert( m_apcPicYuvPred );
537  assert( m_apcPicYuvResi );
[494]538
[313]539  pcPic->setPicYuvPred( m_apcPicYuvPred );
540  pcPic->setPicYuvResi( m_apcPicYuvResi );
541  rpcSlice->setSliceMode            ( m_pcCfg->getSliceMode()            );
542  rpcSlice->setSliceArgument        ( m_pcCfg->getSliceArgument()        );
543  rpcSlice->setSliceSegmentMode     ( m_pcCfg->getSliceSegmentMode()     );
544  rpcSlice->setSliceSegmentArgument ( m_pcCfg->getSliceSegmentArgument() );
545  rpcSlice->setMaxNumMergeCand        ( m_pcCfg->getMaxNumMergeCand()        );
[1128]546
[540]547#if HIGHER_LAYER_IRAP_SKIP_FLAG
[1128]548  if( m_pcCfg->getSkipPictureAtArcSwitch() && m_pcCfg->getAdaptiveResolutionChange() > 0 && rpcSlice->getLayerId() == 1 && rpcSlice->getPOC() == m_pcCfg->getAdaptiveResolutionChange() )
[540]549  {
550    rpcSlice->setMaxNumMergeCand        ( 1 );
551  }
552#endif
[1128]553
[442]554#if SVC_EXTENSION
[313]555  if( layerId > 0 )
556  {
557    if( rpcSlice->getNumILRRefIdx() > 0 )
558    {
[1057]559      rpcSlice->setActiveNumILRRefIdx( m_ppcTEncTop[rpcSlice->getVPS()->getLayerIdxInVps(layerId)]->getNumActiveRefLayers() );
[313]560      for( Int i = 0; i < rpcSlice->getActiveNumILRRefIdx(); i++ )
561      {
[1057]562        rpcSlice->setInterLayerPredLayerIdc( m_ppcTEncTop[rpcSlice->getVPS()->getLayerIdxInVps(layerId)]->getPredLayerIdx(i), i );
[313]563      }
564      rpcSlice->setInterLayerPredEnabledFlag(1);
565    }
[1057]566    rpcSlice->setMFMEnabledFlag(m_ppcTEncTop[rpcSlice->getVPS()->getLayerIdxInVps(layerId)]->getMFMEnabledFlag());
[313]567  }
568
569#endif
570}
571
572Void TEncSlice::resetQP( TComPic* pic, Int sliceQP, Double lambda )
573{
574  TComSlice* slice = pic->getSlice(0);
575
576  // store lambda
577  slice->setSliceQp( sliceQP );
[713]578#if ADAPTIVE_QP_SELECTION
[313]579  slice->setSliceQpBase ( sliceQP );
[713]580#endif
[1029]581  setUpLambda(slice, lambda, sliceQP);
582}
[313]583
584// ====================================================================================================================
585// Public member functions
586// ====================================================================================================================
587
588Void TEncSlice::setSearchRange( TComSlice* pcSlice )
589{
590  Int iCurrPOC = pcSlice->getPOC();
591  Int iRefPOC;
592  Int iGOPSize = m_pcCfg->getGOPSize();
593  Int iOffset = (iGOPSize >> 1);
594  Int iMaxSR = m_pcCfg->getSearchRange();
595  Int iNumPredDir = pcSlice->isInterP() ? 1 : 2;
[494]596
[313]597  for (Int iDir = 0; iDir <= iNumPredDir; iDir++)
598  {
599    //RefPicList e = (RefPicList)iDir;
600    RefPicList  e = ( iDir ? REF_PIC_LIST_1 : REF_PIC_LIST_0 );
601    for (Int iRefIdx = 0; iRefIdx < pcSlice->getNumRefIdx(e); iRefIdx++)
602    {
603      iRefPOC = pcSlice->getRefPic(e, iRefIdx)->getPOC();
604      Int iNewSR = Clip3(8, iMaxSR, (iMaxSR*ADAPT_SR_SCALE*abs(iCurrPOC - iRefPOC)+iOffset)/iGOPSize);
605      m_pcPredSearch->setAdaptiveSearchRange(iDir, iRefIdx, iNewSR);
606    }
607  }
608}
609
610/**
611 - multi-loop slice encoding for different slice QP
612 .
613 \param rpcPic    picture class
614 */
[1029]615Void TEncSlice::precompressSlice( TComPic* pcPic )
[313]616{
617  // if deltaQP RD is not used, simply return
618  if ( m_pcCfg->getDeltaQpRD() == 0 )
619  {
620    return;
621  }
622
623  if ( m_pcCfg->getUseRateCtrl() )
624  {
625    printf( "\nMultiple QP optimization is not allowed when rate control is enabled." );
626    assert(0);
627  }
[494]628
[1029]629  TComSlice* pcSlice        = pcPic->getSlice(getSliceIdx());
[313]630  Double     dPicRdCostBest = MAX_DOUBLE;
631  UInt       uiQpIdxBest = 0;
[494]632
[313]633  Double dFrameLambda;
634#if FULL_NBIT
[1029]635  Int    SHIFT_QP = 12 + 6 * (g_bitDepth[CHANNEL_TYPE_LUMA] - 8);
[313]636#else
637  Int    SHIFT_QP = 12;
638#endif
[494]639
[313]640  // set frame lambda
641  if (m_pcCfg->getGOPSize() > 1)
642  {
643    dFrameLambda = 0.68 * pow (2, (m_piRdPicQp[0]  - SHIFT_QP) / 3.0) * (pcSlice->isInterB()? 2 : 1);
644  }
645  else
646  {
647    dFrameLambda = 0.68 * pow (2, (m_piRdPicQp[0] - SHIFT_QP) / 3.0);
648  }
649  m_pcRdCost      ->setFrameLambda(dFrameLambda);
[494]650
[1029]651  const UInt initialSliceQp=pcSlice->getSliceQp();
[313]652  // for each QP candidate
653  for ( UInt uiQpIdx = 0; uiQpIdx < 2 * m_pcCfg->getDeltaQpRD() + 1; uiQpIdx++ )
654  {
655    pcSlice       ->setSliceQp             ( m_piRdPicQp    [uiQpIdx] );
656#if ADAPTIVE_QP_SELECTION
657    pcSlice       ->setSliceQpBase         ( m_piRdPicQp    [uiQpIdx] );
658#endif
[1029]659    setUpLambda(pcSlice, m_pdRdPicLambda[uiQpIdx], m_piRdPicQp    [uiQpIdx]);
[313]660
661    // try compress
[1029]662    compressSlice   ( pcPic );
[494]663
[313]664    Double dPicRdCost;
665    UInt64 uiPicDist        = m_uiPicDist;
[1029]666    // TODO: will this work if multiple slices are being used? There may not be any reconstruction data yet.
667    //       Will this also be ideal if a byte-restriction is placed on the slice?
668    //         - what if the last CTU was sometimes included, sometimes not, and that had all the distortion?
669    m_pcGOPEncoder->preLoopFilterPicAll( pcPic, uiPicDist );
[494]670
[313]671    // compute RD cost and choose the best
[1029]672    dPicRdCost = m_pcRdCost->calcRdCost64( m_uiPicTotalBits, uiPicDist, true, DF_SSE_FRAME);
[494]673
[313]674    if ( dPicRdCost < dPicRdCostBest )
675    {
676      uiQpIdxBest    = uiQpIdx;
677      dPicRdCostBest = dPicRdCost;
678    }
679  }
[494]680
[1029]681  if (pcSlice->getDependentSliceSegmentFlag() && initialSliceQp!=m_piRdPicQp[uiQpIdxBest] )
682  {
683    // TODO: this won't work with dependent slices: they do not have their own QP.
684    fprintf(stderr,"ERROR - attempt to change QP for a dependent slice-segment, having already coded the slice\n");
685    assert(pcSlice->getDependentSliceSegmentFlag()==false || initialSliceQp==m_piRdPicQp[uiQpIdxBest]);
686  }
[313]687  // set best values
688  pcSlice       ->setSliceQp             ( m_piRdPicQp    [uiQpIdxBest] );
689#if ADAPTIVE_QP_SELECTION
690  pcSlice       ->setSliceQpBase         ( m_piRdPicQp    [uiQpIdxBest] );
691#endif
[1029]692  setUpLambda(pcSlice, m_pdRdPicLambda[uiQpIdxBest], m_piRdPicQp    [uiQpIdxBest]);
[313]693}
694
[1029]695Void TEncSlice::calCostSliceI(TComPic* pcPic)
[313]696{
[1029]697  UInt    ctuRsAddr;
698  UInt    startCtuTsAddr;
699  UInt    boundingCtuTsAddr;
700  Int     iSumHad, shift = g_bitDepth[CHANNEL_TYPE_LUMA]-8, offset = (shift>0)?(1<<(shift-1)):0;;
[313]701  Double  iSumHadSlice = 0;
702
[1029]703  pcPic->getSlice(getSliceIdx())->setSliceSegmentBits(0);
704  TComSlice* pcSlice            = pcPic->getSlice(getSliceIdx());
705  xDetermineStartAndBoundingCtuTsAddr ( startCtuTsAddr, boundingCtuTsAddr, pcPic, false );
[313]706
[1029]707  UInt ctuTsAddr;
708  ctuRsAddr = pcPic->getPicSym()->getCtuTsToRsAddrMap( startCtuTsAddr);
709  for( ctuTsAddr = startCtuTsAddr; ctuTsAddr < boundingCtuTsAddr; ctuRsAddr = pcPic->getPicSym()->getCtuTsToRsAddrMap(++ctuTsAddr) )
[313]710  {
711    // initialize CU encoder
[1029]712    TComDataCU* pCtu = pcPic->getCtu( ctuRsAddr );
713    pCtu->initCtu( pcPic, ctuRsAddr );
[313]714
[1203]715#if SVC_EXTENSION
[1029]716    Int height  = min( pcSlice->getSPS()->getMaxCUHeight(),pcSlice->getPicHeightInLumaSamples() - ctuRsAddr / pcPic->getFrameWidthInCtus() * pcSlice->getSPS()->getMaxCUHeight() );
717    Int width   = min( pcSlice->getSPS()->getMaxCUWidth(),pcSlice->getPicWidthInLumaSamples() - ctuRsAddr % pcPic->getFrameWidthInCtus() * pcSlice->getSPS()->getMaxCUWidth() );
[442]718#else
[1029]719    Int height  = min( pcSlice->getSPS()->getMaxCUHeight(),pcSlice->getSPS()->getPicHeightInLumaSamples() - ctuRsAddr / pcPic->getFrameWidthInCtus() * pcSlice->getSPS()->getMaxCUHeight() );
720    Int width   = min( pcSlice->getSPS()->getMaxCUWidth(),pcSlice->getSPS()->getPicWidthInLumaSamples() - ctuRsAddr % pcPic->getFrameWidthInCtus() * pcSlice->getSPS()->getMaxCUWidth() );
[442]721#endif
[313]722
[1029]723    iSumHad = m_pcCuEncoder->updateCtuDataISlice(pCtu, width, height);
[313]724
[1029]725    (m_pcRateCtrl->getRCPic()->getLCU(ctuRsAddr)).m_costIntra=(iSumHad+offset)>>shift;
726    iSumHadSlice += (m_pcRateCtrl->getRCPic()->getLCU(ctuRsAddr)).m_costIntra;
[313]727
728  }
729  m_pcRateCtrl->getRCPic()->setTotalIntraCost(iSumHadSlice);
730}
731
[1029]732/** \param rpcPic   picture class
733 */
734Void TEncSlice::compressSlice( TComPic* pcPic )
[313]735{
[1029]736  UInt   startCtuTsAddr;
737  UInt   boundingCtuTsAddr;
738  TComSlice* pcSlice            = pcPic->getSlice(getSliceIdx());
739  pcSlice->setSliceSegmentBits(0);
740  xDetermineStartAndBoundingCtuTsAddr ( startCtuTsAddr, boundingCtuTsAddr, pcPic, false );
[494]741
[1029]742  // initialize cost values - these are used by precompressSlice (they should be parameters).
[313]743  m_uiPicTotalBits  = 0;
[1029]744  m_dPicRdCost      = 0; // NOTE: This is a write-only variable!
[313]745  m_uiPicDist       = 0;
[494]746
[1029]747  m_pcEntropyCoder->setEntropyCoder   ( m_pppcRDSbacCoder[0][CI_CURR_BEST], pcSlice );
[595]748  m_pcEntropyCoder->resetEntropy      ();
[1029]749
750  TEncBinCABAC* pRDSbacCoder = (TEncBinCABAC *) m_pppcRDSbacCoder[0][CI_CURR_BEST]->getEncBinIf();
751  pRDSbacCoder->setBinCountingEnableFlag( false );
752  pRDSbacCoder->setBinsCoded( 0 );
753
754  TComBitCounter  tempBitCounter;
755  const UInt      frameWidthInCtus = pcPic->getPicSym()->getFrameWidthInCtus();
756
[313]757  //------------------------------------------------------------------------------
758  //  Weighted Prediction parameters estimation.
759  //------------------------------------------------------------------------------
760  // calculate AC/DC values for current picture
761  if( pcSlice->getPPS()->getUseWP() || pcSlice->getPPS()->getWPBiPred() )
762  {
763    xCalcACDCParamSlice(pcSlice);
764  }
[1201]765#if SVC_EXTENSION
[1057]766  else if( m_ppcTEncTop[pcSlice->getLayerIdx()]->getInterLayerWeightedPredFlag() )
[494]767  {
768    // Calculate for the base layer to be used in EL as Inter layer reference
769    estimateILWpParam( pcSlice );   
770  }
771#endif
[313]772
[1029]773  const Bool bWp_explicit = (pcSlice->getSliceType()==P_SLICE && pcSlice->getPPS()->getUseWP()) || (pcSlice->getSliceType()==B_SLICE && pcSlice->getPPS()->getWPBiPred());
[313]774
775  if ( bWp_explicit )
776  {
777    //------------------------------------------------------------------------------
778    //  Weighted Prediction implemented at Slice level. SliceMode=2 is not supported yet.
779    //------------------------------------------------------------------------------
[1029]780    if ( pcSlice->getSliceMode()==FIXED_NUMBER_OF_BYTES || pcSlice->getSliceSegmentMode()==FIXED_NUMBER_OF_BYTES )
[313]781    {
782      printf("Weighted Prediction is not supported with slice mode determined by max number of bins.\n"); exit(0);
783    }
784
785    xEstimateWPParamSlice( pcSlice );
[1235]786    pcSlice->initWpScaling(pcSlice->getSPS());
[313]787
788    // check WP on/off
789    xCheckWPEnable( pcSlice );
790  }
791
792#if ADAPTIVE_QP_SELECTION
[1029]793  if( m_pcCfg->getUseAdaptQpSelect() && !(pcSlice->getDependentSliceSegmentFlag()))
[313]794  {
[1029]795    // TODO: this won't work with dependent slices: they do not have their own QP. Check fix to mask clause execution with && !(pcSlice->getDependentSliceSegmentFlag())
[313]796    m_pcTrQuant->clearSliceARLCnt();
797    if(pcSlice->getSliceType()!=I_SLICE)
798    {
799      Int qpBase = pcSlice->getSliceQpBase();
800      pcSlice->setSliceQp(qpBase + m_pcTrQuant->getQpDelta(qpBase));
801    }
802  }
803#endif
804
805
[595]806
[1029]807  // Adjust initial state if this is the start of a dependent slice.
[313]808  {
[1029]809    const UInt      ctuRsAddr               = pcPic->getPicSym()->getCtuTsToRsAddrMap( startCtuTsAddr);
810    const UInt      currentTileIdx          = pcPic->getPicSym()->getTileIdxMap(ctuRsAddr);
811    const TComTile *pCurrentTile            = pcPic->getPicSym()->getTComTile(currentTileIdx);
812    const UInt      firstCtuRsAddrOfTile    = pCurrentTile->getFirstCtuRsAddr();
813    if( pcSlice->getDependentSliceSegmentFlag() && ctuRsAddr != firstCtuRsAddrOfTile )
[313]814    {
[1029]815      // This will only occur if dependent slice-segments (m_entropyCodingSyncContextState=true) are being used.
816      if( pCurrentTile->getTileWidthInCtus() >= 2 || !m_pcCfg->getWaveFrontsynchro() )
[313]817      {
[1029]818        m_pppcRDSbacCoder[0][CI_CURR_BEST]->loadContexts( &m_lastSliceSegmentEndContextState );
[313]819      }
820    }
821  }
[1029]822
823  // for every CTU in the slice segment (may terminate sooner if there is a byte limit on the slice-segment)
824
825  for( UInt ctuTsAddr = startCtuTsAddr; ctuTsAddr < boundingCtuTsAddr; ++ctuTsAddr )
[313]826  {
[1029]827    const UInt ctuRsAddr = pcPic->getPicSym()->getCtuTsToRsAddrMap(ctuTsAddr);
828    // initialize CTU encoder
829    TComDataCU* pCtu = pcPic->getCtu( ctuRsAddr );
830    pCtu->initCtu( pcPic, ctuRsAddr );
[313]831
[1029]832    // update CABAC state
833    const UInt firstCtuRsAddrOfTile = pcPic->getPicSym()->getTComTile(pcPic->getPicSym()->getTileIdxMap(ctuRsAddr))->getFirstCtuRsAddr();
834    const UInt tileXPosInCtus = firstCtuRsAddrOfTile % frameWidthInCtus;
835    const UInt ctuXPosInCtus  = ctuRsAddr % frameWidthInCtus;
836   
837    if (ctuRsAddr == firstCtuRsAddrOfTile)
[313]838    {
[1029]839      m_pppcRDSbacCoder[0][CI_CURR_BEST]->resetEntropy();
[595]840    }
[1029]841    else if ( ctuXPosInCtus == tileXPosInCtus && m_pcCfg->getWaveFrontsynchro())
[595]842    {
[1029]843      // reset and then update contexts to the state at the end of the top-right CTU (if within current slice and tile).
844      m_pppcRDSbacCoder[0][CI_CURR_BEST]->resetEntropy();
845      // Sync if the Top-Right is available.
846      TComDataCU *pCtuUp = pCtu->getCtuAbove();
847      if ( pCtuUp && ((ctuRsAddr%frameWidthInCtus+1) < frameWidthInCtus)  )
[313]848      {
[1029]849        TComDataCU *pCtuTR = pcPic->getCtu( ctuRsAddr - frameWidthInCtus + 1 );
850        if ( pCtu->CUIsFromSameSliceAndTile(pCtuTR) )
851        {
852          // Top-Right is available, we use it.
853          m_pppcRDSbacCoder[0][CI_CURR_BEST]->loadContexts( &m_entropyCodingSyncContextState );
854        }
[313]855      }
856    }
857
[1029]858    // set go-on entropy coder (used for all trial encodings - the cu encoder and encoder search also have a copy of the same pointer)
[595]859    m_pcEntropyCoder->setEntropyCoder ( m_pcRDGoOnSbacCoder, pcSlice );
[1029]860    m_pcEntropyCoder->setBitstream( &tempBitCounter );
861    tempBitCounter.resetBits();
862    m_pcRDGoOnSbacCoder->load( m_pppcRDSbacCoder[0][CI_CURR_BEST] ); // this copy is not strictly necessary here, but indicates that the GoOnSbacCoder
863                                                                     // is reset to a known state before every decision process.
[595]864
865    ((TEncBinCABAC*)m_pcRDGoOnSbacCoder->getEncBinIf())->setBinCountingEnableFlag(true);
866
867    Double oldLambda = m_pcRdCost->getLambda();
868    if ( m_pcCfg->getUseRateCtrl() )
[313]869    {
[595]870      Int estQP        = pcSlice->getSliceQp();
871      Double estLambda = -1.0;
872      Double bpp       = -1.0;
[494]873
[1029]874      if ( ( pcPic->getSlice( 0 )->getSliceType() == I_SLICE && m_pcCfg->getForceIntraQP() ) || !m_pcCfg->getLCULevelRC() )
[313]875      {
[595]876        estQP = pcSlice->getSliceQp();
877      }
878      else
879      {
880        bpp = m_pcRateCtrl->getRCPic()->getLCUTargetBpp(pcSlice->getSliceType());
[1029]881        if ( pcPic->getSlice( 0 )->getSliceType() == I_SLICE)
[313]882        {
[595]883          estLambda = m_pcRateCtrl->getRCPic()->getLCUEstLambdaAndQP(bpp, pcSlice->getSliceQp(), &estQP);
[313]884        }
885        else
886        {
[595]887          estLambda = m_pcRateCtrl->getRCPic()->getLCUEstLambda( bpp );
888          estQP     = m_pcRateCtrl->getRCPic()->getLCUEstQP    ( estLambda, pcSlice->getSliceQp() );
889        }
[313]890
[1203]891#if SVC_EXTENSION
[1029]892        estQP     = Clip3( -pcSlice->getQpBDOffsetY(), MAX_QP, estQP );
[442]893#else
[1029]894        estQP     = Clip3( -pcSlice->getSPS()->getQpBDOffset(CHANNEL_TYPE_LUMA), MAX_QP, estQP );
[442]895#endif
[313]896
[1029]897        m_pcRdCost->setLambda(estLambda);
898
[313]899#if RDOQ_CHROMA_LAMBDA
[1029]900        // set lambda for RDOQ
901        const Double chromaLambda = estLambda / m_pcRdCost->getChromaWeight();
902        const Double lambdaArray[MAX_NUM_COMPONENT] = { estLambda, chromaLambda, chromaLambda };
903        m_pcTrQuant->setLambdas( lambdaArray );
[313]904#else
[1029]905        m_pcTrQuant->setLambda( estLambda );
[313]906#endif
907      }
908
[595]909      m_pcRateCtrl->setRCQP( estQP );
[713]910#if ADAPTIVE_QP_SELECTION
[1029]911      pCtu->getSlice()->setSliceQpBase( estQP );
[713]912#endif
[595]913    }
[313]914
[1029]915    // run CTU trial encoder
916    m_pcCuEncoder->compressCtu( pCtu );
[313]917
[1029]918
919    // All CTU decisions have now been made. Restore entropy coder to an initial stage, ready to make a true encode,
920    // which will result in the state of the contexts being correct. It will also count up the number of bits coded,
921    // which is used if there is a limit of the number of bytes per slice-segment.
922
[595]923    m_pcEntropyCoder->setEntropyCoder ( m_pppcRDSbacCoder[0][CI_CURR_BEST], pcSlice );
[1029]924    m_pcEntropyCoder->setBitstream( &tempBitCounter );
925    pRDSbacCoder->setBinCountingEnableFlag( true );
926    m_pppcRDSbacCoder[0][CI_CURR_BEST]->resetBits();
927    pRDSbacCoder->setBinsCoded( 0 );
[494]928
[1029]929    // encode CTU and calculate the true bit counters.
930    m_pcCuEncoder->encodeCtu( pCtu );
931
932
933    pRDSbacCoder->setBinCountingEnableFlag( false );
934
935    const Int numberOfWrittenBits = m_pcEntropyCoder->getNumberOfWrittenBits();
936
937    // Calculate if this CTU puts us over slice bit size.
938    // cannot terminate if current slice/slice-segment would be 0 Ctu in size,
939    const UInt validEndOfSliceCtuTsAddr = ctuTsAddr + (ctuTsAddr == startCtuTsAddr ? 1 : 0);
940    // Set slice end parameter
941    if(pcSlice->getSliceMode()==FIXED_NUMBER_OF_BYTES && pcSlice->getSliceBits()+numberOfWrittenBits > (pcSlice->getSliceArgument()<<3))
[595]942    {
[1029]943      pcSlice->setSliceSegmentCurEndCtuTsAddr(validEndOfSliceCtuTsAddr);
944      pcSlice->setSliceCurEndCtuTsAddr(validEndOfSliceCtuTsAddr);
945      boundingCtuTsAddr=validEndOfSliceCtuTsAddr;
[595]946    }
[1029]947    else if(pcSlice->getSliceSegmentMode()==FIXED_NUMBER_OF_BYTES && pcSlice->getSliceSegmentBits()+numberOfWrittenBits > (pcSlice->getSliceSegmentArgument()<<3))
[595]948    {
[1029]949      pcSlice->setSliceSegmentCurEndCtuTsAddr(validEndOfSliceCtuTsAddr);
950      boundingCtuTsAddr=validEndOfSliceCtuTsAddr;
[595]951    }
[313]952
[1029]953    if (boundingCtuTsAddr <= ctuTsAddr)
954      break;
955
956    pcSlice->setSliceBits( (UInt)(pcSlice->getSliceBits() + numberOfWrittenBits) );
957    pcSlice->setSliceSegmentBits(pcSlice->getSliceSegmentBits()+numberOfWrittenBits);
958
959    // Store probabilities of second CTU in line into buffer - used only if wavefront-parallel-processing is enabled.
960    if ( ctuXPosInCtus == tileXPosInCtus+1 && m_pcCfg->getWaveFrontsynchro())
[595]961    {
[1029]962      m_entropyCodingSyncContextState.loadContexts(m_pppcRDSbacCoder[0][CI_CURR_BEST]);
[595]963    }
[313]964
[1029]965
[595]966    if ( m_pcCfg->getUseRateCtrl() )
967    {
968      Int actualQP        = g_RCInvalidQPValue;
969      Double actualLambda = m_pcRdCost->getLambda();
[1029]970      Int actualBits      = pCtu->getTotalBits();
[595]971      Int numberOfEffectivePixels    = 0;
[1029]972      for ( Int idx = 0; idx < pcPic->getNumPartitionsInCtu(); idx++ )
[595]973      {
[1029]974        if ( pCtu->getPredictionMode( idx ) != NUMBER_OF_PREDICTION_MODES && ( !pCtu->isSkipped( idx ) ) )
[313]975        {
[595]976          numberOfEffectivePixels = numberOfEffectivePixels + 16;
977          break;
[313]978        }
[595]979      }
[313]980
[595]981      if ( numberOfEffectivePixels == 0 )
[313]982      {
[595]983        actualQP = g_RCInvalidQPValue;
[313]984      }
[595]985      else
[313]986      {
[1029]987        actualQP = pCtu->getQP( 0 );
[313]988      }
[595]989      m_pcRdCost->setLambda(oldLambda);
[1029]990      m_pcRateCtrl->getRCPic()->updateAfterCTU( m_pcRateCtrl->getRCPic()->getLCUCoded(), actualBits, actualQP, actualLambda,
991                                                pCtu->getSlice()->getSliceType() == I_SLICE ? 0 : m_pcCfg->getLCULevelRC() );
[313]992    }
[494]993
[1029]994    m_uiPicTotalBits += pCtu->getTotalBits();
995    m_dPicRdCost     += pCtu->getTotalCost();
996    m_uiPicDist      += pCtu->getTotalDistortion();
[313]997  }
[1029]998
999  // store context state at the end of this slice-segment, in case the next slice is a dependent slice and continues using the CABAC contexts.
1000  if( pcSlice->getPPS()->getDependentSliceSegmentsEnabledFlag() )
[313]1001  {
[1029]1002    m_lastSliceSegmentEndContextState.loadContexts( m_pppcRDSbacCoder[0][CI_CURR_BEST] );//ctx end of dep.slice
[313]1003  }
[1029]1004
1005  // stop use of temporary bit counter object.
1006  m_pppcRDSbacCoder[0][CI_CURR_BEST]->setBitstream(NULL);
1007  m_pcRDGoOnSbacCoder->setBitstream(NULL); // stop use of tempBitCounter.
[1235]1008
1009  // TODO: optimise cabac_init during compress slice to improve multi-slice operation
1010  //if (pcSlice->getPPS()->getCabacInitPresentFlag() && !pcSlice->getPPS()->getDependentSliceSegmentsEnabledFlag())
1011  //{
1012  //  m_encCABACTableIdx = m_pcEntropyCoder->determineCabacInitIdx();
1013  //}
1014  //else
1015  //{
1016  //  m_encCABACTableIdx = pcSlice->getSliceType();
1017  //}
[313]1018}
1019
1020/**
1021 \param  rpcPic        picture class
1022 \retval rpcBitstream  bitstream class
1023 */
[1029]1024Void TEncSlice::encodeSlice   ( TComPic* pcPic, TComOutputBitstream* pcSubstreams, UInt &numBinsCoded )
[313]1025{
[1029]1026  TComSlice* pcSlice                 = pcPic->getSlice(getSliceIdx());
[313]1027
[1029]1028  const UInt startCtuTsAddr          = pcSlice->getSliceSegmentCurStartCtuTsAddr();
1029  const UInt boundingCtuTsAddr       = pcSlice->getSliceSegmentCurEndCtuTsAddr();
[494]1030
[1029]1031  const UInt frameWidthInCtus        = pcPic->getPicSym()->getFrameWidthInCtus();
1032  const Bool depSliceSegmentsEnabled = pcSlice->getPPS()->getDependentSliceSegmentsEnabledFlag();
1033  const Bool wavefrontsEnabled       = pcSlice->getPPS()->getEntropyCodingSyncEnabledFlag();
1034
1035  // initialise entropy coder for the slice
1036  m_pcSbacCoder->init( (TEncBinIf*)m_pcBinCABAC );
1037  m_pcEntropyCoder->setEntropyCoder ( m_pcSbacCoder, pcSlice );
1038  m_pcEntropyCoder->resetEntropy      ();
1039
1040  numBinsCoded = 0;
1041  m_pcBinCABAC->setBinCountingEnableFlag( true );
1042  m_pcBinCABAC->setBinsCoded(0);
1043
[313]1044#if ENC_DEC_TRACE
1045  g_bJustDoIt = g_bEncDecTraceEnable;
1046#endif
1047  DTRACE_CABAC_VL( g_nSymbolCounter++ );
1048  DTRACE_CABAC_T( "\tPOC: " );
[1029]1049  DTRACE_CABAC_V( pcPic->getPOC() );
[313]1050  DTRACE_CABAC_T( "\n" );
1051#if ENC_DEC_TRACE
1052  g_bJustDoIt = g_bEncDecTraceDisable;
1053#endif
1054
[1029]1055
1056  if (depSliceSegmentsEnabled)
[313]1057  {
[1029]1058    // modify initial contexts with previous slice segment if this is a dependent slice.
1059    const UInt ctuRsAddr        = pcPic->getPicSym()->getCtuTsToRsAddrMap( startCtuTsAddr );
1060    const UInt currentTileIdx=pcPic->getPicSym()->getTileIdxMap(ctuRsAddr);
1061    const TComTile *pCurrentTile=pcPic->getPicSym()->getTComTile(currentTileIdx);
1062    const UInt firstCtuRsAddrOfTile = pCurrentTile->getFirstCtuRsAddr();
[494]1063
[1029]1064    if( pcSlice->getDependentSliceSegmentFlag() && ctuRsAddr != firstCtuRsAddrOfTile )
[313]1065    {
[1029]1066      if( pCurrentTile->getTileWidthInCtus() >= 2 || !wavefrontsEnabled )
[313]1067      {
[1029]1068        m_pcSbacCoder->loadContexts(&m_lastSliceSegmentEndContextState);
[313]1069      }
1070    }
1071  }
1072
[1029]1073  // for every CTU in the slice segment...
1074
1075  for( UInt ctuTsAddr = startCtuTsAddr; ctuTsAddr < boundingCtuTsAddr; ++ctuTsAddr )
[313]1076  {
[1029]1077    const UInt ctuRsAddr = pcPic->getPicSym()->getCtuTsToRsAddrMap(ctuTsAddr);
1078    const TComTile &currentTile = *(pcPic->getPicSym()->getTComTile(pcPic->getPicSym()->getTileIdxMap(ctuRsAddr)));
1079    const UInt firstCtuRsAddrOfTile = currentTile.getFirstCtuRsAddr();
1080    const UInt tileXPosInCtus       = firstCtuRsAddrOfTile % frameWidthInCtus;
1081    const UInt tileYPosInCtus       = firstCtuRsAddrOfTile / frameWidthInCtus;
1082    const UInt ctuXPosInCtus        = ctuRsAddr % frameWidthInCtus;
1083    const UInt ctuYPosInCtus        = ctuRsAddr / frameWidthInCtus;
1084    const UInt uiSubStrm=pcPic->getSubstreamForCtuAddr(ctuRsAddr, true, pcSlice);
1085    TComDataCU* pCtu = pcPic->getCtu( ctuRsAddr );
[595]1086
1087    m_pcEntropyCoder->setBitstream( &pcSubstreams[uiSubStrm] );
[1029]1088
1089    // set up CABAC contexts' state for this CTU
1090    if (ctuRsAddr == firstCtuRsAddrOfTile)
[595]1091    {
[1029]1092      if (ctuTsAddr != startCtuTsAddr) // if it is the first CTU, then the entropy coder has already been reset
[313]1093      {
[1029]1094        m_pcEntropyCoder->resetEntropy();
[313]1095      }
1096    }
[1029]1097    else if (ctuXPosInCtus == tileXPosInCtus && wavefrontsEnabled)
[313]1098    {
[1029]1099      // Synchronize cabac probabilities with upper-right CTU if it's available and at the start of a line.
1100      if (ctuTsAddr != startCtuTsAddr) // if it is the first CTU, then the entropy coder has already been reset
[313]1101      {
[1029]1102        m_pcEntropyCoder->resetEntropy();
[313]1103      }
[1029]1104      TComDataCU *pCtuUp = pCtu->getCtuAbove();
1105      if ( pCtuUp && ((ctuRsAddr%frameWidthInCtus+1) < frameWidthInCtus)  )
[313]1106      {
[1029]1107        TComDataCU *pCtuTR = pcPic->getCtu( ctuRsAddr - frameWidthInCtus + 1 );
1108        if ( pCtu->CUIsFromSameSliceAndTile(pCtuTR) )
[313]1109        {
[1029]1110          // Top-right is available, so use it.
1111          m_pcSbacCoder->loadContexts( &m_entropyCodingSyncContextState );
[313]1112        }
1113      }
1114    }
1115
[1029]1116
[540]1117    if ( pcSlice->getSPS()->getUseSAO() )
1118    {
[1029]1119      Bool bIsSAOSliceEnabled = false;
1120      Bool sliceEnabled[MAX_NUM_COMPONENT];
1121      for(Int comp=0; comp < MAX_NUM_COMPONENT; comp++)
[540]1122      {
[1029]1123        ComponentID compId=ComponentID(comp);
1124        sliceEnabled[compId] = pcSlice->getSaoEnabledFlag(toChannelType(compId)) && (comp < pcPic->getNumberValidComponents());
1125        if (sliceEnabled[compId]) bIsSAOSliceEnabled=true;
1126      }
1127      if (bIsSAOSliceEnabled)
1128      {
1129        SAOBlkParam& saoblkParam = (pcPic->getPicSym()->getSAOBlkParam())[ctuRsAddr];
[540]1130
1131        Bool leftMergeAvail = false;
1132        Bool aboveMergeAvail= false;
1133        //merge left condition
[1029]1134        Int rx = (ctuRsAddr % frameWidthInCtus);
[540]1135        if(rx > 0)
1136        {
[1029]1137          leftMergeAvail = pcPic->getSAOMergeAvailability(ctuRsAddr, ctuRsAddr-1);
[540]1138        }
1139
1140        //merge up condition
[1029]1141        Int ry = (ctuRsAddr / frameWidthInCtus);
[540]1142        if(ry > 0)
1143        {
[1029]1144          aboveMergeAvail = pcPic->getSAOMergeAvailability(ctuRsAddr, ctuRsAddr-frameWidthInCtus);
[540]1145        }
1146
[644]1147#if SVC_EXTENSION
[1057]1148        m_pcEntropyCoder->encodeSAOBlkParam(saoblkParam, m_ppcTEncTop[pcSlice->getLayerIdx()]->getSAO()->getSaoMaxOffsetQVal(), sliceEnabled, leftMergeAvail, aboveMergeAvail);
[644]1149#else
[1029]1150        m_pcEntropyCoder->encodeSAOBlkParam(saoblkParam, sliceEnabled, leftMergeAvail, aboveMergeAvail);
[644]1151#endif
[540]1152      }
1153    }
1154
[313]1155#if ENC_DEC_TRACE
1156    g_bJustDoIt = g_bEncDecTraceEnable;
1157#endif
[1029]1158      m_pcCuEncoder->encodeCtu( pCtu );
[313]1159#if ENC_DEC_TRACE
1160    g_bJustDoIt = g_bEncDecTraceDisable;
[494]1161#endif
[1029]1162
1163    //Store probabilities of second CTU in line into buffer
1164    if ( ctuXPosInCtus == tileXPosInCtus+1 && wavefrontsEnabled)
[313]1165    {
[1029]1166      m_entropyCodingSyncContextState.loadContexts( m_pcSbacCoder );
[313]1167    }
[1029]1168
1169    // terminate the sub-stream, if required (end of slice-segment, end of tile, end of wavefront-CTU-row):
1170    if (ctuTsAddr+1 == boundingCtuTsAddr ||
1171         (  ctuXPosInCtus + 1 == tileXPosInCtus + currentTile.getTileWidthInCtus() &&
1172          ( ctuYPosInCtus + 1 == tileYPosInCtus + currentTile.getTileHeightInCtus() || wavefrontsEnabled)
1173         )
1174       )
1175    {
1176      m_pcEntropyCoder->encodeTerminatingBit(1);
1177      m_pcEntropyCoder->encodeSliceFinish();
1178      // Byte-alignment in slice_data() when new tile
1179      pcSubstreams[uiSubStrm].writeByteAlignment();
1180
1181      // write sub-stream size
1182      if (ctuTsAddr+1 != boundingCtuTsAddr)
1183      {
1184        pcSlice->addSubstreamSize( (pcSubstreams[uiSubStrm].getNumberOfWrittenBits() >> 3) + pcSubstreams[uiSubStrm].countStartCodeEmulations() );
1185      }
1186    }
1187  } // CTU-loop
1188
[313]1189  if( depSliceSegmentsEnabled )
1190  {
[1029]1191    m_lastSliceSegmentEndContextState.loadContexts( m_pcSbacCoder );//ctx end of dep.slice
[313]1192  }
[1029]1193
[313]1194#if ADAPTIVE_QP_SELECTION
1195  if( m_pcCfg->getUseAdaptQpSelect() )
1196  {
1197    m_pcTrQuant->storeSliceQpNext(pcSlice);
1198  }
1199#endif
[1029]1200
[1235]1201  if (pcSlice->getPPS()->getCabacInitPresentFlag() && !pcSlice->getPPS()->getDependentSliceSegmentsEnabledFlag())
[313]1202  {
[1235]1203    m_encCABACTableIdx = m_pcEntropyCoder->determineCabacInitIdx();
[313]1204  }
[1235]1205  else
1206  {
1207    m_encCABACTableIdx = pcSlice->getSliceType();
1208  }
1209 
[1029]1210  numBinsCoded = m_pcBinCABAC->getBinsCoded();
[313]1211}
1212
[1029]1213Void TEncSlice::calculateBoundingCtuTsAddrForSlice(UInt &startCtuTSAddrSlice, UInt &boundingCtuTSAddrSlice, Bool &haveReachedTileBoundary,
1214                                                   TComPic* pcPic, const Bool encodingSlice, const Int sliceMode, const Int sliceArgument, const UInt sliceCurEndCtuTSAddr)
[313]1215{
[1029]1216  TComSlice* pcSlice = pcPic->getSlice(getSliceIdx());
1217  const UInt numberOfCtusInFrame = pcPic->getNumberOfCtusInFrame();
1218  boundingCtuTSAddrSlice=0;
1219  haveReachedTileBoundary=false;
[313]1220
[1029]1221  switch (sliceMode)
[313]1222  {
[1029]1223    case FIXED_NUMBER_OF_CTU:
1224      {
1225        UInt ctuAddrIncrement    = sliceArgument;
1226        boundingCtuTSAddrSlice  = ((startCtuTSAddrSlice + ctuAddrIncrement) < numberOfCtusInFrame) ? (startCtuTSAddrSlice + ctuAddrIncrement) : numberOfCtusInFrame;
1227      }
[313]1228      break;
1229    case FIXED_NUMBER_OF_BYTES:
[1029]1230      if (encodingSlice)
1231        boundingCtuTSAddrSlice  = sliceCurEndCtuTSAddr;
1232      else
1233        boundingCtuTSAddrSlice  = numberOfCtusInFrame;
[313]1234      break;
1235    case FIXED_NUMBER_OF_TILES:
[1029]1236      {
1237        const UInt tileIdx        = pcPic->getPicSym()->getTileIdxMap( pcPic->getPicSym()->getCtuTsToRsAddrMap(startCtuTSAddrSlice) );
1238        const UInt tileTotalCount = (pcPic->getPicSym()->getNumTileColumnsMinus1()+1) * (pcPic->getPicSym()->getNumTileRowsMinus1()+1);
1239        UInt ctuAddrIncrement   = 0;
[313]1240
[1029]1241        for(UInt tileIdxIncrement = 0; tileIdxIncrement < sliceArgument; tileIdxIncrement++)
[313]1242        {
[1029]1243          if((tileIdx + tileIdxIncrement) < tileTotalCount)
1244          {
1245            UInt tileWidthInCtus   = pcPic->getPicSym()->getTComTile(tileIdx + tileIdxIncrement)->getTileWidthInCtus();
1246            UInt tileHeightInCtus  = pcPic->getPicSym()->getTComTile(tileIdx + tileIdxIncrement)->getTileHeightInCtus();
1247            ctuAddrIncrement    += (tileWidthInCtus * tileHeightInCtus);
1248          }
[313]1249        }
1250
[1029]1251        boundingCtuTSAddrSlice  = ((startCtuTSAddrSlice + ctuAddrIncrement) < numberOfCtusInFrame) ? (startCtuTSAddrSlice + ctuAddrIncrement) : numberOfCtusInFrame;
[313]1252      }
1253      break;
1254    default:
[1029]1255      boundingCtuTSAddrSlice    = numberOfCtusInFrame;
[313]1256      break;
1257  }
1258
[1029]1259  // Adjust for tiles and wavefronts.
1260  if ((sliceMode == FIXED_NUMBER_OF_CTU || sliceMode == FIXED_NUMBER_OF_BYTES) &&
[313]1261      (m_pcCfg->getNumRowsMinus1() > 0 || m_pcCfg->getNumColumnsMinus1() > 0))
1262  {
[1029]1263    const UInt ctuRSAddr                  = pcPic->getPicSym()->getCtuTsToRsAddrMap(startCtuTSAddrSlice);
1264    const UInt startTileIdx               = pcPic->getPicSym()->getTileIdxMap(ctuRSAddr);
1265    const Bool wavefrontsAreEnabled       = m_pcCfg->getWaveFrontsynchro();
[494]1266
[1029]1267    const TComTile *pStartingTile         = pcPic->getPicSym()->getTComTile(startTileIdx);
1268    const UInt tileStartTsAddr            = pcPic->getPicSym()->getCtuRsToTsAddrMap(pStartingTile->getFirstCtuRsAddr());
1269    const UInt tileStartWidth             = pStartingTile->getTileWidthInCtus();
1270    const UInt tileStartHeight            = pStartingTile->getTileHeightInCtus();
1271    const UInt tileLastTsAddr_excl        = tileStartTsAddr + tileStartWidth*tileStartHeight;
1272    const UInt tileBoundingCtuTsAddrSlice = tileLastTsAddr_excl;
[313]1273
[1029]1274    const UInt ctuColumnOfStartingTile    = ((startCtuTSAddrSlice-tileStartTsAddr)%tileStartWidth);
1275    if (wavefrontsAreEnabled && ctuColumnOfStartingTile!=0)
[313]1276    {
[1029]1277      // WPP: if a slice does not start at the beginning of a CTB row, it must end within the same CTB row
1278      const UInt numberOfCTUsToEndOfRow            = tileStartWidth - ctuColumnOfStartingTile;
1279      const UInt wavefrontTileBoundingCtuAddrSlice = startCtuTSAddrSlice + numberOfCTUsToEndOfRow;
1280      if (wavefrontTileBoundingCtuAddrSlice < boundingCtuTSAddrSlice)
[313]1281      {
[1029]1282        boundingCtuTSAddrSlice = wavefrontTileBoundingCtuAddrSlice;
[313]1283      }
[494]1284    }
[313]1285
[1029]1286    if (tileBoundingCtuTsAddrSlice < boundingCtuTSAddrSlice)
[313]1287    {
[1029]1288      boundingCtuTSAddrSlice = tileBoundingCtuTsAddrSlice;
1289      haveReachedTileBoundary = true;
[313]1290    }
1291  }
[1029]1292  else if ((sliceMode == FIXED_NUMBER_OF_CTU || sliceMode == FIXED_NUMBER_OF_BYTES) && pcSlice->getPPS()->getEntropyCodingSyncEnabledFlag() && ((startCtuTSAddrSlice % pcPic->getFrameWidthInCtus()) != 0))
[313]1293  {
[1029]1294    // Adjust for wavefronts (no tiles).
1295    // WPP: if a slice does not start at the beginning of a CTB row, it must end within the same CTB row
1296    boundingCtuTSAddrSlice = min(boundingCtuTSAddrSlice, startCtuTSAddrSlice - (startCtuTSAddrSlice % pcPic->getFrameWidthInCtus()) + (pcPic->getFrameWidthInCtus()));
[313]1297  }
[1029]1298}
[313]1299
[1029]1300/** Determines the starting and bounding CTU address of current slice / dependent slice
1301 * \param bEncodeSlice Identifies if the calling function is compressSlice() [false] or encodeSlice() [true]
1302 * \returns Updates startCtuTsAddr, boundingCtuTsAddr with appropriate CTU address
1303 */
1304Void TEncSlice::xDetermineStartAndBoundingCtuTsAddr  ( UInt& startCtuTsAddr, UInt& boundingCtuTsAddr, TComPic* pcPic, const Bool encodingSlice )
1305{
1306  TComSlice* pcSlice                 = pcPic->getSlice(getSliceIdx());
[313]1307
[1029]1308  // Non-dependent slice
1309  UInt startCtuTsAddrSlice           = pcSlice->getSliceCurStartCtuTsAddr();
1310  Bool haveReachedTileBoundarySlice  = false;
1311  UInt boundingCtuTsAddrSlice;
1312  calculateBoundingCtuTsAddrForSlice(startCtuTsAddrSlice, boundingCtuTsAddrSlice, haveReachedTileBoundarySlice, pcPic,
1313                                     encodingSlice, m_pcCfg->getSliceMode(), m_pcCfg->getSliceArgument(), pcSlice->getSliceCurEndCtuTsAddr());
1314  pcSlice->setSliceCurEndCtuTsAddr(   boundingCtuTsAddrSlice );
1315  pcSlice->setSliceCurStartCtuTsAddr( startCtuTsAddrSlice    );
[494]1316
[1029]1317  // Dependent slice
1318  UInt startCtuTsAddrSliceSegment          = pcSlice->getSliceSegmentCurStartCtuTsAddr();
1319  Bool haveReachedTileBoundarySliceSegment = false;
1320  UInt boundingCtuTsAddrSliceSegment;
1321  calculateBoundingCtuTsAddrForSlice(startCtuTsAddrSliceSegment, boundingCtuTsAddrSliceSegment, haveReachedTileBoundarySliceSegment, pcPic,
1322                                     encodingSlice, m_pcCfg->getSliceSegmentMode(), m_pcCfg->getSliceSegmentArgument(), pcSlice->getSliceSegmentCurEndCtuTsAddr());
1323  if (boundingCtuTsAddrSliceSegment>boundingCtuTsAddrSlice)
[313]1324  {
[1029]1325    boundingCtuTsAddrSliceSegment = boundingCtuTsAddrSlice;
[313]1326  }
[1029]1327  pcSlice->setSliceSegmentCurEndCtuTsAddr( boundingCtuTsAddrSliceSegment );
1328  pcSlice->setSliceSegmentCurStartCtuTsAddr(startCtuTsAddrSliceSegment);
[494]1329
[313]1330  // Make a joint decision based on reconstruction and dependent slice bounds
[1029]1331  startCtuTsAddr    = max(startCtuTsAddrSlice   , startCtuTsAddrSliceSegment   );
1332  boundingCtuTsAddr = boundingCtuTsAddrSliceSegment;
[313]1333}
1334
1335Double TEncSlice::xGetQPValueAccordingToLambda ( Double lambda )
1336{
1337  return 4.2005*log(lambda) + 13.7122;
1338}
1339
[595]1340#if SVC_EXTENSION
[313]1341#if JCTVC_M0259_LAMBDAREFINEMENT
1342Double TEncSlice::xCalEnhLambdaFactor( Double deltaQP , Double beta )
1343{
1344  double tmp = beta * pow( 2.0 , deltaQP / 6 );
1345  double gamma = tmp / ( tmp + 1 );
1346  return( gamma );
1347}
1348#endif
[1201]1349
[494]1350Void TEncSlice::estimateILWpParam( TComSlice* pcSlice )
1351{
1352  xCalcACDCParamSlice(pcSlice);
[1029]1353  WPACDCParam * temp_weightACDCParam;
[494]1354
1355  pcSlice->getWpAcDcParam(temp_weightACDCParam);
1356  g_refWeightACDCParam = (void *) temp_weightACDCParam;
1357}
[595]1358#endif //SVC_EXTENSION
[313]1359//! \}
Note: See TracBrowser for help on using the repository browser.