1*fb1b10abSAndroid Build Coastguard Worker /*
2*fb1b10abSAndroid Build Coastguard Worker * Copyright (c) 2015 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 #include <algorithm>
11*fb1b10abSAndroid Build Coastguard Worker #include "gtest/gtest.h"
12*fb1b10abSAndroid Build Coastguard Worker #include "test/codec_factory.h"
13*fb1b10abSAndroid Build Coastguard Worker #include "test/encode_test_driver.h"
14*fb1b10abSAndroid Build Coastguard Worker #include "test/util.h"
15*fb1b10abSAndroid Build Coastguard Worker #include "test/y4m_video_source.h"
16*fb1b10abSAndroid Build Coastguard Worker
17*fb1b10abSAndroid Build Coastguard Worker namespace {
18*fb1b10abSAndroid Build Coastguard Worker
19*fb1b10abSAndroid Build Coastguard Worker // Check if any pixel in a 16x16 macroblock varies between frames.
CheckMb(const vpx_image_t & current,const vpx_image_t & previous,int mb_r,int mb_c)20*fb1b10abSAndroid Build Coastguard Worker int CheckMb(const vpx_image_t ¤t, const vpx_image_t &previous, int mb_r,
21*fb1b10abSAndroid Build Coastguard Worker int mb_c) {
22*fb1b10abSAndroid Build Coastguard Worker for (int plane = 0; plane < 3; plane++) {
23*fb1b10abSAndroid Build Coastguard Worker int r = 16 * mb_r;
24*fb1b10abSAndroid Build Coastguard Worker int c0 = 16 * mb_c;
25*fb1b10abSAndroid Build Coastguard Worker int r_top = std::min(r + 16, static_cast<int>(current.d_h));
26*fb1b10abSAndroid Build Coastguard Worker int c_top = std::min(c0 + 16, static_cast<int>(current.d_w));
27*fb1b10abSAndroid Build Coastguard Worker r = std::max(r, 0);
28*fb1b10abSAndroid Build Coastguard Worker c0 = std::max(c0, 0);
29*fb1b10abSAndroid Build Coastguard Worker if (plane > 0 && current.x_chroma_shift) {
30*fb1b10abSAndroid Build Coastguard Worker c_top = (c_top + 1) >> 1;
31*fb1b10abSAndroid Build Coastguard Worker c0 >>= 1;
32*fb1b10abSAndroid Build Coastguard Worker }
33*fb1b10abSAndroid Build Coastguard Worker if (plane > 0 && current.y_chroma_shift) {
34*fb1b10abSAndroid Build Coastguard Worker r_top = (r_top + 1) >> 1;
35*fb1b10abSAndroid Build Coastguard Worker r >>= 1;
36*fb1b10abSAndroid Build Coastguard Worker }
37*fb1b10abSAndroid Build Coastguard Worker for (; r < r_top; ++r) {
38*fb1b10abSAndroid Build Coastguard Worker for (int c = c0; c < c_top; ++c) {
39*fb1b10abSAndroid Build Coastguard Worker if (current.planes[plane][current.stride[plane] * r + c] !=
40*fb1b10abSAndroid Build Coastguard Worker previous.planes[plane][previous.stride[plane] * r + c]) {
41*fb1b10abSAndroid Build Coastguard Worker return 1;
42*fb1b10abSAndroid Build Coastguard Worker }
43*fb1b10abSAndroid Build Coastguard Worker }
44*fb1b10abSAndroid Build Coastguard Worker }
45*fb1b10abSAndroid Build Coastguard Worker }
46*fb1b10abSAndroid Build Coastguard Worker return 0;
47*fb1b10abSAndroid Build Coastguard Worker }
48*fb1b10abSAndroid Build Coastguard Worker
GenerateMap(int mb_rows,int mb_cols,const vpx_image_t & current,const vpx_image_t & previous,uint8_t * map)49*fb1b10abSAndroid Build Coastguard Worker void GenerateMap(int mb_rows, int mb_cols, const vpx_image_t ¤t,
50*fb1b10abSAndroid Build Coastguard Worker const vpx_image_t &previous, uint8_t *map) {
51*fb1b10abSAndroid Build Coastguard Worker for (int mb_r = 0; mb_r < mb_rows; ++mb_r) {
52*fb1b10abSAndroid Build Coastguard Worker for (int mb_c = 0; mb_c < mb_cols; ++mb_c) {
53*fb1b10abSAndroid Build Coastguard Worker map[mb_r * mb_cols + mb_c] = CheckMb(current, previous, mb_r, mb_c);
54*fb1b10abSAndroid Build Coastguard Worker }
55*fb1b10abSAndroid Build Coastguard Worker }
56*fb1b10abSAndroid Build Coastguard Worker }
57*fb1b10abSAndroid Build Coastguard Worker
58*fb1b10abSAndroid Build Coastguard Worker const int kAqModeCyclicRefresh = 3;
59*fb1b10abSAndroid Build Coastguard Worker
60*fb1b10abSAndroid Build Coastguard Worker class ActiveMapRefreshTest
61*fb1b10abSAndroid Build Coastguard Worker : public ::libvpx_test::EncoderTest,
62*fb1b10abSAndroid Build Coastguard Worker public ::libvpx_test::CodecTestWith2Params<libvpx_test::TestMode, int> {
63*fb1b10abSAndroid Build Coastguard Worker protected:
ActiveMapRefreshTest()64*fb1b10abSAndroid Build Coastguard Worker ActiveMapRefreshTest() : EncoderTest(GET_PARAM(0)) {}
65*fb1b10abSAndroid Build Coastguard Worker ~ActiveMapRefreshTest() override = default;
66*fb1b10abSAndroid Build Coastguard Worker
SetUp()67*fb1b10abSAndroid Build Coastguard Worker void SetUp() override {
68*fb1b10abSAndroid Build Coastguard Worker InitializeConfig();
69*fb1b10abSAndroid Build Coastguard Worker SetMode(GET_PARAM(1));
70*fb1b10abSAndroid Build Coastguard Worker cpu_used_ = GET_PARAM(2);
71*fb1b10abSAndroid Build Coastguard Worker }
72*fb1b10abSAndroid Build Coastguard Worker
PreEncodeFrameHook(::libvpx_test::VideoSource * video,::libvpx_test::Encoder * encoder)73*fb1b10abSAndroid Build Coastguard Worker void PreEncodeFrameHook(::libvpx_test::VideoSource *video,
74*fb1b10abSAndroid Build Coastguard Worker ::libvpx_test::Encoder *encoder) override {
75*fb1b10abSAndroid Build Coastguard Worker ::libvpx_test::Y4mVideoSource *y4m_video =
76*fb1b10abSAndroid Build Coastguard Worker static_cast<libvpx_test::Y4mVideoSource *>(video);
77*fb1b10abSAndroid Build Coastguard Worker if (video->frame() == 0) {
78*fb1b10abSAndroid Build Coastguard Worker encoder->Control(VP8E_SET_CPUUSED, cpu_used_);
79*fb1b10abSAndroid Build Coastguard Worker encoder->Control(VP9E_SET_AQ_MODE, kAqModeCyclicRefresh);
80*fb1b10abSAndroid Build Coastguard Worker } else if (video->frame() >= 2 && video->img()) {
81*fb1b10abSAndroid Build Coastguard Worker vpx_image_t *current = video->img();
82*fb1b10abSAndroid Build Coastguard Worker vpx_image_t *previous = y4m_holder_->img();
83*fb1b10abSAndroid Build Coastguard Worker ASSERT_NE(previous, nullptr);
84*fb1b10abSAndroid Build Coastguard Worker vpx_active_map_t map = vpx_active_map_t();
85*fb1b10abSAndroid Build Coastguard Worker const int width = static_cast<int>(current->d_w);
86*fb1b10abSAndroid Build Coastguard Worker const int height = static_cast<int>(current->d_h);
87*fb1b10abSAndroid Build Coastguard Worker const int mb_width = (width + 15) / 16;
88*fb1b10abSAndroid Build Coastguard Worker const int mb_height = (height + 15) / 16;
89*fb1b10abSAndroid Build Coastguard Worker uint8_t *active_map = new uint8_t[mb_width * mb_height];
90*fb1b10abSAndroid Build Coastguard Worker GenerateMap(mb_height, mb_width, *current, *previous, active_map);
91*fb1b10abSAndroid Build Coastguard Worker map.cols = mb_width;
92*fb1b10abSAndroid Build Coastguard Worker map.rows = mb_height;
93*fb1b10abSAndroid Build Coastguard Worker map.active_map = active_map;
94*fb1b10abSAndroid Build Coastguard Worker encoder->Control(VP8E_SET_ACTIVEMAP, &map);
95*fb1b10abSAndroid Build Coastguard Worker delete[] active_map;
96*fb1b10abSAndroid Build Coastguard Worker }
97*fb1b10abSAndroid Build Coastguard Worker if (video->img()) {
98*fb1b10abSAndroid Build Coastguard Worker y4m_video->SwapBuffers(y4m_holder_);
99*fb1b10abSAndroid Build Coastguard Worker }
100*fb1b10abSAndroid Build Coastguard Worker }
101*fb1b10abSAndroid Build Coastguard Worker
102*fb1b10abSAndroid Build Coastguard Worker int cpu_used_;
103*fb1b10abSAndroid Build Coastguard Worker ::libvpx_test::Y4mVideoSource *y4m_holder_;
104*fb1b10abSAndroid Build Coastguard Worker };
105*fb1b10abSAndroid Build Coastguard Worker
TEST_P(ActiveMapRefreshTest,Test)106*fb1b10abSAndroid Build Coastguard Worker TEST_P(ActiveMapRefreshTest, Test) {
107*fb1b10abSAndroid Build Coastguard Worker cfg_.g_lag_in_frames = 0;
108*fb1b10abSAndroid Build Coastguard Worker cfg_.g_profile = 1;
109*fb1b10abSAndroid Build Coastguard Worker cfg_.rc_target_bitrate = 600;
110*fb1b10abSAndroid Build Coastguard Worker cfg_.rc_resize_allowed = 0;
111*fb1b10abSAndroid Build Coastguard Worker cfg_.rc_min_quantizer = 8;
112*fb1b10abSAndroid Build Coastguard Worker cfg_.rc_max_quantizer = 30;
113*fb1b10abSAndroid Build Coastguard Worker cfg_.g_pass = VPX_RC_ONE_PASS;
114*fb1b10abSAndroid Build Coastguard Worker cfg_.rc_end_usage = VPX_CBR;
115*fb1b10abSAndroid Build Coastguard Worker cfg_.kf_max_dist = 90000;
116*fb1b10abSAndroid Build Coastguard Worker
117*fb1b10abSAndroid Build Coastguard Worker ::libvpx_test::Y4mVideoSource video("desktop_credits.y4m", 0, 30);
118*fb1b10abSAndroid Build Coastguard Worker ::libvpx_test::Y4mVideoSource video_holder("desktop_credits.y4m", 0, 30);
119*fb1b10abSAndroid Build Coastguard Worker video_holder.Begin();
120*fb1b10abSAndroid Build Coastguard Worker y4m_holder_ = &video_holder;
121*fb1b10abSAndroid Build Coastguard Worker
122*fb1b10abSAndroid Build Coastguard Worker ASSERT_NO_FATAL_FAILURE(RunLoop(&video));
123*fb1b10abSAndroid Build Coastguard Worker }
124*fb1b10abSAndroid Build Coastguard Worker
125*fb1b10abSAndroid Build Coastguard Worker VP9_INSTANTIATE_TEST_SUITE(ActiveMapRefreshTest,
126*fb1b10abSAndroid Build Coastguard Worker ::testing::Values(::libvpx_test::kRealTime),
127*fb1b10abSAndroid Build Coastguard Worker ::testing::Range(5, 6));
128*fb1b10abSAndroid Build Coastguard Worker } // namespace
129