1*fb1b10abSAndroid Build Coastguard Worker /*
2*fb1b10abSAndroid Build Coastguard Worker * Copyright (c) 2017 The WebM project authors. All Rights Reserved.
3*fb1b10abSAndroid Build Coastguard Worker *
4*fb1b10abSAndroid Build Coastguard Worker * Use of this source code is governed by a BSD-style license
5*fb1b10abSAndroid Build Coastguard Worker * that can be found in the LICENSE file in the root of the source
6*fb1b10abSAndroid Build Coastguard Worker * tree. An additional intellectual property rights grant can be found
7*fb1b10abSAndroid Build Coastguard Worker * in the file PATENTS. All contributing project authors may
8*fb1b10abSAndroid Build Coastguard Worker * be found in the AUTHORS file in the root of the source tree.
9*fb1b10abSAndroid Build Coastguard Worker */
10*fb1b10abSAndroid Build Coastguard Worker
11*fb1b10abSAndroid Build Coastguard Worker #include <assert.h>
12*fb1b10abSAndroid Build Coastguard Worker #include <stdio.h>
13*fb1b10abSAndroid Build Coastguard Worker #include <string.h>
14*fb1b10abSAndroid Build Coastguard Worker
15*fb1b10abSAndroid Build Coastguard Worker #include "gtest/gtest.h"
16*fb1b10abSAndroid Build Coastguard Worker
17*fb1b10abSAndroid Build Coastguard Worker #include "./vp9_rtcd.h"
18*fb1b10abSAndroid Build Coastguard Worker #include "./vpx_config.h"
19*fb1b10abSAndroid Build Coastguard Worker #include "./vpx_scale_rtcd.h"
20*fb1b10abSAndroid Build Coastguard Worker #include "test/clear_system_state.h"
21*fb1b10abSAndroid Build Coastguard Worker #include "test/register_state_check.h"
22*fb1b10abSAndroid Build Coastguard Worker #include "test/vpx_scale_test.h"
23*fb1b10abSAndroid Build Coastguard Worker #include "vpx_mem/vpx_mem.h"
24*fb1b10abSAndroid Build Coastguard Worker #include "vpx_ports/vpx_timer.h"
25*fb1b10abSAndroid Build Coastguard Worker #include "vpx_scale/yv12config.h"
26*fb1b10abSAndroid Build Coastguard Worker
27*fb1b10abSAndroid Build Coastguard Worker namespace libvpx_test {
28*fb1b10abSAndroid Build Coastguard Worker
29*fb1b10abSAndroid Build Coastguard Worker typedef void (*ScaleFrameFunc)(const YV12_BUFFER_CONFIG *src,
30*fb1b10abSAndroid Build Coastguard Worker YV12_BUFFER_CONFIG *dst,
31*fb1b10abSAndroid Build Coastguard Worker INTERP_FILTER filter_type, int phase_scaler);
32*fb1b10abSAndroid Build Coastguard Worker
33*fb1b10abSAndroid Build Coastguard Worker class ScaleTest : public VpxScaleBase,
34*fb1b10abSAndroid Build Coastguard Worker public ::testing::TestWithParam<ScaleFrameFunc> {
35*fb1b10abSAndroid Build Coastguard Worker public:
36*fb1b10abSAndroid Build Coastguard Worker ~ScaleTest() override = default;
37*fb1b10abSAndroid Build Coastguard Worker
38*fb1b10abSAndroid Build Coastguard Worker protected:
SetUp()39*fb1b10abSAndroid Build Coastguard Worker void SetUp() override { scale_fn_ = GetParam(); }
40*fb1b10abSAndroid Build Coastguard Worker
ReferenceScaleFrame(INTERP_FILTER filter_type,int phase_scaler)41*fb1b10abSAndroid Build Coastguard Worker void ReferenceScaleFrame(INTERP_FILTER filter_type, int phase_scaler) {
42*fb1b10abSAndroid Build Coastguard Worker vp9_scale_and_extend_frame_c(&img_, &ref_img_, filter_type, phase_scaler);
43*fb1b10abSAndroid Build Coastguard Worker }
44*fb1b10abSAndroid Build Coastguard Worker
ScaleFrame(INTERP_FILTER filter_type,int phase_scaler)45*fb1b10abSAndroid Build Coastguard Worker void ScaleFrame(INTERP_FILTER filter_type, int phase_scaler) {
46*fb1b10abSAndroid Build Coastguard Worker ASM_REGISTER_STATE_CHECK(
47*fb1b10abSAndroid Build Coastguard Worker scale_fn_(&img_, &dst_img_, filter_type, phase_scaler));
48*fb1b10abSAndroid Build Coastguard Worker }
49*fb1b10abSAndroid Build Coastguard Worker
RunTest(INTERP_FILTER filter_type)50*fb1b10abSAndroid Build Coastguard Worker void RunTest(INTERP_FILTER filter_type) {
51*fb1b10abSAndroid Build Coastguard Worker static const int kNumSizesToTest = 22;
52*fb1b10abSAndroid Build Coastguard Worker static const int kNumScaleFactorsToTest = 4;
53*fb1b10abSAndroid Build Coastguard Worker static const int kSizesToTest[] = { 1, 2, 3, 4, 6, 8, 10, 12,
54*fb1b10abSAndroid Build Coastguard Worker 14, 16, 18, 20, 22, 24, 26, 28,
55*fb1b10abSAndroid Build Coastguard Worker 30, 32, 34, 68, 128, 134 };
56*fb1b10abSAndroid Build Coastguard Worker static const int kScaleFactors[] = { 1, 2, 3, 4 };
57*fb1b10abSAndroid Build Coastguard Worker for (int phase_scaler = 0; phase_scaler < 16; ++phase_scaler) {
58*fb1b10abSAndroid Build Coastguard Worker for (int h = 0; h < kNumSizesToTest; ++h) {
59*fb1b10abSAndroid Build Coastguard Worker const int src_height = kSizesToTest[h];
60*fb1b10abSAndroid Build Coastguard Worker for (int w = 0; w < kNumSizesToTest; ++w) {
61*fb1b10abSAndroid Build Coastguard Worker const int src_width = kSizesToTest[w];
62*fb1b10abSAndroid Build Coastguard Worker for (int sf_up_idx = 0; sf_up_idx < kNumScaleFactorsToTest;
63*fb1b10abSAndroid Build Coastguard Worker ++sf_up_idx) {
64*fb1b10abSAndroid Build Coastguard Worker const int sf_up = kScaleFactors[sf_up_idx];
65*fb1b10abSAndroid Build Coastguard Worker for (int sf_down_idx = 0; sf_down_idx < kNumScaleFactorsToTest;
66*fb1b10abSAndroid Build Coastguard Worker ++sf_down_idx) {
67*fb1b10abSAndroid Build Coastguard Worker const int sf_down = kScaleFactors[sf_down_idx];
68*fb1b10abSAndroid Build Coastguard Worker const int dst_width = src_width * sf_up / sf_down;
69*fb1b10abSAndroid Build Coastguard Worker const int dst_height = src_height * sf_up / sf_down;
70*fb1b10abSAndroid Build Coastguard Worker if (sf_up == sf_down && sf_up != 1) {
71*fb1b10abSAndroid Build Coastguard Worker continue;
72*fb1b10abSAndroid Build Coastguard Worker }
73*fb1b10abSAndroid Build Coastguard Worker // I420 frame width and height must be even.
74*fb1b10abSAndroid Build Coastguard Worker if (!dst_width || !dst_height || dst_width & 1 ||
75*fb1b10abSAndroid Build Coastguard Worker dst_height & 1) {
76*fb1b10abSAndroid Build Coastguard Worker continue;
77*fb1b10abSAndroid Build Coastguard Worker }
78*fb1b10abSAndroid Build Coastguard Worker // vpx_convolve8_c() has restriction on the step which cannot
79*fb1b10abSAndroid Build Coastguard Worker // exceed 64 (ratio 1 to 4).
80*fb1b10abSAndroid Build Coastguard Worker if (src_width > 4 * dst_width || src_height > 4 * dst_height) {
81*fb1b10abSAndroid Build Coastguard Worker continue;
82*fb1b10abSAndroid Build Coastguard Worker }
83*fb1b10abSAndroid Build Coastguard Worker ASSERT_NO_FATAL_FAILURE(ResetScaleImages(src_width, src_height,
84*fb1b10abSAndroid Build Coastguard Worker dst_width, dst_height));
85*fb1b10abSAndroid Build Coastguard Worker ReferenceScaleFrame(filter_type, phase_scaler);
86*fb1b10abSAndroid Build Coastguard Worker ScaleFrame(filter_type, phase_scaler);
87*fb1b10abSAndroid Build Coastguard Worker if (memcmp(dst_img_.buffer_alloc, ref_img_.buffer_alloc,
88*fb1b10abSAndroid Build Coastguard Worker ref_img_.frame_size)) {
89*fb1b10abSAndroid Build Coastguard Worker printf(
90*fb1b10abSAndroid Build Coastguard Worker "filter_type = %d, phase_scaler = %d, src_width = %4d, "
91*fb1b10abSAndroid Build Coastguard Worker "src_height = %4d, dst_width = %4d, dst_height = %4d, "
92*fb1b10abSAndroid Build Coastguard Worker "scale factor = %d:%d\n",
93*fb1b10abSAndroid Build Coastguard Worker filter_type, phase_scaler, src_width, src_height, dst_width,
94*fb1b10abSAndroid Build Coastguard Worker dst_height, sf_down, sf_up);
95*fb1b10abSAndroid Build Coastguard Worker PrintDiff();
96*fb1b10abSAndroid Build Coastguard Worker }
97*fb1b10abSAndroid Build Coastguard Worker CompareImages(dst_img_);
98*fb1b10abSAndroid Build Coastguard Worker DeallocScaleImages();
99*fb1b10abSAndroid Build Coastguard Worker }
100*fb1b10abSAndroid Build Coastguard Worker }
101*fb1b10abSAndroid Build Coastguard Worker }
102*fb1b10abSAndroid Build Coastguard Worker }
103*fb1b10abSAndroid Build Coastguard Worker }
104*fb1b10abSAndroid Build Coastguard Worker }
105*fb1b10abSAndroid Build Coastguard Worker
PrintDiffComponent(const uint8_t * const ref,const uint8_t * const opt,const int stride,const int width,const int height,const int plane_idx) const106*fb1b10abSAndroid Build Coastguard Worker void PrintDiffComponent(const uint8_t *const ref, const uint8_t *const opt,
107*fb1b10abSAndroid Build Coastguard Worker const int stride, const int width, const int height,
108*fb1b10abSAndroid Build Coastguard Worker const int plane_idx) const {
109*fb1b10abSAndroid Build Coastguard Worker for (int y = 0; y < height; y++) {
110*fb1b10abSAndroid Build Coastguard Worker for (int x = 0; x < width; x++) {
111*fb1b10abSAndroid Build Coastguard Worker if (ref[y * stride + x] != opt[y * stride + x]) {
112*fb1b10abSAndroid Build Coastguard Worker printf("Plane %d pixel[%d][%d] diff:%6d (ref),%6d (opt)\n", plane_idx,
113*fb1b10abSAndroid Build Coastguard Worker y, x, ref[y * stride + x], opt[y * stride + x]);
114*fb1b10abSAndroid Build Coastguard Worker break;
115*fb1b10abSAndroid Build Coastguard Worker }
116*fb1b10abSAndroid Build Coastguard Worker }
117*fb1b10abSAndroid Build Coastguard Worker }
118*fb1b10abSAndroid Build Coastguard Worker }
119*fb1b10abSAndroid Build Coastguard Worker
PrintDiff() const120*fb1b10abSAndroid Build Coastguard Worker void PrintDiff() const {
121*fb1b10abSAndroid Build Coastguard Worker assert(ref_img_.y_stride == dst_img_.y_stride);
122*fb1b10abSAndroid Build Coastguard Worker assert(ref_img_.y_width == dst_img_.y_width);
123*fb1b10abSAndroid Build Coastguard Worker assert(ref_img_.y_height == dst_img_.y_height);
124*fb1b10abSAndroid Build Coastguard Worker assert(ref_img_.uv_stride == dst_img_.uv_stride);
125*fb1b10abSAndroid Build Coastguard Worker assert(ref_img_.uv_width == dst_img_.uv_width);
126*fb1b10abSAndroid Build Coastguard Worker assert(ref_img_.uv_height == dst_img_.uv_height);
127*fb1b10abSAndroid Build Coastguard Worker
128*fb1b10abSAndroid Build Coastguard Worker if (memcmp(dst_img_.buffer_alloc, ref_img_.buffer_alloc,
129*fb1b10abSAndroid Build Coastguard Worker ref_img_.frame_size)) {
130*fb1b10abSAndroid Build Coastguard Worker PrintDiffComponent(ref_img_.y_buffer, dst_img_.y_buffer,
131*fb1b10abSAndroid Build Coastguard Worker ref_img_.y_stride, ref_img_.y_width, ref_img_.y_height,
132*fb1b10abSAndroid Build Coastguard Worker 0);
133*fb1b10abSAndroid Build Coastguard Worker PrintDiffComponent(ref_img_.u_buffer, dst_img_.u_buffer,
134*fb1b10abSAndroid Build Coastguard Worker ref_img_.uv_stride, ref_img_.uv_width,
135*fb1b10abSAndroid Build Coastguard Worker ref_img_.uv_height, 1);
136*fb1b10abSAndroid Build Coastguard Worker PrintDiffComponent(ref_img_.v_buffer, dst_img_.v_buffer,
137*fb1b10abSAndroid Build Coastguard Worker ref_img_.uv_stride, ref_img_.uv_width,
138*fb1b10abSAndroid Build Coastguard Worker ref_img_.uv_height, 2);
139*fb1b10abSAndroid Build Coastguard Worker }
140*fb1b10abSAndroid Build Coastguard Worker }
141*fb1b10abSAndroid Build Coastguard Worker
142*fb1b10abSAndroid Build Coastguard Worker ScaleFrameFunc scale_fn_;
143*fb1b10abSAndroid Build Coastguard Worker };
144*fb1b10abSAndroid Build Coastguard Worker
TEST_P(ScaleTest,ScaleFrame_EightTap)145*fb1b10abSAndroid Build Coastguard Worker TEST_P(ScaleTest, ScaleFrame_EightTap) { RunTest(EIGHTTAP); }
TEST_P(ScaleTest,ScaleFrame_EightTapSmooth)146*fb1b10abSAndroid Build Coastguard Worker TEST_P(ScaleTest, ScaleFrame_EightTapSmooth) { RunTest(EIGHTTAP_SMOOTH); }
TEST_P(ScaleTest,ScaleFrame_EightTapSharp)147*fb1b10abSAndroid Build Coastguard Worker TEST_P(ScaleTest, ScaleFrame_EightTapSharp) { RunTest(EIGHTTAP_SHARP); }
TEST_P(ScaleTest,ScaleFrame_Bilinear)148*fb1b10abSAndroid Build Coastguard Worker TEST_P(ScaleTest, ScaleFrame_Bilinear) { RunTest(BILINEAR); }
149*fb1b10abSAndroid Build Coastguard Worker
TEST_P(ScaleTest,DISABLED_Speed)150*fb1b10abSAndroid Build Coastguard Worker TEST_P(ScaleTest, DISABLED_Speed) {
151*fb1b10abSAndroid Build Coastguard Worker static const int kCountSpeedTestBlock = 100;
152*fb1b10abSAndroid Build Coastguard Worker static const int kNumScaleFactorsToTest = 4;
153*fb1b10abSAndroid Build Coastguard Worker static const int kScaleFactors[] = { 1, 2, 3, 4 };
154*fb1b10abSAndroid Build Coastguard Worker const int src_width = 1280;
155*fb1b10abSAndroid Build Coastguard Worker const int src_height = 720;
156*fb1b10abSAndroid Build Coastguard Worker for (INTERP_FILTER filter_type = 2; filter_type < 4; ++filter_type) {
157*fb1b10abSAndroid Build Coastguard Worker for (int phase_scaler = 0; phase_scaler < 2; ++phase_scaler) {
158*fb1b10abSAndroid Build Coastguard Worker for (int sf_up_idx = 0; sf_up_idx < kNumScaleFactorsToTest; ++sf_up_idx) {
159*fb1b10abSAndroid Build Coastguard Worker const int sf_up = kScaleFactors[sf_up_idx];
160*fb1b10abSAndroid Build Coastguard Worker for (int sf_down_idx = 0; sf_down_idx < kNumScaleFactorsToTest;
161*fb1b10abSAndroid Build Coastguard Worker ++sf_down_idx) {
162*fb1b10abSAndroid Build Coastguard Worker const int sf_down = kScaleFactors[sf_down_idx];
163*fb1b10abSAndroid Build Coastguard Worker const int dst_width = src_width * sf_up / sf_down;
164*fb1b10abSAndroid Build Coastguard Worker const int dst_height = src_height * sf_up / sf_down;
165*fb1b10abSAndroid Build Coastguard Worker if (sf_up == sf_down && sf_up != 1) {
166*fb1b10abSAndroid Build Coastguard Worker continue;
167*fb1b10abSAndroid Build Coastguard Worker }
168*fb1b10abSAndroid Build Coastguard Worker // I420 frame width and height must be even.
169*fb1b10abSAndroid Build Coastguard Worker if (dst_width & 1 || dst_height & 1) {
170*fb1b10abSAndroid Build Coastguard Worker continue;
171*fb1b10abSAndroid Build Coastguard Worker }
172*fb1b10abSAndroid Build Coastguard Worker ASSERT_NO_FATAL_FAILURE(
173*fb1b10abSAndroid Build Coastguard Worker ResetScaleImages(src_width, src_height, dst_width, dst_height));
174*fb1b10abSAndroid Build Coastguard Worker ASM_REGISTER_STATE_CHECK(
175*fb1b10abSAndroid Build Coastguard Worker ReferenceScaleFrame(filter_type, phase_scaler));
176*fb1b10abSAndroid Build Coastguard Worker
177*fb1b10abSAndroid Build Coastguard Worker vpx_usec_timer timer;
178*fb1b10abSAndroid Build Coastguard Worker vpx_usec_timer_start(&timer);
179*fb1b10abSAndroid Build Coastguard Worker for (int i = 0; i < kCountSpeedTestBlock; ++i) {
180*fb1b10abSAndroid Build Coastguard Worker ScaleFrame(filter_type, phase_scaler);
181*fb1b10abSAndroid Build Coastguard Worker }
182*fb1b10abSAndroid Build Coastguard Worker libvpx_test::ClearSystemState();
183*fb1b10abSAndroid Build Coastguard Worker vpx_usec_timer_mark(&timer);
184*fb1b10abSAndroid Build Coastguard Worker const int elapsed_time =
185*fb1b10abSAndroid Build Coastguard Worker static_cast<int>(vpx_usec_timer_elapsed(&timer) / 1000);
186*fb1b10abSAndroid Build Coastguard Worker CompareImages(dst_img_);
187*fb1b10abSAndroid Build Coastguard Worker DeallocScaleImages();
188*fb1b10abSAndroid Build Coastguard Worker
189*fb1b10abSAndroid Build Coastguard Worker printf(
190*fb1b10abSAndroid Build Coastguard Worker "filter_type = %d, phase_scaler = %d, src_width = %4d, "
191*fb1b10abSAndroid Build Coastguard Worker "src_height = %4d, dst_width = %4d, dst_height = %4d, "
192*fb1b10abSAndroid Build Coastguard Worker "scale factor = %d:%d, scale time: %5d ms\n",
193*fb1b10abSAndroid Build Coastguard Worker filter_type, phase_scaler, src_width, src_height, dst_width,
194*fb1b10abSAndroid Build Coastguard Worker dst_height, sf_down, sf_up, elapsed_time);
195*fb1b10abSAndroid Build Coastguard Worker }
196*fb1b10abSAndroid Build Coastguard Worker }
197*fb1b10abSAndroid Build Coastguard Worker }
198*fb1b10abSAndroid Build Coastguard Worker }
199*fb1b10abSAndroid Build Coastguard Worker }
200*fb1b10abSAndroid Build Coastguard Worker
201*fb1b10abSAndroid Build Coastguard Worker INSTANTIATE_TEST_SUITE_P(C, ScaleTest,
202*fb1b10abSAndroid Build Coastguard Worker ::testing::Values(vp9_scale_and_extend_frame_c));
203*fb1b10abSAndroid Build Coastguard Worker
204*fb1b10abSAndroid Build Coastguard Worker #if HAVE_SSSE3
205*fb1b10abSAndroid Build Coastguard Worker INSTANTIATE_TEST_SUITE_P(SSSE3, ScaleTest,
206*fb1b10abSAndroid Build Coastguard Worker ::testing::Values(vp9_scale_and_extend_frame_ssse3));
207*fb1b10abSAndroid Build Coastguard Worker #endif // HAVE_SSSE3
208*fb1b10abSAndroid Build Coastguard Worker
209*fb1b10abSAndroid Build Coastguard Worker #if HAVE_NEON
210*fb1b10abSAndroid Build Coastguard Worker INSTANTIATE_TEST_SUITE_P(NEON, ScaleTest,
211*fb1b10abSAndroid Build Coastguard Worker ::testing::Values(vp9_scale_and_extend_frame_neon));
212*fb1b10abSAndroid Build Coastguard Worker #endif // HAVE_NEON
213*fb1b10abSAndroid Build Coastguard Worker
214*fb1b10abSAndroid Build Coastguard Worker } // namespace libvpx_test
215