/* The copyright in this software is being made available under the BSD * License, included below. This software may be subject to other third party * and contributor rights, including patent rights, and no such rights are * granted under this license. * * Copyright (c) 2010-2012, ITU/ISO/IEC * All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * * Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * * Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * * Neither the name of the ITU/ISO/IEC nor the names of its contributors may * be used to endorse or promote products derived from this software without * specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF * THE POSSIBILITY OF SUCH DAMAGE. */ /** \file TEncEntropy.cpp \brief entropy encoder class */ #include "TEncEntropy.h" #include "TLibCommon/TypeDef.h" #include "TLibCommon/TComAdaptiveLoopFilter.h" #include "TLibCommon/TComSampleAdaptiveOffset.h" //! \ingroup TLibEncoder //! \{ Void TEncEntropy::setEntropyCoder ( TEncEntropyIf* e, TComSlice* pcSlice ) { m_pcEntropyCoderIf = e; m_pcEntropyCoderIf->setSlice ( pcSlice ); } Void TEncEntropy::encodeSliceHeader ( TComSlice* pcSlice ) { if (pcSlice->getSPS()->getUseSAO()) { pcSlice->setSaoInterleavingFlag(pcSlice->getAPS()->getSaoInterleavingFlag()); pcSlice->setSaoEnabledFlag (pcSlice->getAPS()->getSaoParam()->bSaoFlag[0]); if (pcSlice->getAPS()->getSaoInterleavingFlag()) { pcSlice->setSaoEnabledFlagCb (pcSlice->getAPS()->getSaoParam()->bSaoFlag[1]); pcSlice->setSaoEnabledFlagCr (pcSlice->getAPS()->getSaoParam()->bSaoFlag[2]); } else { pcSlice->setSaoEnabledFlagCb (0); pcSlice->setSaoEnabledFlagCr (0); } } m_pcEntropyCoderIf->codeSliceHeader( pcSlice ); return; } Void TEncEntropy::encodeTilesWPPEntryPoint( TComSlice* pSlice ) { m_pcEntropyCoderIf->codeTilesWPPEntryPoint( pSlice ); } Void TEncEntropy::encodeTerminatingBit ( UInt uiIsLast ) { m_pcEntropyCoderIf->codeTerminatingBit( uiIsLast ); return; } Void TEncEntropy::encodeSliceFinish() { m_pcEntropyCoderIf->codeSliceFinish(); } Void TEncEntropy::encodeFlush() { m_pcEntropyCoderIf->codeFlush(); } Void TEncEntropy::encodeStart() { m_pcEntropyCoderIf->encodeStart(); } Void TEncEntropy::encodeSEI(const SEI& sei) { m_pcEntropyCoderIf->codeSEI(sei); return; } Void TEncEntropy::encodePPS( TComPPS* pcPPS ) { m_pcEntropyCoderIf->codePPS( pcPPS ); return; } #if VIDYO_VPS_INTEGRATION|QC_MVHEVC_B0046 Void TEncEntropy::encodeVPS( TComVPS* pcVPS ) { m_pcEntropyCoderIf->codeVPS( pcVPS ); return; } #endif #if VIDYO_VPS_INTEGRATION|QC_MVHEVC_B0046 Void codeVPS ( TComVPS* pcVPS ); #endif #if HHI_MPI || OL_QTLIMIT_PREDCODING_B0068 Void TEncEntropy::encodeSPS( TComSPS* pcSPS, Bool bIsDepth ) { m_pcEntropyCoderIf->codeSPS( pcSPS, bIsDepth ); return; } #else Void TEncEntropy::encodeSPS( TComSPS* pcSPS ) { m_pcEntropyCoderIf->codeSPS( pcSPS ); return; } #endif Void TEncEntropy::encodeSkipFlag( TComDataCU* pcCU, UInt uiAbsPartIdx, Bool bRD ) { if ( pcCU->getSlice()->isIntra() ) { return; } if( bRD ) { uiAbsPartIdx = 0; } if( !bRD ) { if( pcCU->getLastCUSucIPCMFlag() && pcCU->getIPCMFlag(uiAbsPartIdx) ) { return; } } m_pcEntropyCoderIf->codeSkipFlag( pcCU, uiAbsPartIdx ); } #if LGE_ILLUCOMP_B0045 Void TEncEntropy::encodeICFlag( TComDataCU* pcCU, UInt uiAbsPartIdx, Bool bRD #if LGE_ILLUCOMP_DEPTH_C0046 , UInt uiDepth #endif ) { if (pcCU->isIntra(uiAbsPartIdx) || (pcCU->getSlice()->getViewId() == 0) #if !LGE_ILLUCOMP_DEPTH_C0046 || pcCU->getSlice()->getSPS()->isDepth() #endif ) { return; } if(!pcCU->getSlice()->getApplyIC()) return; if( bRD ) { uiAbsPartIdx = 0; } if(pcCU->isICFlagRequired(uiAbsPartIdx #if LGE_ILLUCOMP_DEPTH_C0046 , uiDepth //This modification is not needed after integrating JCT3V-C0137 #endif )) m_pcEntropyCoderIf->codeICFlag( pcCU, uiAbsPartIdx ); } #endif Void TEncEntropy::codeFiltCountBit(ALFParam* pAlfParam, Int64* ruiRate) { resetEntropy(); resetBits(); codeFilt(pAlfParam); *ruiRate = getNumberOfWrittenBits(); resetEntropy(); resetBits(); } Void TEncEntropy::codeAuxCountBit(ALFParam* pAlfParam, Int64* ruiRate) { resetEntropy(); resetBits(); codeAux(pAlfParam); *ruiRate = getNumberOfWrittenBits(); resetEntropy(); resetBits(); } Void TEncEntropy::codeAux(ALFParam* pAlfParam) { // m_pcEntropyCoderIf->codeAlfUvlc(pAlfParam->realfiltNo); Int noFilters = min(pAlfParam->filters_per_group-1, 2); m_pcEntropyCoderIf->codeAlfUvlc(noFilters); if(noFilters == 1) { m_pcEntropyCoderIf->codeAlfUvlc(pAlfParam->startSecondFilter); } else if (noFilters == 2) { Int numMergeFlags = 16; for (Int i=1; icodeAlfFlag (pAlfParam->filterPattern[i]); } } } Int TEncEntropy::lengthGolomb(int coeffVal, int k) { int m = 2 << (k - 1); int q = coeffVal / m; if(coeffVal != 0) { return(q + 2 + k); } else { return(q + 1 + k); } } Int TEncEntropy::codeFilterCoeff(ALFParam* ALFp) { Int filters_per_group = ALFp->filters_per_group; int sqrFiltLength = ALFp->num_coeff; int i, k, kMin, kStart, minBits, ind, scanPos, maxScanVal, coeffVal, len = 0, *pDepthInt=NULL, kMinTab[MAX_SCAN_VAL], bitsCoeffScan[MAX_SCAN_VAL][MAX_EXP_GOLOMB], minKStart, minBitsKStart, bitsKStart; pDepthInt = pDepthIntTabShapes[ALFp->filter_shape]; maxScanVal = 0; int minScanVal = MIN_SCAN_POS_CROSS; for(i = 0; i < sqrFiltLength; i++) { maxScanVal = max(maxScanVal, pDepthInt[i]); } // vlc for all memset(bitsCoeffScan, 0, MAX_SCAN_VAL * MAX_EXP_GOLOMB * sizeof(int)); for(ind=0; indcoeffmulti[ind][i]); for (k=1; k<15; k++) { bitsCoeffScan[scanPos][k]+=lengthGolomb(coeffVal, k); } } } minBitsKStart = 0; minKStart = -1; for(k = 1; k < 8; k++) { bitsKStart = 0; kStart = k; for(scanPos = minScanVal; scanPos < maxScanVal; scanPos++) { kMin = kStart; minBits = bitsCoeffScan[scanPos][kMin]; if(bitsCoeffScan[scanPos][kStart+1] < minBits) { kMin = kStart + 1; minBits = bitsCoeffScan[scanPos][kMin]; } kStart = kMin; bitsKStart += minBits; } if((bitsKStart < minBitsKStart) || (k == 1)) { minBitsKStart = bitsKStart; minKStart = k; } } kStart = minKStart; for(scanPos = minScanVal; scanPos < maxScanVal; scanPos++) { kMin = kStart; minBits = bitsCoeffScan[scanPos][kMin]; if(bitsCoeffScan[scanPos][kStart+1] < minBits) { kMin = kStart + 1; minBits = bitsCoeffScan[scanPos][kMin]; } kMinTab[scanPos] = kMin; kStart = kMin; } // Coding parameters ALFp->minKStart = minKStart; for(scanPos = minScanVal; scanPos < maxScanVal; scanPos++) { ALFp->kMinTab[scanPos] = kMinTab[scanPos]; } if (ALFp->filters_per_group == 1) { len += writeFilterCoeffs(sqrFiltLength, filters_per_group, pDepthInt, ALFp->coeffmulti, kTableTabShapes[ALF_CROSS9x7_SQUARE3x3]); } else { len += writeFilterCodingParams(minKStart, minScanVal, maxScanVal, kMinTab); // Filter coefficients len += writeFilterCoeffs(sqrFiltLength, filters_per_group, pDepthInt, ALFp->coeffmulti, kMinTab); } return len; } Int TEncEntropy::writeFilterCodingParams(int minKStart, int minScanVal, int maxScanVal, int kMinTab[]) { int scanPos; int golombIndexBit; int kMin; // Golomb parameters m_pcEntropyCoderIf->codeAlfUvlc(minKStart - 1); kMin = minKStart; for(scanPos = minScanVal; scanPos < maxScanVal; scanPos++) { golombIndexBit = (kMinTab[scanPos] != kMin)? 1: 0; assert(kMinTab[scanPos] <= kMin + 1); m_pcEntropyCoderIf->codeAlfFlag(golombIndexBit); kMin = kMinTab[scanPos]; } return 0; } Int TEncEntropy::writeFilterCoeffs(int sqrFiltLength, int filters_per_group, int pDepthInt[], int **FilterCoeff, int kMinTab[]) { int ind, scanPos, i; for(ind = 0; ind < filters_per_group; ++ind) { for(i = 0; i < sqrFiltLength; i++) { scanPos = pDepthInt[i] - 1; Int k = (filters_per_group == 1) ? kMinTab[i] : kMinTab[scanPos]; golombEncode(FilterCoeff[ind][i], k); } } return 0; } Int TEncEntropy::golombEncode(int coeff, int k) { int q, i; int symbol = abs(coeff); q = symbol >> k; for (i = 0; i < q; i++) { m_pcEntropyCoderIf->codeAlfFlag(1); } m_pcEntropyCoderIf->codeAlfFlag(0); // write one zero for(i = 0; i < k; i++) { m_pcEntropyCoderIf->codeAlfFlag(symbol & 0x01); symbol >>= 1; } if(coeff != 0) { int sign = (coeff > 0)? 1: 0; m_pcEntropyCoderIf->codeAlfFlag(sign); } return 0; } Void TEncEntropy::codeFilt(ALFParam* pAlfParam) { if(pAlfParam->filters_per_group > 1) { m_pcEntropyCoderIf->codeAlfFlag (pAlfParam->predMethod); } for(Int ind = 0; ind < pAlfParam->filters_per_group; ++ind) { m_pcEntropyCoderIf->codeAlfFlag (pAlfParam->nbSPred[ind]); } codeFilterCoeff (pAlfParam); } /** encode merge flag * \param pcCU * \param uiAbsPartIdx * \param uiPUIdx * \returns Void */ Void TEncEntropy::encodeMergeFlag( TComDataCU* pcCU, UInt uiAbsPartIdx, UInt uiPUIdx ) { // at least one merge candidate exists m_pcEntropyCoderIf->codeMergeFlag( pcCU, uiAbsPartIdx ); } /** encode merge index * \param pcCU * \param uiAbsPartIdx * \param uiPUIdx * \param bRD * \returns Void */ Void TEncEntropy::encodeMergeIndex( TComDataCU* pcCU, UInt uiAbsPartIdx, UInt uiPUIdx, Bool bRD ) { if( bRD ) { uiAbsPartIdx = 0; assert( pcCU->getPartitionSize(uiAbsPartIdx) == SIZE_2Nx2N ); } UInt uiNumCand = MRG_MAX_NUM_CANDS; if ( uiNumCand > 1 ) { m_pcEntropyCoderIf->codeMergeIndex( pcCU, uiAbsPartIdx ); } } #if H3D_IVRP && !MTK_MDIVRP_C0138 Void TEncEntropy::encodeResPredFlag( TComDataCU* pcCU, UInt uiAbsPartIdx, UInt uiPUIdx, Bool bRD ) { if( bRD ) { uiAbsPartIdx = 0; } // check whether flag is coded ROTVS( pcCU->getSlice()->getSPS()->isDepth () ); ROFVS( pcCU->getSlice()->getSPS()->getViewId () ); ROFVS( pcCU->getSlice()->getSPS()->getMultiviewResPredMode() ); ROTVS( pcCU->isIntra ( uiAbsPartIdx ) ); ROFVS( pcCU->getResPredAvail ( uiAbsPartIdx ) ); #if LG_RESTRICTEDRESPRED_M24766 Int iPUResiPredShift[4]; pcCU->getPUResiPredShift(iPUResiPredShift, uiAbsPartIdx); if(iPUResiPredShift[0] >= 0 || iPUResiPredShift[1] >= 0 || iPUResiPredShift[2] >= 0 || iPUResiPredShift[3] >= 0 ) #endif // encode flag m_pcEntropyCoderIf->codeResPredFlag( pcCU, uiAbsPartIdx ); } #endif /** parse the fixed length code (smaller than one max value) in ALF * \param run: coded value * \param rx: cur addr * \param numLCUInWidth: # of LCU in one LCU * \returns Void */ Void TEncEntropy::encodeAlfFixedLengthRun(UInt run, UInt rx, UInt numLCUInWidth) { assert(numLCUInWidth > rx); UInt maxValue = numLCUInWidth - rx - 1; m_pcEntropyCoderIf->codeAlfFixedLengthIdx(run, maxValue); } /** parse the fixed length code (smaller than one max value) in ALF * \param idx: coded value * \param numFilterSetsInBuffer: max value * \returns Void */ Void TEncEntropy::encodeAlfStoredFilterSetIdx(UInt idx, UInt numFilterSetsInBuffer) { assert(numFilterSetsInBuffer > 0); UInt maxValue = numFilterSetsInBuffer - 1; m_pcEntropyCoderIf->codeAlfFixedLengthIdx(idx, maxValue); } Void TEncEntropy::encodeAlfParam(AlfParamSet* pAlfParamSet, Bool bSentInAPS, Int firstLCUAddr, Bool alfAcrossSlice) { Bool isEnabled[NUM_ALF_COMPONENT]; Bool isUniParam[NUM_ALF_COMPONENT]; isEnabled[ALF_Y] = true; isEnabled[ALF_Cb]= pAlfParamSet->isEnabled[ALF_Cb]; isEnabled[ALF_Cr]= pAlfParamSet->isEnabled[ALF_Cr]; isUniParam[ALF_Y]= pAlfParamSet->isUniParam[ALF_Y]; isUniParam[ALF_Cb]= pAlfParamSet->isUniParam[ALF_Cb]; isUniParam[ALF_Cr]= pAlfParamSet->isUniParam[ALF_Cr]; //alf_cb_enable_flag m_pcEntropyCoderIf->codeAlfFlag(isEnabled[ALF_Cb]?1:0); //alf_cr_enable_flag m_pcEntropyCoderIf->codeAlfFlag(isEnabled[ALF_Cr]?1:0); for(Int compIdx = 0; compIdx< NUM_ALF_COMPONENT; compIdx++) { if(isEnabled[compIdx]) { //alf_one_{luma, cb, cr}_unit_per_slice_flag m_pcEntropyCoderIf->codeAlfFlag(isUniParam[compIdx]?1:0); } } if(bSentInAPS) { //alf_num_lcu_in_width_minus1 m_pcEntropyCoderIf->codeAlfUvlc(pAlfParamSet->numLCUInWidth-1); //alf_num_lcu_in_height_minus1 m_pcEntropyCoderIf->codeAlfUvlc(pAlfParamSet->numLCUInHeight-1); } else //sent in slice header { //alf_num_lcu_in_slice_minus1 m_pcEntropyCoderIf->codeAlfUvlc(pAlfParamSet->numLCU-1); } encodeAlfParamSet(pAlfParamSet, pAlfParamSet->numLCUInWidth, pAlfParamSet->numLCU, firstLCUAddr, alfAcrossSlice, 0, (Int)NUM_ALF_COMPONENT-1); } Bool TEncEntropy::getAlfRepeatRowFlag(Int compIdx, AlfParamSet* pAlfParamSet , Int lcuIdxInSlice, Int lcuPos , Int startlcuPosX, Int endlcuPosX , Int numLCUInWidth ) { assert(startlcuPosX == 0); //only the beginning of one LCU row needs to send repeat_row_flag Int len = endlcuPosX - startlcuPosX +1; Bool isRepeatRow = true; Int curPos; for(Int i= 0; i < len; i++) { curPos = lcuIdxInSlice +i; AlfUnitParam& alfUnitParam = pAlfParamSet->alfUnitParam[compIdx][curPos]; AlfUnitParam& alfUpUnitParam = pAlfParamSet->alfUnitParam[compIdx][curPos-numLCUInWidth]; if ( !(alfUnitParam == alfUpUnitParam) ) { isRepeatRow = false; break; } } return isRepeatRow; } Int TEncEntropy::getAlfRun(Int compIdx, AlfParamSet* pAlfParamSet , Int lcuIdxInSlice, Int lcuPos , Int startlcuPosX, Int endlcuPosX ) { Int alfRun = 0; Int len = endlcuPosX - startlcuPosX +1; AlfUnitParam& alfLeftUnitParam = pAlfParamSet->alfUnitParam[compIdx][lcuIdxInSlice]; for(Int i= 1; i < len; i++) { AlfUnitParam& alfUnitParam = pAlfParamSet->alfUnitParam[compIdx][lcuIdxInSlice+ i]; if (alfUnitParam == alfLeftUnitParam) { alfRun++; } else { break; } } return alfRun; } Void TEncEntropy::encodeAlfParamSet(AlfParamSet* pAlfParamSet, Int numLCUInWidth, Int numLCU, Int firstLCUAddr, Bool alfAcrossSlice, Int startCompIdx, Int endCompIdx) { Int endLCUY = (numLCU -1 + firstLCUAddr)/numLCUInWidth; Int endLCUX = (numLCU -1 + firstLCUAddr)%numLCUInWidth; static Bool isRepeatedRow [NUM_ALF_COMPONENT]; static Int numStoredFilters[NUM_ALF_COMPONENT]; static Int* run [NUM_ALF_COMPONENT]; for(Int compIdx =startCompIdx; compIdx <= endCompIdx; compIdx++) { isRepeatedRow[compIdx] = false; numStoredFilters[compIdx] = 0; run[compIdx] = new Int[numLCU+1]; run[compIdx][0] = -1; } Int ry, rx, addrUp, endrX, lcuPos; for(Int i=0; i< numLCU; i++) { lcuPos= firstLCUAddr+ i; rx = lcuPos% numLCUInWidth; ry = lcuPos/ numLCUInWidth; endrX = ( ry == endLCUY)?( endLCUX ):(numLCUInWidth-1); for(Int compIdx =startCompIdx; compIdx <= endCompIdx; compIdx++) { AlfUnitParam& alfUnitParam = pAlfParamSet->alfUnitParam[compIdx][i]; if(pAlfParamSet->isEnabled[compIdx]) { if(!pAlfParamSet->isUniParam[compIdx]) { addrUp = i-numLCUInWidth; if(rx ==0 && addrUp >=0) { isRepeatedRow[compIdx] = getAlfRepeatRowFlag(compIdx, pAlfParamSet, i, lcuPos, rx, endrX, numLCUInWidth); //alf_repeat_row_flag m_pcEntropyCoderIf->codeAlfFlag(isRepeatedRow[compIdx]?1:0); } if(isRepeatedRow[compIdx]) { assert(addrUp >=0); run[compIdx][i] = run[compIdx][addrUp]; } else { if(rx == 0 || run[compIdx][i] < 0) { run[compIdx][i] = getAlfRun(compIdx, pAlfParamSet, i, lcuPos, rx, endrX); if(addrUp < 0) { //alf_run_diff u(v) encodeAlfFixedLengthRun(run[compIdx][i], rx, numLCUInWidth); } else { //alf_run_diff s(v) m_pcEntropyCoderIf->codeAlfSvlc(run[compIdx][i]- run[compIdx][addrUp]); } if(ry > 0 && (addrUp >=0 || alfAcrossSlice)) { //alf_merge_up_flag m_pcEntropyCoderIf->codeAlfFlag( (alfUnitParam.mergeType == ALF_MERGE_UP)?1:0 ); } if(alfUnitParam.mergeType != ALF_MERGE_UP) { assert(alfUnitParam.mergeType == ALF_MERGE_DISABLED); //alf_lcu_enable_flag m_pcEntropyCoderIf->codeAlfFlag(alfUnitParam.isEnabled ? 1 : 0); if(alfUnitParam.isEnabled) { if(numStoredFilters[compIdx] > 0) { //alf_new_filter_set_flag m_pcEntropyCoderIf->codeAlfFlag(alfUnitParam.isNewFilt ? 1:0); if(!alfUnitParam.isNewFilt) { //alf_stored_filter_set_idx encodeAlfStoredFilterSetIdx(alfUnitParam.storedFiltIdx, numStoredFilters[compIdx]); } } else { assert(alfUnitParam.isNewFilt); } if(alfUnitParam.isNewFilt) { assert(alfUnitParam.alfFiltParam->alf_flag == 1); encodeAlfParam(alfUnitParam.alfFiltParam); numStoredFilters[compIdx]++; } } } } run[compIdx][i+1] = run[compIdx][i] -1; } } else // uni-param { if(i == 0) { //alf_lcu_enable_flag m_pcEntropyCoderIf->codeAlfFlag(alfUnitParam.isEnabled?1:0); if(alfUnitParam.isEnabled) { encodeAlfParam(alfUnitParam.alfFiltParam); } } } } // component enabled/disable } //comp } for(Int compIdx =startCompIdx; compIdx <= endCompIdx; compIdx++) { delete[] run[compIdx]; } } Void TEncEntropy::encodeAlfParam(ALFParam* pAlfParam) { const Int numCoeff = (Int)ALF_MAX_NUM_COEF; switch(pAlfParam->componentID) { case ALF_Cb: case ALF_Cr: { for(Int pos=0; pos< numCoeff; pos++) { m_pcEntropyCoderIf->codeAlfSvlc( pAlfParam->coeffmulti[0][pos]); } } break; case ALF_Y: { codeAux(pAlfParam); codeFilt(pAlfParam); } break; default: { printf("Not a legal component ID\n"); assert(0); exit(-1); } } } Void TEncEntropy::encodeAlfCtrlFlag( TComDataCU* pcCU, UInt uiAbsPartIdx, Bool bRD ) { if( bRD ) { uiAbsPartIdx = 0; } m_pcEntropyCoderIf->codeAlfCtrlFlag( pcCU, uiAbsPartIdx ); } /** Encode ALF CU control flag * \param uiFlag ALF CU control flag: 0 or 1 */ Void TEncEntropy::encodeAlfCtrlFlag(UInt uiFlag) { assert(uiFlag == 0 || uiFlag == 1); m_pcEntropyCoderIf->codeAlfCtrlFlag( uiFlag ); } /** Encode ALF CU control flag parameters * \param pAlfParam ALF parameters */ Void TEncEntropy::encodeAlfCtrlParam(AlfCUCtrlInfo& cAlfParam, Int iNumCUsInPic) { // region control parameters for luma m_pcEntropyCoderIf->codeAlfFlag(cAlfParam.cu_control_flag); if (cAlfParam.cu_control_flag == 0) { return; } m_pcEntropyCoderIf->codeAlfCtrlDepth(); Int iSymbol = ((Int)cAlfParam.num_alf_cu_flag - iNumCUsInPic); m_pcEntropyCoderIf->codeAlfSvlc(iSymbol); for(UInt i=0; i< cAlfParam.num_alf_cu_flag; i++) { m_pcEntropyCoderIf->codeAlfCtrlFlag( cAlfParam.alf_cu_flag[i] ); } } /** encode prediction mode * \param pcCU * \param uiAbsPartIdx * \param bRD * \returns Void */ Void TEncEntropy::encodePredMode( TComDataCU* pcCU, UInt uiAbsPartIdx, Bool bRD ) { if( bRD ) { uiAbsPartIdx = 0; } if( !bRD ) { if( pcCU->getLastCUSucIPCMFlag() && pcCU->getIPCMFlag(uiAbsPartIdx) ) { return; } } #if !RWTH_SDC_DLT_B0036 if ( pcCU->getSlice()->isIntra() ) { return; } #endif m_pcEntropyCoderIf->codePredMode( pcCU, uiAbsPartIdx ); #if RWTH_SDC_DLT_B0036 // if B-Slice, code SDC flag later if( !pcCU->getSlice()->isInterB() && pcCU->getSlice()->getSPS()->isDepth() && pcCU->isIntra(uiAbsPartIdx) ) { // encode SDC flag encodeSDCFlag(pcCU, uiAbsPartIdx, bRD); } #endif } // Split mode Void TEncEntropy::encodeSplitFlag( TComDataCU* pcCU, UInt uiAbsPartIdx, UInt uiDepth, Bool bRD ) { if( bRD ) { uiAbsPartIdx = 0; } if( !bRD ) { if( pcCU->getLastCUSucIPCMFlag() && pcCU->getIPCMFlag(uiAbsPartIdx) ) { return; } } m_pcEntropyCoderIf->codeSplitFlag( pcCU, uiAbsPartIdx, uiDepth ); } /** encode partition size * \param pcCU * \param uiAbsPartIdx * \param uiDepth * \param bRD * \returns Void */ Void TEncEntropy::encodePartSize( TComDataCU* pcCU, UInt uiAbsPartIdx, UInt uiDepth, Bool bRD ) { if( bRD ) { uiAbsPartIdx = 0; } if( !bRD ) { if( pcCU->getLastCUSucIPCMFlag() && pcCU->getIPCMFlag(uiAbsPartIdx) ) { return; } } #if RWTH_SDC_DLT_B0036 if( !pcCU->getSlice()->isInterB() && pcCU->isIntra(uiAbsPartIdx) && pcCU->getSDCFlag(uiAbsPartIdx) ) { assert( pcCU->getPartitionSize(uiAbsPartIdx) == SIZE_2Nx2N ); return; } #endif m_pcEntropyCoderIf->codePartSize( pcCU, uiAbsPartIdx, uiDepth ); #if RWTH_SDC_DLT_B0036 // code SDC flag now! if( pcCU->getSlice()->isInterB() && pcCU->isIntra(uiAbsPartIdx) && pcCU->getSlice()->getSPS()->isDepth() ) { // encode SDC flag encodeSDCFlag(pcCU, uiAbsPartIdx, bRD); if( pcCU->getSDCFlag(uiAbsPartIdx) ) { // part size is also known for SDC intra assert( pcCU->getPartitionSize(uiAbsPartIdx) == SIZE_2Nx2N ); } } #endif } /** Encode I_PCM information. * \param pcCU pointer to CU * \param uiAbsPartIdx CU index * \param bRD flag indicating estimation or encoding * \returns Void */ Void TEncEntropy::encodeIPCMInfo( TComDataCU* pcCU, UInt uiAbsPartIdx, Bool bRD ) { if(!pcCU->getSlice()->getSPS()->getUsePCM() || pcCU->getWidth(uiAbsPartIdx) > (1<getSlice()->getSPS()->getPCMLog2MaxSize()) || pcCU->getWidth(uiAbsPartIdx) < (1<getSlice()->getSPS()->getPCMLog2MinSize())) { return; } #if RWTH_SDC_DLT_B0036 if( pcCU->getSDCFlag(uiAbsPartIdx) ) { return; } #endif if( bRD ) { uiAbsPartIdx = 0; } Int numIPCM = 0; Bool firstIPCMFlag = false; if( pcCU->getIPCMFlag(uiAbsPartIdx) ) { numIPCM = 1; firstIPCMFlag = true; if( !bRD ) { numIPCM = pcCU->getNumSucIPCM(); firstIPCMFlag = !pcCU->getLastCUSucIPCMFlag(); } } m_pcEntropyCoderIf->codeIPCMInfo ( pcCU, uiAbsPartIdx, numIPCM, firstIPCMFlag); } Void TEncEntropy::xEncodeTransform( TComDataCU* pcCU,UInt offsetLuma, UInt offsetChroma, UInt uiAbsPartIdx, UInt absTUPartIdx, UInt uiDepth, UInt width, UInt height, UInt uiTrIdx, UInt uiInnerQuadIdx, UInt& uiYCbfFront3, UInt& uiUCbfFront3, UInt& uiVCbfFront3, Bool& bCodeDQP ) { const UInt uiSubdiv = pcCU->getTransformIdx( uiAbsPartIdx ) + pcCU->getDepth( uiAbsPartIdx ) > uiDepth; const UInt uiLog2TrafoSize = g_aucConvertToBit[pcCU->getSlice()->getSPS()->getMaxCUWidth()]+2 - uiDepth; UInt cbfY = pcCU->getCbf( uiAbsPartIdx, TEXT_LUMA , uiTrIdx ); UInt cbfU = pcCU->getCbf( uiAbsPartIdx, TEXT_CHROMA_U, uiTrIdx ); UInt cbfV = pcCU->getCbf( uiAbsPartIdx, TEXT_CHROMA_V, uiTrIdx ); if(uiTrIdx==0) { m_bakAbsPartIdxCU = uiAbsPartIdx; } if( uiLog2TrafoSize == 2 ) { UInt partNum = pcCU->getPic()->getNumPartInCU() >> ( ( uiDepth - 1 ) << 1 ); if( ( uiAbsPartIdx % partNum ) == 0 ) { m_uiBakAbsPartIdx = uiAbsPartIdx; m_uiBakChromaOffset = offsetChroma; } else if( ( uiAbsPartIdx % partNum ) == (partNum - 1) ) { cbfU = pcCU->getCbf( m_uiBakAbsPartIdx, TEXT_CHROMA_U, uiTrIdx ); cbfV = pcCU->getCbf( m_uiBakAbsPartIdx, TEXT_CHROMA_V, uiTrIdx ); } } {//CABAC if( pcCU->getPredictionMode(uiAbsPartIdx) == MODE_INTRA && pcCU->getPartitionSize(uiAbsPartIdx) == SIZE_NxN && uiDepth == pcCU->getDepth(uiAbsPartIdx) ) { assert( uiSubdiv ); } else if( pcCU->getPredictionMode(uiAbsPartIdx) == MODE_INTER && (pcCU->getPartitionSize(uiAbsPartIdx) != SIZE_2Nx2N) && uiDepth == pcCU->getDepth(uiAbsPartIdx) && (pcCU->getSlice()->getSPS()->getQuadtreeTUMaxDepthInter() == 1) ) { if ( uiLog2TrafoSize > pcCU->getQuadtreeTULog2MinSizeInCU(uiAbsPartIdx) ) { assert( uiSubdiv ); } else { assert(!uiSubdiv ); } } else if( uiLog2TrafoSize > pcCU->getSlice()->getSPS()->getQuadtreeTULog2MaxSize() ) { assert( uiSubdiv ); } else if( uiLog2TrafoSize == pcCU->getSlice()->getSPS()->getQuadtreeTULog2MinSize() ) { assert( !uiSubdiv ); } else if( uiLog2TrafoSize == pcCU->getQuadtreeTULog2MinSizeInCU(uiAbsPartIdx) ) { assert( !uiSubdiv ); } else { assert( uiLog2TrafoSize > pcCU->getQuadtreeTULog2MinSizeInCU(uiAbsPartIdx) ); m_pcEntropyCoderIf->codeTransformSubdivFlag( uiSubdiv, uiDepth ); } } { if( uiLog2TrafoSize <= pcCU->getSlice()->getSPS()->getQuadtreeTULog2MaxSize() ) { const UInt uiTrDepthCurr = uiDepth - pcCU->getDepth( uiAbsPartIdx ); const Bool bFirstCbfOfCU = uiLog2TrafoSize == pcCU->getSlice()->getSPS()->getQuadtreeTULog2MaxSize() || uiTrDepthCurr == 0; if( bFirstCbfOfCU || uiLog2TrafoSize > 2 ) { if( bFirstCbfOfCU || pcCU->getCbf( uiAbsPartIdx, TEXT_CHROMA_U, uiTrDepthCurr - 1 ) ) { if ( uiInnerQuadIdx == 3 && uiUCbfFront3 == 0 && uiLog2TrafoSize < pcCU->getSlice()->getSPS()->getQuadtreeTULog2MaxSize() ) { uiUCbfFront3++; } else { m_pcEntropyCoderIf->codeQtCbf( pcCU, uiAbsPartIdx, TEXT_CHROMA_U, uiTrDepthCurr ); uiUCbfFront3 += pcCU->getCbf( uiAbsPartIdx, TEXT_CHROMA_U, uiTrDepthCurr ); } } if( bFirstCbfOfCU || pcCU->getCbf( uiAbsPartIdx, TEXT_CHROMA_V, uiTrDepthCurr - 1 ) ) { if ( uiInnerQuadIdx == 3 && uiVCbfFront3 == 0 && uiLog2TrafoSize < pcCU->getSlice()->getSPS()->getQuadtreeTULog2MaxSize() ) { uiVCbfFront3++; } else { m_pcEntropyCoderIf->codeQtCbf( pcCU, uiAbsPartIdx, TEXT_CHROMA_V, uiTrDepthCurr ); uiVCbfFront3 += pcCU->getCbf( uiAbsPartIdx, TEXT_CHROMA_V, uiTrDepthCurr ); } } } else if( uiLog2TrafoSize == 2 ) { assert( pcCU->getCbf( uiAbsPartIdx, TEXT_CHROMA_U, uiTrDepthCurr ) == pcCU->getCbf( uiAbsPartIdx, TEXT_CHROMA_U, uiTrDepthCurr - 1 ) ); assert( pcCU->getCbf( uiAbsPartIdx, TEXT_CHROMA_V, uiTrDepthCurr ) == pcCU->getCbf( uiAbsPartIdx, TEXT_CHROMA_V, uiTrDepthCurr - 1 ) ); uiUCbfFront3 += pcCU->getCbf( uiAbsPartIdx, TEXT_CHROMA_U, uiTrDepthCurr ); uiVCbfFront3 += pcCU->getCbf( uiAbsPartIdx, TEXT_CHROMA_V, uiTrDepthCurr ); } } if( uiSubdiv ) { UInt size; width >>= 1; height >>= 1; size = width*height; uiTrIdx++; ++uiDepth; const UInt partNum = pcCU->getPic()->getNumPartInCU() >> (uiDepth << 1); UInt uiCurrentCbfY = 0; UInt uiCurrentCbfU = 0; UInt uiCurrentCbfV = 0; UInt nsAddr = 0; nsAddr = pcCU->getNSAbsPartIdx( uiLog2TrafoSize-1, uiAbsPartIdx, absTUPartIdx, 0, uiDepth - pcCU->getDepth( uiAbsPartIdx ) ); xEncodeTransform( pcCU, offsetLuma, offsetChroma, uiAbsPartIdx, nsAddr, uiDepth, width, height, uiTrIdx, 0, uiCurrentCbfY, uiCurrentCbfU, uiCurrentCbfV, bCodeDQP ); uiAbsPartIdx += partNum; offsetLuma += size; offsetChroma += (size>>2); nsAddr = pcCU->getNSAbsPartIdx( uiLog2TrafoSize-1, uiAbsPartIdx, absTUPartIdx, 1, uiDepth - pcCU->getDepth( uiAbsPartIdx ) ); xEncodeTransform( pcCU, offsetLuma, offsetChroma, uiAbsPartIdx, nsAddr, uiDepth, width, height, uiTrIdx, 1, uiCurrentCbfY, uiCurrentCbfU, uiCurrentCbfV, bCodeDQP ); uiAbsPartIdx += partNum; offsetLuma += size; offsetChroma += (size>>2); nsAddr = pcCU->getNSAbsPartIdx( uiLog2TrafoSize-1, uiAbsPartIdx, absTUPartIdx, 2, uiDepth - pcCU->getDepth( uiAbsPartIdx ) ); xEncodeTransform( pcCU, offsetLuma, offsetChroma, uiAbsPartIdx, nsAddr, uiDepth, width, height, uiTrIdx, 2, uiCurrentCbfY, uiCurrentCbfU, uiCurrentCbfV, bCodeDQP ); uiAbsPartIdx += partNum; offsetLuma += size; offsetChroma += (size>>2); nsAddr = pcCU->getNSAbsPartIdx( uiLog2TrafoSize-1, uiAbsPartIdx, absTUPartIdx, 3, uiDepth - pcCU->getDepth( uiAbsPartIdx ) ); xEncodeTransform( pcCU, offsetLuma, offsetChroma, uiAbsPartIdx, nsAddr, uiDepth, width, height, uiTrIdx, 3, uiCurrentCbfY, uiCurrentCbfU, uiCurrentCbfV, bCodeDQP ); uiYCbfFront3 += uiCurrentCbfY; uiUCbfFront3 += uiCurrentCbfU; uiVCbfFront3 += uiCurrentCbfV; } else { { DTRACE_CABAC_VL( g_nSymbolCounter++ ); DTRACE_CABAC_T( "\tTrIdx: abspart=" ); DTRACE_CABAC_V( uiAbsPartIdx ); DTRACE_CABAC_T( "\tdepth=" ); DTRACE_CABAC_V( uiDepth ); DTRACE_CABAC_T( "\ttrdepth=" ); DTRACE_CABAC_V( pcCU->getTransformIdx( uiAbsPartIdx ) ); DTRACE_CABAC_T( "\n" ); } UInt uiLumaTrMode, uiChromaTrMode; pcCU->convertTransIdx( uiAbsPartIdx, pcCU->getTransformIdx( uiAbsPartIdx ), uiLumaTrMode, uiChromaTrMode ); if(pcCU->getPredictionMode( uiAbsPartIdx ) == MODE_INTER && pcCU->useNonSquarePU( uiAbsPartIdx ) ) { pcCU->setNSQTIdxSubParts( uiLog2TrafoSize, uiAbsPartIdx, absTUPartIdx, uiLumaTrMode ); } if( pcCU->getPredictionMode(uiAbsPartIdx) != MODE_INTRA && uiDepth == pcCU->getDepth( uiAbsPartIdx ) && !pcCU->getCbf( uiAbsPartIdx, TEXT_CHROMA_U, 0 ) && !pcCU->getCbf( uiAbsPartIdx, TEXT_CHROMA_V, 0 ) ) { assert( pcCU->getCbf( uiAbsPartIdx, TEXT_LUMA, 0 ) ); // printf( "saved one bin! " ); } else { const UInt uiLog2CUSize = g_aucConvertToBit[pcCU->getSlice()->getSPS()->getMaxCUWidth()] + 2 - pcCU->getDepth( uiAbsPartIdx ); if ( pcCU->getPredictionMode( uiAbsPartIdx ) != MODE_INTRA && uiInnerQuadIdx == 3 && uiYCbfFront3 == 0 && uiUCbfFront3 == 0 && uiVCbfFront3 == 0 && ( uiLog2CUSize <= pcCU->getSlice()->getSPS()->getQuadtreeTULog2MaxSize() + 1 || uiLog2TrafoSize < pcCU->getSlice()->getSPS()->getQuadtreeTULog2MaxSize() ) ) { uiYCbfFront3++; } else { m_pcEntropyCoderIf->codeQtCbf( pcCU, uiAbsPartIdx, TEXT_LUMA, uiLumaTrMode ); uiYCbfFront3 += pcCU->getCbf( uiAbsPartIdx, TEXT_LUMA, uiLumaTrMode ); } } if ( cbfY || cbfU || cbfV ) { // dQP: only for LCU once if ( pcCU->getSlice()->getPPS()->getUseDQP() ) { if ( bCodeDQP ) { encodeQP( pcCU, m_bakAbsPartIdxCU ); bCodeDQP = false; } } } if( cbfY ) { Int trWidth = width; Int trHeight = height; pcCU->getNSQTSize( uiTrIdx, uiAbsPartIdx, trWidth, trHeight ); m_pcEntropyCoderIf->codeCoeffNxN( pcCU, (pcCU->getCoeffY()+offsetLuma), uiAbsPartIdx, trWidth, trHeight, uiDepth, TEXT_LUMA ); } if( uiLog2TrafoSize > 2 ) { Int trWidth = width >> 1; Int trHeight = height >> 1; pcCU->getNSQTSize( uiTrIdx, uiAbsPartIdx, trWidth, trHeight ); if( cbfU ) { m_pcEntropyCoderIf->codeCoeffNxN( pcCU, (pcCU->getCoeffCb()+offsetChroma), uiAbsPartIdx, trWidth, trHeight, uiDepth, TEXT_CHROMA_U ); } if( cbfV ) { m_pcEntropyCoderIf->codeCoeffNxN( pcCU, (pcCU->getCoeffCr()+offsetChroma), uiAbsPartIdx, trWidth, trHeight, uiDepth, TEXT_CHROMA_V ); } } else { UInt partNum = pcCU->getPic()->getNumPartInCU() >> ( ( uiDepth - 1 ) << 1 ); if( ( uiAbsPartIdx % partNum ) == (partNum - 1) ) { Int trWidth = width; Int trHeight = height; pcCU->getNSQTSize( uiTrIdx - 1, uiAbsPartIdx, trWidth, trHeight ); if( cbfU ) { m_pcEntropyCoderIf->codeCoeffNxN( pcCU, (pcCU->getCoeffCb()+m_uiBakChromaOffset), m_uiBakAbsPartIdx, trWidth, trHeight, uiDepth, TEXT_CHROMA_U ); } if( cbfV ) { m_pcEntropyCoderIf->codeCoeffNxN( pcCU, (pcCU->getCoeffCr()+m_uiBakChromaOffset), m_uiBakAbsPartIdx, trWidth, trHeight, uiDepth, TEXT_CHROMA_V ); } } } } } } // Intra direction for Luma Void TEncEntropy::encodeIntraDirModeLuma ( TComDataCU* pcCU, UInt uiAbsPartIdx ) { m_pcEntropyCoderIf->codeIntraDirLumaAng( pcCU, uiAbsPartIdx ); } // Intra direction for Chroma Void TEncEntropy::encodeIntraDirModeChroma( TComDataCU* pcCU, UInt uiAbsPartIdx, Bool bRD ) { if( bRD ) { uiAbsPartIdx = 0; } m_pcEntropyCoderIf->codeIntraDirChroma( pcCU, uiAbsPartIdx ); } Void TEncEntropy::encodePredInfo( TComDataCU* pcCU, UInt uiAbsPartIdx, Bool bRD ) { if( bRD ) { uiAbsPartIdx = 0; } #if RWTH_SDC_DLT_B0036 if( pcCU->getSDCFlag(uiAbsPartIdx) ) { encodeSDCPredMode(pcCU, uiAbsPartIdx, bRD); return; } #endif PartSize eSize = pcCU->getPartitionSize( uiAbsPartIdx ); if( pcCU->isIntra( uiAbsPartIdx ) ) // If it is Intra mode, encode intra prediction mode. { if( eSize == SIZE_NxN ) // if it is NxN size, encode 4 intra directions. { UInt uiPartOffset = ( pcCU->getPic()->getNumPartInCU() >> ( pcCU->getDepth(uiAbsPartIdx) << 1 ) ) >> 2; // if it is NxN size, this size might be the smallest partition size. encodeIntraDirModeLuma( pcCU, uiAbsPartIdx ); encodeIntraDirModeLuma( pcCU, uiAbsPartIdx + uiPartOffset ); encodeIntraDirModeLuma( pcCU, uiAbsPartIdx + uiPartOffset*2 ); encodeIntraDirModeLuma( pcCU, uiAbsPartIdx + uiPartOffset*3 ); encodeIntraDirModeChroma( pcCU, uiAbsPartIdx, bRD ); } else // if it is not NxN size, encode 1 intra directions { encodeIntraDirModeLuma ( pcCU, uiAbsPartIdx ); encodeIntraDirModeChroma( pcCU, uiAbsPartIdx, bRD ); } } else // if it is Inter mode, encode motion vector and reference index { encodePUWise( pcCU, uiAbsPartIdx, bRD ); } } /** encode motion information for every PU block * \param pcCU * \param uiAbsPartIdx * \param bRD * \returns Void */ Void TEncEntropy::encodePUWise( TComDataCU* pcCU, UInt uiAbsPartIdx, Bool bRD ) { if ( bRD ) { uiAbsPartIdx = 0; } PartSize ePartSize = pcCU->getPartitionSize( uiAbsPartIdx ); UInt uiNumPU = ( ePartSize == SIZE_2Nx2N ? 1 : ( ePartSize == SIZE_NxN ? 4 : 2 ) ); UInt uiDepth = pcCU->getDepth( uiAbsPartIdx ); UInt uiPUOffset = ( g_auiPUOffset[UInt( ePartSize )] << ( ( pcCU->getSlice()->getSPS()->getMaxCUDepth() - uiDepth ) << 1 ) ) >> 4; for ( UInt uiPartIdx = 0, uiSubPartIdx = uiAbsPartIdx; uiPartIdx < uiNumPU; uiPartIdx++, uiSubPartIdx += uiPUOffset ) { encodeMergeFlag( pcCU, uiSubPartIdx, uiPartIdx ); if ( pcCU->getMergeFlag( uiSubPartIdx ) ) { encodeMergeIndex( pcCU, uiSubPartIdx, uiPartIdx ); } else { encodeInterDirPU( pcCU, uiSubPartIdx ); for ( UInt uiRefListIdx = 0; uiRefListIdx < 2; uiRefListIdx++ ) { if ( pcCU->getSlice()->getNumRefIdx( RefPicList( uiRefListIdx ) ) > 0 ) { encodeRefFrmIdxPU ( pcCU, uiSubPartIdx, RefPicList( uiRefListIdx ) ); encodeMvdPU ( pcCU, uiSubPartIdx, RefPicList( uiRefListIdx ) ); encodeMVPIdxPU ( pcCU, uiSubPartIdx, RefPicList( uiRefListIdx ) ); } } } } return; } Void TEncEntropy::encodeInterDirPU( TComDataCU* pcCU, UInt uiAbsPartIdx ) { if ( !pcCU->getSlice()->isInterB() ) { return; } m_pcEntropyCoderIf->codeInterDir( pcCU, uiAbsPartIdx ); return; } /** encode reference frame index for a PU block * \param pcCU * \param uiAbsPartIdx * \param eRefList * \returns Void */ Void TEncEntropy::encodeRefFrmIdxPU( TComDataCU* pcCU, UInt uiAbsPartIdx, RefPicList eRefList ) { assert( !pcCU->isIntra( uiAbsPartIdx ) ); if(pcCU->getSlice()->getNumRefIdx(REF_PIC_LIST_C)>0 && pcCU->getInterDir( uiAbsPartIdx ) != 3) { if ((eRefList== REF_PIC_LIST_1) || ( pcCU->getSlice()->getNumRefIdx( REF_PIC_LIST_C ) == 1 ) ) { return; } if ( pcCU->getSlice()->getNumRefIdx ( REF_PIC_LIST_C ) > 1 ) { m_pcEntropyCoderIf->codeRefFrmIdx( pcCU, uiAbsPartIdx, RefPicList(pcCU->getInterDir( uiAbsPartIdx )-1) ); } } else { if ( ( pcCU->getSlice()->getNumRefIdx( eRefList ) == 1 ) ) { return; } if ( pcCU->getInterDir( uiAbsPartIdx ) & ( 1 << eRefList ) ) { m_pcEntropyCoderIf->codeRefFrmIdx( pcCU, uiAbsPartIdx, eRefList ); } } return; } /** encode motion vector difference for a PU block * \param pcCU * \param uiAbsPartIdx * \param eRefList * \returns Void */ Void TEncEntropy::encodeMvdPU( TComDataCU* pcCU, UInt uiAbsPartIdx, RefPicList eRefList ) { assert( !pcCU->isIntra( uiAbsPartIdx ) ); if ( pcCU->getInterDir( uiAbsPartIdx ) & ( 1 << eRefList ) ) { m_pcEntropyCoderIf->codeMvd( pcCU, uiAbsPartIdx, eRefList ); } return; } Void TEncEntropy::encodeMVPIdxPU( TComDataCU* pcCU, UInt uiAbsPartIdx, RefPicList eRefList ) { if ( (pcCU->getInterDir( uiAbsPartIdx ) & ( 1 << eRefList )) && (pcCU->getAMVPMode(uiAbsPartIdx) == AM_EXPL) ) { #if H3D_IVMP const Int iNumCands = AMVP_MAX_NUM_CANDS + ( pcCU->getSlice()->getSPS()->getMultiviewMvPredMode() ? 1 : 0 ); m_pcEntropyCoderIf->codeMVPIdx( pcCU, uiAbsPartIdx, eRefList, iNumCands ); #else m_pcEntropyCoderIf->codeMVPIdx( pcCU, uiAbsPartIdx, eRefList ); #endif } return; } Void TEncEntropy::encodeQtCbf( TComDataCU* pcCU, UInt uiAbsPartIdx, TextType eType, UInt uiTrDepth ) { m_pcEntropyCoderIf->codeQtCbf( pcCU, uiAbsPartIdx, eType, uiTrDepth ); } Void TEncEntropy::encodeTransformSubdivFlag( UInt uiSymbol, UInt uiCtx ) { m_pcEntropyCoderIf->codeTransformSubdivFlag( uiSymbol, uiCtx ); } Void TEncEntropy::encodeQtRootCbf( TComDataCU* pcCU, UInt uiAbsPartIdx ) { m_pcEntropyCoderIf->codeQtRootCbf( pcCU, uiAbsPartIdx ); } // dQP Void TEncEntropy::encodeQP( TComDataCU* pcCU, UInt uiAbsPartIdx, Bool bRD ) { if( bRD ) { uiAbsPartIdx = 0; } if ( pcCU->getSlice()->getPPS()->getUseDQP() ) { m_pcEntropyCoderIf->codeDeltaQP( pcCU, uiAbsPartIdx ); } } // texture /** encode coefficients * \param pcCU * \param uiAbsPartIdx * \param uiDepth * \param uiWidth * \param uiHeight */ Void TEncEntropy::encodeCoeff( TComDataCU* pcCU, UInt uiAbsPartIdx, UInt uiDepth, UInt uiWidth, UInt uiHeight, Bool& bCodeDQP ) { UInt uiMinCoeffSize = pcCU->getPic()->getMinCUWidth()*pcCU->getPic()->getMinCUHeight(); UInt uiLumaOffset = uiMinCoeffSize*uiAbsPartIdx; UInt uiChromaOffset = uiLumaOffset>>2; UInt uiLumaTrMode, uiChromaTrMode; pcCU->convertTransIdx( uiAbsPartIdx, pcCU->getTransformIdx(uiAbsPartIdx), uiLumaTrMode, uiChromaTrMode ); #if RWTH_SDC_DLT_B0036 if( pcCU->getSDCFlag( uiAbsPartIdx ) ) { assert( pcCU->getPartitionSize(uiAbsPartIdx) == SIZE_2Nx2N ); assert( pcCU->getTransformIdx(uiAbsPartIdx) == 0 ); assert( pcCU->getCbf(uiAbsPartIdx, TEXT_LUMA) == 1 ); assert( pcCU->getCbf(uiAbsPartIdx, TEXT_CHROMA_U) == 1 ); assert( pcCU->getCbf(uiAbsPartIdx, TEXT_CHROMA_V) == 1 ); encodeSDCResidualData(pcCU, uiAbsPartIdx); return; } #endif if( pcCU->isIntra(uiAbsPartIdx) ) { DTRACE_CABAC_VL( g_nSymbolCounter++ ) DTRACE_CABAC_T( "\tdecodeTransformIdx()\tCUDepth=" ) DTRACE_CABAC_V( uiDepth ) DTRACE_CABAC_T( "\n" ) } else { { #if HHI_MPI if( !(pcCU->getMergeFlag( uiAbsPartIdx ) && pcCU->getPartitionSize(uiAbsPartIdx) == SIZE_2Nx2N && ( pcCU->getTextureModeDepth( uiAbsPartIdx ) == -1 || uiDepth == pcCU->getTextureModeDepth( uiAbsPartIdx ) ) ) ) #else if( !(pcCU->getMergeFlag( uiAbsPartIdx ) && pcCU->getPartitionSize(uiAbsPartIdx) == SIZE_2Nx2N ) ) #endif { m_pcEntropyCoderIf->codeQtRootCbf( pcCU, uiAbsPartIdx ); } if ( !pcCU->getQtRootCbf( uiAbsPartIdx ) ) { #if 1 // MW Bug Fix pcCU->setCbfSubParts( 0, 0, 0, uiAbsPartIdx, uiDepth ); pcCU->setTrIdxSubParts( 0 , uiAbsPartIdx, uiDepth ); #endif pcCU->setNSQTIdxSubParts( uiAbsPartIdx, uiDepth ); return; } } } #if FIX_MPI_B0065 if( pcCU->getPredictionMode(uiAbsPartIdx) == MODE_INTER && pcCU->getMergeFlag( uiAbsPartIdx ) && pcCU->getMergeIndex( uiAbsPartIdx ) == 0 && pcCU->getPartitionSize(uiAbsPartIdx) == SIZE_2Nx2N && pcCU->getTextureModeDepth( uiAbsPartIdx ) != -1 ) { TComDataCU *pcTextureCU = pcCU->getSlice()->getTexturePic()->getCU( pcCU->getAddr() ); if( uiDepth == pcTextureCU->getDepth(uiAbsPartIdx)) { PartSize partSize = pcTextureCU->getPartitionSize(uiAbsPartIdx); pcCU->setPartSizeSubParts( partSize, uiAbsPartIdx, uiDepth ); } else { pcCU->setPartSizeSubParts( SIZE_NxN, uiAbsPartIdx, uiDepth ); } } #endif UInt temp = 0; UInt temp1 = 0; UInt temp2 = 0; xEncodeTransform( pcCU, uiLumaOffset, uiChromaOffset, uiAbsPartIdx, uiAbsPartIdx, uiDepth, uiWidth, uiHeight, 0, 0, temp, temp1, temp2, bCodeDQP ); #if FIX_MPI_B0065 if( pcCU->getPredictionMode(uiAbsPartIdx) == MODE_INTER && pcCU->getMergeFlag( uiAbsPartIdx ) && pcCU->getMergeIndex( uiAbsPartIdx ) == 0 && pcCU->getPartitionSize(uiAbsPartIdx) != SIZE_2Nx2N && pcCU->getTextureModeDepth( uiAbsPartIdx ) != -1 ) { pcCU->setPartSizeSubParts( SIZE_2Nx2N, uiAbsPartIdx, uiDepth ); } #endif } Void TEncEntropy::encodeCoeffNxN( TComDataCU* pcCU, TCoeff* pcCoeff, UInt uiAbsPartIdx, UInt uiTrWidth, UInt uiTrHeight, UInt uiDepth, TextType eType ) { // This is for Transform unit processing. This may be used at mode selection stage for Inter. m_pcEntropyCoderIf->codeCoeffNxN( pcCU, pcCoeff, uiAbsPartIdx, uiTrWidth, uiTrHeight, uiDepth, eType ); } Void TEncEntropy::estimateBit (estBitsSbacStruct* pcEstBitsSbac, Int width, Int height, TextType eTType) { eTType = eTType == TEXT_LUMA ? TEXT_LUMA : TEXT_CHROMA; m_pcEntropyCoderIf->estBit ( pcEstBitsSbac, width, height, eTType ); } /** Encode SAO Offset * \param saoLcuParam SAO LCU paramters */ Void TEncEntropy::encodeSaoOffset(SaoLcuParam* saoLcuParam) { UInt uiSymbol; Int i; uiSymbol = saoLcuParam->typeIdx + 1; m_pcEntropyCoderIf->codeSaoTypeIdx(uiSymbol); if (uiSymbol) { if( saoLcuParam->typeIdx == SAO_BO ) { // Code Left Band Index uiSymbol = (UInt) (saoLcuParam->bandPosition); m_pcEntropyCoderIf->codeSaoUflc(uiSymbol); for( i=0; i< saoLcuParam->length; i++) { m_pcEntropyCoderIf->codeSaoSvlc(saoLcuParam->offset[i]); } } else if( saoLcuParam->typeIdx < 4 ) { m_pcEntropyCoderIf->codeSaoUvlc( saoLcuParam->offset[0]); m_pcEntropyCoderIf->codeSaoUvlc( saoLcuParam->offset[1]); m_pcEntropyCoderIf->codeSaoUvlc(-saoLcuParam->offset[2]); m_pcEntropyCoderIf->codeSaoUvlc(-saoLcuParam->offset[3]); } } } /** Encode SAO unit * \param rx * \param ry * \param iCompIdx * \param pSaoParam * \param bRepeatedRow */ Void TEncEntropy::encodeSaoUnit(Int rx, Int ry, Int compIdx, SAOParam* saoParam, Int repeatedRow ) { int addr, addrLeft; int numCuInWidth = saoParam->numCuInWidth; SaoLcuParam* saoOneLcu; Int runLeft; addr = rx + ry*numCuInWidth; addrLeft = (addr%numCuInWidth == 0) ? -1 : addr - 1; if (!repeatedRow) { saoOneLcu = &(saoParam->saoLcuParam[compIdx][addr]); runLeft = (addrLeft>=0 ) ? saoParam->saoLcuParam[compIdx][addrLeft].run : -1; if (rx == 0 || runLeft==0) { if (ry == 0) { m_pcEntropyCoderIf->codeSaoRun(saoOneLcu->runDiff, numCuInWidth-rx-1); saoOneLcu->mergeUpFlag = 0; } else { m_pcEntropyCoderIf->codeSaoSvlc(saoOneLcu->runDiff); m_pcEntropyCoderIf->codeSaoFlag(saoOneLcu->mergeUpFlag); } if (!saoOneLcu->mergeUpFlag) { encodeSaoOffset(saoOneLcu); } } } } /** Encode SAO unit interleaving * \param rx * \param ry * \param pSaoParam * \param pcCU * \param iCUAddrInSlice * \param iCUAddrUpInSlice * \param bLFCrossSliceBoundaryFlag */ Void TEncEntropy::encodeSaoUnitInterleaving(Int rx, Int ry, SAOParam* saoParam, TComDataCU* cu, Int cuAddrInSlice, Int cuAddrUpInSlice, Bool lfCrossSliceBoundaryFlag) { Int addr = cu->getAddr(); for (Int compIdx=0; compIdx<3; compIdx++) { if (saoParam->bSaoFlag[compIdx]) { if (rx>0 && cuAddrInSlice!=0) { m_pcEntropyCoderIf->codeSaoMergeLeft(saoParam->saoLcuParam[compIdx][addr].mergeLeftFlag,compIdx); } else { saoParam->saoLcuParam[compIdx][addr].mergeLeftFlag = 0; } if (saoParam->saoLcuParam[compIdx][addr].mergeLeftFlag == 0) { if ( (ry > 0) && (cuAddrUpInSlice>0||lfCrossSliceBoundaryFlag)) { m_pcEntropyCoderIf->codeSaoMergeUp(saoParam->saoLcuParam[compIdx][addr].mergeUpFlag); } else { saoParam->saoLcuParam[compIdx][addr].mergeUpFlag = 0; } if (!saoParam->saoLcuParam[compIdx][addr].mergeUpFlag) { encodeSaoOffset(&(saoParam->saoLcuParam[compIdx][addr])); } } } } } /** Encode SAO parameter * \param pcAPS */ Void TEncEntropy::encodeSaoParam(TComAPS* aps) { SaoLcuParam* psSaoOneLcu; int i,j,k, compIdx; int numCuInWidth ; int numCuInHeight ; Bool repeatedRow[3]; Int addr; m_pcEntropyCoderIf->codeSaoFlag(aps->getSaoInterleavingFlag()); if(!aps->getSaoInterleavingFlag()) { m_pcEntropyCoderIf->codeSaoFlag(aps->getSaoEnabled()); if (aps->getSaoEnabled()) { SAOParam* pSaoParam = aps->getSaoParam(); numCuInWidth = pSaoParam->numCuInWidth; numCuInHeight = pSaoParam->numCuInHeight; m_pcEntropyCoderIf->codeSaoFlag(pSaoParam->bSaoFlag[1]); m_pcEntropyCoderIf->codeSaoFlag(pSaoParam->bSaoFlag[2]); m_pcEntropyCoderIf->codeSaoUvlc(numCuInWidth-1); m_pcEntropyCoderIf->codeSaoUvlc(numCuInHeight-1); for (compIdx=0;compIdx<3;compIdx++) { if (pSaoParam->bSaoFlag[compIdx]) { m_pcEntropyCoderIf->codeSaoFlag(pSaoParam->oneUnitFlag[compIdx]); if (pSaoParam->oneUnitFlag[compIdx]) { psSaoOneLcu = &(pSaoParam->saoLcuParam[compIdx][0]); encodeSaoOffset(psSaoOneLcu); } } } for (j=0;jsaoLcuParam[compIdx][addr]); if (!psSaoOneLcu->mergeUpFlag || psSaoOneLcu->runDiff) { repeatedRow[compIdx] = false; break; } } } for (i=0;ibSaoFlag[compIdx] && !pSaoParam->oneUnitFlag[compIdx]) { if (j>0 && i==0) { m_pcEntropyCoderIf->codeSaoFlag(repeatedRow[compIdx]); } encodeSaoUnit (i,j, compIdx, pSaoParam, repeatedRow[compIdx]); } } } } } } } Int TEncEntropy::countNonZeroCoeffs( TCoeff* pcCoef, UInt uiSize ) { Int count = 0; for ( Int i = 0; i < uiSize; i++ ) { count += pcCoef[i] != 0; } return count; } /** encode quantization matrix * \param scalingList quantization matrix information */ Void TEncEntropy::encodeScalingList( TComScalingList* scalingList ) { m_pcEntropyCoderIf->codeScalingList( scalingList ); } Void TEncEntropy::encodeDFParams(TComAPS* pcAPS) { m_pcEntropyCoderIf->codeDFFlag(pcAPS->getLoopFilterDisable(), "loop_filter_disable"); if (!pcAPS->getLoopFilterDisable()) { m_pcEntropyCoderIf->codeDFSvlc(pcAPS->getLoopFilterBetaOffset(), "beta_offset_div2"); m_pcEntropyCoderIf->codeDFSvlc(pcAPS->getLoopFilterTcOffset(), "tc_offset_div2"); } } #if RWTH_SDC_DLT_B0036 Void TEncEntropy::encodeSDCPredMode( TComDataCU* pcCU, UInt uiAbsPartIdx, Bool bRD ) { assert( pcCU->getSlice()->getSPS()->isDepth() ); if( bRD ) uiAbsPartIdx = 0; m_pcEntropyCoderIf->codeSDCPredMode(pcCU, uiAbsPartIdx); } Void TEncEntropy::encodeSDCFlag( TComDataCU* pcCU, UInt uiAbsPartIdx, Bool bRD ) { assert( pcCU->getSlice()->getSPS()->isDepth() ); if( bRD ) uiAbsPartIdx = 0; m_pcEntropyCoderIf->codeSDCFlag(pcCU, uiAbsPartIdx); } Void TEncEntropy::encodeSDCResidualData( TComDataCU* pcCU, UInt uiAbsPartIdx, Bool bRD ) { assert( pcCU->getSlice()->getSPS()->isDepth() ); assert( pcCU->getCbf(uiAbsPartIdx, TEXT_LUMA) == 1 ); assert( pcCU->getCbf(uiAbsPartIdx, TEXT_CHROMA_U) == 1 ); assert( pcCU->getCbf(uiAbsPartIdx, TEXT_CHROMA_V) == 1 ); assert( pcCU->getTransformIdx(uiAbsPartIdx) == 0 ); if( bRD ) uiAbsPartIdx = 0; // number of segments depends on prediction mode for INTRA UInt uiNumSegments = 2; UInt uiLumaPredMode = pcCU->getLumaIntraDir( uiAbsPartIdx ); if( uiLumaPredMode == DC_IDX || uiLumaPredMode == PLANAR_IDX ) uiNumSegments = 1; // encode residual data for each segment for( UInt uiSeg = 0; uiSeg < uiNumSegments; uiSeg++ ) m_pcEntropyCoderIf->codeSDCResidualData(pcCU, uiAbsPartIdx, uiSeg); } #endif //! \}