xref: /aosp_15_r20/external/libvpx/test/vp8_ratectrl_rtc_test.cc (revision fb1b10ab9aebc7c7068eedab379b749d7e3900be)
1*fb1b10abSAndroid Build Coastguard Worker /*
2*fb1b10abSAndroid Build Coastguard Worker  *  Copyright (c) 2021 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 <fstream>  // NOLINT
12*fb1b10abSAndroid Build Coastguard Worker #include <string>
13*fb1b10abSAndroid Build Coastguard Worker 
14*fb1b10abSAndroid Build Coastguard Worker #include "./vpx_config.h"
15*fb1b10abSAndroid Build Coastguard Worker #include "gtest/gtest.h"
16*fb1b10abSAndroid Build Coastguard Worker #include "test/codec_factory.h"
17*fb1b10abSAndroid Build Coastguard Worker #include "test/encode_test_driver.h"
18*fb1b10abSAndroid Build Coastguard Worker #include "test/i420_video_source.h"
19*fb1b10abSAndroid Build Coastguard Worker #include "test/util.h"
20*fb1b10abSAndroid Build Coastguard Worker #include "test/video_source.h"
21*fb1b10abSAndroid Build Coastguard Worker #include "vp8/vp8_ratectrl_rtc.h"
22*fb1b10abSAndroid Build Coastguard Worker #include "vpx/vpx_codec.h"
23*fb1b10abSAndroid Build Coastguard Worker #include "vpx_ports/bitops.h"
24*fb1b10abSAndroid Build Coastguard Worker 
25*fb1b10abSAndroid Build Coastguard Worker namespace {
26*fb1b10abSAndroid Build Coastguard Worker 
27*fb1b10abSAndroid Build Coastguard Worker struct Vp8RCTestVideo {
28*fb1b10abSAndroid Build Coastguard Worker   Vp8RCTestVideo() = default;
Vp8RCTestVideo__anon3be2e9f30111::Vp8RCTestVideo29*fb1b10abSAndroid Build Coastguard Worker   Vp8RCTestVideo(const char *name_, int width_, int height_,
30*fb1b10abSAndroid Build Coastguard Worker                  unsigned int frames_)
31*fb1b10abSAndroid Build Coastguard Worker       : name(name_), width(width_), height(height_), frames(frames_) {}
32*fb1b10abSAndroid Build Coastguard Worker 
operator <<(std::ostream & os,const Vp8RCTestVideo & video)33*fb1b10abSAndroid Build Coastguard Worker   friend std::ostream &operator<<(std::ostream &os,
34*fb1b10abSAndroid Build Coastguard Worker                                   const Vp8RCTestVideo &video) {
35*fb1b10abSAndroid Build Coastguard Worker     os << video.name << " " << video.width << " " << video.height << " "
36*fb1b10abSAndroid Build Coastguard Worker        << video.frames;
37*fb1b10abSAndroid Build Coastguard Worker     return os;
38*fb1b10abSAndroid Build Coastguard Worker   }
39*fb1b10abSAndroid Build Coastguard Worker   const char *name;
40*fb1b10abSAndroid Build Coastguard Worker   int width;
41*fb1b10abSAndroid Build Coastguard Worker   int height;
42*fb1b10abSAndroid Build Coastguard Worker   unsigned int frames;
43*fb1b10abSAndroid Build Coastguard Worker };
44*fb1b10abSAndroid Build Coastguard Worker 
45*fb1b10abSAndroid Build Coastguard Worker const Vp8RCTestVideo kVp8RCTestVectors[] = {
46*fb1b10abSAndroid Build Coastguard Worker   Vp8RCTestVideo("niklas_640_480_30.yuv", 640, 480, 470),
47*fb1b10abSAndroid Build Coastguard Worker   Vp8RCTestVideo("desktop_office1.1280_720-020.yuv", 1280, 720, 300),
48*fb1b10abSAndroid Build Coastguard Worker   Vp8RCTestVideo("hantro_collage_w352h288.yuv", 352, 288, 100),
49*fb1b10abSAndroid Build Coastguard Worker };
50*fb1b10abSAndroid Build Coastguard Worker 
51*fb1b10abSAndroid Build Coastguard Worker class Vp8RcInterfaceTest
52*fb1b10abSAndroid Build Coastguard Worker     : public ::libvpx_test::EncoderTest,
53*fb1b10abSAndroid Build Coastguard Worker       public ::libvpx_test::CodecTestWith2Params<int, Vp8RCTestVideo> {
54*fb1b10abSAndroid Build Coastguard Worker  public:
Vp8RcInterfaceTest()55*fb1b10abSAndroid Build Coastguard Worker   Vp8RcInterfaceTest()
56*fb1b10abSAndroid Build Coastguard Worker       : EncoderTest(GET_PARAM(0)), key_interval_(3000), encoder_exit_(false),
57*fb1b10abSAndroid Build Coastguard Worker         frame_drop_thresh_(0) {}
58*fb1b10abSAndroid Build Coastguard Worker   ~Vp8RcInterfaceTest() override = default;
59*fb1b10abSAndroid Build Coastguard Worker 
60*fb1b10abSAndroid Build Coastguard Worker  protected:
SetUp()61*fb1b10abSAndroid Build Coastguard Worker   void SetUp() override {
62*fb1b10abSAndroid Build Coastguard Worker     InitializeConfig();
63*fb1b10abSAndroid Build Coastguard Worker     SetMode(::libvpx_test::kRealTime);
64*fb1b10abSAndroid Build Coastguard Worker   }
65*fb1b10abSAndroid Build Coastguard Worker 
66*fb1b10abSAndroid Build Coastguard Worker   // From error_resilience_test.cc
SetFrameFlags(int frame_num,int num_temp_layers)67*fb1b10abSAndroid Build Coastguard Worker   int SetFrameFlags(int frame_num, int num_temp_layers) {
68*fb1b10abSAndroid Build Coastguard Worker     int frame_flags = 0;
69*fb1b10abSAndroid Build Coastguard Worker     if (num_temp_layers == 2) {
70*fb1b10abSAndroid Build Coastguard Worker       if (frame_num % 2 == 0) {
71*fb1b10abSAndroid Build Coastguard Worker         // Layer 0: predict from L and ARF, update L.
72*fb1b10abSAndroid Build Coastguard Worker         frame_flags =
73*fb1b10abSAndroid Build Coastguard Worker             VP8_EFLAG_NO_REF_GF | VP8_EFLAG_NO_UPD_GF | VP8_EFLAG_NO_UPD_ARF;
74*fb1b10abSAndroid Build Coastguard Worker       } else {
75*fb1b10abSAndroid Build Coastguard Worker         // Layer 1: predict from L, G and ARF, and update G.
76*fb1b10abSAndroid Build Coastguard Worker         frame_flags = VP8_EFLAG_NO_UPD_ARF | VP8_EFLAG_NO_UPD_LAST |
77*fb1b10abSAndroid Build Coastguard Worker                       VP8_EFLAG_NO_UPD_ENTROPY;
78*fb1b10abSAndroid Build Coastguard Worker       }
79*fb1b10abSAndroid Build Coastguard Worker     } else if (num_temp_layers == 3) {
80*fb1b10abSAndroid Build Coastguard Worker       if (frame_num % 4 == 0) {
81*fb1b10abSAndroid Build Coastguard Worker         // Layer 0: predict from L, update L.
82*fb1b10abSAndroid Build Coastguard Worker         frame_flags = VP8_EFLAG_NO_UPD_GF | VP8_EFLAG_NO_UPD_ARF |
83*fb1b10abSAndroid Build Coastguard Worker                       VP8_EFLAG_NO_REF_GF | VP8_EFLAG_NO_REF_ARF;
84*fb1b10abSAndroid Build Coastguard Worker       } else if ((frame_num - 2) % 4 == 0) {
85*fb1b10abSAndroid Build Coastguard Worker         // Layer 1: predict from L, G,  update G.
86*fb1b10abSAndroid Build Coastguard Worker         frame_flags =
87*fb1b10abSAndroid Build Coastguard Worker             VP8_EFLAG_NO_UPD_ARF | VP8_EFLAG_NO_UPD_LAST | VP8_EFLAG_NO_REF_ARF;
88*fb1b10abSAndroid Build Coastguard Worker       } else if ((frame_num - 1) % 2 == 0) {
89*fb1b10abSAndroid Build Coastguard Worker         // Layer 2: predict from L, G, ARF; update ARG.
90*fb1b10abSAndroid Build Coastguard Worker         frame_flags = VP8_EFLAG_NO_UPD_GF | VP8_EFLAG_NO_UPD_LAST;
91*fb1b10abSAndroid Build Coastguard Worker       }
92*fb1b10abSAndroid Build Coastguard Worker     }
93*fb1b10abSAndroid Build Coastguard Worker     return frame_flags;
94*fb1b10abSAndroid Build Coastguard Worker   }
95*fb1b10abSAndroid Build Coastguard Worker 
SetLayerId(int frame_num,int num_temp_layers)96*fb1b10abSAndroid Build Coastguard Worker   int SetLayerId(int frame_num, int num_temp_layers) {
97*fb1b10abSAndroid Build Coastguard Worker     int layer_id = 0;
98*fb1b10abSAndroid Build Coastguard Worker     if (num_temp_layers == 2) {
99*fb1b10abSAndroid Build Coastguard Worker       if (frame_num % 2 == 0) {
100*fb1b10abSAndroid Build Coastguard Worker         layer_id = 0;
101*fb1b10abSAndroid Build Coastguard Worker       } else {
102*fb1b10abSAndroid Build Coastguard Worker         layer_id = 1;
103*fb1b10abSAndroid Build Coastguard Worker       }
104*fb1b10abSAndroid Build Coastguard Worker     } else if (num_temp_layers == 3) {
105*fb1b10abSAndroid Build Coastguard Worker       if (frame_num % 4 == 0) {
106*fb1b10abSAndroid Build Coastguard Worker         layer_id = 0;
107*fb1b10abSAndroid Build Coastguard Worker       } else if ((frame_num - 2) % 4 == 0) {
108*fb1b10abSAndroid Build Coastguard Worker         layer_id = 1;
109*fb1b10abSAndroid Build Coastguard Worker       } else if ((frame_num - 1) % 2 == 0) {
110*fb1b10abSAndroid Build Coastguard Worker         layer_id = 2;
111*fb1b10abSAndroid Build Coastguard Worker       }
112*fb1b10abSAndroid Build Coastguard Worker     }
113*fb1b10abSAndroid Build Coastguard Worker     return layer_id;
114*fb1b10abSAndroid Build Coastguard Worker   }
115*fb1b10abSAndroid Build Coastguard Worker 
PreEncodeFrameHook(::libvpx_test::VideoSource * video,::libvpx_test::Encoder * encoder)116*fb1b10abSAndroid Build Coastguard Worker   void PreEncodeFrameHook(::libvpx_test::VideoSource *video,
117*fb1b10abSAndroid Build Coastguard Worker                           ::libvpx_test::Encoder *encoder) override {
118*fb1b10abSAndroid Build Coastguard Worker     if (rc_cfg_.ts_number_layers > 1) {
119*fb1b10abSAndroid Build Coastguard Worker       const int layer_id = SetLayerId(video->frame(), cfg_.ts_number_layers);
120*fb1b10abSAndroid Build Coastguard Worker       const int frame_flags =
121*fb1b10abSAndroid Build Coastguard Worker           SetFrameFlags(video->frame(), cfg_.ts_number_layers);
122*fb1b10abSAndroid Build Coastguard Worker       frame_params_.temporal_layer_id = layer_id;
123*fb1b10abSAndroid Build Coastguard Worker       if (video->frame() > 0) {
124*fb1b10abSAndroid Build Coastguard Worker         encoder->Control(VP8E_SET_TEMPORAL_LAYER_ID, layer_id);
125*fb1b10abSAndroid Build Coastguard Worker         encoder->Control(VP8E_SET_FRAME_FLAGS, frame_flags);
126*fb1b10abSAndroid Build Coastguard Worker       }
127*fb1b10abSAndroid Build Coastguard Worker     } else {
128*fb1b10abSAndroid Build Coastguard Worker       if (video->frame() == 0) {
129*fb1b10abSAndroid Build Coastguard Worker         encoder->Control(VP8E_SET_CPUUSED, -6);
130*fb1b10abSAndroid Build Coastguard Worker         encoder->Control(VP8E_SET_RTC_EXTERNAL_RATECTRL, 1);
131*fb1b10abSAndroid Build Coastguard Worker         encoder->Control(VP8E_SET_MAX_INTRA_BITRATE_PCT, 1000);
132*fb1b10abSAndroid Build Coastguard Worker         if (rc_cfg_.is_screen) {
133*fb1b10abSAndroid Build Coastguard Worker           encoder->Control(VP8E_SET_SCREEN_CONTENT_MODE, 1);
134*fb1b10abSAndroid Build Coastguard Worker         }
135*fb1b10abSAndroid Build Coastguard Worker       } else if (frame_params_.frame_type == libvpx::RcFrameType::kInterFrame) {
136*fb1b10abSAndroid Build Coastguard Worker         // Disable golden frame update.
137*fb1b10abSAndroid Build Coastguard Worker         frame_flags_ |= VP8_EFLAG_NO_UPD_GF;
138*fb1b10abSAndroid Build Coastguard Worker         frame_flags_ |= VP8_EFLAG_NO_UPD_ARF;
139*fb1b10abSAndroid Build Coastguard Worker       }
140*fb1b10abSAndroid Build Coastguard Worker     }
141*fb1b10abSAndroid Build Coastguard Worker     frame_params_.frame_type = video->frame() % key_interval_ == 0
142*fb1b10abSAndroid Build Coastguard Worker                                    ? libvpx::RcFrameType::kKeyFrame
143*fb1b10abSAndroid Build Coastguard Worker                                    : libvpx::RcFrameType::kInterFrame;
144*fb1b10abSAndroid Build Coastguard Worker     encoder_exit_ = video->frame() == test_video_.frames;
145*fb1b10abSAndroid Build Coastguard Worker   }
146*fb1b10abSAndroid Build Coastguard Worker 
PostEncodeFrameHook(::libvpx_test::Encoder * encoder)147*fb1b10abSAndroid Build Coastguard Worker   void PostEncodeFrameHook(::libvpx_test::Encoder *encoder) override {
148*fb1b10abSAndroid Build Coastguard Worker     if (encoder_exit_) {
149*fb1b10abSAndroid Build Coastguard Worker       return;
150*fb1b10abSAndroid Build Coastguard Worker     }
151*fb1b10abSAndroid Build Coastguard Worker     int qp;
152*fb1b10abSAndroid Build Coastguard Worker     libvpx::UVDeltaQP uv_delta_qp;
153*fb1b10abSAndroid Build Coastguard Worker     encoder->Control(VP8E_GET_LAST_QUANTIZER, &qp);
154*fb1b10abSAndroid Build Coastguard Worker     if (rc_api_->ComputeQP(frame_params_) == libvpx::FrameDropDecision::kOk) {
155*fb1b10abSAndroid Build Coastguard Worker       ASSERT_EQ(rc_api_->GetQP(), qp);
156*fb1b10abSAndroid Build Coastguard Worker       uv_delta_qp = rc_api_->GetUVDeltaQP();
157*fb1b10abSAndroid Build Coastguard Worker       // delta_qp for UV channel is only set for screen.
158*fb1b10abSAndroid Build Coastguard Worker       if (!rc_cfg_.is_screen) {
159*fb1b10abSAndroid Build Coastguard Worker         ASSERT_EQ(uv_delta_qp.uvdc_delta_q, 0);
160*fb1b10abSAndroid Build Coastguard Worker         ASSERT_EQ(uv_delta_qp.uvac_delta_q, 0);
161*fb1b10abSAndroid Build Coastguard Worker       }
162*fb1b10abSAndroid Build Coastguard Worker     } else {
163*fb1b10abSAndroid Build Coastguard Worker       num_drops_++;
164*fb1b10abSAndroid Build Coastguard Worker     }
165*fb1b10abSAndroid Build Coastguard Worker   }
166*fb1b10abSAndroid Build Coastguard Worker 
FramePktHook(const vpx_codec_cx_pkt_t * pkt)167*fb1b10abSAndroid Build Coastguard Worker   void FramePktHook(const vpx_codec_cx_pkt_t *pkt) override {
168*fb1b10abSAndroid Build Coastguard Worker     rc_api_->PostEncodeUpdate(pkt->data.frame.sz);
169*fb1b10abSAndroid Build Coastguard Worker   }
170*fb1b10abSAndroid Build Coastguard Worker 
RunOneLayer()171*fb1b10abSAndroid Build Coastguard Worker   void RunOneLayer() {
172*fb1b10abSAndroid Build Coastguard Worker     test_video_ = GET_PARAM(2);
173*fb1b10abSAndroid Build Coastguard Worker     target_bitrate_ = GET_PARAM(1);
174*fb1b10abSAndroid Build Coastguard Worker     SetConfig();
175*fb1b10abSAndroid Build Coastguard Worker     rc_api_ = libvpx::VP8RateControlRTC::Create(rc_cfg_);
176*fb1b10abSAndroid Build Coastguard Worker     ASSERT_TRUE(rc_api_->UpdateRateControl(rc_cfg_));
177*fb1b10abSAndroid Build Coastguard Worker 
178*fb1b10abSAndroid Build Coastguard Worker     ::libvpx_test::I420VideoSource video(test_video_.name, test_video_.width,
179*fb1b10abSAndroid Build Coastguard Worker                                          test_video_.height, 30, 1, 0,
180*fb1b10abSAndroid Build Coastguard Worker                                          test_video_.frames);
181*fb1b10abSAndroid Build Coastguard Worker 
182*fb1b10abSAndroid Build Coastguard Worker     ASSERT_NO_FATAL_FAILURE(RunLoop(&video));
183*fb1b10abSAndroid Build Coastguard Worker   }
184*fb1b10abSAndroid Build Coastguard Worker 
RunOneLayerScreen()185*fb1b10abSAndroid Build Coastguard Worker   void RunOneLayerScreen() {
186*fb1b10abSAndroid Build Coastguard Worker     test_video_ = GET_PARAM(2);
187*fb1b10abSAndroid Build Coastguard Worker     target_bitrate_ = GET_PARAM(1);
188*fb1b10abSAndroid Build Coastguard Worker     SetConfig();
189*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.is_screen = true;
190*fb1b10abSAndroid Build Coastguard Worker     rc_api_ = libvpx::VP8RateControlRTC::Create(rc_cfg_);
191*fb1b10abSAndroid Build Coastguard Worker     ASSERT_TRUE(rc_api_->UpdateRateControl(rc_cfg_));
192*fb1b10abSAndroid Build Coastguard Worker 
193*fb1b10abSAndroid Build Coastguard Worker     ::libvpx_test::I420VideoSource video(test_video_.name, test_video_.width,
194*fb1b10abSAndroid Build Coastguard Worker                                          test_video_.height, 30, 1, 0,
195*fb1b10abSAndroid Build Coastguard Worker                                          test_video_.frames);
196*fb1b10abSAndroid Build Coastguard Worker 
197*fb1b10abSAndroid Build Coastguard Worker     ASSERT_NO_FATAL_FAILURE(RunLoop(&video));
198*fb1b10abSAndroid Build Coastguard Worker   }
199*fb1b10abSAndroid Build Coastguard Worker 
RunOneLayerDropFrames()200*fb1b10abSAndroid Build Coastguard Worker   void RunOneLayerDropFrames() {
201*fb1b10abSAndroid Build Coastguard Worker     test_video_ = GET_PARAM(2);
202*fb1b10abSAndroid Build Coastguard Worker     target_bitrate_ = GET_PARAM(1);
203*fb1b10abSAndroid Build Coastguard Worker     frame_drop_thresh_ = 30;
204*fb1b10abSAndroid Build Coastguard Worker     num_drops_ = 0;
205*fb1b10abSAndroid Build Coastguard Worker     // Use lower target_bitrate and max_quantizer to trigger drops.
206*fb1b10abSAndroid Build Coastguard Worker     target_bitrate_ = target_bitrate_ >> 2;
207*fb1b10abSAndroid Build Coastguard Worker     SetConfig();
208*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.max_quantizer = 56;
209*fb1b10abSAndroid Build Coastguard Worker     cfg_.rc_max_quantizer = 56;
210*fb1b10abSAndroid Build Coastguard Worker     rc_api_ = libvpx::VP8RateControlRTC::Create(rc_cfg_);
211*fb1b10abSAndroid Build Coastguard Worker     ASSERT_TRUE(rc_api_->UpdateRateControl(rc_cfg_));
212*fb1b10abSAndroid Build Coastguard Worker 
213*fb1b10abSAndroid Build Coastguard Worker     ::libvpx_test::I420VideoSource video(test_video_.name, test_video_.width,
214*fb1b10abSAndroid Build Coastguard Worker                                          test_video_.height, 30, 1, 0,
215*fb1b10abSAndroid Build Coastguard Worker                                          test_video_.frames);
216*fb1b10abSAndroid Build Coastguard Worker 
217*fb1b10abSAndroid Build Coastguard Worker     ASSERT_NO_FATAL_FAILURE(RunLoop(&video));
218*fb1b10abSAndroid Build Coastguard Worker     // Check that some frames were dropped, otherwise test has no value.
219*fb1b10abSAndroid Build Coastguard Worker     ASSERT_GE(num_drops_, 1);
220*fb1b10abSAndroid Build Coastguard Worker   }
221*fb1b10abSAndroid Build Coastguard Worker 
RunPeriodicKey()222*fb1b10abSAndroid Build Coastguard Worker   void RunPeriodicKey() {
223*fb1b10abSAndroid Build Coastguard Worker     test_video_ = GET_PARAM(2);
224*fb1b10abSAndroid Build Coastguard Worker     target_bitrate_ = GET_PARAM(1);
225*fb1b10abSAndroid Build Coastguard Worker     key_interval_ = 100;
226*fb1b10abSAndroid Build Coastguard Worker     frame_drop_thresh_ = 30;
227*fb1b10abSAndroid Build Coastguard Worker     SetConfig();
228*fb1b10abSAndroid Build Coastguard Worker     rc_api_ = libvpx::VP8RateControlRTC::Create(rc_cfg_);
229*fb1b10abSAndroid Build Coastguard Worker     ASSERT_TRUE(rc_api_->UpdateRateControl(rc_cfg_));
230*fb1b10abSAndroid Build Coastguard Worker 
231*fb1b10abSAndroid Build Coastguard Worker     ::libvpx_test::I420VideoSource video(test_video_.name, test_video_.width,
232*fb1b10abSAndroid Build Coastguard Worker                                          test_video_.height, 30, 1, 0,
233*fb1b10abSAndroid Build Coastguard Worker                                          test_video_.frames);
234*fb1b10abSAndroid Build Coastguard Worker 
235*fb1b10abSAndroid Build Coastguard Worker     ASSERT_NO_FATAL_FAILURE(RunLoop(&video));
236*fb1b10abSAndroid Build Coastguard Worker   }
237*fb1b10abSAndroid Build Coastguard Worker 
RunTemporalLayers2TL()238*fb1b10abSAndroid Build Coastguard Worker   void RunTemporalLayers2TL() {
239*fb1b10abSAndroid Build Coastguard Worker     test_video_ = GET_PARAM(2);
240*fb1b10abSAndroid Build Coastguard Worker     target_bitrate_ = GET_PARAM(1);
241*fb1b10abSAndroid Build Coastguard Worker     SetConfigTemporalLayers(2);
242*fb1b10abSAndroid Build Coastguard Worker     rc_api_ = libvpx::VP8RateControlRTC::Create(rc_cfg_);
243*fb1b10abSAndroid Build Coastguard Worker     ASSERT_TRUE(rc_api_->UpdateRateControl(rc_cfg_));
244*fb1b10abSAndroid Build Coastguard Worker 
245*fb1b10abSAndroid Build Coastguard Worker     ::libvpx_test::I420VideoSource video(test_video_.name, test_video_.width,
246*fb1b10abSAndroid Build Coastguard Worker                                          test_video_.height, 30, 1, 0,
247*fb1b10abSAndroid Build Coastguard Worker                                          test_video_.frames);
248*fb1b10abSAndroid Build Coastguard Worker 
249*fb1b10abSAndroid Build Coastguard Worker     ASSERT_NO_FATAL_FAILURE(RunLoop(&video));
250*fb1b10abSAndroid Build Coastguard Worker   }
251*fb1b10abSAndroid Build Coastguard Worker 
RunTemporalLayers3TL()252*fb1b10abSAndroid Build Coastguard Worker   void RunTemporalLayers3TL() {
253*fb1b10abSAndroid Build Coastguard Worker     test_video_ = GET_PARAM(2);
254*fb1b10abSAndroid Build Coastguard Worker     target_bitrate_ = GET_PARAM(1);
255*fb1b10abSAndroid Build Coastguard Worker     SetConfigTemporalLayers(3);
256*fb1b10abSAndroid Build Coastguard Worker     rc_api_ = libvpx::VP8RateControlRTC::Create(rc_cfg_);
257*fb1b10abSAndroid Build Coastguard Worker     ASSERT_TRUE(rc_api_->UpdateRateControl(rc_cfg_));
258*fb1b10abSAndroid Build Coastguard Worker 
259*fb1b10abSAndroid Build Coastguard Worker     ::libvpx_test::I420VideoSource video(test_video_.name, test_video_.width,
260*fb1b10abSAndroid Build Coastguard Worker                                          test_video_.height, 30, 1, 0,
261*fb1b10abSAndroid Build Coastguard Worker                                          test_video_.frames);
262*fb1b10abSAndroid Build Coastguard Worker 
263*fb1b10abSAndroid Build Coastguard Worker     ASSERT_NO_FATAL_FAILURE(RunLoop(&video));
264*fb1b10abSAndroid Build Coastguard Worker   }
265*fb1b10abSAndroid Build Coastguard Worker 
RunTemporalLayers3TLDropFrames()266*fb1b10abSAndroid Build Coastguard Worker   void RunTemporalLayers3TLDropFrames() {
267*fb1b10abSAndroid Build Coastguard Worker     test_video_ = GET_PARAM(2);
268*fb1b10abSAndroid Build Coastguard Worker     target_bitrate_ = GET_PARAM(1);
269*fb1b10abSAndroid Build Coastguard Worker     frame_drop_thresh_ = 30;
270*fb1b10abSAndroid Build Coastguard Worker     num_drops_ = 0;
271*fb1b10abSAndroid Build Coastguard Worker     // Use lower target_bitrate and max_quantizer to trigger drops.
272*fb1b10abSAndroid Build Coastguard Worker     target_bitrate_ = target_bitrate_ >> 2;
273*fb1b10abSAndroid Build Coastguard Worker     SetConfigTemporalLayers(3);
274*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.max_quantizer = 56;
275*fb1b10abSAndroid Build Coastguard Worker     cfg_.rc_max_quantizer = 56;
276*fb1b10abSAndroid Build Coastguard Worker     rc_api_ = libvpx::VP8RateControlRTC::Create(rc_cfg_);
277*fb1b10abSAndroid Build Coastguard Worker     ASSERT_TRUE(rc_api_->UpdateRateControl(rc_cfg_));
278*fb1b10abSAndroid Build Coastguard Worker 
279*fb1b10abSAndroid Build Coastguard Worker     ::libvpx_test::I420VideoSource video(test_video_.name, test_video_.width,
280*fb1b10abSAndroid Build Coastguard Worker                                          test_video_.height, 30, 1, 0,
281*fb1b10abSAndroid Build Coastguard Worker                                          test_video_.frames);
282*fb1b10abSAndroid Build Coastguard Worker 
283*fb1b10abSAndroid Build Coastguard Worker     ASSERT_NO_FATAL_FAILURE(RunLoop(&video));
284*fb1b10abSAndroid Build Coastguard Worker     // Check that some frames were dropped, otherwise test has no value.
285*fb1b10abSAndroid Build Coastguard Worker     ASSERT_GE(num_drops_, 1);
286*fb1b10abSAndroid Build Coastguard Worker   }
287*fb1b10abSAndroid Build Coastguard Worker 
288*fb1b10abSAndroid Build Coastguard Worker  private:
SetConfig()289*fb1b10abSAndroid Build Coastguard Worker   void SetConfig() {
290*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.width = test_video_.width;
291*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.height = test_video_.height;
292*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.max_quantizer = 60;
293*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.min_quantizer = 2;
294*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.target_bandwidth = target_bitrate_;
295*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.buf_initial_sz = 600;
296*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.buf_optimal_sz = 600;
297*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.buf_sz = target_bitrate_;
298*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.undershoot_pct = 50;
299*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.overshoot_pct = 50;
300*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.max_intra_bitrate_pct = 1000;
301*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.framerate = 30.0;
302*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.layer_target_bitrate[0] = target_bitrate_;
303*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.frame_drop_thresh = frame_drop_thresh_;
304*fb1b10abSAndroid Build Coastguard Worker 
305*fb1b10abSAndroid Build Coastguard Worker     // Encoder settings for ground truth.
306*fb1b10abSAndroid Build Coastguard Worker     cfg_.g_w = test_video_.width;
307*fb1b10abSAndroid Build Coastguard Worker     cfg_.g_h = test_video_.height;
308*fb1b10abSAndroid Build Coastguard Worker     cfg_.rc_undershoot_pct = 50;
309*fb1b10abSAndroid Build Coastguard Worker     cfg_.rc_overshoot_pct = 50;
310*fb1b10abSAndroid Build Coastguard Worker     cfg_.rc_buf_initial_sz = 600;
311*fb1b10abSAndroid Build Coastguard Worker     cfg_.rc_buf_optimal_sz = 600;
312*fb1b10abSAndroid Build Coastguard Worker     cfg_.rc_buf_sz = target_bitrate_;
313*fb1b10abSAndroid Build Coastguard Worker     cfg_.rc_dropframe_thresh = 0;
314*fb1b10abSAndroid Build Coastguard Worker     cfg_.rc_min_quantizer = 2;
315*fb1b10abSAndroid Build Coastguard Worker     cfg_.rc_max_quantizer = 60;
316*fb1b10abSAndroid Build Coastguard Worker     cfg_.rc_end_usage = VPX_CBR;
317*fb1b10abSAndroid Build Coastguard Worker     cfg_.g_lag_in_frames = 0;
318*fb1b10abSAndroid Build Coastguard Worker     cfg_.g_error_resilient = 1;
319*fb1b10abSAndroid Build Coastguard Worker     cfg_.rc_target_bitrate = target_bitrate_;
320*fb1b10abSAndroid Build Coastguard Worker     cfg_.kf_min_dist = key_interval_;
321*fb1b10abSAndroid Build Coastguard Worker     cfg_.kf_max_dist = key_interval_;
322*fb1b10abSAndroid Build Coastguard Worker     cfg_.rc_dropframe_thresh = frame_drop_thresh_;
323*fb1b10abSAndroid Build Coastguard Worker   }
324*fb1b10abSAndroid Build Coastguard Worker 
SetConfigTemporalLayers(int temporal_layers)325*fb1b10abSAndroid Build Coastguard Worker   void SetConfigTemporalLayers(int temporal_layers) {
326*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.width = test_video_.width;
327*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.height = test_video_.height;
328*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.max_quantizer = 60;
329*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.min_quantizer = 2;
330*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.target_bandwidth = target_bitrate_;
331*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.buf_initial_sz = 600;
332*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.buf_optimal_sz = 600;
333*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.buf_sz = target_bitrate_;
334*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.undershoot_pct = 50;
335*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.overshoot_pct = 50;
336*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.max_intra_bitrate_pct = 1000;
337*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.framerate = 30.0;
338*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.frame_drop_thresh = frame_drop_thresh_;
339*fb1b10abSAndroid Build Coastguard Worker     if (temporal_layers == 2) {
340*fb1b10abSAndroid Build Coastguard Worker       rc_cfg_.layer_target_bitrate[0] = 60 * target_bitrate_ / 100;
341*fb1b10abSAndroid Build Coastguard Worker       rc_cfg_.layer_target_bitrate[1] = target_bitrate_;
342*fb1b10abSAndroid Build Coastguard Worker       rc_cfg_.ts_rate_decimator[0] = 2;
343*fb1b10abSAndroid Build Coastguard Worker       rc_cfg_.ts_rate_decimator[1] = 1;
344*fb1b10abSAndroid Build Coastguard Worker     } else if (temporal_layers == 3) {
345*fb1b10abSAndroid Build Coastguard Worker       rc_cfg_.layer_target_bitrate[0] = 40 * target_bitrate_ / 100;
346*fb1b10abSAndroid Build Coastguard Worker       rc_cfg_.layer_target_bitrate[1] = 60 * target_bitrate_ / 100;
347*fb1b10abSAndroid Build Coastguard Worker       rc_cfg_.layer_target_bitrate[2] = target_bitrate_;
348*fb1b10abSAndroid Build Coastguard Worker       rc_cfg_.ts_rate_decimator[0] = 4;
349*fb1b10abSAndroid Build Coastguard Worker       rc_cfg_.ts_rate_decimator[1] = 2;
350*fb1b10abSAndroid Build Coastguard Worker       rc_cfg_.ts_rate_decimator[2] = 1;
351*fb1b10abSAndroid Build Coastguard Worker     }
352*fb1b10abSAndroid Build Coastguard Worker 
353*fb1b10abSAndroid Build Coastguard Worker     rc_cfg_.ts_number_layers = temporal_layers;
354*fb1b10abSAndroid Build Coastguard Worker 
355*fb1b10abSAndroid Build Coastguard Worker     // Encoder settings for ground truth.
356*fb1b10abSAndroid Build Coastguard Worker     cfg_.g_w = test_video_.width;
357*fb1b10abSAndroid Build Coastguard Worker     cfg_.g_h = test_video_.height;
358*fb1b10abSAndroid Build Coastguard Worker     cfg_.rc_undershoot_pct = 50;
359*fb1b10abSAndroid Build Coastguard Worker     cfg_.rc_overshoot_pct = 50;
360*fb1b10abSAndroid Build Coastguard Worker     cfg_.rc_buf_initial_sz = 600;
361*fb1b10abSAndroid Build Coastguard Worker     cfg_.rc_buf_optimal_sz = 600;
362*fb1b10abSAndroid Build Coastguard Worker     cfg_.rc_buf_sz = target_bitrate_;
363*fb1b10abSAndroid Build Coastguard Worker     cfg_.rc_dropframe_thresh = 0;
364*fb1b10abSAndroid Build Coastguard Worker     cfg_.rc_min_quantizer = 2;
365*fb1b10abSAndroid Build Coastguard Worker     cfg_.rc_max_quantizer = 60;
366*fb1b10abSAndroid Build Coastguard Worker     cfg_.rc_end_usage = VPX_CBR;
367*fb1b10abSAndroid Build Coastguard Worker     cfg_.g_lag_in_frames = 0;
368*fb1b10abSAndroid Build Coastguard Worker     cfg_.g_error_resilient = 1;
369*fb1b10abSAndroid Build Coastguard Worker     cfg_.rc_target_bitrate = target_bitrate_;
370*fb1b10abSAndroid Build Coastguard Worker     cfg_.kf_min_dist = key_interval_;
371*fb1b10abSAndroid Build Coastguard Worker     cfg_.kf_max_dist = key_interval_;
372*fb1b10abSAndroid Build Coastguard Worker     cfg_.rc_dropframe_thresh = frame_drop_thresh_;
373*fb1b10abSAndroid Build Coastguard Worker     // 2 Temporal layers, no spatial layers, CBR mode.
374*fb1b10abSAndroid Build Coastguard Worker     cfg_.ss_number_layers = 1;
375*fb1b10abSAndroid Build Coastguard Worker     cfg_.ts_number_layers = temporal_layers;
376*fb1b10abSAndroid Build Coastguard Worker     if (temporal_layers == 2) {
377*fb1b10abSAndroid Build Coastguard Worker       cfg_.ts_rate_decimator[0] = 2;
378*fb1b10abSAndroid Build Coastguard Worker       cfg_.ts_rate_decimator[1] = 1;
379*fb1b10abSAndroid Build Coastguard Worker       cfg_.ts_periodicity = 2;
380*fb1b10abSAndroid Build Coastguard Worker       cfg_.ts_target_bitrate[0] = 60 * cfg_.rc_target_bitrate / 100;
381*fb1b10abSAndroid Build Coastguard Worker       cfg_.ts_target_bitrate[1] = cfg_.rc_target_bitrate;
382*fb1b10abSAndroid Build Coastguard Worker     } else if (temporal_layers == 3) {
383*fb1b10abSAndroid Build Coastguard Worker       cfg_.ts_rate_decimator[0] = 4;
384*fb1b10abSAndroid Build Coastguard Worker       cfg_.ts_rate_decimator[1] = 2;
385*fb1b10abSAndroid Build Coastguard Worker       cfg_.ts_rate_decimator[2] = 1;
386*fb1b10abSAndroid Build Coastguard Worker       cfg_.ts_periodicity = 4;
387*fb1b10abSAndroid Build Coastguard Worker       cfg_.ts_target_bitrate[0] = 40 * cfg_.rc_target_bitrate / 100;
388*fb1b10abSAndroid Build Coastguard Worker       cfg_.ts_target_bitrate[1] = 60 * cfg_.rc_target_bitrate / 100;
389*fb1b10abSAndroid Build Coastguard Worker       cfg_.ts_target_bitrate[2] = cfg_.rc_target_bitrate;
390*fb1b10abSAndroid Build Coastguard Worker     }
391*fb1b10abSAndroid Build Coastguard Worker   }
392*fb1b10abSAndroid Build Coastguard Worker 
393*fb1b10abSAndroid Build Coastguard Worker   std::unique_ptr<libvpx::VP8RateControlRTC> rc_api_;
394*fb1b10abSAndroid Build Coastguard Worker   libvpx::VP8RateControlRtcConfig rc_cfg_;
395*fb1b10abSAndroid Build Coastguard Worker   int key_interval_;
396*fb1b10abSAndroid Build Coastguard Worker   int target_bitrate_;
397*fb1b10abSAndroid Build Coastguard Worker   Vp8RCTestVideo test_video_;
398*fb1b10abSAndroid Build Coastguard Worker   libvpx::VP8FrameParamsQpRTC frame_params_;
399*fb1b10abSAndroid Build Coastguard Worker   bool encoder_exit_;
400*fb1b10abSAndroid Build Coastguard Worker   int frame_drop_thresh_;
401*fb1b10abSAndroid Build Coastguard Worker   int num_drops_;
402*fb1b10abSAndroid Build Coastguard Worker };
403*fb1b10abSAndroid Build Coastguard Worker 
TEST_P(Vp8RcInterfaceTest,OneLayer)404*fb1b10abSAndroid Build Coastguard Worker TEST_P(Vp8RcInterfaceTest, OneLayer) { RunOneLayer(); }
405*fb1b10abSAndroid Build Coastguard Worker 
TEST_P(Vp8RcInterfaceTest,OneLayerScreen)406*fb1b10abSAndroid Build Coastguard Worker TEST_P(Vp8RcInterfaceTest, OneLayerScreen) { RunOneLayerScreen(); }
407*fb1b10abSAndroid Build Coastguard Worker 
TEST_P(Vp8RcInterfaceTest,OneLayerDropFrames)408*fb1b10abSAndroid Build Coastguard Worker TEST_P(Vp8RcInterfaceTest, OneLayerDropFrames) { RunOneLayerDropFrames(); }
409*fb1b10abSAndroid Build Coastguard Worker 
TEST_P(Vp8RcInterfaceTest,OneLayerPeriodicKey)410*fb1b10abSAndroid Build Coastguard Worker TEST_P(Vp8RcInterfaceTest, OneLayerPeriodicKey) { RunPeriodicKey(); }
411*fb1b10abSAndroid Build Coastguard Worker 
TEST_P(Vp8RcInterfaceTest,TemporalLayers2TL)412*fb1b10abSAndroid Build Coastguard Worker TEST_P(Vp8RcInterfaceTest, TemporalLayers2TL) { RunTemporalLayers2TL(); }
413*fb1b10abSAndroid Build Coastguard Worker 
TEST_P(Vp8RcInterfaceTest,TemporalLayers3TL)414*fb1b10abSAndroid Build Coastguard Worker TEST_P(Vp8RcInterfaceTest, TemporalLayers3TL) { RunTemporalLayers3TL(); }
415*fb1b10abSAndroid Build Coastguard Worker 
TEST_P(Vp8RcInterfaceTest,TemporalLayers3TLDropFrames)416*fb1b10abSAndroid Build Coastguard Worker TEST_P(Vp8RcInterfaceTest, TemporalLayers3TLDropFrames) {
417*fb1b10abSAndroid Build Coastguard Worker   RunTemporalLayers3TLDropFrames();
418*fb1b10abSAndroid Build Coastguard Worker }
419*fb1b10abSAndroid Build Coastguard Worker 
420*fb1b10abSAndroid Build Coastguard Worker VP8_INSTANTIATE_TEST_SUITE(Vp8RcInterfaceTest,
421*fb1b10abSAndroid Build Coastguard Worker                            ::testing::Values(200, 400, 1000),
422*fb1b10abSAndroid Build Coastguard Worker                            ::testing::ValuesIn(kVp8RCTestVectors));
423*fb1b10abSAndroid Build Coastguard Worker 
424*fb1b10abSAndroid Build Coastguard Worker }  // namespace
425