/* 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-2015, 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 TComPic.cpp \brief picture class */ #include "TComPic.h" #include "SEI.h" //! \ingroup TLibCommon //! \{ // ==================================================================================================================== // Constructor / destructor / create / destroy // ==================================================================================================================== TComPic::TComPic() : m_uiTLayer (0) , m_bUsedByCurr (false) , m_bIsLongTerm (false) , m_pcPicYuvPred (NULL) , m_pcPicYuvResi (NULL) , m_bReconstructed (false) , m_bNeededForOutput (false) #if NH_MV , m_bPicOutputFlag (false) #endif , m_uiCurrSliceIdx (0) , m_bCheckLTMSB (false) #if NH_MV , m_layerId (0) , m_viewId (0) #if NH_3D , m_viewIndex (0) , m_isDepth (false) , m_aaiCodedScale (0) , m_aaiCodedOffset (0) #endif #endif { for(UInt i=0; icreate( iWidth, iHeight, chromaFormatIDC, uiMaxCuWidth, uiMaxCuHeight, uiMaxDepth, true ); m_apcPicYuv[PIC_YUV_TRUE_ORG] = new TComPicYuv; m_apcPicYuv[PIC_YUV_TRUE_ORG]->create( iWidth, iHeight, chromaFormatIDC, uiMaxCuWidth, uiMaxCuHeight, uiMaxDepth, true ); } m_apcPicYuv[PIC_YUV_REC] = new TComPicYuv; m_apcPicYuv[PIC_YUV_REC]->create( iWidth, iHeight, chromaFormatIDC, uiMaxCuWidth, uiMaxCuHeight, uiMaxDepth, true ); // there are no SEI messages associated with this picture initially if (m_SEIs.size() > 0) { deleteSEIs (m_SEIs); } m_bUsedByCurr = false; #if H_3D_FCO /* initialize the texture to depth reference status */ for (int j=0; j<2; j++) { for (int i=0; idestroy(); delete m_apcPicYuv[i]; m_apcPicYuv[i] = NULL; } } deleteSEIs(m_SEIs); } #if NH_3D #if NH_3D_ARP Void TComPic::getCUAddrAndPartIdx( Int iX, Int iY, Int& riCuAddr, Int& riAbsZorderIdx ) { Int iMaxCUWidth = (Int) ( getPicSym()->getSPS().getMaxCUWidth() ); Int iMaxCuHeight = (Int) ( getPicSym()->getSPS().getMaxCUHeight() ); UInt uiMaxTotalCUDepth = getPicSym()->getSPS().getMaxTotalCUDepth(); Int iBaseUnitWidth = iMaxCUWidth >> uiMaxTotalCUDepth; Int iBaseUnitHeight = iMaxCUWidth >> uiMaxTotalCUDepth; Int iNumCuInWidth = getPicYuvRec()->getWidth(COMPONENT_Y) / iMaxCUWidth; iNumCuInWidth += ( getPicYuvRec()->getWidth(COMPONENT_Y) % iMaxCUWidth ) ? 1 : 0; Int iCuX = iX / iMaxCUWidth; Int iCuY = iY / iMaxCuHeight; Int iBaseX = ( iX - iCuX * iMaxCUWidth ) / iBaseUnitWidth; Int iBaseY = ( iY - iCuY * iMaxCuHeight ) / iBaseUnitHeight; Int iCuSizeInBases = iMaxCuHeight / iBaseUnitWidth; riCuAddr = iCuY * iNumCuInWidth + iCuX; Int iRastPartIdx = iBaseY * iCuSizeInBases + iBaseX; riAbsZorderIdx = g_auiRasterToZscan[ iRastPartIdx ]; } #endif Void TComPic::compressMotion(Int scale) #else Void TComPic::compressMotion() #endif { TComPicSym* pPicSym = getPicSym(); for ( UInt uiCUAddr = 0; uiCUAddr < pPicSym->getNumberOfCtusInFrame(); uiCUAddr++ ) { TComDataCU* pCtu = pPicSym->getCtu(uiCUAddr); #if NH_3D pCtu->compressMV(scale); #else pCtu->compressMV(); #endif } } Bool TComPic::getSAOMergeAvailability(Int currAddr, Int mergeAddr) { Bool mergeCtbInSliceSeg = (mergeAddr >= getPicSym()->getCtuTsToRsAddrMap(getCtu(currAddr)->getSlice()->getSliceCurStartCtuTsAddr())); Bool mergeCtbInTile = (getPicSym()->getTileIdxMap(mergeAddr) == getPicSym()->getTileIdxMap(currAddr)); return (mergeCtbInSliceSeg && mergeCtbInTile); } UInt TComPic::getSubstreamForCtuAddr(const UInt ctuAddr, const Bool bAddressInRaster, TComSlice *pcSlice) { UInt subStrm; const bool bWPPEnabled=pcSlice->getPPS()->getEntropyCodingSyncEnabledFlag(); const TComPicSym &picSym = *(getPicSym()); if ((bWPPEnabled && picSym.getFrameHeightInCtus()>1) || (picSym.getNumTiles()>1)) // wavefronts, and possibly tiles being used. { if (bWPPEnabled) { const UInt ctuRsAddr = bAddressInRaster?ctuAddr : picSym.getCtuTsToRsAddrMap(ctuAddr); const UInt frameWidthInCtus = picSym.getFrameWidthInCtus(); const UInt tileIndex = picSym.getTileIdxMap(ctuRsAddr); const UInt numTileColumns = (picSym.getNumTileColumnsMinus1()+1); const TComTile *pTile = picSym.getTComTile(tileIndex); const UInt firstCtuRsAddrOfTile = pTile->getFirstCtuRsAddr(); const UInt tileYInCtus = firstCtuRsAddrOfTile / frameWidthInCtus; // independent tiles => substreams are "per tile" const UInt ctuLine = ctuRsAddr / frameWidthInCtus; const UInt startingSubstreamForTile =(tileYInCtus*numTileColumns) + (pTile->getTileHeightInCtus()*(tileIndex%numTileColumns)); subStrm = startingSubstreamForTile + (ctuLine - tileYInCtus); } else { const UInt ctuRsAddr = bAddressInRaster?ctuAddr : picSym.getCtuTsToRsAddrMap(ctuAddr); const UInt tileIndex = picSym.getTileIdxMap(ctuRsAddr); subStrm=tileIndex; } } else { // dependent tiles => substreams are "per frame". subStrm = 0; } return subStrm; } #if NH_MV Void TComPic::print( Bool legend ) { if ( legend ) std::cout << std::endl << "LId" << "\t" << "POC" << "\t" << "Rec" << "\t" << "Ref" << "\t" << "LT" << "\t" << "OutMark" << "\t" << "OutFlag" << std::endl; else std::cout << getLayerId() << "\t" << getPOC()<< "\t" << getReconMark() << "\t" << getSlice(0)->isReferenced() << "\t" << getIsLongTerm() << "\t" << getOutputMark() << "\t" << getSlice(0)->getPicOutputFlag() <*>::iterator itL = m_lists.begin(); ( itL != m_lists.end() && pcPic == NULL ); itL++) { for(TComList::iterator itP=(*itL)->begin(); ( itP!=(*itL)->end() && pcPic == NULL ); itP++) { TComPic* currPic = (*itP); if ( ( currPic->getPOC() == poc ) && ( currPic->getLayerId() == layerIdInNuh ) ) { pcPic = currPic ; } } } return pcPic; } #if NH_3D TComPic* TComPicLists::getPic( Int viewIndex, Bool depthFlag, Int poc ) { return getPic ( m_vps->getLayerIdInNuh( viewIndex, depthFlag ), poc ); } #endif Void TComPicLists::print() { Bool first = true; for(TComList*>::iterator itL = m_lists.begin(); ( itL != m_lists.end() ); itL++) { for(TComList::iterator itP=(*itL)->begin(); ( itP!=(*itL)->end() ); itP++) { if ( first ) { (*itP)->print( true ); first = false; } (*itP)->print( false ); } } } TComPicYuv* TComPicLists::getPicYuv( Int layerIdInNuh, Int poc, Bool reconFlag ) { TComPic* pcPic = getPic( layerIdInNuh, poc ); TComPicYuv* pcPicYuv = NULL; if (pcPic != NULL) { if( reconFlag ) { if ( pcPic->getReconMark() ) { pcPicYuv = pcPic->getPicYuvRec(); } } else { pcPicYuv = pcPic->getPicYuvOrg(); } }; return pcPicYuv; } #if NH_3D TComPicYuv* TComPicLists::getPicYuv( Int viewIndex, Bool depthFlag, Int poc, Bool recon ) { Int layerIdInNuh = m_vps->getLayerIdInNuh( viewIndex, depthFlag ); return getPicYuv( layerIdInNuh, poc, recon ); } #if NH_3D_ARP TComList* TComPicLists::getPicList( Int layerIdInNuh ) { TComList*>::iterator itL = m_lists.begin(); Int iLayer = 0; assert( layerIdInNuh < m_lists.size() ); while( iLayer != layerIdInNuh ) { itL++; iLayer++; } return *itL; } #endif #endif #endif // NH_MV #if NH_3D_NBDV Int TComPic::getDisCandRefPictures(Int iColPOC) { UInt uiTempLayerCurr = 7; TComSlice* currSlice = getSlice(getCurrSliceIdx()); UInt numDdvCandPics = 0; if(!currSlice->getEnableTMVPFlag()) { return numDdvCandPics; } numDdvCandPics += 1; UInt pocCurr = currSlice->getPOC(); UInt pocDiff = 255; for(UInt lpNr = 0; lpNr < (currSlice->isInterB() ? 2: 1); lpNr++) { Bool x = lpNr ? currSlice->getColFromL0Flag() : 1 - currSlice->getColFromL0Flag(); for(UInt i = 0; i < currSlice->getNumRefIdx(RefPicList(x)); i++) { if(currSlice->getViewIndex() == currSlice->getRefPic((RefPicList)x, i)->getViewIndex() && (x == currSlice->getColFromL0Flag() || currSlice->getRefPOC((RefPicList)x, i) != iColPOC) && numDdvCandPics != 2) { TComSlice* refSlice = currSlice->getRefPic((RefPicList)x, i)->getSlice(getCurrSliceIdx()); Bool bRAP = (refSlice->getViewIndex() && refSlice->isIRAP()) ? 1 : 0; UInt uiTempLayer = currSlice->getRefPic((RefPicList)x, i)->getSlice(getCurrSliceIdx())->getTLayer(); if(bRAP) { this->setRapRefIdx(i); this->setRapRefList((RefPicList)x); numDdvCandPics = 2; return numDdvCandPics; } else if (uiTempLayerCurr > uiTempLayer) { uiTempLayerCurr = uiTempLayer; } } } } UInt z = -1; // GT: Added to make code compile needs to be checked! UInt idx = 0; for(UInt lpNr = 0; lpNr < (currSlice->isInterB() ? 2 : 1); lpNr++) { Bool x = lpNr ? currSlice->getColFromL0Flag() : 1 - currSlice->getColFromL0Flag(); for(UInt i = 0; i < currSlice->getNumRefIdx(RefPicList(x)); i++) { Int iTempPoc = currSlice->getRefPic((RefPicList)x, i)->getPOC(); Int iTempDiff = (iTempPoc > pocCurr) ? (iTempPoc - pocCurr) : (pocCurr - iTempPoc); if(currSlice->getViewIndex() == currSlice->getRefPic((RefPicList)x, i)->getViewIndex() && (x == currSlice->getColFromL0Flag() || currSlice->getRefPOC((RefPicList)x, i) != iColPOC) && currSlice->getRefPic((RefPicList)x, i)->getSlice(getCurrSliceIdx())->getTLayer() == uiTempLayerCurr && pocDiff > iTempDiff) { pocDiff = iTempDiff; z = x; idx = i; } } } if(pocDiff < 255) { this->setRapRefIdx(idx); this->setRapRefList((RefPicList) z); numDdvCandPics = 2; } return numDdvCandPics; } Void TComPic::checkTemporalIVRef() { TComSlice* currSlice = getSlice(getCurrSliceIdx()); const Int numCandPics = this->getNumDdvCandPics(); for(Int curCandPic = 0; curCandPic < numCandPics; curCandPic++) { RefPicList eCurRefPicList = REF_PIC_LIST_0 ; Int curCandPicRefIdx = 0; if(curCandPic == 0) { eCurRefPicList = RefPicList(currSlice->isInterB() ? 1 - currSlice->getColFromL0Flag() : 0); curCandPicRefIdx = currSlice->getColRefIdx(); } else { eCurRefPicList = this->getRapRefList(); curCandPicRefIdx = this->getRapRefIdx(); } TComPic* pcCandColPic = currSlice->getRefPic(eCurRefPicList, curCandPicRefIdx); TComSlice* pcCandColSlice = pcCandColPic->getSlice(0); // currently only support single slice if(!pcCandColSlice->isIntra()) { for(Int iColRefDir = 0; iColRefDir < (pcCandColSlice->isInterB() ? 2 : 1); iColRefDir++) { for(Int iColRefIdx = 0; iColRefIdx < pcCandColSlice->getNumRefIdx((RefPicList)iColRefDir); iColRefIdx++) { m_abTIVRINCurrRL[curCandPic][iColRefDir][iColRefIdx] = false; Int iColViewIdx = pcCandColSlice->getViewIndex(); Int iColRefViewIdx = pcCandColSlice->getRefPic((RefPicList)iColRefDir, iColRefIdx)->getViewIndex(); if(iColViewIdx == iColRefViewIdx) { continue; } for(Int iCurrRefDir = 0; (iCurrRefDir < (currSlice->isInterB() ? 2 : 1)) && (m_abTIVRINCurrRL[curCandPic][iColRefDir][iColRefIdx] == false); iCurrRefDir++) { for(Int iCurrRefIdx = 0; iCurrRefIdx < currSlice->getNumRefIdx((RefPicList)iCurrRefDir); iCurrRefIdx++) { if(currSlice->getRefPic((RefPicList)iCurrRefDir, iCurrRefIdx)->getViewIndex() == iColRefViewIdx) { m_abTIVRINCurrRL[curCandPic][iColRefDir][iColRefIdx] = true; break; } } } } } } } } Bool TComPic::isTempIVRefValid(Int currCandPic, Int iColRefDir, Int iColRefIdx) { return m_abTIVRINCurrRL[currCandPic][iColRefDir][iColRefIdx]; } Void TComPic::checkTextureRef() { TComSlice* pcCurrSlice = getSlice(getCurrSliceIdx()); TComPic* pcTextPic = pcCurrSlice->getIvPic(0, getViewIndex()); #if H_3D_FCO if ( pcTextPic ) { #endif TComSlice* pcTextSlice = pcTextPic->getSlice(0); // currently only support single slice for(Int iTextRefDir = 0; (iTextRefDir < (pcTextSlice->isInterB() ? 2 :1) ) && !pcTextSlice->isIntra(); iTextRefDir++) { for(Int iTextRefIdx =0; iTextRefIdx < pcTextSlice->getNumRefIdx((RefPicList)iTextRefDir); iTextRefIdx++) { Int iTextRefPOC = pcTextSlice->getRefPOC((RefPicList)iTextRefDir, iTextRefIdx); Int iTextRefViewId = pcTextSlice->getRefPic((RefPicList)iTextRefDir, iTextRefIdx)->getViewIndex(); m_aiTexToDepRef[iTextRefDir][iTextRefIdx] = -1; Int iCurrRefDir = iTextRefDir; for(Int iCurrRefIdx = 0; (iCurrRefIdxgetNumRefIdx((RefPicList)iCurrRefDir)) && (m_aiTexToDepRef[iTextRefDir][iTextRefIdx] < 0); iCurrRefIdx++) { if(pcCurrSlice->getRefPOC((RefPicList)iCurrRefDir, iCurrRefIdx ) == iTextRefPOC && pcCurrSlice->getRefPic((RefPicList)iCurrRefDir, iCurrRefIdx)->getViewIndex() == iTextRefViewId) { m_aiTexToDepRef[iTextRefDir][iTextRefIdx] = iCurrRefIdx; } } } } #if H_3D_FCO } #endif } Int TComPic::isTextRefValid(Int iTextRefDir, Int iTextRefIdx) { return m_aiTexToDepRef[iTextRefDir][iTextRefIdx]; } #endif //! \}