sprang@webrtc.org | 131bea8 | 2015-02-18 12:46:06 +0000 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (c) 2015 The WebRTC project authors. All Rights Reserved. |
| 3 | * |
| 4 | * Use of this source code is governed by a BSD-style license |
| 5 | * that can be found in the LICENSE file in the root of the source |
| 6 | * tree. An additional intellectual property rights grant can be found |
| 7 | * in the file PATENTS. All contributing project authors may |
| 8 | * be found in the AUTHORS file in the root of the source tree. |
| 9 | */ |
| 10 | |
| 11 | #include <stdio.h> |
| 12 | |
| 13 | #include <map> |
| 14 | |
| 15 | #include "gflags/gflags.h" |
| 16 | #include "testing/gtest/include/gtest/gtest.h" |
| 17 | |
sprang | d635895 | 2015-07-29 07:58:13 -0700 | [diff] [blame] | 18 | #include "webrtc/base/checks.h" |
sprang@webrtc.org | 131bea8 | 2015-02-18 12:46:06 +0000 | [diff] [blame] | 19 | #include "webrtc/test/field_trial.h" |
| 20 | #include "webrtc/test/frame_generator.h" |
| 21 | #include "webrtc/test/frame_generator_capturer.h" |
| 22 | #include "webrtc/test/run_test.h" |
| 23 | #include "webrtc/test/testsupport/fileutils.h" |
| 24 | #include "webrtc/typedefs.h" |
| 25 | #include "webrtc/video/loopback.h" |
| 26 | #include "webrtc/video/video_send_stream.h" |
| 27 | |
| 28 | namespace webrtc { |
| 29 | namespace flags { |
| 30 | |
sprang | d635895 | 2015-07-29 07:58:13 -0700 | [diff] [blame] | 31 | DEFINE_int32(width, 1850, "Video width (crops source)."); |
sprang@webrtc.org | 131bea8 | 2015-02-18 12:46:06 +0000 | [diff] [blame] | 32 | size_t Width() { |
sprang | d635895 | 2015-07-29 07:58:13 -0700 | [diff] [blame] | 33 | return static_cast<size_t>(FLAGS_width); |
sprang@webrtc.org | 131bea8 | 2015-02-18 12:46:06 +0000 | [diff] [blame] | 34 | } |
sprang | d635895 | 2015-07-29 07:58:13 -0700 | [diff] [blame] | 35 | |
| 36 | DEFINE_int32(height, 1110, "Video height (crops source)."); |
sprang@webrtc.org | 131bea8 | 2015-02-18 12:46:06 +0000 | [diff] [blame] | 37 | size_t Height() { |
sprang | d635895 | 2015-07-29 07:58:13 -0700 | [diff] [blame] | 38 | return static_cast<size_t>(FLAGS_height); |
sprang@webrtc.org | 131bea8 | 2015-02-18 12:46:06 +0000 | [diff] [blame] | 39 | } |
| 40 | |
| 41 | DEFINE_int32(fps, 5, "Frames per second."); |
| 42 | int Fps() { |
| 43 | return static_cast<int>(FLAGS_fps); |
| 44 | } |
| 45 | |
sprang | d635895 | 2015-07-29 07:58:13 -0700 | [diff] [blame] | 46 | DEFINE_int32(slide_change_interval, |
| 47 | 10, |
| 48 | "Interval (in seconds) between simulated slide changes."); |
| 49 | int SlideChangeInterval() { |
| 50 | return static_cast<int>(FLAGS_slide_change_interval); |
| 51 | } |
| 52 | |
| 53 | DEFINE_int32( |
| 54 | scroll_duration, |
| 55 | 0, |
| 56 | "Duration (in seconds) during which a slide will be scrolled into place."); |
| 57 | int ScrollDuration() { |
| 58 | return static_cast<int>(FLAGS_scroll_duration); |
| 59 | } |
| 60 | |
sprang@webrtc.org | 131bea8 | 2015-02-18 12:46:06 +0000 | [diff] [blame] | 61 | DEFINE_int32(min_bitrate, 50, "Minimum video bitrate."); |
| 62 | size_t MinBitrate() { |
| 63 | return static_cast<size_t>(FLAGS_min_bitrate); |
| 64 | } |
| 65 | |
Erik Språng | 2c4c914 | 2015-06-24 11:24:44 +0200 | [diff] [blame] | 66 | DEFINE_int32(tl0_bitrate, 200, "Temporal layer 0 target bitrate."); |
sprang@webrtc.org | 131bea8 | 2015-02-18 12:46:06 +0000 | [diff] [blame] | 67 | size_t StartBitrate() { |
| 68 | return static_cast<size_t>(FLAGS_tl0_bitrate); |
| 69 | } |
| 70 | |
Erik Språng | 2c4c914 | 2015-06-24 11:24:44 +0200 | [diff] [blame] | 71 | DEFINE_int32(tl1_bitrate, 2000, "Temporal layer 1 target bitrate."); |
sprang@webrtc.org | 131bea8 | 2015-02-18 12:46:06 +0000 | [diff] [blame] | 72 | size_t MaxBitrate() { |
| 73 | return static_cast<size_t>(FLAGS_tl1_bitrate); |
| 74 | } |
| 75 | |
sprang | ef7228c | 2015-08-05 02:01:29 -0700 | [diff] [blame^] | 76 | DEFINE_int32(num_temporal_layers, 2, "Number of temporal layers to use."); |
| 77 | int NumTemporalLayers() { |
| 78 | return static_cast<int>(FLAGS_num_temporal_layers); |
| 79 | } |
| 80 | |
sprang@webrtc.org | 131bea8 | 2015-02-18 12:46:06 +0000 | [diff] [blame] | 81 | DEFINE_int32(min_transmit_bitrate, 400, "Min transmit bitrate incl. padding."); |
| 82 | int MinTransmitBitrate() { |
| 83 | return FLAGS_min_transmit_bitrate; |
| 84 | } |
| 85 | |
| 86 | DEFINE_string(codec, "VP8", "Video codec to use."); |
| 87 | std::string Codec() { |
| 88 | return static_cast<std::string>(FLAGS_codec); |
| 89 | } |
| 90 | |
| 91 | DEFINE_int32(loss_percent, 0, "Percentage of packets randomly lost."); |
| 92 | int LossPercent() { |
| 93 | return static_cast<int>(FLAGS_loss_percent); |
| 94 | } |
| 95 | |
| 96 | DEFINE_int32(link_capacity, |
| 97 | 0, |
| 98 | "Capacity (kbps) of the fake link. 0 means infinite."); |
| 99 | int LinkCapacity() { |
| 100 | return static_cast<int>(FLAGS_link_capacity); |
| 101 | } |
| 102 | |
| 103 | DEFINE_int32(queue_size, 0, "Size of the bottleneck link queue in packets."); |
| 104 | int QueueSize() { |
| 105 | return static_cast<int>(FLAGS_queue_size); |
| 106 | } |
| 107 | |
| 108 | DEFINE_int32(avg_propagation_delay_ms, |
| 109 | 0, |
| 110 | "Average link propagation delay in ms."); |
| 111 | int AvgPropagationDelayMs() { |
| 112 | return static_cast<int>(FLAGS_avg_propagation_delay_ms); |
| 113 | } |
| 114 | |
| 115 | DEFINE_int32(std_propagation_delay_ms, |
| 116 | 0, |
| 117 | "Link propagation delay standard deviation in ms."); |
| 118 | int StdPropagationDelayMs() { |
| 119 | return static_cast<int>(FLAGS_std_propagation_delay_ms); |
| 120 | } |
| 121 | |
| 122 | DEFINE_bool(logs, false, "print logs to stderr"); |
| 123 | |
| 124 | DEFINE_string( |
| 125 | force_fieldtrials, |
| 126 | "", |
| 127 | "Field trials control experimental feature code which can be forced. " |
| 128 | "E.g. running with --force_fieldtrials=WebRTC-FooFeature/Enable/" |
| 129 | " will assign the group Enable to field trial WebRTC-FooFeature. Multiple " |
| 130 | "trials are separated by \"/\""); |
| 131 | } // namespace flags |
| 132 | |
| 133 | class ScreenshareLoopback : public test::Loopback { |
| 134 | public: |
Erik Språng | 9d657cf | 2015-04-28 14:00:58 +0200 | [diff] [blame] | 135 | explicit ScreenshareLoopback(const Config& config) : Loopback(config) { |
sprang | ef7228c | 2015-08-05 02:01:29 -0700 | [diff] [blame^] | 136 | CHECK_GE(config.num_temporal_layers, 1u); |
| 137 | CHECK_LE(config.num_temporal_layers, 2u); |
| 138 | |
Erik Språng | 9d657cf | 2015-04-28 14:00:58 +0200 | [diff] [blame] | 139 | vp8_settings_ = VideoEncoder::GetDefaultVp8Settings(); |
| 140 | vp8_settings_.denoisingOn = false; |
| 141 | vp8_settings_.frameDroppingOn = false; |
sprang | ef7228c | 2015-08-05 02:01:29 -0700 | [diff] [blame^] | 142 | vp8_settings_.numberOfTemporalLayers = |
| 143 | static_cast<unsigned char>(config.num_temporal_layers); |
ivica | 028cf48 | 2015-07-30 02:15:56 -0700 | [diff] [blame] | 144 | |
| 145 | vp9_settings_ = VideoEncoder::GetDefaultVp9Settings(); |
| 146 | vp9_settings_.denoisingOn = false; |
| 147 | vp9_settings_.frameDroppingOn = false; |
sprang | ef7228c | 2015-08-05 02:01:29 -0700 | [diff] [blame^] | 148 | vp9_settings_.numberOfTemporalLayers = |
| 149 | static_cast<unsigned char>(config.num_temporal_layers); |
Erik Språng | 9d657cf | 2015-04-28 14:00:58 +0200 | [diff] [blame] | 150 | } |
sprang@webrtc.org | 131bea8 | 2015-02-18 12:46:06 +0000 | [diff] [blame] | 151 | virtual ~ScreenshareLoopback() {} |
| 152 | |
| 153 | protected: |
| 154 | VideoEncoderConfig CreateEncoderConfig() override { |
| 155 | VideoEncoderConfig encoder_config(test::Loopback::CreateEncoderConfig()); |
| 156 | VideoStream* stream = &encoder_config.streams[0]; |
Erik Språng | 143cec1 | 2015-04-28 10:01:41 +0200 | [diff] [blame] | 157 | encoder_config.content_type = VideoEncoderConfig::ContentType::kScreen; |
sprang@webrtc.org | 131bea8 | 2015-02-18 12:46:06 +0000 | [diff] [blame] | 158 | encoder_config.min_transmit_bitrate_bps = flags::MinTransmitBitrate(); |
sprang | ef7228c | 2015-08-05 02:01:29 -0700 | [diff] [blame^] | 159 | int num_temporal_layers; |
ivica | 028cf48 | 2015-07-30 02:15:56 -0700 | [diff] [blame] | 160 | if (config_.codec == "VP8") { |
| 161 | encoder_config.encoder_specific_settings = &vp8_settings_; |
sprang | ef7228c | 2015-08-05 02:01:29 -0700 | [diff] [blame^] | 162 | num_temporal_layers = vp8_settings_.numberOfTemporalLayers; |
ivica | 028cf48 | 2015-07-30 02:15:56 -0700 | [diff] [blame] | 163 | } else if (config_.codec == "VP9") { |
| 164 | encoder_config.encoder_specific_settings = &vp9_settings_; |
sprang | ef7228c | 2015-08-05 02:01:29 -0700 | [diff] [blame^] | 165 | num_temporal_layers = vp9_settings_.numberOfTemporalLayers; |
ivica | 028cf48 | 2015-07-30 02:15:56 -0700 | [diff] [blame] | 166 | } else { |
| 167 | RTC_NOTREACHED() << "Codec not supported!"; |
| 168 | abort(); |
| 169 | } |
sprang@webrtc.org | 131bea8 | 2015-02-18 12:46:06 +0000 | [diff] [blame] | 170 | stream->temporal_layer_thresholds_bps.clear(); |
sprang@webrtc.org | 131bea8 | 2015-02-18 12:46:06 +0000 | [diff] [blame] | 171 | stream->target_bitrate_bps = |
| 172 | static_cast<int>(config_.start_bitrate_kbps) * 1000; |
sprang | ef7228c | 2015-08-05 02:01:29 -0700 | [diff] [blame^] | 173 | if (num_temporal_layers == 2) { |
| 174 | stream->temporal_layer_thresholds_bps.push_back( |
| 175 | stream->target_bitrate_bps); |
| 176 | } |
sprang@webrtc.org | 131bea8 | 2015-02-18 12:46:06 +0000 | [diff] [blame] | 177 | return encoder_config; |
| 178 | } |
| 179 | |
| 180 | test::VideoCapturer* CreateCapturer(VideoSendStream* send_stream) override { |
| 181 | std::vector<std::string> slides; |
| 182 | slides.push_back(test::ResourcePath("web_screenshot_1850_1110", "yuv")); |
| 183 | slides.push_back(test::ResourcePath("presentation_1850_1110", "yuv")); |
| 184 | slides.push_back(test::ResourcePath("photo_1850_1110", "yuv")); |
| 185 | slides.push_back(test::ResourcePath("difficult_photo_1850_1110", "yuv")); |
| 186 | |
sprang | d635895 | 2015-07-29 07:58:13 -0700 | [diff] [blame] | 187 | // Fixed for input resolution for prerecorded screenshare content. |
| 188 | const size_t kWidth = 1850; |
| 189 | const size_t kHeight = 1110; |
| 190 | CHECK_LE(flags::Width(), kWidth); |
| 191 | CHECK_LE(flags::Height(), kHeight); |
| 192 | CHECK_GT(flags::SlideChangeInterval(), 0); |
| 193 | const int kPauseDurationMs = |
| 194 | (flags::SlideChangeInterval() - flags::ScrollDuration()) * 1000; |
| 195 | CHECK_LE(flags::ScrollDuration(), flags::SlideChangeInterval()); |
| 196 | |
sprang@webrtc.org | 131bea8 | 2015-02-18 12:46:06 +0000 | [diff] [blame] | 197 | test::FrameGenerator* frame_generator = |
sprang | d635895 | 2015-07-29 07:58:13 -0700 | [diff] [blame] | 198 | test::FrameGenerator::CreateScrollingInputFromYuvFiles( |
| 199 | Clock::GetRealTimeClock(), slides, kWidth, kHeight, flags::Width(), |
| 200 | flags::Height(), flags::ScrollDuration() * 1000, kPauseDurationMs); |
| 201 | |
sprang@webrtc.org | 131bea8 | 2015-02-18 12:46:06 +0000 | [diff] [blame] | 202 | test::FrameGeneratorCapturer* capturer(new test::FrameGeneratorCapturer( |
| 203 | clock_, send_stream->Input(), frame_generator, flags::Fps())); |
| 204 | EXPECT_TRUE(capturer->Init()); |
| 205 | return capturer; |
| 206 | } |
Erik Språng | 9d657cf | 2015-04-28 14:00:58 +0200 | [diff] [blame] | 207 | |
| 208 | VideoCodecVP8 vp8_settings_; |
ivica | 028cf48 | 2015-07-30 02:15:56 -0700 | [diff] [blame] | 209 | VideoCodecVP9 vp9_settings_; |
sprang@webrtc.org | 131bea8 | 2015-02-18 12:46:06 +0000 | [diff] [blame] | 210 | }; |
| 211 | |
| 212 | void Loopback() { |
| 213 | test::Loopback::Config config{flags::Width(), |
| 214 | flags::Height(), |
| 215 | flags::Fps(), |
| 216 | flags::MinBitrate(), |
| 217 | flags::StartBitrate(), |
| 218 | flags::MaxBitrate(), |
| 219 | flags::MinTransmitBitrate(), |
| 220 | flags::Codec(), |
sprang | ef7228c | 2015-08-05 02:01:29 -0700 | [diff] [blame^] | 221 | flags::NumTemporalLayers(), |
sprang@webrtc.org | 131bea8 | 2015-02-18 12:46:06 +0000 | [diff] [blame] | 222 | flags::LossPercent(), |
| 223 | flags::LinkCapacity(), |
| 224 | flags::QueueSize(), |
| 225 | flags::AvgPropagationDelayMs(), |
| 226 | flags::StdPropagationDelayMs(), |
| 227 | flags::FLAGS_logs}; |
| 228 | ScreenshareLoopback loopback(config); |
| 229 | loopback.Run(); |
| 230 | } |
| 231 | } // namespace webrtc |
| 232 | |
| 233 | int main(int argc, char* argv[]) { |
| 234 | ::testing::InitGoogleTest(&argc, argv); |
| 235 | google::ParseCommandLineFlags(&argc, &argv, true); |
| 236 | webrtc::test::InitFieldTrialsFromString( |
| 237 | webrtc::flags::FLAGS_force_fieldtrials); |
| 238 | webrtc::test::RunTest(webrtc::Loopback); |
| 239 | return 0; |
| 240 | } |