/* 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 TComYuv.cpp \brief general YUV buffer class \todo this should be merged with TComPicYuv */ #include #include #include #include #include "CommonDef.h" #include "TComYuv.h" #include "TComInterpolationFilter.h" //! \ingroup TLibCommon //! \{ TComYuv::TComYuv() { for(Int comp=0; comp>uiPartDepth; const Int iHeight = getHeight(compID)>>uiPartDepth; const Pel* pSrc = getAddr(compID, uiPartIdx, iWidth); Pel* pDst = pcPicYuvDst->getAddr ( compID, ctuRsAddr, uiAbsZorderIdx ); const UInt iSrcStride = getStride(compID); const UInt iDstStride = pcPicYuvDst->getStride(compID); for ( Int y = iHeight; y != 0; y-- ) { ::memcpy( pDst, pSrc, sizeof(Pel)*iWidth); pDst += iDstStride; pSrc += iSrcStride; } } Void TComYuv::copyFromPicYuv ( const TComPicYuv* pcPicYuvSrc, const UInt ctuRsAddr, const UInt uiAbsZorderIdx ) { for(Int comp=0; compgetAddr ( compID, ctuRsAddr, uiAbsZorderIdx ); const UInt iDstStride = getStride(compID); const UInt iSrcStride = pcPicYuvSrc->getStride(compID); const Int iWidth=getWidth(compID); const Int iHeight=getHeight(compID); for (Int y = iHeight; y != 0; y-- ) { ::memcpy( pDst, pSrc, sizeof(Pel)*iWidth); pDst += iDstStride; pSrc += iSrcStride; } } Void TComYuv::copyToPartYuv( TComYuv* pcYuvDst, const UInt uiDstPartIdx ) const { for(Int comp=0; compgetAddr( compID, uiDstPartIdx ); const UInt iSrcStride = getStride(compID); const UInt iDstStride = pcYuvDst->getStride(compID); const Int iWidth=getWidth(compID); const Int iHeight=getHeight(compID); for (Int y = iHeight; y != 0; y-- ) { ::memcpy( pDst, pSrc, sizeof(Pel)*iWidth); pDst += iDstStride; pSrc += iSrcStride; } } Void TComYuv::copyPartToYuv( TComYuv* pcYuvDst, const UInt uiSrcPartIdx ) const { for(Int comp=0; compgetAddr(compID, 0 ); const UInt iSrcStride = getStride(compID); const UInt iDstStride = pcYuvDst->getStride(compID); const UInt uiHeight = pcYuvDst->getHeight(compID); const UInt uiWidth = pcYuvDst->getWidth(compID); for ( UInt y = uiHeight; y != 0; y-- ) { ::memcpy( pDst, pSrc, sizeof(Pel)*uiWidth); pDst += iDstStride; pSrc += iSrcStride; } } Void TComYuv::copyPartToPartYuv ( TComYuv* pcYuvDst, const UInt uiPartIdx, const UInt iWidth, const UInt iHeight ) const { for(Int comp=0; comp>getComponentScaleX(ComponentID(comp)), iHeight>>getComponentScaleY(ComponentID(comp)) ); } } Void TComYuv::copyPartToPartComponent ( const ComponentID compID, TComYuv* pcYuvDst, const UInt uiPartIdx, const UInt iWidthComponent, const UInt iHeightComponent ) const { const Pel* pSrc = getAddr(compID, uiPartIdx); Pel* pDst = pcYuvDst->getAddr(compID, uiPartIdx); if( pSrc == pDst ) { //th not a good idea //th best would be to fix the caller return ; } const UInt iSrcStride = getStride(compID); const UInt iDstStride = pcYuvDst->getStride(compID); for ( UInt y = iHeightComponent; y != 0; y-- ) { ::memcpy( pDst, pSrc, iWidthComponent * sizeof(Pel) ); pSrc += iSrcStride; pDst += iDstStride; } } Void TComYuv::copyPartToPartComponentMxN ( const ComponentID compID, TComYuv* pcYuvDst, const TComRectangle &rect) const { const Pel* pSrc = getAddrPix( compID, rect.x0, rect.y0 ); Pel* pDst = pcYuvDst->getAddrPix( compID, rect.x0, rect.y0 ); if( pSrc == pDst ) { //th not a good idea //th best would be to fix the caller return ; } const UInt iSrcStride = getStride(compID); const UInt iDstStride = pcYuvDst->getStride(compID); const UInt uiHeightComponent=rect.height; const UInt uiWidthComponent=rect.width; for ( UInt y = uiHeightComponent; y != 0; y-- ) { ::memcpy( pDst, pSrc, uiWidthComponent * sizeof( Pel ) ); pSrc += iSrcStride; pDst += iDstStride; } } Void TComYuv::addClip( const TComYuv* pcYuvSrc0, const TComYuv* pcYuvSrc1, const UInt uiTrUnitIdx, const UInt uiPartSize, const BitDepths &clipBitDepths ) { for(Int comp=0; comp>getComponentScaleX(compID); const Int uiPartHeight=uiPartSize>>getComponentScaleY(compID); const Pel* pSrc0 = pcYuvSrc0->getAddr(compID, uiTrUnitIdx, uiPartWidth ); const Pel* pSrc1 = pcYuvSrc1->getAddr(compID, uiTrUnitIdx, uiPartWidth ); Pel* pDst = getAddr(compID, uiTrUnitIdx, uiPartWidth ); const UInt iSrc0Stride = pcYuvSrc0->getStride(compID); const UInt iSrc1Stride = pcYuvSrc1->getStride(compID); const UInt iDstStride = getStride(compID); const Int clipbd = clipBitDepths.recon[toChannelType(compID)]; #if O0043_BEST_EFFORT_DECODING const Int bitDepthDelta = clipBitDepths.stream[toChannelType(compID)] - clipbd; #endif for ( Int y = uiPartHeight-1; y >= 0; y-- ) { for ( Int x = uiPartWidth-1; x >= 0; x-- ) { #if O0043_BEST_EFFORT_DECODING pDst[x] = Pel(ClipBD( Int(pSrc0[x]) + rightShiftEvenRounding(pSrc1[x], bitDepthDelta), clipbd)); #else pDst[x] = Pel(ClipBD( Int(pSrc0[x]) + Int(pSrc1[x]), clipbd)); #endif } pSrc0 += iSrc0Stride; pSrc1 += iSrc1Stride; pDst += iDstStride; } } } Void TComYuv::subtract( const TComYuv* pcYuvSrc0, const TComYuv* pcYuvSrc1, const UInt uiTrUnitIdx, const UInt uiPartSize ) { for(Int comp=0; comp>getComponentScaleX(compID); const Int uiPartHeight=uiPartSize>>getComponentScaleY(compID); const Pel* pSrc0 = pcYuvSrc0->getAddr( compID, uiTrUnitIdx, uiPartWidth ); const Pel* pSrc1 = pcYuvSrc1->getAddr( compID, uiTrUnitIdx, uiPartWidth ); Pel* pDst = getAddr( compID, uiTrUnitIdx, uiPartWidth ); const Int iSrc0Stride = pcYuvSrc0->getStride(compID); const Int iSrc1Stride = pcYuvSrc1->getStride(compID); const Int iDstStride = getStride(compID); for (Int y = uiPartHeight-1; y >= 0; y-- ) { for (Int x = uiPartWidth-1; x >= 0; x-- ) { pDst[x] = pSrc0[x] - pSrc1[x]; } pSrc0 += iSrc0Stride; pSrc1 += iSrc1Stride; pDst += iDstStride; } } } Void TComYuv::addAvg( const TComYuv* pcYuvSrc0, const TComYuv* pcYuvSrc1, const UInt iPartUnitIdx, const UInt uiWidth, const UInt uiHeight, const BitDepths &clipBitDepths ) { for(Int comp=0; compgetAddr( compID, iPartUnitIdx ); const Pel* pSrc1 = pcYuvSrc1->getAddr( compID, iPartUnitIdx ); Pel* pDst = getAddr( compID, iPartUnitIdx ); const UInt iSrc0Stride = pcYuvSrc0->getStride(compID); const UInt iSrc1Stride = pcYuvSrc1->getStride(compID); const UInt iDstStride = getStride(compID); const Int clipbd = clipBitDepths.recon[toChannelType(compID)]; const Int shiftNum = std::max(2, (IF_INTERNAL_PREC - clipbd)) + 1; const Int offset = ( 1 << ( shiftNum - 1 ) ) + 2 * IF_INTERNAL_OFFS; const Int iWidth = uiWidth >> getComponentScaleX(compID); const Int iHeight = uiHeight >> getComponentScaleY(compID); if (iWidth&1) { assert(0); exit(-1); } else if (iWidth&2) { for ( Int y = 0; y < iHeight; y++ ) { for (Int x=0 ; x < iWidth; x+=2 ) { pDst[ x + 0 ] = ClipBD( rightShift(( pSrc0[ x + 0 ] + pSrc1[ x + 0 ] + offset ), shiftNum), clipbd ); pDst[ x + 1 ] = ClipBD( rightShift(( pSrc0[ x + 1 ] + pSrc1[ x + 1 ] + offset ), shiftNum), clipbd ); } pSrc0 += iSrc0Stride; pSrc1 += iSrc1Stride; pDst += iDstStride; } } else { for ( Int y = 0; y < iHeight; y++ ) { for (Int x=0 ; x < iWidth; x+=4 ) { pDst[ x + 0 ] = ClipBD( rightShift(( pSrc0[ x + 0 ] + pSrc1[ x + 0 ] + offset ), shiftNum), clipbd ); pDst[ x + 1 ] = ClipBD( rightShift(( pSrc0[ x + 1 ] + pSrc1[ x + 1 ] + offset ), shiftNum), clipbd ); pDst[ x + 2 ] = ClipBD( rightShift(( pSrc0[ x + 2 ] + pSrc1[ x + 2 ] + offset ), shiftNum), clipbd ); pDst[ x + 3 ] = ClipBD( rightShift(( pSrc0[ x + 3 ] + pSrc1[ x + 3 ] + offset ), shiftNum), clipbd ); } pSrc0 += iSrc0Stride; pSrc1 += iSrc1Stride; pDst += iDstStride; } } } } Void TComYuv::removeHighFreq( const TComYuv* pcYuvSrc, const UInt uiPartIdx, const UInt uiWidth, const UInt uiHeight, const Int bitDepths[MAX_NUM_CHANNEL_TYPE], const Bool bClipToBitDepths ) { for(Int comp=0; compgetAddr(compID, uiPartIdx); Pel* pDst = getAddr(compID, uiPartIdx); const Int iSrcStride = pcYuvSrc->getStride(compID); const Int iDstStride = getStride(compID); const Int iWidth = uiWidth >>getComponentScaleX(compID); const Int iHeight = uiHeight>>getComponentScaleY(compID); if (bClipToBitDepths) { const Int clipBd=bitDepths[toChannelType(compID)]; for ( Int y = iHeight-1; y >= 0; y-- ) { for ( Int x = iWidth-1; x >= 0; x-- ) { pDst[x ] = ClipBD((2 * pDst[x]) - pSrc[x], clipBd); } pSrc += iSrcStride; pDst += iDstStride; } } else { for ( Int y = iHeight-1; y >= 0; y-- ) { for ( Int x = iWidth-1; x >= 0; x-- ) { pDst[x ] = (2 * pDst[x]) - pSrc[x]; } pSrc += iSrcStride; pDst += iDstStride; } } } } //! \}