diff --git a/cfg/encoder_lowdelay_vtm.cfg b/cfg/encoder_lowdelay_vtm.cfg index 2e9644e55279a9aa6b47d7f7869839f1435a35c2..04f5162bb32143178abeb7f528c3831c85cad6c7 100644 --- a/cfg/encoder_lowdelay_vtm.cfg +++ b/cfg/encoder_lowdelay_vtm.cfg @@ -130,6 +130,7 @@ ALF : 1 GBi : 1 GBiFast : 1 MHIntra : 1 +Triangle : 1 # Fast tools PBIntraFast : 1 diff --git a/cfg/encoder_randomaccess_vtm.cfg b/cfg/encoder_randomaccess_vtm.cfg index 0bce5026353f659f813953a3a0bbdb8da98c8345..7e8715098d3ba50270ef06cc844ca593cf4a3272 100644 --- a/cfg/encoder_randomaccess_vtm.cfg +++ b/cfg/encoder_randomaccess_vtm.cfg @@ -145,6 +145,7 @@ GBi : 1 GBiFast : 1 BIO : 1 MHIntra : 1 +Triangle : 1 # Fast tools PBIntraFast : 1 diff --git a/source/App/EncoderApp/EncApp.cpp b/source/App/EncoderApp/EncApp.cpp index 9648f03c03c30349f44bd21e7d3b883e998c3768..92c6553243a3c9a2904de05520b68cc95026367c 100644 --- a/source/App/EncoderApp/EncApp.cpp +++ b/source/App/EncoderApp/EncApp.cpp @@ -265,6 +265,9 @@ void EncApp::xInitLibCfg() #endif #if JVET_L0100_MULTI_HYPOTHESIS_INTRA m_cEncLib.setUseMHIntra ( m_MHIntra ); +#endif +#if JVET_L0124_L0208_TRIANGLE + m_cEncLib.setUseTriangle ( m_Triangle ); #endif // ADD_NEW_TOOL : (encoder app) add setting of tool enabling flags and associated parameters here diff --git a/source/App/EncoderApp/EncAppCfg.cpp b/source/App/EncoderApp/EncAppCfg.cpp index 61ef4ebaa7cd82bbb43a32e39dec4f51c1bd556c..308a838595d2753c8262cf93b45a987f976cb392 100644 --- a/source/App/EncoderApp/EncAppCfg.cpp +++ b/source/App/EncoderApp/EncAppCfg.cpp @@ -870,6 +870,9 @@ bool EncAppCfg::parseCfg( int argc, char* argv[] ) #endif #if JVET_L0100_MULTI_HYPOTHESIS_INTRA ("MHIntra", m_MHIntra, false, "Enable MHIntra mode") +#endif +#if JVET_L0124_L0208_TRIANGLE + ("Triangle", m_Triangle, false, "Enable triangular shape motion vector prediction (0:off, 1:on)") #endif // ADD_NEW_TOOL : (encoder app) add parsing parameters here @@ -1968,6 +1971,9 @@ bool EncAppCfg::xCheckParameter() #if JVET_L0646_GBI xConfirmPara( m_GBi, "GBi is only allowed with NEXT profile" ); xConfirmPara( m_GBiFast, "GBiFast is only allowed with NEXT profile" ); +#endif +#if JVET_L0124_L0208_TRIANGLE + xConfirmPara( m_Triangle, "Triangle is only allowed with NEXT profile" ); #endif // ADD_NEW_TOOL : (parameter check) add a check for next tools here } @@ -3197,6 +3203,9 @@ void EncAppCfg::xPrintParameter() #endif #if JVET_L0100_MULTI_HYPOTHESIS_INTRA msg(VERBOSE, "MHIntra:%d ", m_MHIntra); +#endif +#if JVET_L0124_L0208_TRIANGLE + msg( VERBOSE, "Triangle:%d ", m_Triangle ); #endif } // ADD_NEW_TOOL (add some output indicating the usage of tools) diff --git a/source/App/EncoderApp/EncAppCfg.h b/source/App/EncoderApp/EncAppCfg.h index f71a85b0aacc33f679224e4fef13c831a17c4a59..2ef4e2e32a18aa9a58de8b2e8ca579c47ecb6423 100644 --- a/source/App/EncoderApp/EncAppCfg.h +++ b/source/App/EncoderApp/EncAppCfg.h @@ -242,6 +242,9 @@ protected: #if JVET_L0100_MULTI_HYPOTHESIS_INTRA bool m_MHIntra; +#endif +#if JVET_L0124_L0208_TRIANGLE + bool m_Triangle; #endif // ADD_NEW_TOOL : (encoder app) add tool enabling flags and associated parameters here diff --git a/source/Lib/CommonLib/CodingStatistics.h b/source/Lib/CommonLib/CodingStatistics.h index 3c82f4abe427a601accff05b0727b5253c64edec..a00a49b10802ccd7c733f7be961b5aef31f84d95 100644 --- a/source/Lib/CommonLib/CodingStatistics.h +++ b/source/Lib/CommonLib/CodingStatistics.h @@ -108,6 +108,10 @@ enum CodingStatisticsType STATS__TOOL_EMT, #if JVET_L0100_MULTI_HYPOTHESIS_INTRA STATS__CABAC_BITS__MH_INTRA_FLAG, +#endif +#if JVET_L0124_L0208_TRIANGLE + STATS__CABAC_BITS__TRIANGLE_FLAG, + STATS__CABAC_BITS__TRIANGLE_INDEX, #endif STATS__TOOL_TOTAL, STATS__NUM_STATS @@ -183,6 +187,10 @@ static inline const char* getName(CodingStatisticsType name) "CABAC_BITS__EMT_TU_INDX", #if JVET_L0100_MULTI_HYPOTHESIS_INTRA "CABAC_BITS__MH_INTRA_FLAG", +#endif +#if JVET_L0124_L0208_TRIANGLE + "CABAC_BITS__TRIANGLE_FLAG", + "CABAC_BITS__TRIANGLE_INDEX", #endif "CABAC_BITS__OTHER", "CABAC_BITS__INVALID", diff --git a/source/Lib/CommonLib/CommonDef.h b/source/Lib/CommonLib/CommonDef.h index a75d7eb89385e4b98595249c57766ddfb3d74245..7e979bfa9d3f6291760e964bff6e4334285d968f 100644 --- a/source/Lib/CommonLib/CommonDef.h +++ b/source/Lib/CommonLib/CommonDef.h @@ -421,6 +421,14 @@ static const int NTAPS_BILINEAR = 2; ///< Numb #if JVET_L0198_L0468_L0104_ATMVP_8x8SUB_BLOCK static const int ATMVP_SUB_BLOCK_SIZE = 3; ///< sub-block size for ATMVP #endif +#if JVET_L0124_L0208_TRIANGLE +static const int TRIANGLE_MAX_NUM_UNI_CANDS = 5; +static const int TRIANGLE_MAX_NUM_CANDS_MEM = 7; +static const int TRIANGLE_MAX_NUM_CANDS = 40; +static const int TRIANGLE_MAX_NUM_SATD_CANDS = 3; +static const int TRIANGLE_MIN_SIZE = 8 * 8; +#endif + // ==================================================================================================================== // Macro functions // ==================================================================================================================== diff --git a/source/Lib/CommonLib/ContextModelling.cpp b/source/Lib/CommonLib/ContextModelling.cpp index 300e861c353bcc1df0c66440dbf597703be6218d..f2787ae42e8e46772458ec93d9761e096f0212f2 100644 --- a/source/Lib/CommonLib/ContextModelling.cpp +++ b/source/Lib/CommonLib/ContextModelling.cpp @@ -359,6 +359,22 @@ unsigned DeriveCtx::CtxBTsplit(const CodingStructure& cs, Partitioner& partition return ctx; } +#if JVET_L0124_L0208_TRIANGLE +unsigned DeriveCtx::CtxTriangleFlag( const CodingUnit& cu ) +{ + const CodingStructure *cs = cu.cs; + unsigned ctxId = 0; + + const CodingUnit *cuLeft = cs->getCURestricted( cu.lumaPos().offset( -1, 0 ), cu, CH_L ); + ctxId = ( cuLeft && cuLeft->triangle ) ? 1 : 0; + + const CodingUnit *cuAbove = cs->getCURestricted( cu.lumaPos().offset( 0, -1 ), cu, CH_L ); + ctxId += ( cuAbove && cuAbove->triangle ) ? 1 : 0; + + return ctxId; +} +#endif + void MergeCtx::setMergeInfo( PredictionUnit& pu, int candIdx ) { diff --git a/source/Lib/CommonLib/ContextModelling.h b/source/Lib/CommonLib/ContextModelling.h index 2e03bc1b47438c1a70d8f388a8d7c2e89d624ea8..806b6cc577d9747f07f4d6b07e8cad960ee6420d 100644 --- a/source/Lib/CommonLib/ContextModelling.h +++ b/source/Lib/CommonLib/ContextModelling.h @@ -353,6 +353,9 @@ unsigned CtxInterDir ( const PredictionUnit& pu ); unsigned CtxSkipFlag ( const CodingUnit& cu ); unsigned CtxIMVFlag ( const CodingUnit& cu ); unsigned CtxAffineFlag( const CodingUnit& cu ); +#if JVET_L0124_L0208_TRIANGLE +unsigned CtxTriangleFlag( const CodingUnit& cu ); +#endif } #endif // __CONTEXTMODELLING__ diff --git a/source/Lib/CommonLib/Contexts.cpp b/source/Lib/CommonLib/Contexts.cpp index 7fd0d1de7e2fa8afcfe2ed1927042b49666cdd88..4130a64624c5ce1c4569267d6aebd5cf197a2358 100644 --- a/source/Lib/CommonLib/Contexts.cpp +++ b/source/Lib/CommonLib/Contexts.cpp @@ -823,6 +823,22 @@ const CtxSet ContextSetCfg::MHIntraPredMode = ContextSetCfg::addCtxSet }); #endif +#if JVET_L0124_L0208_TRIANGLE +const CtxSet ContextSetCfg::TriangleFlag = ContextSetCfg::addCtxSet +({ + { 151, 137, 154, }, + { 151, 137, 154, }, + { CNU, CNU, CNU, }, +}); + +const CtxSet ContextSetCfg::TriangleIdx = ContextSetCfg::addCtxSet +({ + { 140, }, + { 140, }, + { CNU, }, +}); +#endif + const unsigned ContextSetCfg::NumberOfContexts = (unsigned)ContextSetCfg::sm_InitTables[0].size(); diff --git a/source/Lib/CommonLib/Contexts.h b/source/Lib/CommonLib/Contexts.h index 31b4bafe13982f50e6aa9f528b0ad637b5486737..718a8b2341f47cffe95ab659dc08f0122cfa964e 100644 --- a/source/Lib/CommonLib/Contexts.h +++ b/source/Lib/CommonLib/Contexts.h @@ -206,6 +206,10 @@ public: #if JVET_L0100_MULTI_HYPOTHESIS_INTRA static const CtxSet MHIntraFlag; static const CtxSet MHIntraPredMode; +#endif +#if JVET_L0124_L0208_TRIANGLE + static const CtxSet TriangleFlag; + static const CtxSet TriangleIdx; #endif static const unsigned NumberOfContexts; diff --git a/source/Lib/CommonLib/InterPrediction.cpp b/source/Lib/CommonLib/InterPrediction.cpp index 52eb710aa8a30bf8df8e98cf73da557540c0c9e0..bfc4b47599020c6ea68b829935566cd0c6c56758 100644 --- a/source/Lib/CommonLib/InterPrediction.cpp +++ b/source/Lib/CommonLib/InterPrediction.cpp @@ -120,6 +120,10 @@ void InterPrediction::destroy() } } +#if JVET_L0124_L0208_TRIANGLE + m_triangleBuf.destroy(); +#endif + #if JVET_L0265_AFF_MINIMUM4X4 if (m_storedMv != nullptr) { @@ -175,6 +179,9 @@ void InterPrediction::init( RdCost* pcRdCost, ChromaFormat chromaFormatIDC ) } } +#if JVET_L0124_L0208_TRIANGLE + m_triangleBuf.create(UnitArea(chromaFormatIDC, Area(0, 0, MAX_CU_SIZE, MAX_CU_SIZE))); +#endif m_iRefListIdx = -1; @@ -488,7 +495,11 @@ void InterPrediction::xPredInterBi(PredictionUnit& pu, PelUnitBuf &pcYuvPred) } else { +#if JVET_L0124_L0208_TRIANGLE + xPredInterUni ( pu, eRefPicList, pcMbBuf, pu.cu->triangle, false ); +#else xPredInterUni ( pu, eRefPicList, pcMbBuf, false ); +#endif } } } @@ -1076,10 +1087,24 @@ void InterPrediction::xWeightedAverage( const PredictionUnit& pu, const CPelUnit } else if( iRefIdx0 >= 0 && iRefIdx1 < 0 ) { +#if JVET_L0124_L0208_TRIANGLE + if( pu.cu->triangle ) + { + pcYuvDst.copyFrom( pcYuvSrc0 ); + } + else +#endif pcYuvDst.copyClip( pcYuvSrc0, clpRngs ); } else if( iRefIdx0 < 0 && iRefIdx1 >= 0 ) { +#if JVET_L0124_L0208_TRIANGLE + if( pu.cu->triangle ) + { + pcYuvDst.copyFrom( pcYuvSrc1 ); + } + else +#endif pcYuvDst.copyClip( pcYuvSrc1, clpRngs ); } } @@ -1159,6 +1184,118 @@ int InterPrediction::rightShiftMSB(int numer, int denom) } #endif +#if JVET_L0124_L0208_TRIANGLE +void InterPrediction::motionCompensation4Triangle( CodingUnit &cu, MergeCtx &triangleMrgCtx, const bool splitDir, const uint8_t candIdx0, const uint8_t candIdx1 ) +{ + for( auto &pu : CU::traversePUs( cu ) ) + { + const UnitArea localUnitArea( cu.cs->area.chromaFormat, Area( 0, 0, pu.lwidth(), pu.lheight() ) ); + PelUnitBuf tmpTriangleBuf = m_triangleBuf.getBuf( localUnitArea ); + PelUnitBuf predBuf = cu.cs->getPredBuf( pu ); + + triangleMrgCtx.setMergeInfo( pu, candIdx0 ); + PU::spanMotionInfo( pu ); + motionCompensation( pu, tmpTriangleBuf ); + + triangleMrgCtx.setMergeInfo( pu, candIdx1 ); + PU::spanMotionInfo( pu ); + motionCompensation( pu, predBuf ); + + weightedTriangleBlk( pu, PU::getTriangleWeights(pu, triangleMrgCtx, candIdx0, candIdx1), splitDir, MAX_NUM_CHANNEL_TYPE, predBuf, tmpTriangleBuf, predBuf ); + } +} + +void InterPrediction::weightedTriangleBlk( PredictionUnit &pu, bool weights, const bool splitDir, int32_t channel, PelUnitBuf& predDst, PelUnitBuf& predSrc0, PelUnitBuf& predSrc1 ) +{ + if( channel == CHANNEL_TYPE_LUMA ) + { + xWeightedTriangleBlk( pu, pu.lumaSize().width, pu.lumaSize().height, COMPONENT_Y, splitDir, weights, predDst, predSrc0, predSrc1 ); + } + else if( channel == CHANNEL_TYPE_CHROMA ) + { + xWeightedTriangleBlk( pu, pu.chromaSize().width, pu.chromaSize().height, COMPONENT_Cb, splitDir, weights, predDst, predSrc0, predSrc1 ); + xWeightedTriangleBlk( pu, pu.chromaSize().width, pu.chromaSize().height, COMPONENT_Cr, splitDir, weights, predDst, predSrc0, predSrc1 ); + } + else + { + xWeightedTriangleBlk( pu, pu.lumaSize().width, pu.lumaSize().height, COMPONENT_Y, splitDir, weights, predDst, predSrc0, predSrc1 ); + xWeightedTriangleBlk( pu, pu.chromaSize().width, pu.chromaSize().height, COMPONENT_Cb, splitDir, weights, predDst, predSrc0, predSrc1 ); + xWeightedTriangleBlk( pu, pu.chromaSize().width, pu.chromaSize().height, COMPONENT_Cr, splitDir, weights, predDst, predSrc0, predSrc1 ); + } +} + +void InterPrediction::xWeightedTriangleBlk( const PredictionUnit &pu, const uint32_t width, const uint32_t height, const ComponentID compIdx, const bool splitDir, const bool weights, PelUnitBuf& predDst, PelUnitBuf& predSrc0, PelUnitBuf& predSrc1 ) +{ + Pel* dst = predDst .get(compIdx).buf; + Pel* src0 = predSrc0.get(compIdx).buf; + Pel* src1 = predSrc1.get(compIdx).buf; + int32_t strideDst = predDst .get(compIdx).stride - width; + int32_t strideSrc0 = predSrc0.get(compIdx).stride - width; + int32_t strideSrc1 = predSrc1.get(compIdx).stride - width; + + const char log2WeightBase = 3; + const ClpRng clipRng = pu.cu->slice->clpRngs().comp[compIdx]; + const int32_t clipbd = clipRng.bd; + const int32_t shiftDefault = std::max<int>(2, (IF_INTERNAL_PREC - clipbd)); + const int32_t offsetDefault = (1<<(shiftDefault-1)) + IF_INTERNAL_OFFS; + const int32_t shiftWeighted = std::max<int>(2, (IF_INTERNAL_PREC - clipbd)) + log2WeightBase; + const int32_t offsetWeighted = (1 << (shiftWeighted - 1)) + (IF_INTERNAL_OFFS << log2WeightBase); + + const int32_t ratioWH = (width > height) ? (width / height) : 1; + const int32_t ratioHW = (width > height) ? 1 : (height / width); + const Pel* pelWeighted = (compIdx == COMPONENT_Y) ? g_trianglePelWeightedLuma[splitDir][weights] : g_trianglePelWeightedChroma[predDst.chromaFormat == CHROMA_444 ? 0 : 1][splitDir][weights]; + const int32_t weightedLength = (compIdx == COMPONENT_Y) ? g_triangleWeightLengthLuma[weights] : g_triangleWeightLengthChroma[predDst.chromaFormat == CHROMA_444 ? 0 : 1][weights]; + int32_t weightedStartPos = ( splitDir == 0 ) ? ( 0 - (weightedLength >> 1) * ratioWH ) : ( width - ((weightedLength + 1) >> 1) * ratioWH ); + int32_t weightedEndPos = weightedStartPos + weightedLength * ratioWH - 1; + int32_t weightedPosoffset =( splitDir == 0 ) ? ratioWH : -ratioWH; + + const Pel* tmpPelWeighted; + int32_t x, y, tmpX, tmpY, tmpWeightedStart, tmpWeightedEnd; + + for( y = 0; y < height; y+= ratioHW ) + { + for( tmpY = ratioHW; tmpY > 0; tmpY-- ) + { + for( x = 0; x < weightedStartPos; x++ ) + { + *dst++ = ClipPel( rightShift( (splitDir == 0 ? *src1 : *src0) + offsetDefault, shiftDefault), clipRng ); + src0++; + src1++; + } + + tmpWeightedStart = std::max((int32_t)0, weightedStartPos); + tmpWeightedEnd = std::min(weightedEndPos, (int32_t)(width - 1)); + tmpPelWeighted = pelWeighted; + if( weightedStartPos < 0 ) + { + tmpPelWeighted += abs(weightedStartPos) / ratioWH; + } + for( x = tmpWeightedStart; x <= tmpWeightedEnd; x+= ratioWH ) + { + for( tmpX = ratioWH; tmpX > 0; tmpX-- ) + { + *dst++ = ClipPel( rightShift( ((*tmpPelWeighted)*(*src0++) + ((8 - (*tmpPelWeighted)) * (*src1++)) + offsetWeighted), shiftWeighted ), clipRng ); + } + tmpPelWeighted++; + } + + for( x = weightedEndPos + 1; x < width; x++ ) + { + *dst++ = ClipPel( rightShift( (splitDir == 0 ? *src0 : *src1) + offsetDefault, shiftDefault ), clipRng ); + src0++; + src1++; + } + + dst += strideDst; + src0 += strideSrc0; + src1 += strideSrc1; + } + weightedStartPos += weightedPosoffset; + weightedEndPos += weightedPosoffset; + } +} +#endif + #if JVET_J0090_MEMORY_BANDWITH_MEASURE void InterPrediction::cacheAssign( CacheModel *cache ) { diff --git a/source/Lib/CommonLib/InterPrediction.h b/source/Lib/CommonLib/InterPrediction.h index eb7cf48ddb00103e77765fe5bdb7aab4d35ef198..0f61defe0ddfcb662aac1e3924354d91f11b7936 100644 --- a/source/Lib/CommonLib/InterPrediction.h +++ b/source/Lib/CommonLib/InterPrediction.h @@ -91,6 +91,9 @@ protected: RdCost* m_pcRdCost; int m_iRefListIdx; +#if JVET_L0124_L0208_TRIANGLE + PelStorage m_triangleBuf; +#endif #if JVET_L0265_AFF_MINIMUM4X4 Mv* m_storedMv; #endif @@ -127,6 +130,10 @@ protected: #endif void xPredAffineBlk( const ComponentID& compID, const PredictionUnit& pu, const Picture* refPic, const Mv* _mv, PelUnitBuf& dstPic, const bool& bi, const ClpRng& clpRng ); +#if JVET_L0124_L0208_TRIANGLE + void xWeightedTriangleBlk ( const PredictionUnit &pu, const uint32_t width, const uint32_t height, const ComponentID compIdx, const bool splitDir, const bool weights, PelUnitBuf& predDst, PelUnitBuf& predSrc0, PelUnitBuf& predSrc1 ); +#endif + static bool xCheckIdenticalMotion( const PredictionUnit& pu ); void xSubPuMC(PredictionUnit& pu, PelUnitBuf& predBuf, const RefPicList &eRefPicList = REF_PIC_LIST_X); @@ -149,6 +156,11 @@ public: void motionCompensation (CodingUnit &cu, const RefPicList &eRefPicList = REF_PIC_LIST_X ); +#if JVET_L0124_L0208_TRIANGLE + void motionCompensation4Triangle( CodingUnit &cu, MergeCtx &triangleMrgCtx, const bool splitDir, const uint8_t candIdx0, const uint8_t candIdx1 ); + void weightedTriangleBlk ( PredictionUnit &pu, bool weights, const bool splitDir, int32_t channel, PelUnitBuf& predDst, PelUnitBuf& predSrc0, PelUnitBuf& predSrc1 ); +#endif + #if JVET_J0090_MEMORY_BANDWITH_MEASURE void cacheAssign( CacheModel *cache ); #endif diff --git a/source/Lib/CommonLib/Rom.cpp b/source/Lib/CommonLib/Rom.cpp index e9972bf93357d10e5c46ee4c78d857561c3b3a3b..2c25d72c14c8d4d72970135808ab4f3d6272862e 100644 --- a/source/Lib/CommonLib/Rom.cpp +++ b/source/Lib/CommonLib/Rom.cpp @@ -528,6 +528,30 @@ void initROM() //-------------------------------------------------------------------------------------------------- } } + +#if JVET_L0124_L0208_TRIANGLE + for( int idxH = MAX_CU_DEPTH - MIN_CU_LOG2; idxH >= 0; --idxH ) + { + for( int idxW = MAX_CU_DEPTH - MIN_CU_LOG2; idxW >= 0; --idxW ) + { + int numW = 1 << idxW; + int numH = 1 << idxH; + int ratioW = std::max( 0, idxW - idxH ); + int ratioH = std::max( 0, idxH - idxW ); + int sum = std::max( (numW >> ratioW), (numH >> ratioH) ) - 1; + for( int y = 0; y < numH; y++ ) + { + int idxY = y >> ratioH; + for( int x = 0; x < numW; x++ ) + { + int idxX = x >> ratioW; + g_triangleMvStorage[TRIANGLE_DIR_135][idxH][idxW][y][x] = (idxX == idxY) ? 2 : (idxX > idxY ? 0 : 1); + g_triangleMvStorage[TRIANGLE_DIR_45][idxH][idxW][y][x] = (idxX + idxY == sum) ? 2 : (idxX + idxY > sum ? 1 : 0); + } + } + } + } +#endif } void destroyROM() @@ -883,5 +907,65 @@ const uint8_t g_NonMPM[257] = { 0, 0, 1, 1, 2, 2, 2, 2, 3, 3, 3, 3, 3, 3, 3, 3, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 7, 8 }; +#if JVET_L0124_L0208_TRIANGLE +const Pel g_trianglePelWeightedLuma[TRIANGLE_DIR_NUM][2][7] = +{ + { // TRIANGLE_DIR_135 + { 1, 2, 4, 6, 7, 0, 0 }, + { 1, 2, 3, 4, 5, 6, 7 } + }, + { // TRIANGLE_DIR_45 + { 7, 6, 4, 2, 1, 0, 0 }, + { 7, 6, 5, 4, 3, 2, 1 } + } +}; +const Pel g_trianglePelWeightedChroma[2][TRIANGLE_DIR_NUM][2][7] = +{ + { // 444 format + { // TRIANGLE_DIR_135 + { 1, 2, 4, 6, 7, 0, 0 }, + { 1, 2, 3, 4, 5, 6, 7 } + }, + { // TRIANGLE_DIR_45 + { 7, 6, 4, 2, 1, 0, 0 }, + { 7, 6, 5, 4, 3, 2, 1 } + } + }, + { // 420 format + { // TRIANGLE_DIR_135 + { 1, 4, 7, 0, 0, 0, 0 }, + { 2, 4, 6, 0, 0, 0, 0 } + }, + { // TRIANGLE_DIR_45 + { 7, 4, 1, 0, 0, 0, 0 }, + { 6, 4, 2, 0, 0, 0, 0 } + } + } +}; + +const uint8_t g_triangleWeightLengthLuma[2] = { 5, 7 }; +const uint8_t g_triangleWeightLengthChroma[2][2] = { { 5, 7 }, { 3, 3 } }; + + uint8_t g_triangleMvStorage[TRIANGLE_DIR_NUM][MAX_CU_DEPTH - MIN_CU_LOG2 + 1][MAX_CU_DEPTH - MIN_CU_LOG2 + 1][MAX_CU_SIZE >> MIN_CU_LOG2][MAX_CU_SIZE >> MIN_CU_LOG2]; +const uint8_t g_triangleCombination[TRIANGLE_MAX_NUM_CANDS][3] = +{ + { 0, 1, 0 }, { 1, 0, 1 }, { 1, 0, 2 }, { 0, 0, 1 }, { 0, 2, 0 }, + { 1, 0, 3 }, { 1, 0, 4 }, { 1, 1, 0 }, { 0, 3, 0 }, { 0, 4, 0 }, + { 0, 0, 2 }, { 0, 1, 2 }, { 1, 1, 2 }, { 0, 0, 4 }, { 0, 0, 3 }, + { 0, 1, 3 }, { 0, 1, 4 }, { 1, 1, 4 }, { 1, 1, 3 }, { 1, 2, 1 }, + { 1, 2, 0 }, { 0, 2, 1 }, { 0, 4, 3 }, { 1, 3, 0 }, { 1, 3, 2 }, + { 1, 3, 4 }, { 1, 4, 0 }, { 1, 3, 1 }, { 1, 2, 3 }, { 1, 4, 1 }, + { 0, 4, 1 }, { 0, 2, 3 }, { 1, 4, 2 }, { 0, 3, 2 }, { 1, 4, 3 }, + { 0, 3, 1 }, { 0, 2, 4 }, { 1, 2, 4 }, { 0, 4, 2 }, { 0, 3, 4 }, +}; + +const uint8_t g_triangleIdxBins[TRIANGLE_MAX_NUM_CANDS] = +{ + 2, 2, 4, 4, 4, 4, 6, 6, 6, 6, + 6, 6, 6, 6, 8, 8, 8, 8, 8, 8, + 8, 8, 8, 8, 8, 8, 8, 8, 8, 8, + 10, 10, 10, 10, 10, 10, 10, 10, 10, 10 +}; +#endif //! \} diff --git a/source/Lib/CommonLib/Rom.h b/source/Lib/CommonLib/Rom.h index ba99b1608285405bd837b834e17481f9adc61fc5..825126095943bbfaba9379e3424549c7224873ea 100644 --- a/source/Lib/CommonLib/Rom.h +++ b/source/Lib/CommonLib/Rom.h @@ -270,5 +270,15 @@ constexpr uint8_t g_tbMax[257] = { 0, 0, 1, 1, 2, 2, 2, 2, 3, 3, 3, 3, 3, 3, 3, //! \} +#if JVET_L0124_L0208_TRIANGLE +extern const Pel g_trianglePelWeightedLuma[TRIANGLE_DIR_NUM][2][7]; +extern const Pel g_trianglePelWeightedChroma[2][TRIANGLE_DIR_NUM][2][7]; +extern const uint8_t g_triangleWeightLengthLuma[2]; +extern const uint8_t g_triangleWeightLengthChroma[2][2]; +extern uint8_t g_triangleMvStorage[TRIANGLE_DIR_NUM][MAX_CU_DEPTH - MIN_CU_LOG2 + 1][MAX_CU_DEPTH - MIN_CU_LOG2 + 1][MAX_CU_SIZE >> MIN_CU_LOG2][MAX_CU_SIZE >> MIN_CU_LOG2]; +extern const uint8_t g_triangleCombination[TRIANGLE_MAX_NUM_CANDS][3]; +extern const uint8_t g_triangleIdxBins[TRIANGLE_MAX_NUM_CANDS]; +#endif + #endif //__TCOMROM__ diff --git a/source/Lib/CommonLib/Slice.cpp b/source/Lib/CommonLib/Slice.cpp index d0010f49a2ea6a7d3e264304b413459a5d4a89a2..9b0b776ba301d23b102322761c404bc9578f8789 100644 --- a/source/Lib/CommonLib/Slice.cpp +++ b/source/Lib/CommonLib/Slice.cpp @@ -1657,6 +1657,9 @@ void Slice::updateMotionLUTs(LutMotionCand* lutMC, CodingUnit & cu) { PredictionUnit *selectedPU = cu.firstPU; if (cu.affine) { return; } +#if JVET_L0124_L0208_TRIANGLE + if (cu.triangle) { return; } +#endif MotionInfo newMi = selectedPU->getMotionInfo(); addMotionInfoToLUTs(lutMC, newMi); @@ -1763,6 +1766,9 @@ SPSNext::SPSNext( SPS& sps ) #if JVET_L0100_MULTI_HYPOTHESIS_INTRA , m_MHIntra ( false ) #endif +#if JVET_L0124_L0208_TRIANGLE + , m_Triangle ( false ) +#endif #if ENABLE_WPP_PARALLELISM , m_NextDQP ( false ) #endif diff --git a/source/Lib/CommonLib/Slice.h b/source/Lib/CommonLib/Slice.h index c0dc3d739b36c354cfdcd6e35823643fb2085ce0..5bda75a751cf51c8b5a54d96c930f658a18a5471 100644 --- a/source/Lib/CommonLib/Slice.h +++ b/source/Lib/CommonLib/Slice.h @@ -823,6 +823,9 @@ private: #if JVET_L0100_MULTI_HYPOTHESIS_INTRA bool m_MHIntra; #endif +#if JVET_L0124_L0208_TRIANGLE + bool m_Triangle; +#endif #if ENABLE_WPP_PARALLELISM bool m_NextDQP; #endif @@ -976,6 +979,10 @@ public: #if JVET_L0100_MULTI_HYPOTHESIS_INTRA void setUseMHIntra ( bool b ) { m_MHIntra = b; } bool getUseMHIntra () const { return m_MHIntra; } +#endif +#if JVET_L0124_L0208_TRIANGLE + void setUseTriangle ( bool b ) { m_Triangle = b; } + bool getUseTriangle () const { return m_Triangle; } #endif // ADD_NEW_TOOL : (sps extension) add access functions for tool enabling flags and associated parameters here diff --git a/source/Lib/CommonLib/TypeDef.h b/source/Lib/CommonLib/TypeDef.h index 3e1ab50bfc0cd9f50e07ed17b0a458aa5b2c4fb8..e6e7ff8eb8f2eedc447bd8bcd469d196a64489d8 100644 --- a/source/Lib/CommonLib/TypeDef.h +++ b/source/Lib/CommonLib/TypeDef.h @@ -53,6 +53,7 @@ #define JVET_L0410_TC_TAB 1 // Change TC table for QP 51-63 #define JVET_L0136_L0085_LM_RESTRICTED_LINEBUFFER 1 // line buffer restriction in LM prediction +#define JVET_L0124_L0208_TRIANGLE 1 // triangular shape prediction unit #define JVET_L0059_MTS_SIMP 1 // Simpification on MTS signaling #define JVET_L0100_MULTI_HYPOTHESIS_INTRA 1 // Combine intra mode with an extra merge indexed prediction @@ -950,6 +951,14 @@ enum MergeType NUM_MRG_TYPE // 5 }; +#if JVET_L0124_L0208_TRIANGLE +enum TriangleSplit +{ + TRIANGLE_DIR_135 = 0, + TRIANGLE_DIR_45, + TRIANGLE_DIR_NUM +}; +#endif ////////////////////////////////////////////////////////////////////////// // Encoder modes to try out diff --git a/source/Lib/CommonLib/Unit.cpp b/source/Lib/CommonLib/Unit.cpp index 710868551a3ff96a04b571618ed7848f20bb0760..098732295cef633a568eb54b362b13bc77f64139 100644 --- a/source/Lib/CommonLib/Unit.cpp +++ b/source/Lib/CommonLib/Unit.cpp @@ -258,6 +258,9 @@ CodingUnit& CodingUnit::operator=( const CodingUnit& other ) #endif affine = other.affine; affineType = other.affineType; +#if JVET_L0124_L0208_TRIANGLE + triangle = other.triangle; +#endif transQuantBypass = other.transQuantBypass; ipcm = other.ipcm; qp = other.qp; @@ -292,6 +295,9 @@ void CodingUnit::initData() #endif affine = false; affineType = 0; +#if JVET_L0124_L0208_TRIANGLE + triangle = false; +#endif transQuantBypass = false; ipcm = false; qp = 0; diff --git a/source/Lib/CommonLib/Unit.h b/source/Lib/CommonLib/Unit.h index 48496a8e620479ec8ae890433148b4a1d52044e3..976dfdf1f9620351bfc109b7717aa33c7e3c499c 100644 --- a/source/Lib/CommonLib/Unit.h +++ b/source/Lib/CommonLib/Unit.h @@ -300,6 +300,9 @@ struct CodingUnit : public UnitArea #endif bool affine; int affineType; +#if JVET_L0124_L0208_TRIANGLE + bool triangle; +#endif bool transQuantBypass; bool ipcm; uint8_t imv; diff --git a/source/Lib/CommonLib/UnitTools.cpp b/source/Lib/CommonLib/UnitTools.cpp index fec4d894ded4b6edbd08623701c9266fc03d8775..509d3e7971aed4f02a573e1a61a27f4ac2419397 100644 --- a/source/Lib/CommonLib/UnitTools.cpp +++ b/source/Lib/CommonLib/UnitTools.cpp @@ -4073,6 +4073,495 @@ void PU::restrictBiPredMergeCands( const PredictionUnit &pu, MergeCtx& mergeCtx } } +#if JVET_L0124_L0208_TRIANGLE +void PU::getTriangleMergeCandidates( const PredictionUnit &pu, MergeCtx& triangleMrgCtx ) +{ + const CodingStructure &cs = *pu.cs; + const Slice &slice = *pu.cs->slice; + const int32_t maxNumMergeCand = TRIANGLE_MAX_NUM_UNI_CANDS; + triangleMrgCtx.numValidMergeCand = 0; + + for( int32_t i = 0; i < maxNumMergeCand; i++ ) + { + triangleMrgCtx.interDirNeighbours[i] = 0; + triangleMrgCtx.mrgTypeNeighbours [i] = MRG_TYPE_DEFAULT_N; + triangleMrgCtx.mvFieldNeighbours[(i << 1) ].refIdx = NOT_VALID; + triangleMrgCtx.mvFieldNeighbours[(i << 1) + 1].refIdx = NOT_VALID; + triangleMrgCtx.mvFieldNeighbours[(i << 1) ].mv = Mv(); + triangleMrgCtx.mvFieldNeighbours[(i << 1) + 1].mv = Mv(); + } + + MotionInfo candidate[TRIANGLE_MAX_NUM_CANDS_MEM]; + int32_t candCount = 0; + + const Position posLT = pu.Y().topLeft(); + const Position posRT = pu.Y().topRight(); + const Position posLB = pu.Y().bottomLeft(); + + MotionInfo miAbove, miLeft, miAboveLeft, miAboveRight, miBelowLeft; + + //left + const PredictionUnit* puLeft = cs.getPURestricted( posLB.offset( -1, 0 ), pu, pu.chType ); + const bool isAvailableA1 = puLeft && isDiffMER( pu, *puLeft ) && pu.cu != puLeft->cu && CU::isInter( *puLeft->cu ); + if( isAvailableA1 ) + { + miLeft = puLeft->getMotionInfo( posLB.offset(-1, 0) ); + candidate[candCount].isInter = true; + candidate[candCount].interDir = miLeft.interDir; + candidate[candCount].mv[0] = miLeft.mv[0]; + candidate[candCount].mv[1] = miLeft.mv[1]; + candidate[candCount].refIdx[0] = miLeft.refIdx[0]; + candidate[candCount].refIdx[1] = miLeft.refIdx[1]; + candCount++; + } + + // above + const PredictionUnit *puAbove = cs.getPURestricted( posRT.offset( 0, -1 ), pu, pu.chType ); + bool isAvailableB1 = puAbove && isDiffMER( pu, *puAbove ) && pu.cu != puAbove->cu && CU::isInter( *puAbove->cu ); + if( isAvailableB1 ) + { + miAbove = puAbove->getMotionInfo( posRT.offset( 0, -1 ) ); + + if( !isAvailableA1 || ( miAbove != miLeft ) ) + { + candidate[candCount].isInter = true; + candidate[candCount].interDir = miAbove.interDir; + candidate[candCount].mv[0] = miAbove.mv[0]; + candidate[candCount].mv[1] = miAbove.mv[1]; + candidate[candCount].refIdx[0] = miAbove.refIdx[0]; + candidate[candCount].refIdx[1] = miAbove.refIdx[1]; + candCount++; + } + } + + // above right + const PredictionUnit *puAboveRight = cs.getPURestricted( posRT.offset( 1, -1 ), pu, pu.chType ); + bool isAvailableB0 = puAboveRight && isDiffMER( pu, *puAboveRight ) && CU::isInter( *puAboveRight->cu ); + + if( isAvailableB0 ) + { + miAboveRight = puAboveRight->getMotionInfo( posRT.offset( 1, -1 ) ); + + if( ( !isAvailableB1 || ( miAbove != miAboveRight ) ) && ( !isAvailableA1 || ( miLeft != miAboveRight ) ) ) + { + candidate[candCount].isInter = true; + candidate[candCount].interDir = miAboveRight.interDir; + candidate[candCount].mv[0] = miAboveRight.mv[0]; + candidate[candCount].mv[1] = miAboveRight.mv[1]; + candidate[candCount].refIdx[0] = miAboveRight.refIdx[0]; + candidate[candCount].refIdx[1] = miAboveRight.refIdx[1]; + candCount++; + } + } + + //left bottom + const PredictionUnit *puLeftBottom = cs.getPURestricted( posLB.offset( -1, 1 ), pu, pu.chType ); + bool isAvailableA0 = puLeftBottom && isDiffMER( pu, *puLeftBottom ) && CU::isInter( *puLeftBottom->cu ); + if( isAvailableA0 ) + { + miBelowLeft = puLeftBottom->getMotionInfo( posLB.offset( -1, 1 ) ); + + if( ( !isAvailableA1 || ( miBelowLeft != miLeft ) ) && ( !isAvailableB1 || ( miBelowLeft != miAbove ) ) && ( !isAvailableB0 || ( miBelowLeft != miAboveRight ) ) ) + { + candidate[candCount].isInter = true; + candidate[candCount].interDir = miBelowLeft.interDir; + candidate[candCount].mv[0] = miBelowLeft.mv[0]; + candidate[candCount].mv[1] = miBelowLeft.mv[1]; + candidate[candCount].refIdx[0] = miBelowLeft.refIdx[0]; + candidate[candCount].refIdx[1] = miBelowLeft.refIdx[1]; + candCount++; + } + } + + // above left + const PredictionUnit *puAboveLeft = cs.getPURestricted( posLT.offset( -1, -1 ), pu, pu.chType ); + bool isAvailableB2 = puAboveLeft && isDiffMER( pu, *puAboveLeft ) && CU::isInter( *puAboveLeft->cu ); + + if( isAvailableB2 ) + { + miAboveLeft = puAboveLeft->getMotionInfo( posLT.offset( -1, -1 ) ); + + if( ( !isAvailableA1 || ( miLeft != miAboveLeft ) ) && ( !isAvailableB1 || ( miAbove != miAboveLeft ) ) && ( !isAvailableA0 || ( miBelowLeft != miAboveLeft ) ) && ( !isAvailableB0 || ( miAboveRight != miAboveLeft ) ) ) + { + candidate[candCount].isInter = true; + candidate[candCount].interDir = miAboveLeft.interDir; + candidate[candCount].mv[0] = miAboveLeft.mv[0]; + candidate[candCount].mv[1] = miAboveLeft.mv[1]; + candidate[candCount].refIdx[0] = miAboveLeft.refIdx[0]; + candidate[candCount].refIdx[1] = miAboveLeft.refIdx[1]; + candCount++; + } + } + + if( slice.getEnableTMVPFlag() ) + { + Position posRB = pu.Y().bottomRight().offset(-3, -3); + + const PreCalcValues& pcv = *cs.pcv; + + Position posC0; + Position posC1 = pu.Y().center(); + bool isAvailableC0 = false; + + if (((posRB.x + pcv.minCUWidth) < pcv.lumaWidth) && ((posRB.y + pcv.minCUHeight) < pcv.lumaHeight)) + { + Position posInCtu( posRB.x & pcv.maxCUWidthMask, posRB.y & pcv.maxCUHeightMask ); + + if( ( posInCtu.x + 4 < pcv.maxCUWidth ) && // is not at the last column of CTU + ( posInCtu.y + 4 < pcv.maxCUHeight ) ) // is not at the last row of CTU + { + posC0 = posRB.offset( 4, 4 ); + isAvailableC0 = true; + } + else if( posInCtu.x + 4 < pcv.maxCUWidth ) // is not at the last column of CTU But is last row of CTU + { + posC0 = posRB.offset( 4, 4 ); + // in the reference the CTU address is not set - thus probably resulting in no using this C0 possibility + } + else if( posInCtu.y + 4 < pcv.maxCUHeight ) // is not at the last row of CTU But is last column of CTU + { + posC0 = posRB.offset( 4, 4 ); + isAvailableC0 = true; + } + else //is the right bottom corner of CTU + { + posC0 = posRB.offset( 4, 4 ); + // same as for last column but not last row + } + } + + // C0 + Mv cColMv; + int32_t refIdx = 0; + bool existMV = ( isAvailableC0 && getColocatedMVP( pu, REF_PIC_LIST_0, posC0, cColMv, refIdx ) ); + MotionInfo temporalMv; + temporalMv.interDir = 0; + if( existMV ) + { + temporalMv.isInter = true; + temporalMv.interDir |= 1; + temporalMv.mv[0] = cColMv; + temporalMv.refIdx[0] = refIdx; + } + existMV = ( isAvailableC0 && getColocatedMVP( pu, REF_PIC_LIST_1, posC0, cColMv, refIdx ) ); + if( existMV ) + { + temporalMv.interDir |= 2; + temporalMv.mv[1] = cColMv; + temporalMv.refIdx[1] = refIdx; + } + + if( temporalMv.interDir != 0 ) + { + candidate[candCount].isInter = true; + candidate[candCount].interDir = temporalMv.interDir; + candidate[candCount].mv[0] = temporalMv.mv[0]; + candidate[candCount].mv[1] = temporalMv.mv[1]; + candidate[candCount].refIdx[0] = temporalMv.refIdx[0]; + candidate[candCount].refIdx[1] = temporalMv.refIdx[1]; + candCount++; + } + + // C1 + temporalMv.interDir = 0; + existMV = getColocatedMVP(pu, REF_PIC_LIST_0, posC1, cColMv, refIdx ); + if( existMV ) + { + temporalMv.isInter = true; + temporalMv.interDir |= 1; + temporalMv.mv[0] = cColMv; + temporalMv.refIdx[0] = refIdx; + } + existMV = getColocatedMVP(pu, REF_PIC_LIST_1, posC1, cColMv, refIdx ); + if( existMV ) + { + temporalMv.interDir |= 2; + temporalMv.mv[1] = cColMv; + temporalMv.refIdx[1] = refIdx; + } + + if( temporalMv.interDir != 0 ) + { + candidate[candCount].isInter = true; + candidate[candCount].interDir = temporalMv.interDir; + candidate[candCount].mv[0] = temporalMv.mv[0]; + candidate[candCount].mv[1] = temporalMv.mv[1]; + candidate[candCount].refIdx[0] = temporalMv.refIdx[0]; + candidate[candCount].refIdx[1] = temporalMv.refIdx[1]; + candCount++; + } + } + + // put uni-prediction candidate to the triangle candidate list + for( int32_t i = 0; i < candCount; i++ ) + { + if( candidate[i].interDir != 3 ) + { + triangleMrgCtx.interDirNeighbours[triangleMrgCtx.numValidMergeCand] = candidate[i].interDir; + triangleMrgCtx.mrgTypeNeighbours [triangleMrgCtx.numValidMergeCand] = MRG_TYPE_DEFAULT_N; + triangleMrgCtx.mvFieldNeighbours [(triangleMrgCtx.numValidMergeCand << 1) ].mv = candidate[i].mv[0]; + triangleMrgCtx.mvFieldNeighbours [(triangleMrgCtx.numValidMergeCand << 1) + 1].mv = candidate[i].mv[1]; + triangleMrgCtx.mvFieldNeighbours [(triangleMrgCtx.numValidMergeCand << 1) ].refIdx = candidate[i].refIdx[0]; + triangleMrgCtx.mvFieldNeighbours [(triangleMrgCtx.numValidMergeCand << 1) + 1].refIdx = candidate[i].refIdx[1]; + triangleMrgCtx.numValidMergeCand += isUniqueTriangleCandidates(pu, triangleMrgCtx); + if( triangleMrgCtx.numValidMergeCand == TRIANGLE_MAX_NUM_UNI_CANDS ) + { + return; + } + } + } + + // put L0 mv of bi-prediction candidate to the triangle candidate list + for( int32_t i = 0; i < candCount; i++ ) + { + if( candidate[i].interDir == 3 ) + { + triangleMrgCtx.interDirNeighbours[triangleMrgCtx.numValidMergeCand] = 1; + triangleMrgCtx.mrgTypeNeighbours [triangleMrgCtx.numValidMergeCand] = MRG_TYPE_DEFAULT_N; + triangleMrgCtx.mvFieldNeighbours [(triangleMrgCtx.numValidMergeCand << 1) ].mv = candidate[i].mv[0]; + triangleMrgCtx.mvFieldNeighbours [(triangleMrgCtx.numValidMergeCand << 1) + 1].mv = Mv(0, 0); + triangleMrgCtx.mvFieldNeighbours [(triangleMrgCtx.numValidMergeCand << 1) ].refIdx = candidate[i].refIdx[0]; + triangleMrgCtx.mvFieldNeighbours [(triangleMrgCtx.numValidMergeCand << 1) + 1].refIdx = -1; + triangleMrgCtx.numValidMergeCand += isUniqueTriangleCandidates(pu, triangleMrgCtx); + if( triangleMrgCtx.numValidMergeCand == TRIANGLE_MAX_NUM_UNI_CANDS ) + { + return; + } + } + } + + // put L1 mv of bi-prediction candidate to the triangle candidate list + for( int32_t i = 0; i < candCount; i++ ) + { + if( candidate[i].interDir == 3 ) + { + triangleMrgCtx.interDirNeighbours[triangleMrgCtx.numValidMergeCand] = 2; + triangleMrgCtx.mrgTypeNeighbours [triangleMrgCtx.numValidMergeCand] = MRG_TYPE_DEFAULT_N; + triangleMrgCtx.mvFieldNeighbours [(triangleMrgCtx.numValidMergeCand << 1) ].mv = Mv(0, 0); + triangleMrgCtx.mvFieldNeighbours [(triangleMrgCtx.numValidMergeCand << 1) + 1].mv = candidate[i].mv[1]; + triangleMrgCtx.mvFieldNeighbours [(triangleMrgCtx.numValidMergeCand << 1) ].refIdx = -1; + triangleMrgCtx.mvFieldNeighbours [(triangleMrgCtx.numValidMergeCand << 1) + 1].refIdx = candidate[i].refIdx[1]; + triangleMrgCtx.numValidMergeCand += isUniqueTriangleCandidates(pu, triangleMrgCtx); + if( triangleMrgCtx.numValidMergeCand == TRIANGLE_MAX_NUM_UNI_CANDS ) + { + return; + } + } + } + + // put average of L0 and L1 mvs of bi-prediction candidate to the triangle candidate list + for( int32_t i = 0; i < candCount; i++ ) + { + if( candidate[i].interDir == 3 ) + { + int32_t curPicPoc = slice.getPOC(); + int32_t refPicPocL0 = slice.getRefPOC(REF_PIC_LIST_0, candidate[i].refIdx[0]); + int32_t refPicPocL1 = slice.getRefPOC(REF_PIC_LIST_1, candidate[i].refIdx[1]); + Mv aveMv = candidate[i].mv[1]; + aveMv = aveMv.scaleMv( xGetDistScaleFactor( curPicPoc, refPicPocL0, curPicPoc, refPicPocL1 ) ); // scaling to L0 + aveMv.setHor( ( aveMv.getHor() + candidate[i].mv[0].getHor() + 1 ) >> 1 ); + aveMv.setVer( ( aveMv.getVer() + candidate[i].mv[0].getVer() + 1 ) >> 1 ); + + triangleMrgCtx.interDirNeighbours[triangleMrgCtx.numValidMergeCand] = 1; + triangleMrgCtx.mrgTypeNeighbours [triangleMrgCtx.numValidMergeCand] = MRG_TYPE_DEFAULT_N; + triangleMrgCtx.mvFieldNeighbours [(triangleMrgCtx.numValidMergeCand << 1) ].mv = aveMv; + triangleMrgCtx.mvFieldNeighbours [(triangleMrgCtx.numValidMergeCand << 1) + 1].mv = Mv(0, 0); + triangleMrgCtx.mvFieldNeighbours [(triangleMrgCtx.numValidMergeCand << 1) ].refIdx = candidate[i].refIdx[0]; + triangleMrgCtx.mvFieldNeighbours [(triangleMrgCtx.numValidMergeCand << 1) + 1].refIdx = -1; + triangleMrgCtx.numValidMergeCand += isUniqueTriangleCandidates(pu, triangleMrgCtx); + if( triangleMrgCtx.numValidMergeCand == TRIANGLE_MAX_NUM_UNI_CANDS ) + { + return; + } + } + } + + // fill with Mv(0, 0) + int32_t numRefIdx = std::min( slice.getNumRefIdx(REF_PIC_LIST_0), slice.getNumRefIdx(REF_PIC_LIST_1) ); + int32_t cnt = 0; + while( triangleMrgCtx.numValidMergeCand < TRIANGLE_MAX_NUM_UNI_CANDS ) + { + if( cnt < numRefIdx ) + { + triangleMrgCtx.interDirNeighbours[triangleMrgCtx.numValidMergeCand] = 1; + triangleMrgCtx.mvFieldNeighbours[triangleMrgCtx.numValidMergeCand << 1].setMvField(Mv(0, 0), cnt); + triangleMrgCtx.numValidMergeCand++; + + if( triangleMrgCtx.numValidMergeCand == TRIANGLE_MAX_NUM_UNI_CANDS ) + { + return; + } + + triangleMrgCtx.interDirNeighbours[triangleMrgCtx.numValidMergeCand] = 2; + triangleMrgCtx.mvFieldNeighbours [(triangleMrgCtx.numValidMergeCand << 1) + 1 ].setMvField(Mv(0, 0), cnt); + triangleMrgCtx.numValidMergeCand++; + + cnt = (cnt + 1) % numRefIdx; + } + } +} + +bool PU::isUniqueTriangleCandidates( const PredictionUnit &pu, MergeCtx& triangleMrgCtx ) +{ + int newCand = triangleMrgCtx.numValidMergeCand; + for( int32_t i = 0; i < newCand; i++ ) + { + int32_t predFlagCur = triangleMrgCtx.interDirNeighbours[i] == 1 ? 0 : 1; + int32_t predFlagNew = triangleMrgCtx.interDirNeighbours[newCand] == 1 ? 0 : 1; + int32_t refPicPocCur = pu.cs->slice->getRefPOC( (RefPicList)predFlagCur, triangleMrgCtx.mvFieldNeighbours[(i << 1) + predFlagCur].refIdx ); + int32_t refPicPocNew = pu.cs->slice->getRefPOC( (RefPicList)predFlagNew, triangleMrgCtx.mvFieldNeighbours[(newCand << 1) + predFlagNew].refIdx); + if( refPicPocCur == refPicPocNew && triangleMrgCtx.mvFieldNeighbours[(i << 1) + predFlagCur].mv == triangleMrgCtx.mvFieldNeighbours[(newCand << 1) + predFlagNew].mv ) + { + return false; + } + } + return true; +} + +bool PU::getTriangleWeights( const PredictionUnit& pu, MergeCtx &triangleMrgCtx, const uint8_t candIdx0, const uint8_t candIdx1 ) +{ + RefPicList refPicListCand0 = triangleMrgCtx.interDirNeighbours[candIdx0] == 1 ? REF_PIC_LIST_0 : REF_PIC_LIST_1; + RefPicList refPicListCand1 = triangleMrgCtx.interDirNeighbours[candIdx1] == 1 ? REF_PIC_LIST_0 : REF_PIC_LIST_1; + int32_t refPicPoc0 = pu.cs->slice->getRefPOC( refPicListCand0, triangleMrgCtx.mvFieldNeighbours[ (candIdx0 << 1) + refPicListCand0 ].refIdx ); + int32_t refPicPoc1 = pu.cs->slice->getRefPOC( refPicListCand1, triangleMrgCtx.mvFieldNeighbours[ (candIdx1 << 1) + refPicListCand1 ].refIdx ); + + if( refPicPoc0 != refPicPoc1 ) + { + // different reference picture + return true; + } + + // same reference picture, but mv difference is larger than 16 pel + int32_t threshold = 16 << 4; + Mv diffMv = triangleMrgCtx.mvFieldNeighbours[(candIdx0 << 1) + refPicListCand0].mv - triangleMrgCtx.mvFieldNeighbours[(candIdx1 << 1) + refPicListCand1].mv; + + if( diffMv.getAbsHor() > threshold || diffMv.getAbsVer() > threshold ) + { + return true; + } + + return false; +} + +void PU::spanTriangleMotionInfo( PredictionUnit &pu, MergeCtx &triangleMrgCtx, const uint8_t mergeIdx, const bool splitDir, const uint8_t candIdx0, const uint8_t candIdx1 ) +{ + pu.mergeIdx = mergeIdx; + MotionBuf mb = pu.getMotionBuf(); + + MotionInfo biMv; + biMv.isInter = true; + + if( triangleMrgCtx.interDirNeighbours[candIdx0] == 1 && triangleMrgCtx.interDirNeighbours[candIdx1] == 2 ) + { + biMv.interDir = 3; + biMv.mv[0] = triangleMrgCtx.mvFieldNeighbours[ candIdx0 << 1 ].mv; + biMv.mv[1] = triangleMrgCtx.mvFieldNeighbours[(candIdx1 << 1) + 1].mv; + biMv.refIdx[0] = triangleMrgCtx.mvFieldNeighbours[ candIdx0 << 1 ].refIdx; + biMv.refIdx[1] = triangleMrgCtx.mvFieldNeighbours[(candIdx1 << 1) + 1].refIdx; + } + else if( triangleMrgCtx.interDirNeighbours[candIdx0] == 2 && triangleMrgCtx.interDirNeighbours[candIdx1] == 1 ) + { + biMv.interDir = 3; + biMv.mv[0] = triangleMrgCtx.mvFieldNeighbours[ candIdx1 << 1 ].mv; + biMv.mv[1] = triangleMrgCtx.mvFieldNeighbours[(candIdx0 << 1) + 1].mv; + biMv.refIdx[0] = triangleMrgCtx.mvFieldNeighbours[ candIdx1 << 1 ].refIdx; + biMv.refIdx[1] = triangleMrgCtx.mvFieldNeighbours[(candIdx0 << 1) + 1].refIdx; + } + else if( triangleMrgCtx.interDirNeighbours[candIdx0] == 1 && triangleMrgCtx.interDirNeighbours[candIdx1] == 1 ) + { + int32_t refIdx = mappingRefPic( pu, pu.cs->slice->getRefPOC( REF_PIC_LIST_0, triangleMrgCtx.mvFieldNeighbours[candIdx1 << 1].refIdx ), REF_PIC_LIST_1 ); + if( refIdx != -1 ) + { + biMv.interDir = 3; + biMv.mv[0] = triangleMrgCtx.mvFieldNeighbours[candIdx0 << 1].mv; + biMv.mv[1] = triangleMrgCtx.mvFieldNeighbours[candIdx1 << 1].mv; + biMv.refIdx[0] = triangleMrgCtx.mvFieldNeighbours[candIdx0 << 1].refIdx; + biMv.refIdx[1] = refIdx; + } + else + { + refIdx = mappingRefPic( pu, pu.cs->slice->getRefPOC( REF_PIC_LIST_0, triangleMrgCtx.mvFieldNeighbours[candIdx0 << 1].refIdx), REF_PIC_LIST_1 ); + biMv.interDir = ( refIdx != -1 ) ? 3 : 1; + biMv.mv[0] = ( refIdx != -1 ) ? triangleMrgCtx.mvFieldNeighbours[candIdx1 << 1].mv : triangleMrgCtx.mvFieldNeighbours[candIdx0 << 1].mv; + biMv.mv[1] = ( refIdx != -1 ) ? triangleMrgCtx.mvFieldNeighbours[candIdx0 << 1].mv : Mv(0, 0); + biMv.refIdx[0] = ( refIdx != -1 ) ? triangleMrgCtx.mvFieldNeighbours[candIdx1 << 1].refIdx : triangleMrgCtx.mvFieldNeighbours[candIdx0 << 1].refIdx; + biMv.refIdx[1] = ( refIdx != -1 ) ? refIdx : -1; + } + } + else if( triangleMrgCtx.interDirNeighbours[candIdx0] == 2 && triangleMrgCtx.interDirNeighbours[candIdx1] == 2 ) + { + int32_t refIdx = mappingRefPic( pu, pu.cs->slice->getRefPOC( REF_PIC_LIST_1, triangleMrgCtx.mvFieldNeighbours[(candIdx1 << 1) + 1].refIdx ), REF_PIC_LIST_0 ); + if( refIdx != -1 ) + { + biMv.interDir = 3; + biMv.mv[0] = triangleMrgCtx.mvFieldNeighbours[(candIdx1 << 1) + 1].mv; + biMv.mv[1] = triangleMrgCtx.mvFieldNeighbours[(candIdx0 << 1) + 1].mv; + biMv.refIdx[0] = refIdx; + biMv.refIdx[1] = triangleMrgCtx.mvFieldNeighbours[(candIdx0 << 1) + 1].refIdx; + } + else + { + refIdx = mappingRefPic( pu, pu.cs->slice->getRefPOC( REF_PIC_LIST_1, triangleMrgCtx.mvFieldNeighbours[(candIdx0 << 1) + 1].refIdx ), REF_PIC_LIST_0 ); + biMv.interDir = ( refIdx != -1 ) ? 3 : 2; + biMv.mv[0] = ( refIdx != -1 ) ? triangleMrgCtx.mvFieldNeighbours[(candIdx0 << 1) + 1].mv : Mv(0, 0); + biMv.mv[1] = ( refIdx != -1 ) ? triangleMrgCtx.mvFieldNeighbours[(candIdx1 << 1) + 1].mv : triangleMrgCtx.mvFieldNeighbours[(candIdx0 << 1) + 1].mv; + biMv.refIdx[0] = ( refIdx != -1 ) ? refIdx : -1; + biMv.refIdx[1] = ( refIdx != -1 ) ? triangleMrgCtx.mvFieldNeighbours[(candIdx1 << 1) + 1].refIdx : triangleMrgCtx.mvFieldNeighbours[(candIdx0 << 1) + 1].refIdx; + } + } + + int32_t idxW = (int32_t)(g_aucLog2[pu.lwidth() ] - MIN_CU_LOG2); + int32_t idxH = (int32_t)(g_aucLog2[pu.lheight()] - MIN_CU_LOG2); + for( int32_t y = 0; y < mb.height; y++ ) + { + for( int32_t x = 0; x < mb.width; x++ ) + { + if( g_triangleMvStorage[splitDir][idxH][idxW][y][x] == 2 ) + { + mb.at( x, y ).isInter = true; + mb.at( x, y ).interDir = biMv.interDir; + mb.at( x, y ).refIdx[0] = biMv.refIdx[0]; + mb.at( x, y ).refIdx[1] = biMv.refIdx[1]; + mb.at( x, y ).mv [0] = biMv.mv [0]; + mb.at( x, y ).mv [1] = biMv.mv [1]; + } + else if( g_triangleMvStorage[splitDir][idxH][idxW][y][x] == 0 ) + { + mb.at( x, y ).isInter = true; + mb.at( x, y ).interDir = triangleMrgCtx.interDirNeighbours[candIdx0]; + mb.at( x, y ).refIdx[0] = triangleMrgCtx.mvFieldNeighbours[ candIdx0 << 1 ].refIdx; + mb.at( x, y ).refIdx[1] = triangleMrgCtx.mvFieldNeighbours[(candIdx0 << 1) + 1].refIdx; + mb.at( x, y ).mv [0] = triangleMrgCtx.mvFieldNeighbours[ candIdx0 << 1 ].mv; + mb.at( x, y ).mv [1] = triangleMrgCtx.mvFieldNeighbours[(candIdx0 << 1) + 1].mv; + } + else + { + mb.at( x, y ).isInter = true; + mb.at( x, y ).interDir = triangleMrgCtx.interDirNeighbours[candIdx1]; + mb.at( x, y ).refIdx[0] = triangleMrgCtx.mvFieldNeighbours[ candIdx1 << 1 ].refIdx; + mb.at( x, y ).refIdx[1] = triangleMrgCtx.mvFieldNeighbours[(candIdx1 << 1) + 1].refIdx; + mb.at( x, y ).mv [0] = triangleMrgCtx.mvFieldNeighbours[ candIdx1 << 1 ].mv; + mb.at( x, y ).mv [1] = triangleMrgCtx.mvFieldNeighbours[(candIdx1 << 1) + 1].mv; + } + } + } +} + +int32_t PU::mappingRefPic( const PredictionUnit &pu, int32_t refPicPoc, bool targetRefPicList ) +{ + int32_t numRefIdx = pu.cs->slice->getNumRefIdx( (RefPicList)targetRefPicList ); + + for( int32_t i = 0; i < numRefIdx; i++ ) + { + if( pu.cs->slice->getRefPOC( (RefPicList)targetRefPicList, i ) == refPicPoc ) + { + return i; + } + } + return -1; +} +#endif + void CU::resetMVDandMV2Int( CodingUnit& cu, InterPrediction *interPred ) { for( auto &pu : CU::traversePUs( cu ) ) diff --git a/source/Lib/CommonLib/UnitTools.h b/source/Lib/CommonLib/UnitTools.h index 9645cd77397d84a13a370566eaf63c325870e918..2b517549724865c6326351b0fc351d64a1876d2b 100644 --- a/source/Lib/CommonLib/UnitTools.h +++ b/source/Lib/CommonLib/UnitTools.h @@ -173,6 +173,13 @@ namespace PU int getMHIntraMPMs (const PredictionUnit &pu, unsigned *mpm, const ChannelType &channelType = CHANNEL_TYPE_LUMA, const bool isChromaMDMS = false, const unsigned startIdx = 0); int getNarrowShape (const int width, const int height); #endif +#if JVET_L0124_L0208_TRIANGLE + void getTriangleMergeCandidates (const PredictionUnit &pu, MergeCtx &triangleMrgCtx); + bool isUniqueTriangleCandidates (const PredictionUnit &pu, MergeCtx &triangleMrgCtx); + bool getTriangleWeights (const PredictionUnit &pu, MergeCtx &triangleMrgCtx, const uint8_t candIdx0, const uint8_t candIdx1); + void spanTriangleMotionInfo ( PredictionUnit &pu, MergeCtx &triangleMrgCtx, const uint8_t mergeIdx, const bool splitDir, const uint8_t candIdx0, const uint8_t candIdx1); + int32_t mappingRefPic (const PredictionUnit &pu, int32_t refPicPoc, bool targetRefPicList); +#endif } // TU tools diff --git a/source/Lib/DecoderLib/CABACReader.cpp b/source/Lib/DecoderLib/CABACReader.cpp index 10b6febe124a4a91f4935cc80f1109534bb8d144..a4404254d4dcc5279e51691228bf5b248a260418 100644 --- a/source/Lib/DecoderLib/CABACReader.cpp +++ b/source/Lib/DecoderLib/CABACReader.cpp @@ -1217,6 +1217,9 @@ void CABACReader::prediction_unit( PredictionUnit& pu, MergeCtx& mrgCtx ) pu.intraDir[1] = DM_CHROMA_IDX; } #endif +#if JVET_L0124_L0208_TRIANGLE + triangle_mode( *pu.cu ); +#endif #if JVET_L0054_MMVD if (pu.mmvdMergeFlag) { @@ -1458,6 +1461,25 @@ void CABACReader::merge_idx( PredictionUnit& pu ) #endif int numCandminus1 = int( pu.cs->slice->getMaxNumMergeCand() ) - 1; pu.mergeIdx = 0; + +#if JVET_L0124_L0208_TRIANGLE + if( pu.cu->triangle ) + { + RExt__DECODER_DEBUG_BIT_STATISTICS_CREATE_SET( STATS__CABAC_BITS__TRIANGLE_INDEX ); + if( m_BinDecoder.decodeBin( Ctx::TriangleIdx() ) == 0 ) + { + pu.mergeIdx += m_BinDecoder.decodeBinEP(); + } + else + { + pu.mergeIdx += exp_golomb_eqprob( 2 ) + 2; + } + + DTRACE( g_trace_ctx, D_SYNTAX, "merge_idx() triangle_idx=%d\n", pu.mergeIdx ); + return; + } +#endif + if( numCandminus1 > 0 ) { if( m_BinDecoder.decodeBin( Ctx::MergeIdx() ) ) @@ -1778,6 +1800,23 @@ void CABACReader::MHIntra_luma_pred_modes(CodingUnit &cu) } #endif +#if JVET_L0124_L0208_TRIANGLE +void CABACReader::triangle_mode( CodingUnit& cu ) +{ + RExt__DECODER_DEBUG_BIT_STATISTICS_CREATE_SET( STATS__CABAC_BITS__TRIANGLE_FLAG ); + + if( !cu.cs->slice->getSPS()->getSpsNext().getUseTriangle() || !cu.cs->slice->isInterB() || cu.lwidth() * cu.lheight() < TRIANGLE_MIN_SIZE || cu.affine ) + { + return; + } + + unsigned flag_idx = DeriveCtx::CtxTriangleFlag( cu ); + cu.triangle = m_BinDecoder.decodeBin( Ctx::TriangleFlag(flag_idx) ); + + + DTRACE( g_trace_ctx, D_SYNTAX, "triangle_mode() triangle_mode=%d pos=(%d,%d) size: %dx%d\n", cu.triangle, cu.Y().x, cu.Y().y, cu.lumaSize().width, cu.lumaSize().height ); +} +#endif //================================================================================ // clause 7.3.8.7 diff --git a/source/Lib/DecoderLib/CABACReader.h b/source/Lib/DecoderLib/CABACReader.h index 153f356be96d8cb92c5342ee6badeba54c5ddd40..16699dcdf16d6bfa201ac7caebb89e50e8e5264a 100644 --- a/source/Lib/DecoderLib/CABACReader.h +++ b/source/Lib/DecoderLib/CABACReader.h @@ -111,6 +111,9 @@ public: void MHIntra_flag ( PredictionUnit& pu ); void MHIntra_luma_pred_modes ( CodingUnit& cu ); #endif +#if JVET_L0124_L0208_TRIANGLE + void triangle_mode ( CodingUnit& cu ); +#endif // pcm samples (clause 7.3.8.7) void pcm_samples ( TransformUnit& tu ); diff --git a/source/Lib/DecoderLib/DecCu.cpp b/source/Lib/DecoderLib/DecCu.cpp index 7cfd907efdd0dfd4754c04aea147d654868cd947..f1ccaae1b0830c3314dd0d66a0d43730d30b586b 100644 --- a/source/Lib/DecoderLib/DecCu.cpp +++ b/source/Lib/DecoderLib/DecCu.cpp @@ -314,12 +314,28 @@ void DecCu::xFillPCMBuffer(CodingUnit &cu) void DecCu::xReconInter(CodingUnit &cu) { +#if JVET_L0124_L0208_TRIANGLE + if( cu.triangle ) + { + const uint8_t mergeIdx = cu.firstPU->mergeIdx; + const bool splitDir = g_triangleCombination[mergeIdx][0]; + const uint8_t candIdx0 = g_triangleCombination[mergeIdx][1]; + const uint8_t candIdx1 = g_triangleCombination[mergeIdx][2]; + m_pcInterPred->motionCompensation4Triangle( cu, m_triangleMrgCtx, splitDir, candIdx0, candIdx1 ); + PU::spanTriangleMotionInfo( *cu.firstPU, m_triangleMrgCtx, mergeIdx, splitDir, candIdx0, candIdx1 ); + } + else + { +#endif #if JVET_L0100_MULTI_HYPOTHESIS_INTRA m_pcIntraPred->geneIntrainterPred(cu); #endif // inter prediction m_pcInterPred->motionCompensation( cu ); +#if JVET_L0124_L0208_TRIANGLE + } +#endif #if JVET_L0266_HMVP cu.slice->updateMotionLUTs(cu.slice->getMotionLUTs(), cu); #endif @@ -486,6 +502,14 @@ void DecCu::xDeriveCUMV( CodingUnit &cu ) { #endif { +#if JVET_L0124_L0208_TRIANGLE + if( pu.cu->triangle ) + { + PU::getTriangleMergeCandidates( pu, m_triangleMrgCtx ); + } + else + { +#endif if( pu.cu->affine ) { #if JVET_L0632_AFFINE_MERGE @@ -593,6 +617,9 @@ void DecCu::xDeriveCUMV( CodingUnit &cu ) PU::spanMotionInfo( pu, mrgCtx ); } +#if JVET_L0124_L0208_TRIANGLE + } +#endif } #if JVET_L0054_MMVD } diff --git a/source/Lib/DecoderLib/DecCu.h b/source/Lib/DecoderLib/DecCu.h index 2a61f26cc368bace2522e105a58fec042edc1c89..3a9f46d909e2ff7c9a1ba27dcecba802977a71f9 100644 --- a/source/Lib/DecoderLib/DecCu.h +++ b/source/Lib/DecoderLib/DecCu.h @@ -92,6 +92,10 @@ private: MotionInfo m_SubPuMiBuf[(MAX_CU_SIZE * MAX_CU_SIZE) >> (MIN_CU_LOG2 << 1)]; + +#if JVET_L0124_L0208_TRIANGLE + MergeCtx m_triangleMrgCtx; +#endif }; //! \} diff --git a/source/Lib/DecoderLib/VLCReader.cpp b/source/Lib/DecoderLib/VLCReader.cpp index aa7b342be6fd1b706fb30a77d078b1a08b606e07..992ae745a17a7ea28f9da241513b6dd0c6095817 100644 --- a/source/Lib/DecoderLib/VLCReader.cpp +++ b/source/Lib/DecoderLib/VLCReader.cpp @@ -818,6 +818,9 @@ void HLSyntaxReader::parseSPSNext( SPSNext& spsNext, const bool usePCM ) #if JVET_L0100_MULTI_HYPOTHESIS_INTRA READ_FLAG( symbol, "mhintra_flag" ); spsNext.setUseMHIntra ( symbol != 0 ); #endif +#if JVET_L0124_L0208_TRIANGLE + READ_FLAG( symbol, "triangle_flag" ); spsNext.setUseTriangle ( symbol != 0 ); +#endif #if ENABLE_WPP_PARALLELISM READ_FLAG( symbol, "next_dqp_enabled_flag" ); spsNext.setUseNextDQP ( symbol != 0 ); #else diff --git a/source/Lib/EncoderLib/CABACWriter.cpp b/source/Lib/EncoderLib/CABACWriter.cpp index d7b8ce61ed09b9157887c4754918f32d2bd45655..4f3ec1ddd969d0daa4b297a34b5fb1606eba148b 100644 --- a/source/Lib/EncoderLib/CABACWriter.cpp +++ b/source/Lib/EncoderLib/CABACWriter.cpp @@ -1209,6 +1209,9 @@ void CABACWriter::prediction_unit( const PredictionUnit& pu ) MHIntra_luma_pred_modes( *pu.cu ); } #endif +#if JVET_L0124_L0208_TRIANGLE + triangle_mode( *pu.cu ); +#endif #if JVET_L0054_MMVD if (pu.mmvdMergeFlag) { @@ -1431,6 +1434,24 @@ void CABACWriter::merge_idx( const PredictionUnit& pu ) } else { +#endif +#if JVET_L0124_L0208_TRIANGLE + if( pu.cu->triangle ) + { + if( pu.mergeIdx < 2 ) + { + m_BinEncoder.encodeBin( 0, Ctx::TriangleIdx() ); + m_BinEncoder.encodeBinEP( pu.mergeIdx ); + } + else + { + m_BinEncoder.encodeBin( 1, Ctx::TriangleIdx() ); + exp_golomb_eqprob( pu.mergeIdx - 2, 2 ); + } + + DTRACE( g_trace_ctx, D_SYNTAX, "merge_idx() triangle_idx=%d\n", pu.mergeIdx ); + return; + } #endif int numCandminus1 = int( pu.cs->slice->getMaxNumMergeCand() ) - 1; if( numCandminus1 > 0 ) @@ -1700,6 +1721,22 @@ void CABACWriter::MHIntra_luma_pred_modes(const CodingUnit& cu) } #endif +#if JVET_L0124_L0208_TRIANGLE +void CABACWriter::triangle_mode( const CodingUnit& cu ) +{ + if( !cu.cs->slice->getSPS()->getSpsNext().getUseTriangle() || !cu.cs->slice->isInterB() || cu.lwidth() * cu.lheight() < TRIANGLE_MIN_SIZE || cu.affine ) + { + return; + } + + unsigned flag_idx = DeriveCtx::CtxTriangleFlag( cu ); + + m_BinEncoder.encodeBin( cu.triangle, Ctx::TriangleFlag(flag_idx) ); + + DTRACE( g_trace_ctx, D_SYNTAX, "triangle_mode() triangle_mode=%d pos=(%d,%d) size: %dx%d\n", cu.triangle, cu.Y().x, cu.Y().y, cu.lumaSize().width, cu.lumaSize().height ); +} +#endif + //================================================================================ // clause 7.3.8.7 //-------------------------------------------------------------------------------- diff --git a/source/Lib/EncoderLib/CABACWriter.h b/source/Lib/EncoderLib/CABACWriter.h index d01a8d06017066384df72a61fd6581ceef97b0ac..2e25dcac24ec79910d933bb84b769415002a7a14 100644 --- a/source/Lib/EncoderLib/CABACWriter.h +++ b/source/Lib/EncoderLib/CABACWriter.h @@ -125,6 +125,9 @@ public: void MHIntra_flag ( const PredictionUnit& pu ); void MHIntra_luma_pred_modes ( const CodingUnit& cu ); #endif +#if JVET_L0124_L0208_TRIANGLE + void triangle_mode ( const CodingUnit& cu ); +#endif // pcm samples (clause 7.3.8.7) void pcm_samples ( const TransformUnit& tu ); diff --git a/source/Lib/EncoderLib/EncCfg.h b/source/Lib/EncoderLib/EncCfg.h index d346ec04c147b9679c350633b2ba812a1b3e9b7a..610e00bb93dca60a3536f7deb5988caae6e82b10 100644 --- a/source/Lib/EncoderLib/EncCfg.h +++ b/source/Lib/EncoderLib/EncCfg.h @@ -225,6 +225,9 @@ protected: #if JVET_L0100_MULTI_HYPOTHESIS_INTRA bool m_MHIntra; +#endif +#if JVET_L0124_L0208_TRIANGLE + bool m_Triangle; #endif // ADD_NEW_TOOL : (encoder lib) add tool enabling flags and associated parameters here @@ -699,6 +702,10 @@ public: void setUseMHIntra ( bool b ) { m_MHIntra = b; } bool getUseMHIntra () const { return m_MHIntra; } #endif +#if JVET_L0124_L0208_TRIANGLE + void setUseTriangle ( bool b ) { m_Triangle = b; } + bool getUseTriangle () const { return m_Triangle; } +#endif // ADD_NEW_TOOL : (encoder lib) add access functions here diff --git a/source/Lib/EncoderLib/EncCu.cpp b/source/Lib/EncoderLib/EncCu.cpp index 23a755c29d257852cb5aef781eed7826eda1fecf..d459e98d14eca69498d786bf70caa3090f33f9be 100644 --- a/source/Lib/EncoderLib/EncCu.cpp +++ b/source/Lib/EncoderLib/EncCu.cpp @@ -197,6 +197,12 @@ void EncCu::create( EncCfg* encCfg ) m_acRealMergeBuffer[ui].create(chromaFormat, Area(0, 0, uiMaxWidth, uiMaxHeight)); } #endif +#if JVET_L0124_L0208_TRIANGLE + for( unsigned ui = 0; ui < TRIANGLE_MAX_NUM_CANDS; ui++ ) + { + m_acTriangleWeightedBuffer[ui].create( chromaFormat, Area( 0, 0, uiMaxWidth, uiMaxHeight ) ); + } +#endif m_CtxBuffer.resize( maxDepth ); m_CurrCtx = 0; @@ -301,6 +307,12 @@ void EncCu::destroy() m_acRealMergeBuffer[ui].destroy(); } #endif +#if JVET_L0124_L0208_TRIANGLE + for( unsigned ui = 0; ui < TRIANGLE_MAX_NUM_CANDS; ui++ ) + { + m_acTriangleWeightedBuffer[ui].destroy(); + } +#endif } @@ -741,6 +753,12 @@ void EncCu::xCompressCU( CodingStructure *&tempCS, CodingStructure *&bestCS, Par cu->mmvdSkip = cu->skip == false ? false : cu->mmvdSkip; #endif } +#if JVET_L0124_L0208_TRIANGLE + else if( currTestMode.type == ETM_MERGE_TRIANGLE ) + { + xCheckRDCostMergeTriangle2Nx2N( tempCS, bestCS, partitioner, currTestMode ); + } +#endif else if( currTestMode.type == ETM_INTRA ) { xCheckRDCostIntra( tempCS, bestCS, partitioner, currTestMode ); @@ -1611,6 +1629,10 @@ void EncCu::xCheckRDCostMerge2Nx2N( CodingStructure *&tempCS, CodingStructure *& mergeCtx.subPuMvpMiBuf = MotionBuf( m_SubPuMiBuf, bufSize ); } +#if JVET_L0124_L0208_TRIANGLE + setMergeBestSATDCost( MAX_DOUBLE ); +#endif + { // first get merge candidates CodingUnit cu( tempCS->area ); @@ -1781,6 +1803,9 @@ void EncCu::xCheckRDCostMerge2Nx2N( CodingStructure *&tempCS, CodingStructure *& cu.skip = false; #if JVET_L0054_MMVD cu.mmvdSkip = false; +#endif +#if JVET_L0124_L0208_TRIANGLE + cu.triangle = false; #endif cu.partSize = SIZE_2Nx2N; //cu.affine @@ -2075,6 +2100,10 @@ void EncCu::xCheckRDCostMerge2Nx2N( CodingStructure *&tempCS, CodingStructure *& } } +#if JVET_L0124_L0208_TRIANGLE + setMergeBestSATDCost( candCostList[0] ); +#endif + #if JVET_L0100_MULTI_HYPOTHESIS_INTRA if (isIntrainterEnabled) { @@ -2172,6 +2201,9 @@ void EncCu::xCheckRDCostMerge2Nx2N( CodingStructure *&tempCS, CodingStructure *& cu.skip = false; #if JVET_L0054_MMVD cu.mmvdSkip = false; +#endif +#if JVET_L0124_L0208_TRIANGLE + cu.triangle = false; #endif cu.partSize = SIZE_2Nx2N; //cu.affine @@ -2338,6 +2370,233 @@ void EncCu::xCheckRDCostMerge2Nx2N( CodingStructure *&tempCS, CodingStructure *& } } +#if JVET_L0124_L0208_TRIANGLE +void EncCu::xCheckRDCostMergeTriangle2Nx2N( CodingStructure *&tempCS, CodingStructure *&bestCS, Partitioner &partitioner, const EncTestMode& encTestMode ) +{ + const Slice &slice = *tempCS->slice; + const SPS &sps = *tempCS->sps; + + CHECK( slice.getSliceType() != B_SLICE, "Triangle mode is only applied to B-slices" ); + + tempCS->initStructData( encTestMode.qp, encTestMode.lossless ); + + bool trianglecandHasNoResidual[TRIANGLE_MAX_NUM_CANDS]; + for( int mergeCand = 0; mergeCand < TRIANGLE_MAX_NUM_CANDS; mergeCand++ ) + { + trianglecandHasNoResidual[mergeCand] = false; + } + + bool bestIsSkip = m_pcEncCfg->getUseFastDecisionForMerge() ? bestCS->getCU( partitioner.chType )->rootCbf == 0 : false; + uint8_t numTriangleCandidate = TRIANGLE_MAX_NUM_CANDS; + uint8_t triangleNumMrgSATDCand = TRIANGLE_MAX_NUM_SATD_CANDS; + PelUnitBuf triangleBuffer[TRIANGLE_MAX_NUM_UNI_CANDS]; + PelUnitBuf triangleWeightedBuffer[TRIANGLE_MAX_NUM_CANDS]; + static_vector<uint8_t, TRIANGLE_MAX_NUM_CANDS> triangleRdModeList; + static_vector<double, TRIANGLE_MAX_NUM_CANDS> tianglecandCostList; + + if( auto blkCache = dynamic_cast< CacheBlkInfoCtrl* >( m_modeCtrl ) ) + { + bestIsSkip |= blkCache->isSkip( tempCS->area ); + } + + DistParam distParam; + const bool useHadamard = !encTestMode.lossless; + m_pcRdCost->setDistParam( distParam, tempCS->getOrgBuf().Y(), m_acMergeBuffer[0].Y(), sps.getBitDepth( CHANNEL_TYPE_LUMA ), COMPONENT_Y, useHadamard ); + + const UnitArea localUnitArea( tempCS->area.chromaFormat, Area( 0, 0, tempCS->area.Y().width, tempCS->area.Y().height) ); + + const double sqrtLambdaForFirstPass = m_pcRdCost->getMotionLambda(encTestMode.lossless); + + MergeCtx triangleMrgCtx; + { + CodingUnit cu( tempCS->area ); + cu.cs = tempCS; + cu.partSize = SIZE_2Nx2N; + cu.predMode = MODE_INTER; + cu.slice = tempCS->slice; + cu.triangle = true; +#if JVET_L0054_MMVD + cu.mmvdSkip = false; +#endif +#if JVET_L0646_GBI + cu.GBiIdx = GBI_DEFAULT; +#endif + + PredictionUnit pu( tempCS->area ); + pu.cu = &cu; + pu.cs = tempCS; + + + PU::getTriangleMergeCandidates( pu, triangleMrgCtx ); + for( uint8_t mergeCand = 0; mergeCand < TRIANGLE_MAX_NUM_UNI_CANDS; mergeCand++ ) + { + triangleBuffer[mergeCand] = m_acMergeBuffer[mergeCand].getBuf(localUnitArea); + triangleMrgCtx.setMergeInfo( pu, mergeCand ); + PU::spanMotionInfo( pu, triangleMrgCtx ); + + m_pcInterSearch->motionCompensation( pu, triangleBuffer[mergeCand] ); + } + } + + bool tempBufSet = bestIsSkip ? false : true; + triangleNumMrgSATDCand = bestIsSkip ? TRIANGLE_MAX_NUM_CANDS : TRIANGLE_MAX_NUM_SATD_CANDS; + if( bestIsSkip ) + { + for( uint8_t i = 0; i < TRIANGLE_MAX_NUM_CANDS; i++ ) + { + triangleRdModeList.push_back(i); + } + } + else + { + CodingUnit &cu = tempCS->addCU( tempCS->area, partitioner.chType ); + + partitioner.setCUData( cu ); + cu.slice = tempCS->slice; + cu.skip = false; + cu.partSize = SIZE_2Nx2N; + cu.predMode = MODE_INTER; + cu.transQuantBypass = encTestMode.lossless; + cu.chromaQpAdj = cu.transQuantBypass ? 0 : m_cuChromaQpOffsetIdxPlus1; + cu.qp = encTestMode.qp; + cu.triangle = true; +#if JVET_L0054_MMVD + cu.mmvdSkip = false; +#endif +#if JVET_L0646_GBI + cu.GBiIdx = GBI_DEFAULT; +#endif + + PredictionUnit &pu = tempCS->addPU( cu, partitioner.chType ); + + if( abs(g_aucLog2[cu.lwidth()] - g_aucLog2[cu.lheight()]) >= 2 ) + { + numTriangleCandidate = 30; + } + else + { + numTriangleCandidate = TRIANGLE_MAX_NUM_CANDS; + } + + for( uint8_t mergeCand = 0; mergeCand < numTriangleCandidate; mergeCand++ ) + { + bool splitDir = g_triangleCombination[mergeCand][0]; + uint8_t candIdx0 = g_triangleCombination[mergeCand][1]; + uint8_t candIdx1 = g_triangleCombination[mergeCand][2]; + + pu.mergeIdx = mergeCand; + pu.mergeFlag = true; + triangleWeightedBuffer[mergeCand] = m_acTriangleWeightedBuffer[mergeCand].getBuf( localUnitArea ); + triangleBuffer[candIdx0] = m_acMergeBuffer[candIdx0].getBuf( localUnitArea ); + triangleBuffer[candIdx1] = m_acMergeBuffer[candIdx1].getBuf( localUnitArea ); + + m_pcInterSearch->weightedTriangleBlk( pu, PU::getTriangleWeights(pu, triangleMrgCtx, candIdx0, candIdx1), splitDir, CHANNEL_TYPE_LUMA, triangleWeightedBuffer[mergeCand], triangleBuffer[candIdx0], triangleBuffer[candIdx1] ); + + distParam.cur = triangleWeightedBuffer[mergeCand].Y(); + + Distortion uiSad = distParam.distFunc( distParam ); + + uint32_t uiBitsCand = g_triangleIdxBins[mergeCand]; + + double cost = (double)uiSad + (double)uiBitsCand * sqrtLambdaForFirstPass; + + updateCandList( mergeCand, cost, triangleRdModeList, tianglecandCostList, triangleNumMrgSATDCand ); + } + + // limit number of candidates using SATD-costs + for( uint8_t i = 0; i < triangleNumMrgSATDCand; i++ ) + { + if( tianglecandCostList[i] > MRG_FAST_RATIO * tianglecandCostList[0] || tianglecandCostList[i] > getMergeBestSATDCost() ) + { + triangleNumMrgSATDCand = i; + break; + } + } + + // perform chroma weighting process + for( uint8_t i = 0; i < triangleNumMrgSATDCand; i++ ) + { + uint8_t mergeCand = triangleRdModeList[i]; + bool splitDir = g_triangleCombination[mergeCand][0]; + uint8_t candIdx0 = g_triangleCombination[mergeCand][1]; + uint8_t candIdx1 = g_triangleCombination[mergeCand][2]; + + pu.mergeIdx = mergeCand; + pu.mergeFlag = true; + + m_pcInterSearch->weightedTriangleBlk( pu, PU::getTriangleWeights(pu, triangleMrgCtx, candIdx0, candIdx1), splitDir, CHANNEL_TYPE_CHROMA, triangleWeightedBuffer[mergeCand], triangleBuffer[candIdx0], triangleBuffer[candIdx1] ); + } + + tempCS->initStructData( encTestMode.qp, encTestMode.lossless ); + } + + { + const uint8_t iteration = encTestMode.lossless ? 1 : 2; + for( uint8_t noResidualPass = 0; noResidualPass < iteration; noResidualPass++ ) + { + for( uint8_t mrgHADIdx = 0; mrgHADIdx < triangleNumMrgSATDCand; mrgHADIdx++ ) + { + uint8_t mergeCand = triangleRdModeList[mrgHADIdx]; + + if ( ( (noResidualPass != 0) && trianglecandHasNoResidual[mergeCand] ) + || ( (noResidualPass == 0) && bestIsSkip ) ) + { + continue; + } + + bool splitDir = g_triangleCombination[mergeCand][0]; + uint8_t candIdx0 = g_triangleCombination[mergeCand][1]; + uint8_t candIdx1 = g_triangleCombination[mergeCand][2]; + + CodingUnit &cu = tempCS->addCU(tempCS->area, partitioner.chType); + + partitioner.setCUData(cu); + cu.slice = tempCS->slice; + cu.skip = false; + cu.partSize = SIZE_2Nx2N; + cu.predMode = MODE_INTER; + cu.transQuantBypass = encTestMode.lossless; + cu.chromaQpAdj = cu.transQuantBypass ? 0 : m_cuChromaQpOffsetIdxPlus1; + cu.qp = encTestMode.qp; + cu.triangle = true; +#if JVET_L0054_MMVD + cu.mmvdSkip = false; +#endif +#if JVET_L0646_GBI + cu.GBiIdx = GBI_DEFAULT; +#endif + PredictionUnit &pu = tempCS->addPU(cu, partitioner.chType); + + pu.mergeIdx = mergeCand; + pu.mergeFlag = true; + + PU::spanTriangleMotionInfo(pu, triangleMrgCtx, mergeCand, splitDir, candIdx0, candIdx1 ); + + if( tempBufSet ) + { + tempCS->getPredBuf().copyFrom( triangleWeightedBuffer[mergeCand] ); + } + else + { + triangleBuffer[candIdx0] = m_acMergeBuffer[candIdx0].getBuf( localUnitArea ); + triangleBuffer[candIdx1] = m_acMergeBuffer[candIdx1].getBuf( localUnitArea ); + PelUnitBuf predBuf = tempCS->getPredBuf(); + m_pcInterSearch->weightedTriangleBlk( pu, PU::getTriangleWeights(pu, triangleMrgCtx, candIdx0, candIdx1), splitDir, MAX_NUM_CHANNEL_TYPE, predBuf, triangleBuffer[candIdx0], triangleBuffer[candIdx1] ); + } + + xEncodeInterResidual( tempCS, bestCS, partitioner, encTestMode, noResidualPass, NULL, true, ( (noResidualPass == 0 ) ? &trianglecandHasNoResidual[mergeCand] : NULL ) ); + + if (m_pcEncCfg->getUseFastDecisionForMerge() && !bestIsSkip) + { + bestIsSkip = bestCS->getCU(partitioner.chType)->rootCbf == 0; + } + tempCS->initStructData(encTestMode.qp, encTestMode.lossless); + }// end loop mrgHADIdx + } + } +} +#endif + void EncCu::xCheckRDCostAffineMerge2Nx2N( CodingStructure *&tempCS, CodingStructure *&bestCS, Partitioner &partitioner, const EncTestMode& encTestMode ) { if( m_modeCtrl->getFastDeltaQp() ) diff --git a/source/Lib/EncoderLib/EncCu.h b/source/Lib/EncoderLib/EncCu.h index fbc2b7603b387a6a7672c4409e4a34090a5f00f5..570dd4bdd0886a4ad6bfd4030fc51e294f2e7148 100644 --- a/source/Lib/EncoderLib/EncCu.h +++ b/source/Lib/EncoderLib/EncCu.h @@ -119,6 +119,10 @@ private: #endif #else PelStorage m_acMergeBuffer[MRG_MAX_NUM_CANDS]; +#endif +#if JVET_L0124_L0208_TRIANGLE + PelStorage m_acTriangleWeightedBuffer[TRIANGLE_MAX_NUM_CANDS]; // to store weighted prediction pixles + double m_mergeBestSATDCost; #endif MotionInfo m_SubPuMiBuf [( MAX_CU_SIZE * MAX_CU_SIZE ) >> ( MIN_CU_LOG2 << 1 )]; unsigned int m_subMergeBlkSize[10]; @@ -177,6 +181,11 @@ public: bool getClearSubMergeStatic() { return m_clearSubMergeStatic; } #endif +#if JVET_L0124_L0208_TRIANGLE + void setMergeBestSATDCost(double cost) { m_mergeBestSATDCost = cost; } + double getMergeBestSATDCost() { return m_mergeBestSATDCost; } +#endif + ~EncCu(); protected: @@ -221,6 +230,10 @@ protected: void xCheckRDCostMerge2Nx2N ( CodingStructure *&tempCS, CodingStructure *&bestCS, Partitioner &pm, const EncTestMode& encTestMode ); +#if JVET_L0124_L0208_TRIANGLE + void xCheckRDCostMergeTriangle2Nx2N( CodingStructure *&tempCS, CodingStructure *&bestCS, Partitioner &pm, const EncTestMode& encTestMode ); +#endif + void xEncodeInterResidual ( CodingStructure *&tempCS, CodingStructure *&bestCS, Partitioner &partitioner, const EncTestMode& encTestMode, int residualPass = 0 , CodingStructure* imvCS = NULL , int emtMode = 1 diff --git a/source/Lib/EncoderLib/EncLib.cpp b/source/Lib/EncoderLib/EncLib.cpp index d46b5e17bc818c45b301fe91159d4d277a20de2b..82ea0cdae7a1c354ef863dc52346a34afa627c82 100644 --- a/source/Lib/EncoderLib/EncLib.cpp +++ b/source/Lib/EncoderLib/EncLib.cpp @@ -880,6 +880,9 @@ void EncLib::xInitSPS(SPS &sps) #if JVET_L0100_MULTI_HYPOTHESIS_INTRA sps.getSpsNext().setUseMHIntra ( m_MHIntra ); #endif +#if JVET_L0124_L0208_TRIANGLE + sps.getSpsNext().setUseTriangle ( m_Triangle ); +#endif // ADD_NEW_TOOL : (encoder lib) set tool enabling flags and associated parameters here diff --git a/source/Lib/EncoderLib/EncModeCtrl.cpp b/source/Lib/EncoderLib/EncModeCtrl.cpp index 0aba5bc5b6dbe667cfa422f3f1ba358b6aa06bb6..c4e07119aacb5dfa41221d77a18effcb99b80c22 100644 --- a/source/Lib/EncoderLib/EncModeCtrl.cpp +++ b/source/Lib/EncoderLib/EncModeCtrl.cpp @@ -1077,6 +1077,12 @@ void EncModeCtrlMTnoRQT::initCULevel( Partitioner &partitioner, const CodingStru // add inter modes if( m_pcEncCfg->getUseEarlySkipDetection() ) { +#if JVET_L0124_L0208_TRIANGLE + if( cs.sps->getSpsNext().getUseTriangle() && cs.slice->isInterB() ) + { + m_ComprCUCtxList.back().testModes.push_back( { ETM_MERGE_TRIANGLE, SIZE_2Nx2N, ETO_STANDARD, qp, lossless } ); + } +#endif m_ComprCUCtxList.back().testModes.push_back( { ETM_MERGE_SKIP, SIZE_2Nx2N, ETO_STANDARD, qp, lossless } ); #if JVET_L0369_SUBBLOCK_MERGE if ( cs.sps->getSpsNext().getUseAffine() || cs.sps->getSpsNext().getUseSubPuMvp() ) @@ -1091,7 +1097,12 @@ void EncModeCtrlMTnoRQT::initCULevel( Partitioner &partitioner, const CodingStru else { m_ComprCUCtxList.back().testModes.push_back( { ETM_INTER_ME, SIZE_2Nx2N, ETO_STANDARD, qp, lossless } ); - +#if JVET_L0124_L0208_TRIANGLE + if( cs.sps->getSpsNext().getUseTriangle() && cs.slice->isInterB() ) + { + m_ComprCUCtxList.back().testModes.push_back( { ETM_MERGE_TRIANGLE, SIZE_2Nx2N, ETO_STANDARD, qp, lossless } ); + } +#endif m_ComprCUCtxList.back().testModes.push_back( { ETM_MERGE_SKIP, SIZE_2Nx2N, ETO_STANDARD, qp, lossless } ); #if JVET_L0369_SUBBLOCK_MERGE if ( cs.sps->getSpsNext().getUseAffine() || cs.sps->getSpsNext().getUseSubPuMvp() ) @@ -1313,6 +1324,12 @@ bool EncModeCtrlMTnoRQT::tryMode( const EncTestMode& encTestmode, const CodingSt { return false; } +#if JVET_L0124_L0208_TRIANGLE + if( encTestmode.type == ETM_MERGE_TRIANGLE && ( partitioner.currArea().lumaSize().area() < TRIANGLE_MIN_SIZE || relatedCU.isIntra ) ) + { + return false; + } +#endif return true; } else if( isModeSplit( encTestmode ) ) diff --git a/source/Lib/EncoderLib/EncModeCtrl.h b/source/Lib/EncoderLib/EncModeCtrl.h index f56fceb06b1435025bced67fcf32150c1a4c7253..988ba51c4d8821c484d076b6d726748b810a81a3 100644 --- a/source/Lib/EncoderLib/EncModeCtrl.h +++ b/source/Lib/EncoderLib/EncModeCtrl.h @@ -57,6 +57,9 @@ enum EncTestModeType ETM_MERGE_SKIP, ETM_INTER_ME, ETM_AFFINE, +#if JVET_L0124_L0208_TRIANGLE + ETM_MERGE_TRIANGLE, +#endif ETM_INTRA, ETM_IPCM, ETM_SPLIT_QT, @@ -134,6 +137,9 @@ inline bool isModeInter( const EncTestMode& encTestmode ) // perhaps remove return ( encTestmode.type == ETM_INTER_ME || encTestmode.type == ETM_MERGE_SKIP || encTestmode.type == ETM_AFFINE +#if JVET_L0124_L0208_TRIANGLE + || encTestmode.type == ETM_MERGE_TRIANGLE +#endif ); } diff --git a/source/Lib/EncoderLib/InterSearch.cpp b/source/Lib/EncoderLib/InterSearch.cpp index 8d6f8cb2462ae1c88aacd74d48f0a01fd1c15452..5258a13df78387ee96e99054846377000876789e 100644 --- a/source/Lib/EncoderLib/InterSearch.cpp +++ b/source/Lib/EncoderLib/InterSearch.cpp @@ -4860,6 +4860,9 @@ void InterSearch::encodeResAndCalcRdInterCU(CodingStructure &cs, Partitioner &pa #else m_CABACEstimator->affine_flag( cu ); #endif +#if JVET_L0124_L0208_TRIANGLE + m_CABACEstimator->triangle_mode ( cu ); +#endif #if JVET_L0054_MMVD if (cu.mmvdSkip) { @@ -4995,6 +4998,9 @@ uint64_t InterSearch::xGetSymbolFracBitsInter(CodingStructure &cs, Partitioner & #else m_CABACEstimator->affine_flag ( cu ); #endif +#if JVET_L0124_L0208_TRIANGLE + m_CABACEstimator->triangle_mode ( cu ); +#endif #if JVET_L0054_MMVD if (cu.mmvdSkip) { diff --git a/source/Lib/EncoderLib/VLCWriter.cpp b/source/Lib/EncoderLib/VLCWriter.cpp index 4feff09a66c72f945903b1a9e48562bc1887806d..685fd3ab5005cd5160ce5b2fd84d44ded21b4ba5 100644 --- a/source/Lib/EncoderLib/VLCWriter.cpp +++ b/source/Lib/EncoderLib/VLCWriter.cpp @@ -559,6 +559,9 @@ void HLSWriter::codeSPSNext( const SPSNext& spsNext, const bool usePCM ) #if JVET_L0100_MULTI_HYPOTHESIS_INTRA WRITE_FLAG( spsNext.getUseMHIntra() ? 1 : 0, "mhintra_flag" ); #endif +#if JVET_L0124_L0208_TRIANGLE + WRITE_FLAG( spsNext.getUseTriangle() ? 1: 0, "triangle_flag" ); +#endif #if ENABLE_WPP_PARALLELISM WRITE_FLAG( spsNext.getUseNextDQP(), "next_dqp_enabled_flag" ); #else