2014-07-01 09:55:04 +02:00
|
|
|
#include <gtest/gtest.h>
|
2014-04-24 04:55:38 +02:00
|
|
|
|
|
|
|
#include "wels_common_basis.h"
|
2015-01-30 09:44:04 +01:00
|
|
|
#include "memory_align.h"
|
2014-04-24 04:55:38 +02:00
|
|
|
#include "error_concealment.h"
|
|
|
|
#include "ls_defines.h"
|
|
|
|
#include "cpu.h"
|
|
|
|
|
|
|
|
using namespace WelsDec;
|
|
|
|
|
|
|
|
#define MAX_MB_WIDTH 260
|
|
|
|
#define MAX_MB_HEIGHT 130
|
|
|
|
|
|
|
|
typedef struct TagECInputCtx {
|
|
|
|
int32_t iMbWidth;
|
|
|
|
int32_t iMbHeight;
|
|
|
|
uint32_t iLinesize[3];
|
|
|
|
bool* pMbCorrectlyDecodedFlag; //actual memory
|
|
|
|
PWelsDecoderContext pCtx;
|
|
|
|
SDqLayer sDqLayer;
|
|
|
|
SPicture sAncPic; //Anc picture for comparison
|
|
|
|
SPicture sSrcPic; //Src picture as common input picture data
|
|
|
|
SPicture sWelsPic; //Wels picture to be compared
|
|
|
|
} SECInputCtx, *PECInputCtx;
|
|
|
|
|
|
|
|
void FreeInputData (PECInputCtx pECCtx) {
|
|
|
|
if (pECCtx != NULL) {
|
|
|
|
if (pECCtx->pCtx != NULL) {
|
|
|
|
WELS_SAFE_FREE (pECCtx->pCtx->pSps, "pECCtx->pCtx->pSps");
|
|
|
|
WELS_SAFE_FREE (pECCtx->pCtx, "pECCtx->pCtx");
|
|
|
|
}
|
|
|
|
|
|
|
|
WELS_SAFE_FREE (pECCtx->pMbCorrectlyDecodedFlag, "pECCtx->pMbCorrectlyDecodedFlag");
|
|
|
|
WELS_SAFE_FREE (pECCtx->sSrcPic.pData[0], "pECCtx->sSrcPic.pData");
|
|
|
|
WELS_SAFE_FREE (pECCtx->sAncPic.pData[0], "pECCtx->sAncPic.pData");
|
|
|
|
WELS_SAFE_FREE (pECCtx->sWelsPic.pData[0], "pECCtx->sWelsPic.pData");
|
|
|
|
|
|
|
|
WELS_SAFE_FREE (pECCtx, "pECCtx");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
int32_t InitAndAllocInputData (PECInputCtx& pECCtx) {
|
|
|
|
FreeInputData (pECCtx);
|
|
|
|
|
2015-01-30 09:37:48 +01:00
|
|
|
pECCtx = (PECInputCtx) WelsMallocz (sizeof (SECInputCtx), "pECCtx");
|
2014-04-24 04:55:38 +02:00
|
|
|
if (pECCtx == NULL)
|
|
|
|
return 1;
|
|
|
|
memset (pECCtx, 0, sizeof (SECInputCtx));
|
|
|
|
|
2014-11-11 02:56:26 +01:00
|
|
|
pECCtx->iMbWidth = rand() % (MAX_MB_WIDTH - 1) + 1; //give a constrained max width
|
|
|
|
pECCtx->iMbHeight = rand() % (MAX_MB_HEIGHT - 1) + 1; //give a constrained max height
|
2014-04-24 04:55:38 +02:00
|
|
|
pECCtx->iLinesize[0] = pECCtx->iMbWidth << 4;
|
|
|
|
pECCtx->iLinesize[1] = pECCtx->iLinesize[2] = pECCtx->iLinesize[0] >> 1;
|
|
|
|
|
|
|
|
const uint32_t kiLumaSize = pECCtx->iMbWidth * pECCtx->iMbHeight * 256;
|
|
|
|
|
|
|
|
//allocate picture data
|
2015-01-30 09:37:48 +01:00
|
|
|
pECCtx->sWelsPic.pData[0] = (uint8_t*) WelsMallocz (kiLumaSize * 3 / 2 * sizeof (uint8_t), "pECCtx->sWelsPic.pData");
|
2014-04-24 04:55:38 +02:00
|
|
|
if (pECCtx->sWelsPic.pData[0] == NULL)
|
|
|
|
return 1;
|
|
|
|
pECCtx->sWelsPic.pData[1] = pECCtx->sWelsPic.pData[0] + kiLumaSize;
|
|
|
|
pECCtx->sWelsPic.pData[2] = pECCtx->sWelsPic.pData[1] + (kiLumaSize >> 2);
|
|
|
|
|
2015-01-30 09:37:48 +01:00
|
|
|
pECCtx->sAncPic.pData[0] = (uint8_t*) WelsMallocz (kiLumaSize * 3 / 2 * sizeof (uint8_t), "pECCtx->sAncPic.pData");
|
2014-04-24 04:55:38 +02:00
|
|
|
if (pECCtx->sAncPic.pData[0] == NULL)
|
|
|
|
return 1;
|
|
|
|
pECCtx->sAncPic.pData[1] = pECCtx->sAncPic.pData[0] + kiLumaSize;
|
|
|
|
pECCtx->sAncPic.pData[2] = pECCtx->sAncPic.pData[1] + (kiLumaSize >> 2);
|
|
|
|
|
2015-01-30 09:37:48 +01:00
|
|
|
pECCtx->sSrcPic.pData[0] = (uint8_t*) WelsMallocz (kiLumaSize * 3 / 2 * sizeof (uint8_t), "pECCtx->sSrcPic.pData");
|
2014-04-29 05:34:00 +02:00
|
|
|
if (pECCtx->sSrcPic.pData[0] == NULL)
|
2014-04-24 04:55:38 +02:00
|
|
|
return 1;
|
|
|
|
pECCtx->sSrcPic.pData[1] = pECCtx->sSrcPic.pData[0] + kiLumaSize;
|
|
|
|
pECCtx->sSrcPic.pData[2] = pECCtx->sSrcPic.pData[1] + (kiLumaSize >> 2);
|
|
|
|
|
|
|
|
pECCtx->sWelsPic.iLinesize[0] = pECCtx->sAncPic.iLinesize[0] = pECCtx->sSrcPic.iLinesize[0] = pECCtx->iLinesize[0];
|
|
|
|
pECCtx->sWelsPic.iLinesize[1] = pECCtx->sAncPic.iLinesize[1] = pECCtx->sSrcPic.iLinesize[1] = pECCtx->iLinesize[1];
|
|
|
|
pECCtx->sWelsPic.iLinesize[2] = pECCtx->sAncPic.iLinesize[2] = pECCtx->sSrcPic.iLinesize[2] = pECCtx->iLinesize[2];
|
|
|
|
|
2015-01-30 09:37:48 +01:00
|
|
|
pECCtx->pMbCorrectlyDecodedFlag = (bool*) WelsMallocz (pECCtx->iMbWidth * pECCtx->iMbHeight * sizeof (bool),
|
2014-04-24 04:55:38 +02:00
|
|
|
"pECCtx->pMbCorrectlyDecodedFlag");
|
|
|
|
if (pECCtx->pMbCorrectlyDecodedFlag == NULL)
|
|
|
|
return 1;
|
|
|
|
|
2015-01-30 09:37:48 +01:00
|
|
|
pECCtx->pCtx = (PWelsDecoderContext) WelsMallocz (sizeof (SWelsDecoderContext), "pECCtx->pCtx");
|
2014-04-24 04:55:38 +02:00
|
|
|
if (pECCtx->pCtx == NULL)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
pECCtx->pCtx->pDec = &pECCtx->sWelsPic;
|
|
|
|
pECCtx->pCtx->pCurDqLayer = &pECCtx->sDqLayer;
|
|
|
|
pECCtx->pCtx->pCurDqLayer->pMbCorrectlyDecodedFlag = pECCtx->pMbCorrectlyDecodedFlag;
|
|
|
|
|
2015-01-30 09:37:48 +01:00
|
|
|
pECCtx->pCtx->pSps = (PSps) WelsMallocz (sizeof (SSps), "pECCtx->pCtx->pSps");
|
2014-04-24 04:55:38 +02:00
|
|
|
if (pECCtx->pCtx->pSps == NULL)
|
|
|
|
return 1;
|
|
|
|
pECCtx->pCtx->pSps->iMbWidth = pECCtx->iMbWidth;
|
|
|
|
pECCtx->pCtx->pSps->iMbHeight = pECCtx->iMbHeight;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
void InitECCopyData (PECInputCtx pECCtx) {
|
|
|
|
const int32_t kiMbNum = pECCtx->iMbWidth * pECCtx->iMbHeight;
|
|
|
|
int i;
|
|
|
|
//init pMbCorrectlyDecodedFlag
|
|
|
|
for (i = 0; i < kiMbNum; ++i) {
|
|
|
|
pECCtx->pMbCorrectlyDecodedFlag[i] = !! (rand() & 1);
|
|
|
|
}
|
|
|
|
//init Data
|
|
|
|
const int32_t iPixNum = kiMbNum * 256 * 3 / 2;
|
|
|
|
for (i = 0; i < iPixNum; ++i) {
|
|
|
|
pECCtx->sSrcPic.pData[0][i] = rand() & 0xff;
|
|
|
|
}
|
|
|
|
int32_t iCpuCores = 1;
|
|
|
|
pECCtx->pCtx->uiCpuFlag = WelsCPUFeatureDetect (&iCpuCores);
|
|
|
|
InitErrorCon (pECCtx->pCtx);
|
|
|
|
}
|
|
|
|
|
|
|
|
void DoAncErrorConSliceCopy (PECInputCtx pECCtx) {
|
|
|
|
int32_t iMbWidth = (int32_t) pECCtx->iMbWidth;
|
|
|
|
int32_t iMbHeight = (int32_t) pECCtx->iMbHeight;
|
|
|
|
PPicture pDstPic = &pECCtx->sAncPic;
|
|
|
|
PPicture pSrcPic = pECCtx->pCtx->pPreviousDecodedPictureInDpb;
|
2014-11-04 03:17:32 +01:00
|
|
|
if ((pECCtx->pCtx->eErrorConMethod == ERROR_CON_SLICE_COPY)
|
|
|
|
&& (pECCtx->pCtx->pCurDqLayer->sLayerInfo.sNalHeaderExt.bIdrFlag))
|
|
|
|
pSrcPic = NULL;
|
2014-04-24 04:55:38 +02:00
|
|
|
|
|
|
|
//uint8_t *pDstData[3], *pSrcData[3];
|
|
|
|
bool* pMbCorrectlyDecodedFlag = pECCtx->pMbCorrectlyDecodedFlag;
|
|
|
|
|
|
|
|
//Do slice copy late
|
|
|
|
int32_t iMbXyIndex, i;
|
|
|
|
uint8_t* pSrcData, *pDstData;
|
|
|
|
uint32_t iSrcStride = pECCtx->iLinesize[0];
|
|
|
|
uint32_t iDstStride = pECCtx->iLinesize[0];
|
|
|
|
for (int32_t iMbY = 0; iMbY < iMbHeight; ++iMbY) {
|
|
|
|
for (int32_t iMbX = 0; iMbX < iMbWidth; ++iMbX) {
|
|
|
|
iMbXyIndex = iMbY * iMbWidth + iMbX;
|
|
|
|
if (!pMbCorrectlyDecodedFlag[iMbXyIndex]) {
|
|
|
|
if (pSrcPic != NULL) {
|
|
|
|
//Y component
|
|
|
|
pDstData = pDstPic->pData[0] + iMbY * 16 * iDstStride + iMbX * 16;
|
|
|
|
pSrcData = pSrcPic->pData[0] + iMbY * 16 * iSrcStride + iMbX * 16;
|
|
|
|
for (i = 0; i < 16; ++i) {
|
|
|
|
memcpy (pDstData, pSrcData, 16);
|
|
|
|
pDstData += iDstStride;
|
|
|
|
pSrcData += iSrcStride;
|
|
|
|
}
|
|
|
|
//U component
|
|
|
|
pDstData = pDstPic->pData[1] + iMbY * 8 * iDstStride / 2 + iMbX * 8;
|
|
|
|
pSrcData = pSrcPic->pData[1] + iMbY * 8 * iSrcStride / 2 + iMbX * 8;
|
|
|
|
for (i = 0; i < 8; ++i) {
|
|
|
|
memcpy (pDstData, pSrcData, 8);
|
2014-04-29 05:34:00 +02:00
|
|
|
pDstData += iDstStride / 2;
|
|
|
|
pSrcData += iSrcStride / 2;
|
2014-04-24 04:55:38 +02:00
|
|
|
}
|
|
|
|
//V component
|
|
|
|
pDstData = pDstPic->pData[2] + iMbY * 8 * iDstStride / 2 + iMbX * 8;
|
|
|
|
pSrcData = pSrcPic->pData[2] + iMbY * 8 * iSrcStride / 2 + iMbX * 8;
|
|
|
|
for (i = 0; i < 8; ++i) {
|
|
|
|
memcpy (pDstData, pSrcData, 8);
|
2014-04-29 05:34:00 +02:00
|
|
|
pDstData += iDstStride / 2;
|
|
|
|
pSrcData += iSrcStride / 2;
|
2014-04-24 04:55:38 +02:00
|
|
|
}
|
|
|
|
} else { //pSrcPic == NULL
|
|
|
|
//Y component
|
|
|
|
pDstData = pDstPic->pData[0] + iMbY * 16 * iDstStride + iMbX * 16;
|
2014-04-29 05:34:00 +02:00
|
|
|
for (i = 0; i < 16; ++i) {
|
2014-07-07 09:02:39 +02:00
|
|
|
memset (pDstData, 128, 16);
|
2014-04-24 04:55:38 +02:00
|
|
|
pDstData += iDstStride;
|
|
|
|
}
|
|
|
|
//U component
|
|
|
|
pDstData = pDstPic->pData[1] + iMbY * 8 * iDstStride / 2 + iMbX * 8;
|
2014-04-29 05:34:00 +02:00
|
|
|
for (i = 0; i < 8; ++i) {
|
2014-07-07 09:02:39 +02:00
|
|
|
memset (pDstData, 128, 8);
|
2014-04-24 04:55:38 +02:00
|
|
|
pDstData += iDstStride / 2;
|
|
|
|
}
|
|
|
|
//V component
|
|
|
|
pDstData = pDstPic->pData[2] + iMbY * 8 * iDstStride / 2 + iMbX * 8;
|
2014-04-29 05:34:00 +02:00
|
|
|
for (i = 0; i < 8; ++i) {
|
2014-07-07 09:02:39 +02:00
|
|
|
memset (pDstData, 128, 8);
|
2014-04-24 04:55:38 +02:00
|
|
|
pDstData += iDstStride / 2;
|
|
|
|
}
|
|
|
|
} //
|
|
|
|
} //!pMbCorrectlyDecodedFlag[iMbXyIndex]
|
|
|
|
} //iMbX
|
|
|
|
} //iMbY
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
bool ComparePictureDataI420 (uint8_t* pSrcData, uint8_t* pDstData, const uint32_t kiStride, const int32_t kiHeight) {
|
|
|
|
bool bSame = true;
|
|
|
|
uint8_t* pAncData; // = pECCtx->sAncPic.pData[0];
|
|
|
|
uint8_t* pCompData;
|
|
|
|
int32_t iStride;
|
|
|
|
int32_t iCurHeight;
|
|
|
|
int32_t iHeight = kiHeight;
|
|
|
|
|
|
|
|
//Y component
|
|
|
|
iStride = kiStride;
|
|
|
|
pAncData = pSrcData;
|
|
|
|
pCompData = pDstData;
|
|
|
|
for (iCurHeight = 0; bSame && (iCurHeight < kiHeight); ++iCurHeight) {
|
|
|
|
bSame = (memcmp (pAncData, pCompData, iStride * sizeof (uint8_t)) == 0);
|
|
|
|
pAncData += iStride;
|
|
|
|
pCompData += iStride;
|
|
|
|
}
|
|
|
|
//chroma component
|
|
|
|
iHeight >>= 1;
|
|
|
|
iStride >>= 1;
|
|
|
|
//U component
|
|
|
|
for (iCurHeight = 0; bSame && (iCurHeight < kiHeight / 2); ++iCurHeight) {
|
|
|
|
bSame = (memcmp (pAncData, pCompData, iStride * sizeof (uint8_t)) == 0);
|
|
|
|
pAncData += iStride;
|
|
|
|
pCompData += iStride;
|
|
|
|
}
|
|
|
|
//V component
|
|
|
|
for (iCurHeight = 0; bSame && (iCurHeight < kiHeight / 2); ++iCurHeight) {
|
|
|
|
bSame = (memcmp (pAncData, pCompData, iStride * sizeof (uint8_t)) == 0);
|
|
|
|
pAncData += iStride;
|
|
|
|
pCompData += iStride;
|
|
|
|
}
|
|
|
|
|
|
|
|
return bSame;
|
|
|
|
}
|
|
|
|
|
|
|
|
//TEST cases followed
|
|
|
|
TEST (ErrorConTest, DoErrorConFrameCopy) {
|
|
|
|
bool bOK = true;
|
|
|
|
PECInputCtx pECCtx = NULL;
|
|
|
|
if (InitAndAllocInputData (pECCtx)) {
|
|
|
|
FreeInputData (pECCtx);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2014-11-04 03:17:32 +01:00
|
|
|
for (int iEC = 0; iEC < 2; ++ iEC) { //ERROR_CON_FRAME_COPY, ERROR_CON_FRAME_COPY_CROSS_IDR
|
|
|
|
pECCtx->pCtx->eErrorConMethod = iEC > 0 ? ERROR_CON_FRAME_COPY_CROSS_IDR : ERROR_CON_FRAME_COPY;
|
|
|
|
InitECCopyData (pECCtx);
|
|
|
|
int32_t iLumaSize = pECCtx->iMbWidth * pECCtx->iMbHeight * 256;
|
|
|
|
|
|
|
|
for (int iRef = 0; iRef < 2; ++ iRef) { //no ref, with ref
|
|
|
|
pECCtx->pCtx->pPreviousDecodedPictureInDpb = iRef ? &pECCtx->sSrcPic : NULL;
|
|
|
|
for (int iIDR = 0; iIDR < 2; ++ iIDR) { //non IDR, IDR
|
|
|
|
pECCtx->pCtx->pCurDqLayer->sLayerInfo.sNalHeaderExt.bIdrFlag = (iIDR > 0);
|
|
|
|
//Do reference code method
|
|
|
|
DoErrorConFrameCopy (pECCtx->pCtx);
|
|
|
|
//Do anchor method
|
|
|
|
if (iRef && ! ((pECCtx->pCtx->eErrorConMethod == ERROR_CON_FRAME_COPY)
|
|
|
|
&& (pECCtx->pCtx->pCurDqLayer->sLayerInfo.sNalHeaderExt.bIdrFlag)))
|
|
|
|
memcpy (pECCtx->sAncPic.pData[0], pECCtx->sSrcPic.pData[0], iLumaSize * 3 / 2);
|
|
|
|
else
|
|
|
|
memset (pECCtx->sAncPic.pData[0], 128, iLumaSize * 3 / 2); //should be the same as known EC method, here all 128
|
|
|
|
//Compare results
|
|
|
|
bOK = ComparePictureDataI420 (pECCtx->sAncPic.pData[0], pECCtx->sWelsPic.pData[0], pECCtx->iLinesize[0],
|
|
|
|
pECCtx->iMbHeight * 16);
|
|
|
|
EXPECT_EQ (bOK, true);
|
|
|
|
} //non IDR, IDR
|
|
|
|
} // no ref, with ref
|
|
|
|
} //FRAME_COPY methods
|
2014-04-24 04:55:38 +02:00
|
|
|
|
|
|
|
FreeInputData (pECCtx);
|
|
|
|
}
|
|
|
|
|
|
|
|
//TEST cases followed
|
|
|
|
TEST (ErrorConTest, DoErrorConSliceCopy) {
|
|
|
|
bool bOK = true;
|
|
|
|
PECInputCtx pECCtx = NULL;
|
|
|
|
if (InitAndAllocInputData (pECCtx)) {
|
|
|
|
FreeInputData (pECCtx);
|
|
|
|
return;
|
|
|
|
}
|
2014-11-04 03:17:32 +01:00
|
|
|
|
|
|
|
for (int iEC = 0; iEC < 2; ++ iEC) { //ERROR_CON_SLICE_COPY, ERROR_CON_SLICE_COPY_CROSS_IDR
|
|
|
|
pECCtx->pCtx->eErrorConMethod = iEC > 0 ? ERROR_CON_SLICE_COPY_CROSS_IDR : ERROR_CON_SLICE_COPY;
|
|
|
|
InitECCopyData (pECCtx);
|
|
|
|
for (int iRef = 0; iRef < 2; ++ iRef) { //no ref, with ref
|
|
|
|
pECCtx->pCtx->pPreviousDecodedPictureInDpb = iRef ? &pECCtx->sSrcPic : NULL;
|
|
|
|
for (int iIDR = 0; iIDR < 2; ++ iIDR) { //non IDR, IDR
|
|
|
|
pECCtx->pCtx->pCurDqLayer->sLayerInfo.sNalHeaderExt.bIdrFlag = (iIDR > 0);
|
|
|
|
//Do reference code method
|
|
|
|
DoErrorConSliceCopy (pECCtx->pCtx);
|
|
|
|
//Do anchor method
|
|
|
|
DoAncErrorConSliceCopy (pECCtx);
|
|
|
|
//Compare results
|
|
|
|
bOK = ComparePictureDataI420 (pECCtx->sAncPic.pData[0], pECCtx->sWelsPic.pData[0], pECCtx->iLinesize[0],
|
|
|
|
pECCtx->iMbHeight * 16);
|
|
|
|
EXPECT_EQ (bOK, true);
|
|
|
|
} //non IDR, IDR
|
|
|
|
} // no ref, with ref
|
|
|
|
} //FRAME_COPY methods
|
2014-04-24 04:55:38 +02:00
|
|
|
|
|
|
|
FreeInputData (pECCtx);
|
|
|
|
}
|