blob: 46e97cd2a5352ad97b9f6901bb5e7c6bda92fb24 [file] [log] [blame]
pbos@webrtc.org744fbc72013-09-10 09:26:25 +00001/*
2 * Copyright (c) 2013 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 */
pbos@webrtc.org744fbc72013-09-10 09:26:25 +000010
stefanff483612015-12-21 03:14:00 -080011#include "webrtc/call/rampup_tests.h"
12
pbos@webrtc.org2b4ce3a2015-03-23 13:12:24 +000013#include "webrtc/base/checks.h"
pbos12411ef2015-11-23 14:47:56 -080014#include "webrtc/base/platform_thread.h"
perkjfa10b552016-10-02 23:45:26 -070015#include "webrtc/test/encoder_settings.h"
kwibergac9f8762016-09-30 22:29:43 -070016#include "webrtc/test/gtest.h"
stefan@webrtc.org7e9315b2013-12-04 10:24:26 +000017#include "webrtc/test/testsupport/perf_test.h"
pbos@webrtc.org744fbc72013-09-10 09:26:25 +000018
19namespace webrtc {
pbos@webrtc.org29023282013-09-11 10:14:56 +000020namespace {
stefan@webrtc.org3d7da882014-07-08 13:59:46 +000021
Stefan Holmer723dff12015-10-05 14:59:41 +020022static const int64_t kPollIntervalMs = 20;
stefandb752f92016-12-05 08:23:40 -080023static const int kExpectedHighVideoBitrateBps = 60000;
24static const int kExpectedHighAudioBitrateBps = 30000;
25static const int kLowBandwidthLimitBps = 20000;
26static const int kExpectedLowBitrateBps = 20000;
pbos@webrtc.org29023282013-09-11 10:14:56 +000027
stefanff483612015-12-21 03:14:00 -080028std::vector<uint32_t> GenerateSsrcs(size_t num_streams, uint32_t ssrc_offset) {
stefan@webrtc.org3d7da882014-07-08 13:59:46 +000029 std::vector<uint32_t> ssrcs;
30 for (size_t i = 0; i != num_streams; ++i)
31 ssrcs.push_back(static_cast<uint32_t>(ssrc_offset + i));
32 return ssrcs;
33}
mflodman@webrtc.orgeb16b812014-06-16 08:57:39 +000034} // namespace
henrik.lundin@webrtc.org845862f2014-03-06 07:19:28 +000035
stefanff483612015-12-21 03:14:00 -080036RampUpTester::RampUpTester(size_t num_video_streams,
37 size_t num_audio_streams,
stefan4fbd1452015-09-28 03:57:14 -070038 unsigned int start_bitrate_bps,
stefan5a2c5062017-01-27 06:43:18 -080039 int64_t min_run_time_ms,
stefan4fbd1452015-09-28 03:57:14 -070040 const std::string& extension_type,
41 bool rtx,
stefan5a2c5062017-01-27 06:43:18 -080042 bool red,
43 bool report_perf_stats)
stefan4fbd1452015-09-28 03:57:14 -070044 : EndToEndTest(test::CallTest::kLongTimeoutMs),
45 event_(false, false),
46 clock_(Clock::GetRealTimeClock()),
stefanff483612015-12-21 03:14:00 -080047 num_video_streams_(num_video_streams),
48 num_audio_streams_(num_audio_streams),
stefan4fbd1452015-09-28 03:57:14 -070049 rtx_(rtx),
50 red_(red),
mflodman86cc6ff2016-07-26 04:44:06 -070051 sender_call_(nullptr),
stefan4fbd1452015-09-28 03:57:14 -070052 send_stream_(nullptr),
53 start_bitrate_bps_(start_bitrate_bps),
stefan5a2c5062017-01-27 06:43:18 -080054 min_run_time_ms_(min_run_time_ms),
55 report_perf_stats_(report_perf_stats),
stefan@webrtc.org3d7da882014-07-08 13:59:46 +000056 expected_bitrate_bps_(0),
Erik Språngf3a7c9d2015-10-05 14:03:22 +020057 test_start_ms_(-1),
stefan4fbd1452015-09-28 03:57:14 -070058 ramp_up_finished_ms_(-1),
59 extension_type_(extension_type),
stefanff483612015-12-21 03:14:00 -080060 video_ssrcs_(GenerateSsrcs(num_video_streams_, 100)),
61 video_rtx_ssrcs_(GenerateSsrcs(num_video_streams_, 200)),
62 audio_ssrcs_(GenerateSsrcs(num_audio_streams_, 300)),
Peter Boström8c38e8b2015-11-26 17:45:47 +010063 poller_thread_(&BitrateStatsPollingThread,
64 this,
mflodman86cc6ff2016-07-26 04:44:06 -070065 "BitrateStatsPollingThread") {
Stefan Holmerff2a6352016-01-14 10:00:21 +010066 EXPECT_LE(num_audio_streams_, 1u);
stefan4fbd1452015-09-28 03:57:14 -070067}
68
69RampUpTester::~RampUpTester() {
70 event_.Set();
stefan4fbd1452015-09-28 03:57:14 -070071}
72
73Call::Config RampUpTester::GetSenderCallConfig() {
skvlad11a9cbf2016-10-07 11:53:05 -070074 Call::Config call_config(&event_log_);
stefan4fbd1452015-09-28 03:57:14 -070075 if (start_bitrate_bps_ != 0) {
76 call_config.bitrate_config.start_bitrate_bps = start_bitrate_bps_;
77 }
78 call_config.bitrate_config.min_bitrate_bps = 10000;
79 return call_config;
80}
81
stefanff483612015-12-21 03:14:00 -080082void RampUpTester::OnVideoStreamsCreated(
stefan4fbd1452015-09-28 03:57:14 -070083 VideoSendStream* send_stream,
84 const std::vector<VideoReceiveStream*>& receive_streams) {
85 send_stream_ = send_stream;
86}
87
stefane74eef12016-01-08 06:47:13 -080088test::PacketTransport* RampUpTester::CreateSendTransport(Call* sender_call) {
89 send_transport_ = new test::PacketTransport(sender_call, this,
90 test::PacketTransport::kSender,
91 forward_transport_config_);
92 return send_transport_;
stefanf116bd02015-10-27 08:29:42 -070093}
94
Stefan Holmerd20e6512016-01-12 15:51:22 +010095size_t RampUpTester::GetNumVideoStreams() const {
96 return num_video_streams_;
97}
98
Stefan Holmerff2a6352016-01-14 10:00:21 +010099size_t RampUpTester::GetNumAudioStreams() const {
100 return num_audio_streams_;
101}
102
perkjfa10b552016-10-02 23:45:26 -0700103class RampUpTester::VideoStreamFactory
104 : public VideoEncoderConfig::VideoStreamFactoryInterface {
105 public:
106 VideoStreamFactory() {}
107
108 private:
109 std::vector<VideoStream> CreateEncoderStreams(
110 int width,
111 int height,
112 const VideoEncoderConfig& encoder_config) override {
113 std::vector<VideoStream> streams =
114 test::CreateVideoStreams(width, height, encoder_config);
115 if (encoder_config.number_of_streams == 1) {
116 streams[0].target_bitrate_bps = streams[0].max_bitrate_bps = 2000000;
117 }
118 return streams;
119 }
120};
121
stefanff483612015-12-21 03:14:00 -0800122void RampUpTester::ModifyVideoConfigs(
stefan4fbd1452015-09-28 03:57:14 -0700123 VideoSendStream::Config* send_config,
124 std::vector<VideoReceiveStream::Config>* receive_configs,
125 VideoEncoderConfig* encoder_config) {
126 send_config->suspend_below_min_bitrate = true;
perkjfa10b552016-10-02 23:45:26 -0700127 encoder_config->number_of_streams = num_video_streams_;
128 encoder_config->max_bitrate_bps = 2000000;
129 encoder_config->video_stream_factory =
130 new rtc::RefCountedObject<RampUpTester::VideoStreamFactory>();
stefanff483612015-12-21 03:14:00 -0800131 if (num_video_streams_ == 1) {
stefan4fbd1452015-09-28 03:57:14 -0700132 // For single stream rampup until 1mbps
133 expected_bitrate_bps_ = kSingleStreamTargetBps;
134 } else {
135 // For multi stream rampup until all streams are being sent. That means
perkjfa10b552016-10-02 23:45:26 -0700136 // enough bitrate to send all the target streams plus the min bitrate of
stefan4fbd1452015-09-28 03:57:14 -0700137 // the last one.
perkjfa10b552016-10-02 23:45:26 -0700138 std::vector<VideoStream> streams = test::CreateVideoStreams(
139 test::CallTest::kDefaultWidth, test::CallTest::kDefaultHeight,
140 *encoder_config);
141 expected_bitrate_bps_ = streams.back().min_bitrate_bps;
142 for (size_t i = 0; i < streams.size() - 1; ++i) {
143 expected_bitrate_bps_ += streams[i].target_bitrate_bps;
stefan@webrtc.org7e9315b2013-12-04 10:24:26 +0000144 }
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000145 }
stefan@webrtc.org7e9315b2013-12-04 10:24:26 +0000146
stefan4fbd1452015-09-28 03:57:14 -0700147 send_config->rtp.extensions.clear();
148
149 bool remb;
stefan43edf0f2015-11-20 18:05:48 -0800150 bool transport_cc;
isheriff6f8d6862016-05-26 11:24:55 -0700151 if (extension_type_ == RtpExtension::kAbsSendTimeUri) {
stefan4fbd1452015-09-28 03:57:14 -0700152 remb = true;
stefan43edf0f2015-11-20 18:05:48 -0800153 transport_cc = false;
stefan4fbd1452015-09-28 03:57:14 -0700154 send_config->rtp.extensions.push_back(
155 RtpExtension(extension_type_.c_str(), kAbsSendTimeExtensionId));
isheriff6f8d6862016-05-26 11:24:55 -0700156 } else if (extension_type_ == RtpExtension::kTransportSequenceNumberUri) {
stefan4fbd1452015-09-28 03:57:14 -0700157 remb = false;
stefan43edf0f2015-11-20 18:05:48 -0800158 transport_cc = true;
stefan4fbd1452015-09-28 03:57:14 -0700159 send_config->rtp.extensions.push_back(RtpExtension(
160 extension_type_.c_str(), kTransportSequenceNumberExtensionId));
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000161 } else {
stefan4fbd1452015-09-28 03:57:14 -0700162 remb = true;
stefan43edf0f2015-11-20 18:05:48 -0800163 transport_cc = false;
stefan4fbd1452015-09-28 03:57:14 -0700164 send_config->rtp.extensions.push_back(RtpExtension(
165 extension_type_.c_str(), kTransmissionTimeOffsetExtensionId));
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000166 }
stefan4fbd1452015-09-28 03:57:14 -0700167
168 send_config->rtp.nack.rtp_history_ms = test::CallTest::kNackRtpHistoryMs;
stefanff483612015-12-21 03:14:00 -0800169 send_config->rtp.ssrcs = video_ssrcs_;
stefan4fbd1452015-09-28 03:57:14 -0700170 if (rtx_) {
171 send_config->rtp.rtx.payload_type = test::CallTest::kSendRtxPayloadType;
stefanff483612015-12-21 03:14:00 -0800172 send_config->rtp.rtx.ssrcs = video_rtx_ssrcs_;
stefan4fbd1452015-09-28 03:57:14 -0700173 }
174 if (red_) {
brandtrb5f2c3f2016-10-04 23:28:39 -0700175 send_config->rtp.ulpfec.ulpfec_payload_type =
stefan4fbd1452015-09-28 03:57:14 -0700176 test::CallTest::kUlpfecPayloadType;
brandtrb5f2c3f2016-10-04 23:28:39 -0700177 send_config->rtp.ulpfec.red_payload_type = test::CallTest::kRedPayloadType;
brandtrfbfb5362016-11-17 04:18:37 -0800178 if (rtx_) {
179 send_config->rtp.ulpfec.red_rtx_payload_type =
180 test::CallTest::kRtxRedPayloadType;
181 }
stefan4fbd1452015-09-28 03:57:14 -0700182 }
183
184 size_t i = 0;
185 for (VideoReceiveStream::Config& recv_config : *receive_configs) {
186 recv_config.rtp.remb = remb;
stefan43edf0f2015-11-20 18:05:48 -0800187 recv_config.rtp.transport_cc = transport_cc;
stefan4fbd1452015-09-28 03:57:14 -0700188 recv_config.rtp.extensions = send_config->rtp.extensions;
189
stefanff483612015-12-21 03:14:00 -0800190 recv_config.rtp.remote_ssrc = video_ssrcs_[i];
stefan4fbd1452015-09-28 03:57:14 -0700191 recv_config.rtp.nack.rtp_history_ms = send_config->rtp.nack.rtp_history_ms;
192
193 if (red_) {
brandtrb5f2c3f2016-10-04 23:28:39 -0700194 recv_config.rtp.ulpfec.red_payload_type =
195 send_config->rtp.ulpfec.red_payload_type;
196 recv_config.rtp.ulpfec.ulpfec_payload_type =
197 send_config->rtp.ulpfec.ulpfec_payload_type;
brandtrfbfb5362016-11-17 04:18:37 -0800198 if (rtx_) {
199 recv_config.rtp.ulpfec.red_rtx_payload_type =
200 send_config->rtp.ulpfec.red_rtx_payload_type;
201 }
stefan4fbd1452015-09-28 03:57:14 -0700202 }
203
204 if (rtx_) {
brandtr14742122017-01-27 04:53:07 -0800205 recv_config.rtp.rtx_ssrc = video_rtx_ssrcs_[i];
206 recv_config.rtp
207 .rtx_payload_types[send_config->encoder_settings.payload_type] =
208 send_config->rtp.rtx.payload_type;
stefan4fbd1452015-09-28 03:57:14 -0700209 }
210 ++i;
211 }
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000212}
213
Stefan Holmerff2a6352016-01-14 10:00:21 +0100214void RampUpTester::ModifyAudioConfigs(
215 AudioSendStream::Config* send_config,
216 std::vector<AudioReceiveStream::Config>* receive_configs) {
217 if (num_audio_streams_ == 0)
218 return;
219
isheriff6f8d6862016-05-26 11:24:55 -0700220 EXPECT_NE(RtpExtension::kTimestampOffsetUri, extension_type_)
Stefan Holmerff2a6352016-01-14 10:00:21 +0100221 << "Audio BWE not supported with toffset.";
222
223 send_config->rtp.ssrc = audio_ssrcs_[0];
224 send_config->rtp.extensions.clear();
225
minyue10cbb462016-11-07 09:29:22 -0800226 send_config->min_bitrate_bps = 6000;
227 send_config->max_bitrate_bps = 60000;
mflodman86cc6ff2016-07-26 04:44:06 -0700228
Stefan Holmerff2a6352016-01-14 10:00:21 +0100229 bool transport_cc = false;
isheriff6f8d6862016-05-26 11:24:55 -0700230 if (extension_type_ == RtpExtension::kAbsSendTimeUri) {
Stefan Holmerff2a6352016-01-14 10:00:21 +0100231 transport_cc = false;
232 send_config->rtp.extensions.push_back(
233 RtpExtension(extension_type_.c_str(), kAbsSendTimeExtensionId));
isheriff6f8d6862016-05-26 11:24:55 -0700234 } else if (extension_type_ == RtpExtension::kTransportSequenceNumberUri) {
Stefan Holmerff2a6352016-01-14 10:00:21 +0100235 transport_cc = true;
236 send_config->rtp.extensions.push_back(RtpExtension(
237 extension_type_.c_str(), kTransportSequenceNumberExtensionId));
238 }
239
240 for (AudioReceiveStream::Config& recv_config : *receive_configs) {
Stefan Holmerff2a6352016-01-14 10:00:21 +0100241 recv_config.rtp.transport_cc = transport_cc;
242 recv_config.rtp.extensions = send_config->rtp.extensions;
243 recv_config.rtp.remote_ssrc = send_config->rtp.ssrc;
244 }
245}
246
stefan4fbd1452015-09-28 03:57:14 -0700247void RampUpTester::OnCallsCreated(Call* sender_call, Call* receiver_call) {
248 sender_call_ = sender_call;
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000249}
250
stefan4fbd1452015-09-28 03:57:14 -0700251bool RampUpTester::BitrateStatsPollingThread(void* obj) {
252 return static_cast<RampUpTester*>(obj)->PollStats();
pbos@webrtc.org32452b22014-10-22 12:15:24 +0000253}
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000254
stefan4fbd1452015-09-28 03:57:14 -0700255bool RampUpTester::PollStats() {
256 if (sender_call_) {
257 Call::Stats stats = sender_call_->GetStats();
258
stefan5a2c5062017-01-27 06:43:18 -0800259 EXPECT_GE(stats.send_bandwidth_bps, start_bitrate_bps_);
260 EXPECT_GE(expected_bitrate_bps_, 0);
261 if (stats.send_bandwidth_bps >= expected_bitrate_bps_ &&
262 (min_run_time_ms_ == -1 ||
263 clock_->TimeInMilliseconds() - test_start_ms_ >= min_run_time_ms_)) {
stefan4fbd1452015-09-28 03:57:14 -0700264 ramp_up_finished_ms_ = clock_->TimeInMilliseconds();
Peter Boström5811a392015-12-10 13:02:50 +0100265 observation_complete_.Set();
stefan4fbd1452015-09-28 03:57:14 -0700266 }
267 }
268
269 return !event_.Wait(kPollIntervalMs);
Erik Språng468e62a2015-07-06 10:50:47 +0200270}
271
stefan4fbd1452015-09-28 03:57:14 -0700272void RampUpTester::ReportResult(const std::string& measurement,
273 size_t value,
274 const std::string& units) const {
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000275 webrtc::test::PrintResult(
276 measurement, "",
stefanff483612015-12-21 03:14:00 -0800277 ::testing::UnitTest::GetInstance()->current_test_info()->name(), value,
278 units, false);
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000279}
280
stefan092508a2015-09-29 02:26:42 -0700281void RampUpTester::AccumulateStats(const VideoSendStream::StreamStats& stream,
282 size_t* total_packets_sent,
283 size_t* total_sent,
284 size_t* padding_sent,
285 size_t* media_sent) const {
stefan4fbd1452015-09-28 03:57:14 -0700286 *total_packets_sent += stream.rtp_stats.transmitted.packets +
287 stream.rtp_stats.retransmitted.packets +
288 stream.rtp_stats.fec.packets;
289 *total_sent += stream.rtp_stats.transmitted.TotalBytes() +
290 stream.rtp_stats.retransmitted.TotalBytes() +
291 stream.rtp_stats.fec.TotalBytes();
292 *padding_sent += stream.rtp_stats.transmitted.padding_bytes +
293 stream.rtp_stats.retransmitted.padding_bytes +
294 stream.rtp_stats.fec.padding_bytes;
295 *media_sent += stream.rtp_stats.MediaPayloadBytes();
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000296}
297
stefan4fbd1452015-09-28 03:57:14 -0700298void RampUpTester::TriggerTestDone() {
Erik Språngf3a7c9d2015-10-05 14:03:22 +0200299 RTC_DCHECK_GE(test_start_ms_, 0);
300
Stefan Holmerff2a6352016-01-14 10:00:21 +0100301 // TODO(holmer): Add audio send stats here too when those APIs are available.
mflodman86cc6ff2016-07-26 04:44:06 -0700302 if (!send_stream_)
303 return;
304
stefan4fbd1452015-09-28 03:57:14 -0700305 VideoSendStream::Stats send_stats = send_stream_->GetStats();
306
307 size_t total_packets_sent = 0;
308 size_t total_sent = 0;
309 size_t padding_sent = 0;
310 size_t media_sent = 0;
stefanff483612015-12-21 03:14:00 -0800311 for (uint32_t ssrc : video_ssrcs_) {
stefan092508a2015-09-29 02:26:42 -0700312 AccumulateStats(send_stats.substreams[ssrc], &total_packets_sent,
313 &total_sent, &padding_sent, &media_sent);
stefan4fbd1452015-09-28 03:57:14 -0700314 }
315
316 size_t rtx_total_packets_sent = 0;
317 size_t rtx_total_sent = 0;
318 size_t rtx_padding_sent = 0;
319 size_t rtx_media_sent = 0;
stefanff483612015-12-21 03:14:00 -0800320 for (uint32_t rtx_ssrc : video_rtx_ssrcs_) {
stefan092508a2015-09-29 02:26:42 -0700321 AccumulateStats(send_stats.substreams[rtx_ssrc], &rtx_total_packets_sent,
322 &rtx_total_sent, &rtx_padding_sent, &rtx_media_sent);
stefan4fbd1452015-09-28 03:57:14 -0700323 }
324
stefan5a2c5062017-01-27 06:43:18 -0800325 if (report_perf_stats_) {
326 ReportResult("ramp-up-total-packets-sent", total_packets_sent, "packets");
327 ReportResult("ramp-up-total-sent", total_sent, "bytes");
328 ReportResult("ramp-up-media-sent", media_sent, "bytes");
329 ReportResult("ramp-up-padding-sent", padding_sent, "bytes");
330 ReportResult("ramp-up-rtx-total-packets-sent", rtx_total_packets_sent,
331 "packets");
332 ReportResult("ramp-up-rtx-total-sent", rtx_total_sent, "bytes");
333 ReportResult("ramp-up-rtx-media-sent", rtx_media_sent, "bytes");
334 ReportResult("ramp-up-rtx-padding-sent", rtx_padding_sent, "bytes");
335 if (ramp_up_finished_ms_ >= 0) {
336 ReportResult("ramp-up-time", ramp_up_finished_ms_ - test_start_ms_,
337 "milliseconds");
338 }
339 ReportResult("ramp-up-average-network-latency",
340 send_transport_->GetAverageDelayMs(), "milliseconds");
stefan4fbd1452015-09-28 03:57:14 -0700341 }
342}
343
344void RampUpTester::PerformTest() {
Erik Språngf3a7c9d2015-10-05 14:03:22 +0200345 test_start_ms_ = clock_->TimeInMilliseconds();
Peter Boström8c38e8b2015-11-26 17:45:47 +0100346 poller_thread_.Start();
Peter Boström5811a392015-12-10 13:02:50 +0100347 EXPECT_TRUE(Wait()) << "Timed out while waiting for ramp-up to complete.";
stefan4fbd1452015-09-28 03:57:14 -0700348 TriggerTestDone();
Peter Boström8c38e8b2015-11-26 17:45:47 +0100349 poller_thread_.Stop();
stefan4fbd1452015-09-28 03:57:14 -0700350}
351
Stefan Holmerff2a6352016-01-14 10:00:21 +0100352RampUpDownUpTester::RampUpDownUpTester(size_t num_video_streams,
353 size_t num_audio_streams,
stefan4fbd1452015-09-28 03:57:14 -0700354 unsigned int start_bitrate_bps,
355 const std::string& extension_type,
356 bool rtx,
357 bool red)
Stefan Holmerff2a6352016-01-14 10:00:21 +0100358 : RampUpTester(num_video_streams,
359 num_audio_streams,
360 start_bitrate_bps,
stefan5a2c5062017-01-27 06:43:18 -0800361 0,
Stefan Holmerff2a6352016-01-14 10:00:21 +0100362 extension_type,
363 rtx,
stefan5a2c5062017-01-27 06:43:18 -0800364 red,
365 true),
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000366 test_state_(kFirstRampup),
367 state_start_ms_(clock_->TimeInMilliseconds()),
stefan4fbd1452015-09-28 03:57:14 -0700368 interval_start_ms_(clock_->TimeInMilliseconds()),
369 sent_bytes_(0) {
stefan38d8b3c2017-01-09 04:19:24 -0800370 forward_transport_config_.link_capacity_kbps = GetHighLinkCapacity();
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000371}
372
stefan4fbd1452015-09-28 03:57:14 -0700373RampUpDownUpTester::~RampUpDownUpTester() {}
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000374
stefan4fbd1452015-09-28 03:57:14 -0700375bool RampUpDownUpTester::PollStats() {
376 if (send_stream_) {
377 webrtc::VideoSendStream::Stats stats = send_stream_->GetStats();
378 int transmit_bitrate_bps = 0;
379 for (auto it : stats.substreams) {
380 transmit_bitrate_bps += it.second.total_bitrate_bps;
mflodman@webrtc.orgeb16b812014-06-16 08:57:39 +0000381 }
stefan4fbd1452015-09-28 03:57:14 -0700382 EvolveTestState(transmit_bitrate_bps, stats.suspended);
mflodman86cc6ff2016-07-26 04:44:06 -0700383 } else if (num_audio_streams_ > 0 && sender_call_ != nullptr) {
384 // An audio send stream doesn't have bitrate stats, so the call send BW is
385 // currently used instead.
386 int transmit_bitrate_bps = sender_call_->GetStats().send_bandwidth_bps;
387 EvolveTestState(transmit_bitrate_bps, false);
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000388 }
stefan4fbd1452015-09-28 03:57:14 -0700389
390 return !event_.Wait(kPollIntervalMs);
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000391}
stefan@webrtc.org7e9315b2013-12-04 10:24:26 +0000392
stefan4fbd1452015-09-28 03:57:14 -0700393Call::Config RampUpDownUpTester::GetReceiverCallConfig() {
skvlad11a9cbf2016-10-07 11:53:05 -0700394 Call::Config config(&event_log_);
stefan4fbd1452015-09-28 03:57:14 -0700395 config.bitrate_config.min_bitrate_bps = 10000;
396 return config;
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000397}
stefan@webrtc.org7e9315b2013-12-04 10:24:26 +0000398
stefan4fbd1452015-09-28 03:57:14 -0700399std::string RampUpDownUpTester::GetModifierString() const {
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000400 std::string str("_");
stefanff483612015-12-21 03:14:00 -0800401 if (num_video_streams_ > 0) {
402 std::ostringstream s;
403 s << num_video_streams_;
404 str += s.str();
405 str += "stream";
406 str += (num_video_streams_ > 1 ? "s" : "");
407 str += "_";
408 }
409 if (num_audio_streams_ > 0) {
410 std::ostringstream s;
411 s << num_audio_streams_;
412 str += s.str();
413 str += "stream";
414 str += (num_audio_streams_ > 1 ? "s" : "");
415 str += "_";
416 }
stefan4fbd1452015-09-28 03:57:14 -0700417 str += (rtx_ ? "" : "no");
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000418 str += "rtx";
419 return str;
420}
andresp@webrtc.orgc1480792014-03-20 03:23:55 +0000421
stefandb752f92016-12-05 08:23:40 -0800422int RampUpDownUpTester::GetExpectedHighBitrate() const {
stefan38d8b3c2017-01-09 04:19:24 -0800423 int expected_bitrate_bps = 0;
424 if (num_audio_streams_ > 0)
425 expected_bitrate_bps += kExpectedHighAudioBitrateBps;
426 if (num_video_streams_ > 0)
427 expected_bitrate_bps += kExpectedHighVideoBitrateBps;
428 return expected_bitrate_bps;
429}
430
431int RampUpDownUpTester::GetHighLinkCapacity() const {
432 return 4 * GetExpectedHighBitrate() / (3 * 1000);
stefandb752f92016-12-05 08:23:40 -0800433}
434
stefan4fbd1452015-09-28 03:57:14 -0700435void RampUpDownUpTester::EvolveTestState(int bitrate_bps, bool suspended) {
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000436 int64_t now = clock_->TimeInMilliseconds();
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000437 switch (test_state_) {
438 case kFirstRampup: {
stefan4fbd1452015-09-28 03:57:14 -0700439 EXPECT_FALSE(suspended);
stefandb752f92016-12-05 08:23:40 -0800440 if (bitrate_bps >= GetExpectedHighBitrate()) {
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000441 // The first ramp-up has reached the target bitrate. Change the
442 // channel limit, and move to the next test state.
443 forward_transport_config_.link_capacity_kbps =
444 kLowBandwidthLimitBps / 1000;
stefanf116bd02015-10-27 08:29:42 -0700445 send_transport_->SetConfig(forward_transport_config_);
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000446 test_state_ = kLowRate;
stefanff483612015-12-21 03:14:00 -0800447 webrtc::test::PrintResult("ramp_up_down_up", GetModifierString(),
448 "first_rampup", now - state_start_ms_, "ms",
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000449 false);
450 state_start_ms_ = now;
451 interval_start_ms_ = now;
452 sent_bytes_ = 0;
andresp@webrtc.orgc1480792014-03-20 03:23:55 +0000453 }
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000454 break;
pbos@webrtc.orgf577ae92014-03-19 08:43:57 +0000455 }
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000456 case kLowRate: {
mflodman86cc6ff2016-07-26 04:44:06 -0700457 // Audio streams are never suspended.
458 bool check_suspend_state = num_video_streams_ > 0;
459 if (bitrate_bps < kExpectedLowBitrateBps &&
460 suspended == check_suspend_state) {
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000461 // The ramp-down was successful. Change the channel limit back to a
462 // high value, and move to the next test state.
stefan38d8b3c2017-01-09 04:19:24 -0800463 forward_transport_config_.link_capacity_kbps = GetHighLinkCapacity();
stefanf116bd02015-10-27 08:29:42 -0700464 send_transport_->SetConfig(forward_transport_config_);
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000465 test_state_ = kSecondRampup;
stefanff483612015-12-21 03:14:00 -0800466 webrtc::test::PrintResult("ramp_up_down_up", GetModifierString(),
467 "rampdown", now - state_start_ms_, "ms",
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000468 false);
469 state_start_ms_ = now;
470 interval_start_ms_ = now;
471 sent_bytes_ = 0;
472 }
473 break;
474 }
475 case kSecondRampup: {
stefandb752f92016-12-05 08:23:40 -0800476 if (bitrate_bps >= GetExpectedHighBitrate() && !suspended) {
stefanff483612015-12-21 03:14:00 -0800477 webrtc::test::PrintResult("ramp_up_down_up", GetModifierString(),
478 "second_rampup", now - state_start_ms_, "ms",
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000479 false);
Stefan Holmerff2a6352016-01-14 10:00:21 +0100480 ReportResult("ramp-up-down-up-average-network-latency",
481 send_transport_->GetAverageDelayMs(), "milliseconds");
Peter Boström5811a392015-12-10 13:02:50 +0100482 observation_complete_.Set();
stefan@webrtc.org3d7da882014-07-08 13:59:46 +0000483 }
484 break;
485 }
486 }
487}
pbos@webrtc.orgf577ae92014-03-19 08:43:57 +0000488
stefan4fbd1452015-09-28 03:57:14 -0700489class RampUpTest : public test::CallTest {
Erik Språng6b8d3552015-09-24 15:06:57 +0200490 public:
stefan4fbd1452015-09-28 03:57:14 -0700491 RampUpTest() {}
Erik Språng6b8d3552015-09-24 15:06:57 +0200492
stefan4fbd1452015-09-28 03:57:14 -0700493 virtual ~RampUpTest() {
stefanff483612015-12-21 03:14:00 -0800494 EXPECT_EQ(nullptr, video_send_stream_);
495 EXPECT_TRUE(video_receive_streams_.empty());
Erik Språng6b8d3552015-09-24 15:06:57 +0200496 }
Erik Språng6b8d3552015-09-24 15:06:57 +0200497};
498
Stefan Holmerff2a6352016-01-14 10:00:21 +0100499static const uint32_t kStartBitrateBps = 60000;
500
stefan38d8b3c2017-01-09 04:19:24 -0800501TEST_F(RampUpTest, UpDownUpAbsSendTimeSimulcastRedRtx) {
isheriff6f8d6862016-05-26 11:24:55 -0700502 RampUpDownUpTester test(3, 0, kStartBitrateBps, RtpExtension::kAbsSendTimeUri,
Stefan Holmerff2a6352016-01-14 10:00:21 +0100503 true, true);
stefane74eef12016-01-08 06:47:13 -0800504 RunBaseTest(&test);
Shao Changbine62202f2015-04-21 20:24:50 +0800505}
Stefan Holmerff2a6352016-01-14 10:00:21 +0100506
stefan38d8b3c2017-01-09 04:19:24 -0800507TEST_F(RampUpTest, UpDownUpTransportSequenceNumberRtx) {
Stefan Holmerff2a6352016-01-14 10:00:21 +0100508 RampUpDownUpTester test(3, 0, kStartBitrateBps,
isheriff6f8d6862016-05-26 11:24:55 -0700509 RtpExtension::kTransportSequenceNumberUri, true,
510 false);
Stefan Holmerff2a6352016-01-14 10:00:21 +0100511 RunBaseTest(&test);
512}
513
stefan38d8b3c2017-01-09 04:19:24 -0800514TEST_F(RampUpTest, UpDownUpAudioVideoTransportSequenceNumberRtx) {
Stefan Holmerff2a6352016-01-14 10:00:21 +0100515 RampUpDownUpTester test(3, 1, kStartBitrateBps,
isheriff6f8d6862016-05-26 11:24:55 -0700516 RtpExtension::kTransportSequenceNumberUri, true,
517 false);
Stefan Holmerff2a6352016-01-14 10:00:21 +0100518 RunBaseTest(&test);
519}
520
stefan38d8b3c2017-01-09 04:19:24 -0800521TEST_F(RampUpTest, UpDownUpAudioTransportSequenceNumberRtx) {
mflodman86cc6ff2016-07-26 04:44:06 -0700522 RampUpDownUpTester test(0, 1, kStartBitrateBps,
523 RtpExtension::kTransportSequenceNumberUri, true,
524 false);
525 RunBaseTest(&test);
526}
527
stefan38d8b3c2017-01-09 04:19:24 -0800528TEST_F(RampUpTest, TOffsetSimulcastRedRtx) {
stefan5a2c5062017-01-27 06:43:18 -0800529 RampUpTester test(3, 0, 0, 0, RtpExtension::kTimestampOffsetUri, true, true,
530 true);
stefan38d8b3c2017-01-09 04:19:24 -0800531 RunBaseTest(&test);
532}
533
534TEST_F(RampUpTest, AbsSendTime) {
stefan5a2c5062017-01-27 06:43:18 -0800535 RampUpTester test(1, 0, 0, 0, RtpExtension::kAbsSendTimeUri, false, false,
536 true);
stefane74eef12016-01-08 06:47:13 -0800537 RunBaseTest(&test);
pbos@webrtc.org85bd53e2014-12-10 10:36:20 +0000538}
539
stefan38d8b3c2017-01-09 04:19:24 -0800540TEST_F(RampUpTest, AbsSendTimeSimulcastRedRtx) {
stefan5a2c5062017-01-27 06:43:18 -0800541 RampUpTester test(3, 0, 0, 0, RtpExtension::kAbsSendTimeUri, true, true,
542 true);
stefane74eef12016-01-08 06:47:13 -0800543 RunBaseTest(&test);
pbos@webrtc.org85bd53e2014-12-10 10:36:20 +0000544}
545
stefan38d8b3c2017-01-09 04:19:24 -0800546TEST_F(RampUpTest, TransportSequenceNumber) {
stefan5a2c5062017-01-27 06:43:18 -0800547 RampUpTester test(1, 0, 0, 0, RtpExtension::kTransportSequenceNumberUri,
548 false, false, true);
stefane74eef12016-01-08 06:47:13 -0800549 RunBaseTest(&test);
Erik Språng6b8d3552015-09-24 15:06:57 +0200550}
551
552TEST_F(RampUpTest, TransportSequenceNumberSimulcast) {
stefan5a2c5062017-01-27 06:43:18 -0800553 RampUpTester test(3, 0, 0, 0, RtpExtension::kTransportSequenceNumberUri,
554 false, false, true);
stefane74eef12016-01-08 06:47:13 -0800555 RunBaseTest(&test);
Erik Språng6b8d3552015-09-24 15:06:57 +0200556}
557
stefan38d8b3c2017-01-09 04:19:24 -0800558TEST_F(RampUpTest, TransportSequenceNumberSimulcastRedRtx) {
stefan5a2c5062017-01-27 06:43:18 -0800559 RampUpTester test(3, 0, 0, 0, RtpExtension::kTransportSequenceNumberUri,
560 true, true, true);
561 RunBaseTest(&test);
562}
563
564TEST_F(RampUpTest, AudioTransportSequenceNumber) {
565 RampUpTester test(0, 1, 300000, 10000,
566 RtpExtension::kTransportSequenceNumberUri, false, false,
567 false);
stefane74eef12016-01-08 06:47:13 -0800568 RunBaseTest(&test);
Erik Språng6b8d3552015-09-24 15:06:57 +0200569}
pbos@webrtc.org744fbc72013-09-10 09:26:25 +0000570} // namespace webrtc