📄 motioncompensation.cpp
字号:
Int iRefIdx0 = rcMbDataAccess.getMbMotionData( LIST_0 ).getRefIdx( ePar8x8 );
Int iRefIdx1 = rcMbDataAccess.getMbMotionData( LIST_1 ).getRefIdx( ePar8x8 );
//TMM_EC {{
if( rcMbDataAccess.getSH().m_eErrorConceal == EC_TEMPORAL_DIRECT)
{
Bool bOneMv = false;
ROFRS( rcMbDataAccess.getMvPredictorDirectVirtual( c8x8Idx.b8x8(), bOneMv, true, rcRefFrameListL0, rcRefFrameListL1 ), Err::m_nOK );
iRefIdx0 = rcMbDataAccess.getMbMotionData( LIST_0).getRefIdx( c8x8Idx.b8x8());
iRefIdx1 = rcMbDataAccess.getMbMotionData( LIST_1).getRefIdx( c8x8Idx.b8x8());
}
//TMM_EC }}
IntFrame* pcRefFrame0 = ( iRefIdx0 > 0 ? rcRefFrameListL0[ iRefIdx0 ] : NULL );
IntFrame* pcRefFrame1 = ( iRefIdx1 > 0 ? rcRefFrameListL1[ iRefIdx1 ] : NULL );
if( rcMbDataAccess.getSH().getSPS().getDirect8x8InferenceFlag() )
{
xGetBlkPredData( rcMbDataAccess, pcRefFrame0, pcRefFrame1, cMC8x8D, BLK_8x8 );
m_pcSampleWeighting->getTargetBuffers( apcTarBuffer, pcRecBuffer, cMC8x8D.m_apcPW[LIST_0], cMC8x8D.m_apcPW[LIST_1] );
xPredLuma( apcTarBuffer, 8, 8, cMC8x8D, SPART_4x4_0, bSR );
xPredChroma( apcTarBuffer, 4, 4, cMC8x8D, SPART_4x4_0 );
}
else
{
xGetBlkPredData( rcMbDataAccess, pcRefFrame0, pcRefFrame1, cMC8x8D, BLK_4x4 );
m_pcSampleWeighting->getTargetBuffers( apcTarBuffer, pcRecBuffer, cMC8x8D.m_apcPW[LIST_0], cMC8x8D.m_apcPW[LIST_1] );
xPredLuma( apcTarBuffer, 4, 4, cMC8x8D, SPART_4x4_0, bSR );
xPredLuma( apcTarBuffer, 4, 4, cMC8x8D, SPART_4x4_1, bSR );
xPredLuma( apcTarBuffer, 4, 4, cMC8x8D, SPART_4x4_2, bSR );
xPredLuma( apcTarBuffer, 4, 4, cMC8x8D, SPART_4x4_3, bSR );
xPredChroma( apcTarBuffer, 2, 2, cMC8x8D, SPART_4x4_0 );
xPredChroma( apcTarBuffer, 2, 2, cMC8x8D, SPART_4x4_1 );
xPredChroma( apcTarBuffer, 2, 2, cMC8x8D, SPART_4x4_2 );
xPredChroma( apcTarBuffer, 2, 2, cMC8x8D, SPART_4x4_3 );
}
m_pcSampleWeighting->weightLumaSamples ( pcRecBuffer, 8, 8, c8x8Idx, cMC8x8D.m_apcPW[LIST_0], cMC8x8D.m_apcPW[LIST_1] );
m_pcSampleWeighting->weightChromaSamples( pcRecBuffer, 4, 4, c8x8Idx, cMC8x8D.m_apcPW[LIST_0], cMC8x8D.m_apcPW[LIST_1] );
return Err::m_nOK;
}
ErrVal MotionCompensation::initMb( UInt uiMbY, UInt uiMbX, MbDataAccess& rcMbDataAccess )// TMM_INTERLACE
{
UInt uiMbInFrameY = m_uiMbInFrameY;
if( rcMbDataAccess.getMbData().getFieldFlag() )
{
uiMbInFrameY >>= 1;
uiMbY >>= 1;
}
m_cMin.setHor( (Short) max( (Int)MSYS_SHORT_MIN, (Int)((-(Int)uiMbX << 4) - (16+8) ) << 2 ) );
m_cMin.setVer( (Short) max( (Int)MSYS_SHORT_MIN, (Int)((-(Int)uiMbY << 4) - (16+8) ) << 2 ) );
m_cMax.setHor( (Short) min( (Int)MSYS_SHORT_MAX, (Int)(((m_uiMbInFrameX - uiMbX) << 4) + 8 ) << 2 ) );
m_cMax.setVer( (Short) min( (Int)MSYS_SHORT_MAX, (Int)((( uiMbInFrameY - uiMbY) << 4) + 8 ) << 2 ) );
return Err::m_nOK;
}
__inline Void MotionCompensation::xPredChroma( YuvMbBuffer* pcDesBuffer, YuvPicBuffer* pcSrcBuffer, LumaIdx cIdx, Mv cMv, Int iSizeY, Int iSizeX)
{
const Int iDesStride = pcDesBuffer->getCStride();
const Int iSrcStride = pcSrcBuffer->getCStride();
cMv.limitComponents( m_cMin, m_cMax );
Int iOffset = (cMv.getHor() >> 3) + (cMv.getVer() >> 3) * iSrcStride;
if( m_bRCDOC )
{
iOffset = ( ( cMv.getHor() + ( (Int)m_uiFrameNum & 1 ) ) >> 3 ) + ( ( cMv.getVer() + ( (Int)m_uiFrameNum & 1 ) ) >> 3 ) * iSrcStride;
}
xPredChromaPel( pcDesBuffer->getUBlk( cIdx ), iDesStride,
pcSrcBuffer->getUBlk( cIdx )+ iOffset, iSrcStride,
cMv, iSizeY, iSizeX );
xPredChromaPel( pcDesBuffer->getVBlk( cIdx ), iDesStride,
pcSrcBuffer->getVBlk( cIdx )+ iOffset, iSrcStride,
cMv, iSizeY, iSizeX );
}
/*
Void MotionCompensation::xGetMbPredData( MbDataAccess& rcMbDataAccess,
const IntFrame* pcRefFrame,
IntMC8x8D& rcMC8x8D )
{
rcMC8x8D.clear();
const SliceHeader& rcSH = rcMbDataAccess.getSH();
Int nMax = rcSH.isInterB() ? 2:1;
Int iPredCount = 0;
for( Int n = 0; n < nMax; n++)
{
ListIdx eLstIdx = ListIdx( n );
Mv3D& rcMv3D = rcMC8x8D.m_aacMv[eLstIdx][0];
rcMbDataAccess.getMbMotionData( eLstIdx ).getMv3D( rcMv3D, rcMC8x8D.m_cIdx );
if( BLOCK_NOT_PREDICTED != rcMv3D.getRef() )
{
rcMv3D.limitComponents( m_cMin, m_cMax );
rcMC8x8D.m_sChromaOffset[eLstIdx] = xCorrectChromaMv( rcMbDataAccess, pcRefFrame->getPicType() );
rcMC8x8D.m_apcRefBuffer[eLstIdx] = const_cast<IntFrame*>(pcRefFrame)->getFullPelYuvBuffer();
iPredCount++;
}
}
if( ( 2 == iPredCount ) && ( 2 == rcSH.getPPS().getWeightedBiPredIdc() ) )
{
Int iScal = rcSH.getDistScaleFactorWP( apcFrame[0], apcFrame[1] );
rcMC8x8D.m_acPW[LIST_1].scaleL1Weight( iScal );
rcMC8x8D.m_acPW[LIST_0].scaleL0Weight( rcMC8x8D.m_acPW[LIST_1] );
rcMC8x8D.m_apcPW[LIST_1] = &rcMC8x8D.m_acPW[LIST_1];
rcMC8x8D.m_apcPW[LIST_0] = &rcMC8x8D.m_acPW[LIST_0];
}
}
__inline Void MotionCompensation::xGetBlkPredData( MbDataAccess& rcMbDataAccess, const IntFrame* pcRefFrame, IntMC8x8D& rcMC8x8D, BlkMode eBlkMode )
{
rcMC8x8D.clear();
Int nMax = rcMbDataAccess.getSH().isInterB() ? 2:1;
Int iPredCount = 0;
for( Int n = 0; n < nMax; n++)
{
ListIdx eLstIdx = ListIdx( n );
Mv3D& rcMv3D = rcMC8x8D.m_aacMv[eLstIdx][SPART_4x4_0];
const MbMotionData& rcMbMotionData = rcMbDataAccess.getMbMotionData( eLstIdx );
rcMbMotionData.getMv3D( rcMv3D, rcMC8x8D.m_cIdx + SPART_4x4_0 );
rcMv3D.limitComponents( m_cMin, m_cMax );
if( BLOCK_NOT_PREDICTED != rcMv3D.getRef() )
{
iPredCount++;
rcMC8x8D.m_sChromaOffset[eLstIdx] = xCorrectChromaMv( rcMbDataAccess, pcRefFrame->getPicType() );
rcMC8x8D.m_apcRefBuffer[eLstIdx] = const_cast<IntFrame*>(pcRefFrame)->getFullPelYuvBuffer();
switch( eBlkMode )
{
case BLK_8x8:
break;
case BLK_8x4:
{
rcMC8x8D.m_aacMv[eLstIdx][SPART_4x4_2] = rcMbMotionData.getMv( rcMC8x8D.m_cIdx + SPART_4x4_2 );
rcMC8x8D.m_aacMv[eLstIdx][SPART_4x4_2].limitComponents( m_cMin, m_cMax );
}
break;
case BLK_4x8:
{
rcMC8x8D.m_aacMv[eLstIdx][SPART_4x4_1] = rcMbMotionData.getMv( rcMC8x8D.m_cIdx + SPART_4x4_1 );
rcMC8x8D.m_aacMv[eLstIdx][SPART_4x4_1].limitComponents( m_cMin, m_cMax );
}
break;
case BLK_SKIP:
case BLK_4x4:
{
rcMC8x8D.m_aacMv[eLstIdx][SPART_4x4_1] = rcMbMotionData.getMv( rcMC8x8D.m_cIdx + SPART_4x4_1 );
rcMC8x8D.m_aacMv[eLstIdx][SPART_4x4_1].limitComponents( m_cMin, m_cMax );
rcMC8x8D.m_aacMv[eLstIdx][SPART_4x4_2] = rcMbMotionData.getMv( rcMC8x8D.m_cIdx + SPART_4x4_2 );
rcMC8x8D.m_aacMv[eLstIdx][SPART_4x4_2].limitComponents( m_cMin, m_cMax );
rcMC8x8D.m_aacMv[eLstIdx][SPART_4x4_3] = rcMbMotionData.getMv( rcMC8x8D.m_cIdx + SPART_4x4_3 );
rcMC8x8D.m_aacMv[eLstIdx][SPART_4x4_3].limitComponents( m_cMin, m_cMax );
}
break;
default:
{
AOT(1);
}
}
}
}
if( ( 2 == iPredCount ) && ( 2 == rcSH.getPPS().getWeightedBiPredIdc() ) )
{
Int iScal = rcSH.getDistScaleFactorWP( apcFrame[0], apcFrame[1] );
rcMC8x8D.m_acPW[LIST_1].scaleL1Weight( iScal );
rcMC8x8D.m_acPW[LIST_0].scaleL0Weight( rcMC8x8D.m_acPW[LIST_1] );
rcMC8x8D.m_apcPW[LIST_1] = &rcMC8x8D.m_acPW[LIST_1];
rcMC8x8D.m_apcPW[LIST_0] = &rcMC8x8D.m_acPW[LIST_0];
}
}*/
Void MotionCompensation::xPredLuma( IntYuvMbBuffer* pcRecBuffer, Int iSizeX, Int iSizeY, IntMC8x8D& rcMc8x8D, Bool bSR )
{
IntYuvMbBuffer* apcTarBuffer[2];
m_pcSampleWeighting->getTargetBuffers( apcTarBuffer, pcRecBuffer, rcMc8x8D.m_apcPW[LIST_0], rcMc8x8D.m_apcPW[LIST_1] );
for( Int n = 0; n < 2; n++ )
{
IntYuvPicBuffer* pcRefBuffer = rcMc8x8D.m_apcRefBuffer[n];
if( NULL != pcRefBuffer )
{
const Mv cMv = rcMc8x8D.m_aacMv[n][0];
if( bSR )
m_pcQuarterPelFilter->predBlkSR ( apcTarBuffer[n], pcRefBuffer, rcMc8x8D.m_cIdx, cMv, iSizeY, iSizeX );
else
m_pcQuarterPelFilter->predBlk( apcTarBuffer[n], pcRefBuffer, rcMc8x8D.m_cIdx, cMv, iSizeY, iSizeX );
}
}
m_pcSampleWeighting->weightLumaSamples( pcRecBuffer, iSizeX, iSizeY, rcMc8x8D.m_cIdx, rcMc8x8D.m_apcPW[LIST_0], rcMc8x8D.m_apcPW[LIST_1] );
}
Void MotionCompensation::xPredLuma( IntYuvMbBuffer* apcTarBuffer[2], Int iSizeX, Int iSizeY, IntMC8x8D& rcMc8x8D, SParIdx4x4 eSParIdx, Bool bSR )
{
B4x4Idx cIdx( rcMc8x8D.m_cIdx + eSParIdx );
for( Int n = 0; n < 2; n++ )
{
IntYuvPicBuffer* pcRefBuffer = rcMc8x8D.m_apcRefBuffer[n];
if( NULL != pcRefBuffer )
{
const Mv cMv = rcMc8x8D.m_aacMv[n][eSParIdx];
if( bSR )
m_pcQuarterPelFilter->predBlkSR ( apcTarBuffer[n], pcRefBuffer, cIdx, cMv, iSizeY, iSizeX );
else
m_pcQuarterPelFilter->predBlk( apcTarBuffer[n], pcRefBuffer, cIdx, cMv, iSizeY, iSizeX );
}
}
}
__inline Void MotionCompensation::xPredChromaPel( XPel* pucDest, Int iDestStride, XPel* pucSrc, Int iSrcStride, Mv cMv, Int iSizeY, Int iSizeX )
{
if( m_bRCDOC )
{
xPredChromaPelRCDO( pucDest, iDestStride, pucSrc, iSrcStride, cMv, iSizeY, iSizeX );
return;
}
Int xF1 = cMv.getHor();
Int yF1 = cMv.getVer();
xF1 &= 0x7;
yF1 &= 0x7;
Int x, y;
if( 0 == xF1 )
{
if( 0 == yF1 )
{
for( y = 0; y < iSizeY; y++ )
{
for( x = 0; x < iSizeX; x++ )
{
pucDest[x ] = pucSrc[x ];
}
pucDest += iDestStride;
pucSrc += iSrcStride;
}
return;
}
Int yF0 = 8 - yF1;
for( y = 0; y < iSizeY; y++ )
{
for( x = 0; x < iSizeX; x++ )
{
#if AR_FGS_COMPENSATE_SIGNED_FRAME
pucDest[x ] = SIGNED_ROUNDING( yF0 * pucSrc[x ] + yF1 * pucSrc[x+iSrcStride ], 4, 3 );
#else
pucDest[x ] = (yF0 * pucSrc[x ] + yF1 * pucSrc[x+iSrcStride ] + 4) >> 3;
#endif
}
pucDest += iDestStride;
pucSrc += iSrcStride;
}
return;
}
if( 0 == yF1 )
{
Int xF0 = 8 - xF1;
for( y = 0; y < iSizeY; y++ )
{
for( x = 0; x < iSizeX; x++ )
{
#if AR_FGS_COMPENSATE_SIGNED_FRAME
pucDest[x] = SIGNED_ROUNDING( xF0 * pucSrc[x] + xF1 * pucSrc[x+1], 4, 3 );
#else
pucDest[x] = (xF0 * pucSrc[x] + xF1 * pucSrc[x+1] + 4) >> 3;
#endif
}
pucDest += iDestStride;
pucSrc += iSrcStride;
}
return;
}
Int xF0 = 8 - xF1;
Int yF0 = 8 - yF1;
for( y = 0; y < iSizeY; y++ )
{
Int a = xF0* ( yF0 * pucSrc[0] + yF1 * pucSrc[iSrcStride]);
for( x = 0; x < iSizeX; x++ )
{
Int b = yF0 * pucSrc[x+1] + yF1 * pucSrc[iSrcStride+x+1];
Int c = xF1 * b;
#if AR_FGS_COMPENSATE_SIGNED_FRAME
pucDest[x] = SIGNED_ROUNDING( a + c, 0x20, 6 );
#else
pucDest[x] = (a + c + 0x20) >> 6;
#endif
a = (b<<3) - c;
}
pucDest += iDestStride;
pucSrc += iSrcStride;
}
}
__inline Void MotionCompensation::xPredChromaPelRCDO( XPel* pucDest, Int iDestStride, XPel* pucSrc, Int iSrcStride, Mv cMv, Int iSizeY, Int iSizeX )
{
Int iDx = ( ( cMv.getHor() + ( (Int)m_uiFrameNum & 1 ) ) >> 1 ) & 3;
Int iDy = ( ( cMv.getVer() + ( (Int)m_uiFrameNum & 1 ) ) >> 1 ) & 3;
for( Int y = 0; y < iSizeY; y++ )
{
for( Int x = 0; x < iSizeX; x++ )
{
if( iDx == 0 && iDy == 0 )
{
pucDest[x] = pucSrc[x];
}
else if( iDy == 0 )
{
XPel A = pucSrc[x];
XPel B = pucSrc[x+1];
XPel b = ( A + B ) >> 1;
pucDest[x] = ( iDx == 1 ? (A+b+1)>>1 : iDx == 2 ? b : (b+B+1)>> 1 );
}
else if( iDx == 0 )
{
XPel A = pucSrc[x];
XPel C = pucSrc[x+iSrcStride];
XPel h = ( A + C ) >> 1;
pucDest[x] = ( iDy == 1 ? (A+h+1)>>1 : iDy == 2 ? h : (h+C+1)>> 1 );
}
else
{
XPel A = pucSrc[x];
XPel B = pucSrc[x+1];
XPel C = pucSrc[x+iSrcStride];
XPel D = pucSrc[x+iSrcStride+1];
XPel b = ( A + B ) >> 1;
XPel h = ( A + C ) >> 1;
XPel j = ( B + C ) >> 1;
XPel m = ( B + D ) >> 1;
XPel s = ( C + D ) >> 1;
if ( iDy == 1 ) pucDest[x] = ( iDx == 1 ? (b+h+1)>>1 : iDx == 2 ? (A+m+1)>>1 : (b+m+1)>> 1 );
else if( iDy == 2 ) pucDest[x] = ( iDx == 1 ? (C+b+1)>>1 : iDx == 2 ? j : (s+B+1)>> 1 );
else pucDest[x] = ( iDx == 1 ? (s+h+1)>>1 : iDx == 2 ? (D+h+1)>>1 : (s+m+1)>> 1 );
}
}
pucDest += iDestStride;
pucSrc += iSrcStride;
}
}
__inline Void MotionCompensation::xPredChromaPelRCDO( Pel* pucDest, Int iDestStride, Pel* pucSrc, Int iSrcStride, Mv cMv, Int iSizeY, Int iSizeX )
{
Int iDx = ( ( cMv.getHor() + ( (Int)m_uiFrameNum & 1 ) ) >> 1 ) & 3;
Int iDy = ( ( cMv.getVer() + ( (Int)m_uiFrameNum & 1 ) ) >> 1 ) & 3;
for( Int y = 0; y < iSizeY; y++ )
{
for( Int x = 0; x < iSizeX; x++ )
{
if( iDx == 0 && iDy == 0 )
{
pucDest[x] = pucSrc[x];
}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -