mirror of
https://chromium.googlesource.com/libyuv/libyuv
synced 2025-12-07 09:16:48 +08:00
BUG=391 TESTED=TestYUV R=harryjin@google.com Review URL: https://webrtc-codereview.appspot.com/36819004 git-svn-id: http://libyuv.googlecode.com/svn/trunk@1232 16f28f9a-4ce2-e073-06de-1de4eb20be90
291 lines
13 KiB
C++
291 lines
13 KiB
C++
/*
|
|
* Copyright 2015 The LibYuv Project Authors. All rights reserved.
|
|
*
|
|
* Use of this source code is governed by a BSD-style license
|
|
* that can be found in the LICENSE file in the root of the source
|
|
* tree. An additional intellectual property rights grant can be found
|
|
* in the file PATENTS. All contributing project authors may
|
|
* be found in the AUTHORS file in the root of the source tree.
|
|
*/
|
|
|
|
#include <math.h> // For round
|
|
#include <stdlib.h>
|
|
|
|
#include "libyuv/convert.h"
|
|
#include "libyuv/convert_argb.h"
|
|
#include "libyuv/convert_from.h"
|
|
#include "libyuv/convert_from_argb.h"
|
|
#include "libyuv/cpu_id.h"
|
|
#include "libyuv/row.h" // For Sobel
|
|
#include "../unit_test/unit_test.h"
|
|
|
|
namespace libyuv {
|
|
|
|
#define TESTCS(TESTNAME, YUVTOARGB, ARGBTOYUV, HS1, HS, HN, DIFF) \
|
|
TEST_F(libyuvTest, TESTNAME) { \
|
|
const int kPixels = benchmark_width_ * benchmark_height_; \
|
|
const int kHalfPixels = ((benchmark_width_ + 1) / 2) * \
|
|
((benchmark_height_ + HS1) / HS); \
|
|
align_buffer_64(orig_y, kPixels); \
|
|
align_buffer_64(orig_u, kHalfPixels); \
|
|
align_buffer_64(orig_v, kHalfPixels); \
|
|
align_buffer_64(orig_pixels, kPixels * 4); \
|
|
align_buffer_64(temp_y, kPixels); \
|
|
align_buffer_64(temp_u, kHalfPixels); \
|
|
align_buffer_64(temp_v, kHalfPixels); \
|
|
align_buffer_64(dst_pixels_opt, kPixels * 4); \
|
|
align_buffer_64(dst_pixels_c, kPixels * 4); \
|
|
\
|
|
MemRandomize(orig_pixels, kPixels * 4); \
|
|
MemRandomize(orig_y, kPixels); \
|
|
MemRandomize(orig_u, kHalfPixels); \
|
|
MemRandomize(orig_v, kHalfPixels); \
|
|
MemRandomize(temp_y, kPixels); \
|
|
MemRandomize(temp_u, kHalfPixels); \
|
|
MemRandomize(temp_v, kHalfPixels); \
|
|
MemRandomize(dst_pixels_opt, kPixels * 4); \
|
|
MemRandomize(dst_pixels_c, kPixels * 4); \
|
|
\
|
|
/* The test is overall for color conversion matrix being reversible, so */ \
|
|
/* this initializes the pixel with 2x2 blocks to eliminate subsampling. */ \
|
|
uint8* p = orig_y; \
|
|
for (int y = 0; y < benchmark_height_ - HS1; y += HS) { \
|
|
for (int x = 0; x < benchmark_width_ - 1; x += 2) { \
|
|
uint8 r = static_cast<uint8>(random()); \
|
|
p[0] = r; \
|
|
p[1] = r; \
|
|
p[HN] = r; \
|
|
p[HN + 1] = r; \
|
|
p += 2; \
|
|
} \
|
|
p += HN; \
|
|
} \
|
|
\
|
|
/* Start with YUV converted to ARGB. */ \
|
|
YUVTOARGB(orig_y, benchmark_width_, \
|
|
orig_u, (benchmark_width_ + 1) / 2, \
|
|
orig_v, (benchmark_width_ + 1) / 2, \
|
|
orig_pixels, benchmark_width_ * 4, \
|
|
benchmark_width_, benchmark_height_); \
|
|
\
|
|
ARGBTOYUV(orig_pixels, benchmark_width_ * 4, \
|
|
temp_y, benchmark_width_, \
|
|
temp_u, (benchmark_width_ + 1) / 2, \
|
|
temp_v, (benchmark_width_ + 1) / 2, \
|
|
benchmark_width_, benchmark_height_); \
|
|
\
|
|
MaskCpuFlags(0); \
|
|
YUVTOARGB(temp_y, benchmark_width_, \
|
|
temp_u, (benchmark_width_ + 1) / 2, \
|
|
temp_v, (benchmark_width_ + 1) / 2, \
|
|
dst_pixels_c, benchmark_width_ * 4, \
|
|
benchmark_width_, benchmark_height_); \
|
|
MaskCpuFlags(-1); \
|
|
\
|
|
for (int i = 0; i < benchmark_iterations_; ++i) { \
|
|
YUVTOARGB(temp_y, benchmark_width_, \
|
|
temp_u, (benchmark_width_ + 1) / 2, \
|
|
temp_v, (benchmark_width_ + 1) / 2, \
|
|
dst_pixels_opt, benchmark_width_ * 4, \
|
|
benchmark_width_, benchmark_height_); \
|
|
} \
|
|
/* Test C and SIMD match. */ \
|
|
for (int i = 0; i < kPixels * 4; ++i) { \
|
|
EXPECT_EQ(dst_pixels_c[i], dst_pixels_opt[i]); \
|
|
} \
|
|
/* Test SIMD is close to original. */ \
|
|
for (int i = 0; i < kPixels * 4; ++i) { \
|
|
EXPECT_NEAR(static_cast<int>(orig_pixels[i]), \
|
|
static_cast<int>(dst_pixels_opt[i]), DIFF); \
|
|
} \
|
|
\
|
|
free_aligned_buffer_64(orig_pixels); \
|
|
free_aligned_buffer_64(orig_y); \
|
|
free_aligned_buffer_64(orig_u); \
|
|
free_aligned_buffer_64(orig_v); \
|
|
free_aligned_buffer_64(temp_y); \
|
|
free_aligned_buffer_64(temp_u); \
|
|
free_aligned_buffer_64(temp_v); \
|
|
free_aligned_buffer_64(dst_pixels_opt); \
|
|
free_aligned_buffer_64(dst_pixels_c); \
|
|
} \
|
|
|
|
TESTCS(TestI420, I420ToARGB, ARGBToI420, 1, 2, benchmark_width_, 7)
|
|
TESTCS(TestI422, I422ToARGB, ARGBToI422, 0, 1, 0, 7)
|
|
TESTCS(TestJ420, J420ToARGB, ARGBToJ420, 1, 2, benchmark_width_, 3)
|
|
TESTCS(TestJ422, J422ToARGB, ARGBToJ422, 0, 1, 0, 4)
|
|
|
|
void TestYUVToRGB(int y, int u, int v, int &r, int &g, int &b,
|
|
int benchmark_width_, int benchmark_height_) {
|
|
const int kPixels = benchmark_width_ * benchmark_height_;
|
|
const int kHalfPixels = ((benchmark_width_ + 1) / 2) *
|
|
((benchmark_height_ + 1) / 2);
|
|
align_buffer_64(orig_y, kPixels);
|
|
align_buffer_64(orig_u, kHalfPixels);
|
|
align_buffer_64(orig_v, kHalfPixels);
|
|
align_buffer_64(orig_pixels, kPixels * 4);
|
|
|
|
memset(orig_y, y, kPixels);
|
|
memset(orig_u, u, kHalfPixels);
|
|
memset(orig_v, v, kHalfPixels);
|
|
MemRandomize(orig_pixels, kPixels * 4);
|
|
|
|
/* YUV converted to ARGB. */
|
|
I420ToARGB(orig_y, benchmark_width_,
|
|
orig_u, (benchmark_width_ + 1) / 2,
|
|
orig_v, (benchmark_width_ + 1) / 2,
|
|
orig_pixels, benchmark_width_ * 4,
|
|
benchmark_width_, benchmark_height_);
|
|
|
|
b = orig_pixels[0];
|
|
g = orig_pixels[1];
|
|
r = orig_pixels[2];
|
|
|
|
free_aligned_buffer_64(orig_pixels);
|
|
free_aligned_buffer_64(orig_y);
|
|
free_aligned_buffer_64(orig_u);
|
|
free_aligned_buffer_64(orig_v);
|
|
}
|
|
|
|
int Clamp(double f) {
|
|
int i = static_cast<int>(round(f));
|
|
if (i < 0) {
|
|
i = 0;
|
|
}
|
|
if (i > 255) {
|
|
i = 255;
|
|
}
|
|
return i;
|
|
}
|
|
|
|
void TestYUVToRGBReference(int y, int u, int v, int &r, int &g, int &b) {
|
|
r = Clamp((y - 16) * 1.164 + (v - 128) * 1.596);
|
|
g = Clamp((y - 16) * 1.164 + (u - 128) * -0.391 + (v - 128) * -0.813);
|
|
b = Clamp((y - 16) * 1.164 + (u - 128) * 2.018);
|
|
}
|
|
|
|
// C prototype code
|
|
|
|
#define YG 4901241 /* round(1.164 * 64 * 256 * 257) */
|
|
#define YGB 1192 /* round(1.164 * 64 * 16 ) */
|
|
|
|
#define UB -128 /* -min(128, round(2.018 * 64)) */
|
|
#define UG 25 /* -round(-0.391 * 64) */
|
|
#define UR 0
|
|
|
|
#define VB 0
|
|
#define VG 52 /* -round(-0.813 * 64) */
|
|
#define VR -102 /* -round(1.596 * 64) */
|
|
|
|
// Bias
|
|
#define BB (UB * 128 + VB * 128 - YGB)
|
|
#define BG (UG * 128 + VG * 128 - YGB)
|
|
#define BR (UR * 128 + VR * 128 - YGB)
|
|
|
|
void TestYUVToRGBInt(int y, int u, int v, int &r, int &g, int &b) {
|
|
uint32 y1 = (uint32)(y * YG) >> 16;
|
|
b = Clamp((int32)(y1 - (v * VB + u * UB) + BB) >> 6);
|
|
g = Clamp((int32)(y1 - (v * VG + u * UG) + BG) >> 6);
|
|
r = Clamp((int32)(y1 - (v * VR + u * UR) + BR) >> 6);
|
|
}
|
|
|
|
TEST_F(libyuvTest, TestYUV) {
|
|
int r0, g0, b0;
|
|
|
|
// black
|
|
TestYUVToRGBReference(16, 128, 128, r0, g0, b0);
|
|
EXPECT_EQ(0, r0);
|
|
EXPECT_EQ(0, g0);
|
|
EXPECT_EQ(0, b0);
|
|
|
|
int r1, g1, b1;
|
|
TestYUVToRGB(16, 128, 128, r1, g1, b1, benchmark_width_, benchmark_height_);
|
|
EXPECT_EQ(0, r1);
|
|
EXPECT_EQ(0, g1);
|
|
EXPECT_EQ(0, b1);
|
|
|
|
// white
|
|
TestYUVToRGBReference(240, 128, 128, r0, g0, b0);
|
|
EXPECT_EQ(255, r0);
|
|
EXPECT_EQ(255, g0);
|
|
EXPECT_EQ(255, b0);
|
|
|
|
TestYUVToRGB(240, 128, 128, r1, g1, b1, benchmark_width_, benchmark_height_);
|
|
EXPECT_EQ(255, r1);
|
|
EXPECT_EQ(255, g1);
|
|
EXPECT_EQ(255, b1);
|
|
|
|
// cyan (less red)
|
|
TestYUVToRGBReference(240, 255, 0, r0, g0, b0);
|
|
EXPECT_EQ(56, r0);
|
|
EXPECT_EQ(255, g0);
|
|
EXPECT_EQ(255, b0);
|
|
|
|
TestYUVToRGB(240, 255, 0, r1, g1, b1, benchmark_width_, benchmark_height_);
|
|
EXPECT_EQ(55, r1);
|
|
EXPECT_EQ(255, g1);
|
|
EXPECT_EQ(255, b1);
|
|
|
|
// green (less red and blue)
|
|
TestYUVToRGBReference(240, 0, 0, r0, g0, b0);
|
|
EXPECT_EQ(56, r0);
|
|
EXPECT_EQ(255, g0);
|
|
EXPECT_EQ(2, b0);
|
|
|
|
TestYUVToRGB(240, 0, 0, r1, g1, b1, benchmark_width_, benchmark_height_);
|
|
EXPECT_EQ(55, r1);
|
|
EXPECT_EQ(255, g1);
|
|
EXPECT_EQ(5, b1);
|
|
|
|
int r2, g2, b2;
|
|
for (int i = 0; i < 255; ++i) {
|
|
TestYUVToRGBReference(i, 128, 128, r0, g0, b0);
|
|
TestYUVToRGB(i, 128, 128, r1, g1, b1, benchmark_width_, benchmark_height_);
|
|
TestYUVToRGBInt(i, 128, 128, r2, g2, b2);
|
|
EXPECT_NEAR(r0, r1, 3);
|
|
EXPECT_NEAR(g0, g1, 3);
|
|
EXPECT_NEAR(b0, b1, 3);
|
|
EXPECT_NEAR(r0, r2, 1);
|
|
EXPECT_NEAR(g0, g2, 1);
|
|
EXPECT_NEAR(b0, b2, 1);
|
|
|
|
TestYUVToRGBReference(i, 0, 0, r0, g0, b0);
|
|
TestYUVToRGB(i, 0, 0, r1, g1, b1, benchmark_width_, benchmark_height_);
|
|
TestYUVToRGBInt(i, 0, 0, r2, g2, b2);
|
|
EXPECT_NEAR(r0, r1, 3);
|
|
EXPECT_NEAR(g0, g1, 3);
|
|
EXPECT_NEAR(b0, b1, 3);
|
|
EXPECT_NEAR(r0, r2, 1);
|
|
EXPECT_NEAR(g0, g2, 1);
|
|
EXPECT_NEAR(b0, b2, 3);
|
|
|
|
TestYUVToRGBReference(i, 0, 255, r0, g0, b0);
|
|
TestYUVToRGB(i, 0, 255, r1, g1, b1, benchmark_width_, benchmark_height_);
|
|
TestYUVToRGBInt(i, 0, 255, r2, g2, b2);
|
|
EXPECT_NEAR(r0, r1, 3);
|
|
EXPECT_NEAR(g0, g1, 3);
|
|
EXPECT_NEAR(b0, b1, 3);
|
|
EXPECT_NEAR(r0, r2, 1);
|
|
EXPECT_NEAR(g0, g2, 1);
|
|
EXPECT_NEAR(b0, b2, 3);
|
|
}
|
|
|
|
for (int i = 0; i < 1000; ++i) {
|
|
int yr = random() & 255;
|
|
int ur = random() & 255;
|
|
int vr = random() & 255;
|
|
|
|
TestYUVToRGBReference(yr, ur, vr, r0, g0, b0);
|
|
TestYUVToRGB(yr, ur, vr, r1, g1, b1, benchmark_width_, benchmark_height_);
|
|
TestYUVToRGBInt(yr, ur, vr, r2, g2, b2);
|
|
EXPECT_NEAR(r0, r1, 3);
|
|
EXPECT_NEAR(g0, g1, 3);
|
|
EXPECT_NEAR(b0, b1, 5);
|
|
EXPECT_NEAR(r0, r2, 1);
|
|
EXPECT_NEAR(g0, g2, 1);
|
|
EXPECT_NEAR(b0, b2, 3);
|
|
}
|
|
}
|
|
|
|
} // namespace libyuv
|