2014-07-01 09:55:04 +02:00
|
|
|
#include <gtest/gtest.h>
|
2016-01-19 14:27:46 +01:00
|
|
|
#include "cpu.h"
|
2014-04-03 10:25:06 +02:00
|
|
|
#include "ls_defines.h"
|
|
|
|
#include "encode_mb_aux.h"
|
|
|
|
#include "wels_common_basis.h"
|
|
|
|
|
2014-08-11 10:08:49 +02:00
|
|
|
using namespace WelsEnc;
|
2014-04-03 10:25:06 +02:00
|
|
|
|
|
|
|
#define ALLOC_MEMORY(type, name, num) type* name = (type*)cMemoryAlign.WelsMalloc(num*sizeof(type), #name);
|
|
|
|
#define FREE_MEMORY(name) cMemoryAlign.WelsFree(name, #name);
|
2014-06-26 03:50:41 +02:00
|
|
|
TEST (EncodeMbAuxTest, TestScan_4x4_ac_c) {
|
|
|
|
CMemoryAlign cMemoryAlign (0);
|
|
|
|
ALLOC_MEMORY (int16_t, iLevel, 16);
|
|
|
|
ALLOC_MEMORY (int16_t, iDctA, 16);
|
|
|
|
ALLOC_MEMORY (int16_t, iDctB, 16);
|
|
|
|
for (int i = 0; i < 16; i++) {
|
|
|
|
iDctA[i] = rand() % 256 + 1;
|
|
|
|
iDctB[i] = iDctA[i];
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
2014-06-26 03:50:41 +02:00
|
|
|
WelsScan4x4Ac_c (iLevel, iDctA);
|
|
|
|
EXPECT_EQ (iLevel[0], iDctB[1]);
|
|
|
|
EXPECT_EQ (iLevel[1], iDctB[4]);
|
|
|
|
EXPECT_EQ (iLevel[2], iDctB[8]);
|
|
|
|
EXPECT_EQ (iLevel[3], iDctB[5]);
|
|
|
|
EXPECT_EQ (iLevel[4], iDctB[2]);
|
|
|
|
EXPECT_EQ (iLevel[5], iDctB[3]);
|
|
|
|
EXPECT_EQ (iLevel[6], iDctB[6]);
|
|
|
|
EXPECT_EQ (iLevel[7], iDctB[9]);
|
|
|
|
EXPECT_EQ (iLevel[8], iDctB[12]);
|
|
|
|
EXPECT_EQ (iLevel[9], iDctB[13]);
|
|
|
|
EXPECT_EQ (iLevel[10], iDctB[10]);
|
|
|
|
EXPECT_EQ (iLevel[11], iDctB[7]);
|
|
|
|
EXPECT_EQ (iLevel[12], iDctB[11]);
|
|
|
|
EXPECT_EQ (iLevel[13], iDctB[14]);
|
|
|
|
EXPECT_EQ (iLevel[14], iDctB[15]);
|
|
|
|
EXPECT_EQ (iLevel[15], 0);
|
|
|
|
FREE_MEMORY (iLevel);
|
|
|
|
FREE_MEMORY (iDctA);
|
|
|
|
FREE_MEMORY (iDctB);
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef X86_ASM
|
2014-06-26 03:50:41 +02:00
|
|
|
TEST (EncodeMbAuxTest, TestScan_4x4_ac_sse2) {
|
|
|
|
CMemoryAlign cMemoryAlign (0);
|
|
|
|
ALLOC_MEMORY (int16_t, iLevelA, 16);
|
|
|
|
ALLOC_MEMORY (int16_t, iLevelB, 16);
|
|
|
|
ALLOC_MEMORY (int16_t, iDct, 16);
|
|
|
|
for (int i = 0; i < 16; i++) {
|
|
|
|
iDct[i] = rand() % 256 + 1;
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
2014-06-26 03:50:41 +02:00
|
|
|
WelsScan4x4Ac_c (iLevelA, iDct);
|
|
|
|
WelsScan4x4Ac_sse2 (iLevelB, iDct);
|
|
|
|
for (int j = 0; j < 16; j++)
|
|
|
|
EXPECT_EQ (iLevelA[j], iLevelB[j]);
|
|
|
|
FREE_MEMORY (iLevelA);
|
|
|
|
FREE_MEMORY (iLevelB);
|
|
|
|
FREE_MEMORY (iDct);
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
2014-06-26 03:50:41 +02:00
|
|
|
TEST (EncodeMbAuxTest, WelsScan4x4DcAc_sse2) {
|
|
|
|
CMemoryAlign cMemoryAlign (0);
|
|
|
|
ALLOC_MEMORY (int16_t, iLevelA, 32);
|
|
|
|
ALLOC_MEMORY (int16_t, iLevelB, 32);
|
|
|
|
ALLOC_MEMORY (int16_t, iDct, 32);
|
|
|
|
for (int i = 0; i < 32; i++)
|
2014-04-03 10:25:06 +02:00
|
|
|
iDct[i] = (rand() & 32767) - 16384;
|
2014-06-26 03:50:41 +02:00
|
|
|
WelsScan4x4DcAc_sse2 (iLevelA, iDct);
|
|
|
|
WelsScan4x4DcAc_c (iLevelB, iDct);
|
|
|
|
for (int i = 0; i < 16; i++)
|
|
|
|
EXPECT_EQ (iLevelA[i], iLevelB[i]);
|
|
|
|
FREE_MEMORY (iLevelA);
|
|
|
|
FREE_MEMORY (iLevelB);
|
|
|
|
FREE_MEMORY (iDct);
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
|
|
|
#endif
|
2014-06-26 03:50:41 +02:00
|
|
|
TEST (EncodeMbAuxTest, TestScan_4x4_dcc) {
|
|
|
|
CMemoryAlign cMemoryAlign (0);
|
|
|
|
ALLOC_MEMORY (int16_t, iLevel, 16);
|
|
|
|
ALLOC_MEMORY (int16_t, iDctA, 16);
|
|
|
|
ALLOC_MEMORY (int16_t, iDctB, 16);
|
|
|
|
for (int i = 0; i < 16; i++)
|
|
|
|
iDctA[i] = iDctB[i] = rand() % 256 + 1;
|
|
|
|
WelsScan4x4Dc (iLevel, iDctA);
|
|
|
|
EXPECT_EQ (iLevel[0], iDctB[0]);
|
|
|
|
EXPECT_EQ (iLevel[1], iDctB[1]);
|
|
|
|
EXPECT_EQ (iLevel[2], iDctB[4]);
|
|
|
|
EXPECT_EQ (iLevel[3], iDctB[8]);
|
|
|
|
EXPECT_EQ (iLevel[4], iDctB[5]);
|
|
|
|
EXPECT_EQ (iLevel[5], iDctB[2]);
|
|
|
|
EXPECT_EQ (iLevel[6], iDctB[3]);
|
|
|
|
EXPECT_EQ (iLevel[7], iDctB[6]);
|
|
|
|
EXPECT_EQ (iLevel[8], iDctB[9]);
|
|
|
|
EXPECT_EQ (iLevel[9], iDctB[12]);
|
|
|
|
EXPECT_EQ (iLevel[10], iDctB[13]);
|
|
|
|
EXPECT_EQ (iLevel[11], iDctB[10]);
|
|
|
|
EXPECT_EQ (iLevel[12], iDctB[7]);
|
|
|
|
EXPECT_EQ (iLevel[13], iDctB[11]);
|
|
|
|
EXPECT_EQ (iLevel[14], iDctB[14]);
|
|
|
|
EXPECT_EQ (iLevel[15], iDctB[15]);
|
|
|
|
FREE_MEMORY (iLevel);
|
|
|
|
FREE_MEMORY (iDctA);
|
|
|
|
FREE_MEMORY (iDctB);
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
2014-06-26 03:50:41 +02:00
|
|
|
static inline void PixelSubWH (int16_t* iDiff, int iSize, uint8_t* pPix1, int iStride1, uint8_t* pPix2, int iStride2) {
|
2014-04-03 10:25:06 +02:00
|
|
|
int y, x;
|
2014-06-26 03:50:41 +02:00
|
|
|
for (y = 0; y < iSize; y++) {
|
|
|
|
for (x = 0; x < iSize; x++)
|
|
|
|
iDiff[x + y * iSize] = pPix1[x] - pPix2[x];
|
|
|
|
pPix1 += iStride1;
|
|
|
|
pPix2 += iStride2;
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
2014-06-26 03:50:41 +02:00
|
|
|
}
|
2014-04-03 10:25:06 +02:00
|
|
|
|
|
|
|
#define FENC_STRIDE 16
|
|
|
|
#define FDEC_STRIDE 32
|
2014-06-26 03:50:41 +02:00
|
|
|
static void Sub4x4DctAnchor (int16_t iDct[4][4], uint8_t* pPix1, uint8_t* pPix2) {
|
2014-04-03 10:25:06 +02:00
|
|
|
int16_t iDiff[4][4];
|
|
|
|
int16_t tmp[4][4];
|
|
|
|
int i;
|
2014-06-26 03:50:41 +02:00
|
|
|
PixelSubWH ((int16_t*)iDiff, 4, pPix1, FENC_STRIDE, pPix2, FDEC_STRIDE);
|
|
|
|
for (i = 0; i < 4; i++) {
|
2014-04-03 10:25:06 +02:00
|
|
|
const int a03 = iDiff[i][0] + iDiff[i][3];
|
|
|
|
const int a12 = iDiff[i][1] + iDiff[i][2];
|
|
|
|
const int s03 = iDiff[i][0] - iDiff[i][3];
|
|
|
|
const int s12 = iDiff[i][1] - iDiff[i][2];
|
|
|
|
tmp[0][i] = a03 + a12;
|
2014-06-26 03:50:41 +02:00
|
|
|
tmp[1][i] = 2 * s03 + s12;
|
2014-04-03 10:25:06 +02:00
|
|
|
tmp[2][i] = a03 - a12;
|
2014-06-26 03:50:41 +02:00
|
|
|
tmp[3][i] = s03 - 2 * s12;
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
2014-06-26 03:50:41 +02:00
|
|
|
for (i = 0; i < 4; i++) {
|
2014-04-03 10:25:06 +02:00
|
|
|
const int a03 = tmp[i][0] + tmp[i][3];
|
|
|
|
const int a12 = tmp[i][1] + tmp[i][2];
|
|
|
|
const int s03 = tmp[i][0] - tmp[i][3];
|
|
|
|
const int s12 = tmp[i][1] - tmp[i][2];
|
|
|
|
iDct[i][0] = a03 + a12;
|
2014-06-26 03:50:41 +02:00
|
|
|
iDct[i][1] = 2 * s03 + s12;
|
2014-04-03 10:25:06 +02:00
|
|
|
iDct[i][2] = a03 - a12;
|
2014-06-26 03:50:41 +02:00
|
|
|
iDct[i][3] = s03 - 2 * s12;
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-26 03:50:41 +02:00
|
|
|
static void Sub8x8DctAnchor (int16_t iDct[4][4][4], uint8_t* pPix1, uint8_t* pPix2) {
|
|
|
|
Sub4x4DctAnchor (iDct[0], &pPix1[0], &pPix2[0]);
|
|
|
|
Sub4x4DctAnchor (iDct[1], &pPix1[4], &pPix2[4]);
|
|
|
|
Sub4x4DctAnchor (iDct[2], &pPix1[4 * FENC_STRIDE + 0], &pPix2[4 * FDEC_STRIDE + 0]);
|
|
|
|
Sub4x4DctAnchor (iDct[3], &pPix1[4 * FENC_STRIDE + 4], &pPix2[4 * FDEC_STRIDE + 4]);
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
2016-01-19 22:00:24 +01:00
|
|
|
static void TestDctT4 (PDctFunc func) {
|
2014-04-03 10:25:06 +02:00
|
|
|
int16_t iDctRef[4][4];
|
2016-01-27 13:18:51 +01:00
|
|
|
CMemoryAlign cMemoryAlign (0);
|
|
|
|
ALLOC_MEMORY (uint8_t, uiPix1, 16 * FENC_STRIDE);
|
|
|
|
ALLOC_MEMORY (uint8_t, uiPix2, 16 * FDEC_STRIDE);
|
|
|
|
ALLOC_MEMORY (int16_t, iDct, 16);
|
2016-01-18 20:14:47 +01:00
|
|
|
for (int i = 0; i < 4; i++) {
|
|
|
|
for (int j = 0; j < 4; j++) {
|
|
|
|
uiPix1[i * FENC_STRIDE + j] = rand() & 255;
|
|
|
|
uiPix2[i * FDEC_STRIDE + j] = rand() & 255;
|
|
|
|
}
|
|
|
|
}
|
2014-06-26 03:50:41 +02:00
|
|
|
Sub4x4DctAnchor (iDctRef, uiPix1, uiPix2);
|
2016-01-18 20:14:47 +01:00
|
|
|
func (iDct, uiPix1, FENC_STRIDE, uiPix2, FDEC_STRIDE);
|
2014-06-26 03:50:41 +02:00
|
|
|
for (int i = 0; i < 4; i++)
|
|
|
|
for (int j = 0; j < 4; j++)
|
|
|
|
EXPECT_EQ (iDctRef[j][i], iDct[i * 4 + j]);
|
2016-01-27 13:18:51 +01:00
|
|
|
FREE_MEMORY (uiPix1);
|
|
|
|
FREE_MEMORY (uiPix2);
|
|
|
|
FREE_MEMORY (iDct);
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
2016-01-19 22:00:24 +01:00
|
|
|
static void TestDctFourT4 (PDctFunc func) {
|
2014-06-26 03:50:41 +02:00
|
|
|
int16_t iDctRef[4][4][4];
|
2016-01-18 20:14:47 +01:00
|
|
|
CMemoryAlign cMemoryAlign (0);
|
|
|
|
ALLOC_MEMORY (uint8_t, uiPix1, 16 * FENC_STRIDE);
|
|
|
|
ALLOC_MEMORY (uint8_t, uiPix2, 16 * FDEC_STRIDE);
|
|
|
|
ALLOC_MEMORY (int16_t, iDct, 16 * 4);
|
|
|
|
for (int i = 0; i < 8; i++) {
|
|
|
|
for (int j = 0; j < 8; j++) {
|
|
|
|
uiPix1[i * FENC_STRIDE + j] = rand() & 255;
|
|
|
|
uiPix2[i * FDEC_STRIDE + j] = rand() & 255;
|
|
|
|
}
|
|
|
|
}
|
2014-06-26 03:50:41 +02:00
|
|
|
Sub8x8DctAnchor (iDctRef, uiPix1, uiPix2);
|
2016-01-18 20:14:47 +01:00
|
|
|
func (iDct, uiPix1, FENC_STRIDE, uiPix2, FDEC_STRIDE);
|
2014-06-26 03:50:41 +02:00
|
|
|
for (int k = 0; k < 4; k++)
|
|
|
|
for (int i = 0; i < 4; i++)
|
|
|
|
for (int j = 0; j < 4; j++)
|
|
|
|
EXPECT_EQ (iDctRef[k][j][i], iDct[k * 16 + i * 4 + j]);
|
2016-01-18 20:14:47 +01:00
|
|
|
FREE_MEMORY (uiPix1);
|
|
|
|
FREE_MEMORY (uiPix2);
|
|
|
|
FREE_MEMORY (iDct);
|
|
|
|
}
|
|
|
|
TEST (EncodeMbAuxTest, WelsDctT4_c) {
|
|
|
|
TestDctT4 (WelsDctT4_c);
|
|
|
|
}
|
|
|
|
TEST (EncodeMbAuxTest, WelsDctFourT4_c) {
|
|
|
|
TestDctFourT4 (WelsDctFourT4_c);
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#ifdef X86_ASM
|
2014-06-26 03:50:41 +02:00
|
|
|
TEST (EncodeMbAuxTest, WelsDctT4_mmx) {
|
2016-01-18 20:14:47 +01:00
|
|
|
TestDctT4 (WelsDctT4_mmx);
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
|
|
|
|
2016-01-27 13:18:51 +01:00
|
|
|
TEST (EncodeMbAuxTest, WelsDctT4_sse2) {
|
|
|
|
TestDctT4 (WelsDctT4_sse2);
|
|
|
|
}
|
|
|
|
|
2014-06-26 03:50:41 +02:00
|
|
|
TEST (EncodeMbAuxTest, WelsDctFourT4_sse2) {
|
2016-01-18 20:14:47 +01:00
|
|
|
TestDctFourT4 (WelsDctFourT4_sse2);
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
|
|
|
|
2016-01-27 13:23:52 +01:00
|
|
|
TEST (EncodeMbAuxTest, WelsDctT4_avx2) {
|
|
|
|
if (WelsCPUFeatureDetect (0) & WELS_CPU_AVX2)
|
|
|
|
TestDctT4 (WelsDctT4_avx2);
|
|
|
|
}
|
|
|
|
|
2016-01-18 20:43:28 +01:00
|
|
|
TEST (EncodeMbAuxTest, WelsDctFourT4_avx2) {
|
2016-01-19 14:27:46 +01:00
|
|
|
if (WelsCPUFeatureDetect (0) & WELS_CPU_AVX2)
|
|
|
|
TestDctFourT4 (WelsDctFourT4_avx2);
|
2016-01-18 20:43:28 +01:00
|
|
|
}
|
|
|
|
|
2014-06-26 03:50:41 +02:00
|
|
|
TEST (EncodeMbAuxTest, WelsCalculateSingleCtr4x4_sse2) {
|
|
|
|
CMemoryAlign cMemoryAlign (0);
|
|
|
|
ALLOC_MEMORY (int16_t, iDctC, 16);
|
|
|
|
ALLOC_MEMORY (int16_t, iDctS, 16);
|
|
|
|
for (int i = 0; i < 16; i++)
|
|
|
|
iDctC[i] = iDctS[i] = (rand() & 65535) - 32768;
|
|
|
|
WelsCalculateSingleCtr4x4_c (iDctC);
|
|
|
|
WelsCalculateSingleCtr4x4_sse2 (iDctS);
|
|
|
|
for (int i = 0; i < 16; i++)
|
|
|
|
EXPECT_EQ (iDctC[i], iDctS[i]);
|
|
|
|
FREE_MEMORY (iDctC);
|
|
|
|
FREE_MEMORY (iDctS);
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2014-06-26 03:50:41 +02:00
|
|
|
void copy (uint8_t* pDst, int32_t iDStride, uint8_t* pSrc, int32_t iSStride, int32_t iWidth, int32_t iHeight) {
|
|
|
|
for (int i = 0; i < iHeight; i++)
|
|
|
|
memcpy (pDst + i * iDStride, pSrc + i * iSStride, iWidth);
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#define GENERATE_UT_FOR_COPY(width, height, function) \
|
|
|
|
TEST(EncodeMbAuxTest, function) { \
|
|
|
|
const int iSStride = 64; \
|
|
|
|
const int iDStride = 64; \
|
2014-06-28 23:51:58 +02:00
|
|
|
ENFORCE_STACK_ALIGN_1D (uint8_t, ref_src, iSStride*height, 16); \
|
|
|
|
ENFORCE_STACK_ALIGN_1D (uint8_t, ref_dst, iDStride*height, 16); \
|
|
|
|
ENFORCE_STACK_ALIGN_1D (uint8_t, dst, iDStride*height, 16); \
|
2014-04-03 10:25:06 +02:00
|
|
|
for(int i = 0; i < height; i++) \
|
|
|
|
for(int j = 0; j < width; j++) \
|
|
|
|
ref_src[i*iSStride+j] = rand() & 255; \
|
|
|
|
function(dst, iDStride, ref_src, iSStride); \
|
|
|
|
copy(ref_dst, iDStride, ref_src, iSStride, width, height); \
|
|
|
|
for(int i = 0; i < height; i++) \
|
|
|
|
for(int j = 0; j < width; j++) \
|
|
|
|
EXPECT_EQ(ref_dst[i*iDStride+j], dst[i*iDStride+j]); \
|
|
|
|
}
|
|
|
|
|
2015-06-01 09:11:20 +02:00
|
|
|
GENERATE_UT_FOR_COPY (4, 4, WelsCopy4x4_c);
|
|
|
|
GENERATE_UT_FOR_COPY (8, 4, WelsCopy8x4_c);
|
|
|
|
GENERATE_UT_FOR_COPY (4, 8, WelsCopy4x8_c);
|
2014-06-26 03:50:41 +02:00
|
|
|
GENERATE_UT_FOR_COPY (8, 8, WelsCopy8x8_c);
|
|
|
|
GENERATE_UT_FOR_COPY (8, 16, WelsCopy8x16_c);
|
|
|
|
GENERATE_UT_FOR_COPY (16, 8, WelsCopy16x8_c);
|
|
|
|
GENERATE_UT_FOR_COPY (16, 16, WelsCopy16x16_c);
|
2014-04-03 10:25:06 +02:00
|
|
|
#ifdef X86_ASM
|
2014-06-26 03:50:41 +02:00
|
|
|
GENERATE_UT_FOR_COPY (16, 8, WelsCopy16x8NotAligned_sse2);
|
|
|
|
GENERATE_UT_FOR_COPY (16, 16, WelsCopy16x16NotAligned_sse2);
|
|
|
|
GENERATE_UT_FOR_COPY (16, 16, WelsCopy16x16_sse2);
|
2014-04-03 10:25:06 +02:00
|
|
|
#endif
|
2014-06-26 03:50:41 +02:00
|
|
|
TEST (EncodeMbAuxTest, WelsGetNoneZeroCount_c) {
|
2014-06-28 23:51:58 +02:00
|
|
|
ENFORCE_STACK_ALIGN_1D (int16_t, pLevel, 16, 16);
|
2014-04-03 10:25:06 +02:00
|
|
|
int32_t result = 0;
|
2014-06-26 03:50:41 +02:00
|
|
|
for (int i = 0; i < 16; i++) {
|
2014-04-03 10:25:06 +02:00
|
|
|
pLevel[i] = (rand() & 0x07) - 4;
|
2014-06-26 03:50:41 +02:00
|
|
|
if (pLevel[i]) result ++;
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
2014-06-26 03:50:41 +02:00
|
|
|
int32_t nnz = WelsGetNoneZeroCount_c (pLevel);
|
|
|
|
EXPECT_EQ (nnz, result);
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
|
|
|
#ifdef X86_ASM
|
2014-06-26 03:50:41 +02:00
|
|
|
TEST (EncodeMbAuxTest, WelsGetNoneZeroCount_sse2) {
|
2014-06-28 23:51:58 +02:00
|
|
|
ENFORCE_STACK_ALIGN_1D (int16_t, pLevel, 16, 16);
|
2014-04-03 10:25:06 +02:00
|
|
|
int32_t result = 0;
|
2014-06-26 03:50:41 +02:00
|
|
|
for (int i = 0; i < 16; i++) {
|
2014-04-03 10:25:06 +02:00
|
|
|
pLevel[i] = (rand() & 0x07) - 4;
|
2014-06-26 03:50:41 +02:00
|
|
|
if (pLevel[i]) result ++;
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
2014-06-26 03:50:41 +02:00
|
|
|
int32_t nnz = WelsGetNoneZeroCount_sse2 (pLevel);
|
|
|
|
EXPECT_EQ (nnz, result);
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
|
|
|
#endif
|
|
|
|
#define WELS_ABS_LC(a) ((sign ^ (int32_t)(a)) - sign)
|
|
|
|
#define NEW_QUANT(pDct, ff, mf) (((ff)+ WELS_ABS_LC(pDct))*(mf)) >>16
|
2015-06-03 08:54:03 +02:00
|
|
|
#define WELS_NEW_QUANT(pDct,ff,mf) WELS_ABS_LC(NEW_QUANT(pDct, ff, mf))
|
2014-06-26 03:50:41 +02:00
|
|
|
void WelsQuantFour4x4MaxAnchor (int16_t* pDct, int16_t* ff, int16_t* mf, int16_t* max) {
|
2014-04-03 10:25:06 +02:00
|
|
|
int32_t i, j, k, sign;
|
|
|
|
int16_t max_abs;
|
2014-06-26 03:50:41 +02:00
|
|
|
for (k = 0; k < 4; k++) {
|
2014-04-03 10:25:06 +02:00
|
|
|
max_abs = 0;
|
2014-06-26 03:50:41 +02:00
|
|
|
for (i = 0; i < 16; i++) {
|
2014-04-03 10:25:06 +02:00
|
|
|
j = i & 0x07;
|
2014-06-26 03:50:41 +02:00
|
|
|
sign = WELS_SIGN (pDct[i]);
|
|
|
|
pDct[i] = NEW_QUANT (pDct[i], ff[j], mf[j]);
|
|
|
|
if (max_abs < pDct[i]) max_abs = pDct[i];
|
|
|
|
pDct[i] = WELS_ABS_LC (pDct[i]);
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
|
|
|
pDct += 16;
|
|
|
|
max[k] = max_abs;
|
|
|
|
}
|
|
|
|
}
|
2014-06-26 03:50:41 +02:00
|
|
|
TEST (EncodeMbAuxTest, WelsQuantFour4x4Max_c) {
|
2014-04-03 10:25:06 +02:00
|
|
|
int16_t ff[8], mf[8];
|
|
|
|
int16_t iDctA[64], iMaxA[16];
|
|
|
|
int16_t iDctC[64], iMaxC[16];
|
2014-06-26 03:50:41 +02:00
|
|
|
for (int i = 0; i < 8; i++) {
|
2014-04-03 10:25:06 +02:00
|
|
|
ff[i] = rand() & 32767;
|
|
|
|
mf[i] = rand() & 32767;
|
|
|
|
}
|
2014-06-26 03:50:41 +02:00
|
|
|
for (int i = 0; i < 64; i++)
|
2014-04-03 10:25:06 +02:00
|
|
|
iDctA[i] = iDctC[i] = (rand() & 65535) - 32767;
|
2014-06-26 03:50:41 +02:00
|
|
|
WelsQuantFour4x4MaxAnchor (iDctA, ff, mf, iMaxA);
|
|
|
|
WelsQuantFour4x4Max_c (iDctC, ff, mf, iMaxC);
|
|
|
|
for (int i = 0; i < 64; i++)
|
|
|
|
EXPECT_EQ (iDctA[i], iDctC[i]);
|
|
|
|
for (int i = 0; i < 4; i++)
|
|
|
|
EXPECT_EQ (iMaxA[i], iMaxC[i]);
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
|
|
|
#ifdef X86_ASM
|
2014-06-26 03:50:41 +02:00
|
|
|
TEST (EncodeMbAuxTest, WelsQuantFour4x4Max_sse2) {
|
|
|
|
CMemoryAlign cMemoryAlign (0);
|
|
|
|
ALLOC_MEMORY (int16_t, ff, 8);
|
|
|
|
ALLOC_MEMORY (int16_t, mf, 8);
|
|
|
|
ALLOC_MEMORY (int16_t, iDctC, 64);
|
|
|
|
ALLOC_MEMORY (int16_t, iDctS, 64);
|
|
|
|
ALLOC_MEMORY (int16_t, iMaxC, 16);
|
|
|
|
ALLOC_MEMORY (int16_t, iMaxS, 16);
|
|
|
|
for (int i = 0; i < 8; i++) {
|
2014-04-03 10:25:06 +02:00
|
|
|
ff[i] = rand() & 32767;
|
|
|
|
mf[i] = rand() & 32767;
|
|
|
|
}
|
2014-06-26 03:50:41 +02:00
|
|
|
for (int i = 0; i < 64; i++)
|
2014-04-03 10:25:06 +02:00
|
|
|
iDctC[i] = iDctS[i] = (rand() & 65535) - 32767;
|
2014-06-26 03:50:41 +02:00
|
|
|
WelsQuantFour4x4Max_c (iDctC, ff, mf, iMaxC);
|
|
|
|
WelsQuantFour4x4Max_sse2 (iDctS, ff, mf, iMaxS);
|
|
|
|
for (int i = 0; i < 64; i++)
|
|
|
|
EXPECT_EQ (iDctC[i], iDctS[i]);
|
|
|
|
for (int i = 0; i < 4; i++)
|
|
|
|
EXPECT_EQ (iMaxC[i], iMaxS[i]);
|
|
|
|
FREE_MEMORY (ff);
|
|
|
|
FREE_MEMORY (mf);
|
|
|
|
FREE_MEMORY (iDctC);
|
|
|
|
FREE_MEMORY (iDctS);
|
|
|
|
FREE_MEMORY (iMaxC);
|
|
|
|
FREE_MEMORY (iMaxS);
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
|
|
|
#endif
|
2014-06-26 03:50:41 +02:00
|
|
|
int32_t WelsHadamardQuant2x2SkipAnchor (int16_t* rs, int16_t ff, int16_t mf) {
|
2014-04-03 10:25:06 +02:00
|
|
|
int16_t pDct[4], s[4];
|
2014-06-26 03:50:41 +02:00
|
|
|
int16_t threshold = ((1 << 16) - 1) / mf - ff;
|
2014-04-03 10:25:06 +02:00
|
|
|
s[0] = rs[0] + rs[32];
|
|
|
|
s[1] = rs[0] - rs[32];
|
|
|
|
s[2] = rs[16] + rs[48];
|
|
|
|
s[3] = rs[16] - rs[48];
|
|
|
|
pDct[0] = s[0] + s[2];
|
|
|
|
pDct[1] = s[0] - s[2];
|
|
|
|
pDct[2] = s[1] + s[3];
|
|
|
|
pDct[3] = s[1] - s[3];
|
2014-06-26 03:50:41 +02:00
|
|
|
return ((WELS_ABS (pDct[0]) > threshold) || (WELS_ABS (pDct[1]) > threshold) || (WELS_ABS (pDct[2]) > threshold)
|
|
|
|
|| (WELS_ABS (pDct[3]) > threshold));
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
|
|
|
|
2014-06-26 03:50:41 +02:00
|
|
|
TEST (EncodeMbAuxTest, WelsHadamardQuant2x2Skip_c) {
|
2014-04-03 10:25:06 +02:00
|
|
|
int16_t iRS[64];
|
|
|
|
int16_t ff, mf;
|
2014-06-26 03:50:41 +02:00
|
|
|
for (int i = 0; i < 64; i++)
|
2014-04-03 10:25:06 +02:00
|
|
|
iRS[i] = (rand() & 32767) - 16384;
|
|
|
|
ff = rand() & 32767;
|
|
|
|
mf = rand() & 32767;
|
2014-06-26 03:50:41 +02:00
|
|
|
EXPECT_EQ (WelsHadamardQuant2x2Skip_c (iRS, ff, mf), WelsHadamardQuant2x2SkipAnchor (iRS, ff, mf));
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
|
|
|
|
2014-06-26 03:50:41 +02:00
|
|
|
int32_t WelsHadamardQuant2x2Anchor (int16_t* rs, const int16_t ff, int16_t mf, int16_t* pDct, int16_t* block) {
|
2014-04-03 10:25:06 +02:00
|
|
|
int16_t s[4];
|
|
|
|
int32_t sign, i, dc_nzc = 0;
|
|
|
|
|
|
|
|
s[0] = rs[0] + rs[32];
|
|
|
|
s[1] = rs[0] - rs[32];
|
|
|
|
s[2] = rs[16] + rs[48];
|
|
|
|
s[3] = rs[16] - rs[48];
|
|
|
|
|
|
|
|
rs[0] = 0;
|
|
|
|
rs[16] = 0;
|
|
|
|
rs[32] = 0;
|
|
|
|
rs[48] = 0;
|
|
|
|
|
|
|
|
pDct[0] = s[0] + s[2];
|
|
|
|
pDct[1] = s[0] - s[2];
|
|
|
|
pDct[2] = s[1] + s[3];
|
|
|
|
pDct[3] = s[1] - s[3];
|
|
|
|
|
2014-06-26 03:50:41 +02:00
|
|
|
sign = WELS_SIGN (pDct[0]);
|
|
|
|
pDct[0] = WELS_NEW_QUANT (pDct[0], ff, mf);
|
|
|
|
sign = WELS_SIGN (pDct[1]);
|
|
|
|
pDct[1] = WELS_NEW_QUANT (pDct[1], ff, mf);
|
|
|
|
sign = WELS_SIGN (pDct[2]);
|
|
|
|
pDct[2] = WELS_NEW_QUANT (pDct[2], ff, mf);
|
|
|
|
sign = WELS_SIGN (pDct[3]);
|
|
|
|
pDct[3] = WELS_NEW_QUANT (pDct[3], ff, mf);
|
|
|
|
ST64 (block, LD64 (pDct));
|
|
|
|
for (i = 0; i < 4; i++)
|
2014-04-03 10:25:06 +02:00
|
|
|
dc_nzc += (block[i] != 0);
|
|
|
|
return dc_nzc;
|
|
|
|
}
|
|
|
|
|
2014-06-26 03:50:41 +02:00
|
|
|
TEST (EncodeMbAuxTest, WelsHadamardQuant2x2_c) {
|
2014-04-03 10:25:06 +02:00
|
|
|
int16_t iRsC[64], iRsA[64];
|
|
|
|
int16_t ff, mf;
|
|
|
|
int16_t iBlockA[16], iBlockC[16], iDctA[4], iDctC[4];
|
2014-06-26 03:50:41 +02:00
|
|
|
for (int i = 0; i < 64; i++)
|
2014-04-03 10:25:06 +02:00
|
|
|
iRsA[i] = iRsC[i] = (rand() & 32767) - 16384;
|
2014-06-26 03:50:41 +02:00
|
|
|
for (int i = 0; i < 4; i++)
|
2014-04-03 10:25:06 +02:00
|
|
|
iDctA[i] = iDctC[i] = (rand() & 32767) - 16384;
|
|
|
|
ff = rand() & 32767;
|
|
|
|
mf = rand() & 32767;
|
|
|
|
|
2014-06-26 03:50:41 +02:00
|
|
|
int32_t iRetA = WelsHadamardQuant2x2Anchor (iRsA, ff, mf, iDctA, iBlockA);
|
|
|
|
int32_t iRetC = WelsHadamardQuant2x2_c (iRsC, ff, mf, iDctC, iBlockC);
|
|
|
|
EXPECT_EQ (iRetA, iRetC);
|
|
|
|
for (int i = 0; i < 4; i++)
|
|
|
|
EXPECT_EQ (iDctA[i], iDctC[i]);
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
|
|
|
|
2014-06-26 03:50:41 +02:00
|
|
|
void WelsHadamardT4DcAnchor (int16_t* pLumaDc, int16_t* pDct) {
|
2014-04-03 10:25:06 +02:00
|
|
|
int32_t p[16], s[4];
|
|
|
|
int32_t i, iIdx;
|
2014-06-26 03:50:41 +02:00
|
|
|
for (i = 0 ; i < 16 ; i += 4) {
|
|
|
|
iIdx = ((i & 0x08) << 4) + ((i & 0x04) << 3);
|
2015-05-15 10:45:20 +02:00
|
|
|
s[0] = pDct[iIdx ] + pDct[iIdx + 80];
|
|
|
|
s[3] = pDct[iIdx ] - pDct[iIdx + 80];
|
2014-06-26 03:50:41 +02:00
|
|
|
s[1] = pDct[iIdx + 16] + pDct[iIdx + 64];
|
|
|
|
s[2] = pDct[iIdx + 16] - pDct[iIdx + 64];
|
2014-04-03 10:25:06 +02:00
|
|
|
p[i ] = s[0] + s[1];
|
2014-06-26 03:50:41 +02:00
|
|
|
p[i + 2] = s[0] - s[1];
|
|
|
|
p[i + 1] = s[3] + s[2];
|
|
|
|
p[i + 3] = s[3] - s[2];
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
2014-06-26 03:50:41 +02:00
|
|
|
for (i = 0 ; i < 4 ; i ++) {
|
|
|
|
s[0] = p[i ] + p[i + 12];
|
|
|
|
s[3] = p[i ] - p[i + 12];
|
|
|
|
s[1] = p[i + 4] + p[i + 8];
|
|
|
|
s[2] = p[i + 4] - p[i + 8];
|
|
|
|
pLumaDc[i ] = WELS_CLIP3 ((s[0] + s[1] + 1) >> 1, -32768, 32767);
|
|
|
|
pLumaDc[i + 8 ] = WELS_CLIP3 ((s[0] - s[1] + 1) >> 1, -32768, 32767);
|
|
|
|
pLumaDc[i + 4 ] = WELS_CLIP3 ((s[3] + s[2] + 1) >> 1, -32768, 32767);
|
|
|
|
pLumaDc[i + 12] = WELS_CLIP3 ((s[3] - s[2] + 1) >> 1, -32768, 32767);
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
|
|
|
}
|
2014-06-26 03:50:41 +02:00
|
|
|
TEST (EncodeMbAuxTest, WelsHadamardT4Dc_c) {
|
|
|
|
CMemoryAlign cMemoryAlign (0);
|
|
|
|
ALLOC_MEMORY (int16_t, iDct, 128 * 16);
|
|
|
|
ALLOC_MEMORY (int16_t, iLumaDcR, 16);
|
|
|
|
ALLOC_MEMORY (int16_t, iLumaDcC, 16);
|
|
|
|
for (int i = 0; i < 128 * 16; i++)
|
2014-04-03 10:25:06 +02:00
|
|
|
iDct[i] = (rand() & 32767) - 16384;
|
2014-06-26 03:50:41 +02:00
|
|
|
WelsHadamardT4DcAnchor (iLumaDcR, iDct);
|
|
|
|
WelsHadamardT4Dc_c (iLumaDcC, iDct);
|
|
|
|
for (int i = 0; i < 16; i++)
|
|
|
|
EXPECT_EQ (iLumaDcR[i], iLumaDcC[i]);
|
|
|
|
FREE_MEMORY (iDct);
|
|
|
|
FREE_MEMORY (iLumaDcR);
|
|
|
|
FREE_MEMORY (iLumaDcC);
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
|
|
|
#ifdef X86_ASM
|
2014-06-26 03:50:41 +02:00
|
|
|
TEST (EncodeMbAuxTest, WelsHadamardT4Dc_sse2) {
|
|
|
|
CMemoryAlign cMemoryAlign (0);
|
|
|
|
ALLOC_MEMORY (int16_t, iDct, 128 * 16);
|
|
|
|
ALLOC_MEMORY (int16_t, iLumaDcC, 16);
|
|
|
|
ALLOC_MEMORY (int16_t, iLumaDcS, 16);
|
|
|
|
for (int i = 0; i < 128 * 16; i++)
|
2014-04-03 10:25:06 +02:00
|
|
|
iDct[i] = (rand() & 32767) - 16384;
|
2014-06-26 03:50:41 +02:00
|
|
|
WelsHadamardT4Dc_c (iLumaDcC, iDct);
|
|
|
|
WelsHadamardT4Dc_sse2 (iLumaDcS, iDct);
|
|
|
|
for (int i = 0; i < 16; i++)
|
|
|
|
EXPECT_EQ (iLumaDcC[i], iLumaDcS[i]);
|
|
|
|
FREE_MEMORY (iDct);
|
|
|
|
FREE_MEMORY (iLumaDcC);
|
|
|
|
FREE_MEMORY (iLumaDcS);
|
2014-04-03 10:25:06 +02:00
|
|
|
}
|
|
|
|
#endif
|