blob: de9d5d4b799ca994802a321bf6eadb2e20e348c6 [file] [log] [blame]
sprang@webrtc.org70e2d112014-09-24 14:06:56 +00001/* Copyright (c) 2014 The WebRTC project authors. All Rights Reserved.
2 *
3 * Use of this source code is governed by a BSD-style license
4 * that can be found in the LICENSE file in the root of the source
5 * tree. An additional intellectual property rights grant can be found
6 * in the file PATENTS. All contributing project authors may
7 * be found in the AUTHORS file in the root of the source tree.
8 */
9
Yves Gerey3e707812018-11-28 16:47:49 +010010#include <stdint.h>
Jonas Olssona4d87372019-07-05 19:08:33 +020011
Yves Gerey3e707812018-11-28 16:47:49 +010012#include <memory>
13#include <vector>
sprang@webrtc.org70e2d112014-09-24 14:06:56 +000014
Yves Gerey3e707812018-11-28 16:47:49 +010015#include "absl/memory/memory.h"
16#include "absl/types/optional.h"
17#include "api/array_view.h"
18#include "modules/video_coding/jitter_estimator.h"
Erik Språngb1e031a2018-11-01 11:20:49 +010019#include "rtc_base/experiments/jitter_upper_bound_experiment.h"
Erik Språngb1e031a2018-11-01 11:20:49 +010020#include "rtc_base/numerics/histogram_percentile_counter.h"
Yves Gerey3e707812018-11-28 16:47:49 +010021#include "rtc_base/strings/string_builder.h"
Steve Anton10542f22019-01-11 09:11:00 -080022#include "rtc_base/time_utils.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020023#include "system_wrappers/include/clock.h"
Erik Språngb1e031a2018-11-01 11:20:49 +010024#include "test/field_trial.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020025#include "test/gtest.h"
sprang@webrtc.org70e2d112014-09-24 14:06:56 +000026
27namespace webrtc {
28
Erik Språngb1e031a2018-11-01 11:20:49 +010029class TestVCMJitterEstimator : public ::testing::Test {
30 protected:
31 TestVCMJitterEstimator() : fake_clock_(0) {}
sprang@webrtc.org70e2d112014-09-24 14:06:56 +000032
Erik Språngb1e031a2018-11-01 11:20:49 +010033 virtual void SetUp() {
Åsa Persson3fcc5be2019-04-04 09:40:27 +020034 estimator_ = absl::make_unique<VCMJitterEstimator>(&fake_clock_);
Erik Språngb1e031a2018-11-01 11:20:49 +010035 }
sprang@webrtc.org70e2d112014-09-24 14:06:56 +000036
37 void AdvanceClock(int64_t microseconds) {
38 fake_clock_.AdvanceTimeMicroseconds(microseconds);
39 }
40
sprang@webrtc.org70e2d112014-09-24 14:06:56 +000041 SimulatedClock fake_clock_;
Erik Språngb1e031a2018-11-01 11:20:49 +010042 std::unique_ptr<VCMJitterEstimator> estimator_;
sprang@webrtc.org70e2d112014-09-24 14:06:56 +000043};
44
45// Generates some simple test data in the form of a sawtooth wave.
46class ValueGenerator {
47 public:
oprypin60c56682017-03-24 03:22:49 -070048 explicit ValueGenerator(int32_t amplitude)
49 : amplitude_(amplitude), counter_(0) {}
sprang@webrtc.org70e2d112014-09-24 14:06:56 +000050 virtual ~ValueGenerator() {}
51
Åsa Persson3fcc5be2019-04-04 09:40:27 +020052 int64_t Delay() const { return ((counter_ % 11) - 5) * amplitude_; }
sprang@webrtc.org70e2d112014-09-24 14:06:56 +000053
Åsa Persson3fcc5be2019-04-04 09:40:27 +020054 uint32_t FrameSize() const { return 1000 + Delay(); }
sprang@webrtc.org70e2d112014-09-24 14:06:56 +000055
56 void Advance() { ++counter_; }
57
58 private:
59 const int32_t amplitude_;
60 int64_t counter_;
61};
62
63// 5 fps, disable jitter delay altogether.
64TEST_F(TestVCMJitterEstimator, TestLowRate) {
65 ValueGenerator gen(10);
Erik Språngb1e031a2018-11-01 11:20:49 +010066 uint64_t time_delta_us = rtc::kNumMicrosecsPerSec / 5;
sprang@webrtc.org70e2d112014-09-24 14:06:56 +000067 for (int i = 0; i < 60; ++i) {
Erik Språngb1e031a2018-11-01 11:20:49 +010068 estimator_->UpdateEstimate(gen.Delay(), gen.FrameSize());
69 AdvanceClock(time_delta_us);
sprang@webrtc.org70e2d112014-09-24 14:06:56 +000070 if (i > 2)
“Michaele0f37042019-06-04 10:04:12 -050071 EXPECT_EQ(estimator_->GetJitterEstimate(0, absl::nullopt), 0);
sprang@webrtc.org70e2d112014-09-24 14:06:56 +000072 gen.Advance();
73 }
74}
75
Erik Språngb1e031a2018-11-01 11:20:49 +010076TEST_F(TestVCMJitterEstimator, TestUpperBound) {
77 struct TestContext {
“Michaele0f37042019-06-04 10:04:12 -050078 TestContext()
79 : upper_bound(0.0),
80 rtt_mult(0),
81 rtt_mult_add_cap_ms(absl::nullopt),
82 percentiles(1000) {}
Erik Språngb1e031a2018-11-01 11:20:49 +010083 double upper_bound;
“Michaele0f37042019-06-04 10:04:12 -050084 double rtt_mult;
85 absl::optional<double> rtt_mult_add_cap_ms;
Erik Språngb1e031a2018-11-01 11:20:49 +010086 rtc::HistogramPercentileCounter percentiles;
87 };
“Michaele0f37042019-06-04 10:04:12 -050088 std::vector<TestContext> test_cases(4);
sprang@webrtc.org70e2d112014-09-24 14:06:56 +000089
“Michaele0f37042019-06-04 10:04:12 -050090 // Large upper bound, rtt_mult = 0, and nullopt for rtt_mult addition cap.
91 test_cases[0].upper_bound = 100.0;
92 test_cases[0].rtt_mult = 0;
93 test_cases[0].rtt_mult_add_cap_ms = absl::nullopt;
94 // Small upper bound, rtt_mult = 0, and nullopt for rtt_mult addition cap.
95 test_cases[1].upper_bound = 3.5;
96 test_cases[1].rtt_mult = 0;
97 test_cases[1].rtt_mult_add_cap_ms = absl::nullopt;
98 // Large upper bound, rtt_mult = 1, and large rtt_mult addition cap value.
99 test_cases[2].upper_bound = 1000.0;
100 test_cases[2].rtt_mult = 1.0;
101 test_cases[2].rtt_mult_add_cap_ms = 200.0;
102 // Large upper bound, rtt_mult = 1, and small rtt_mult addition cap value.
103 test_cases[3].upper_bound = 1000.0;
104 test_cases[3].rtt_mult = 1.0;
105 test_cases[3].rtt_mult_add_cap_ms = 10.0;
sprang@webrtc.org70e2d112014-09-24 14:06:56 +0000106
“Michaele0f37042019-06-04 10:04:12 -0500107 // Test jitter buffer upper_bound and rtt_mult addition cap sizes.
Erik Språngb1e031a2018-11-01 11:20:49 +0100108 for (TestContext& context : test_cases) {
109 // Set up field trial and reset jitter estimator.
110 char string_buf[64];
111 rtc::SimpleStringBuilder ssb(string_buf);
112 ssb << JitterUpperBoundExperiment::kJitterUpperBoundExperimentName
113 << "/Enabled-" << context.upper_bound << "/";
114 test::ScopedFieldTrials field_trials(ssb.str());
115 SetUp();
sprang@webrtc.org70e2d112014-09-24 14:06:56 +0000116
Erik Språngb1e031a2018-11-01 11:20:49 +0100117 ValueGenerator gen(50);
118 uint64_t time_delta_us = rtc::kNumMicrosecsPerSec / 30;
“Michaele0f37042019-06-04 10:04:12 -0500119 constexpr int64_t kRttMs = 250;
Erik Språngb1e031a2018-11-01 11:20:49 +0100120 for (int i = 0; i < 100; ++i) {
121 estimator_->UpdateEstimate(gen.Delay(), gen.FrameSize());
122 AdvanceClock(time_delta_us);
“Michaele0f37042019-06-04 10:04:12 -0500123 estimator_->FrameNacked(); // To test rtt_mult.
124 estimator_->UpdateRtt(kRttMs); // To test rtt_mult.
Erik Språngb1e031a2018-11-01 11:20:49 +0100125 context.percentiles.Add(
“Michaele0f37042019-06-04 10:04:12 -0500126 static_cast<uint32_t>(estimator_->GetJitterEstimate(
127 context.rtt_mult, context.rtt_mult_add_cap_ms)));
Erik Språngb1e031a2018-11-01 11:20:49 +0100128 gen.Advance();
sprang@webrtc.org70e2d112014-09-24 14:06:56 +0000129 }
sprang@webrtc.org70e2d112014-09-24 14:06:56 +0000130 }
131
Erik Språngb1e031a2018-11-01 11:20:49 +0100132 // Median should be similar after three seconds. Allow 5% error margin.
133 uint32_t median_unbound = *test_cases[0].percentiles.GetPercentile(0.5);
134 uint32_t median_bounded = *test_cases[1].percentiles.GetPercentile(0.5);
135 EXPECT_NEAR(median_unbound, median_bounded, (median_unbound * 5) / 100);
136
137 // Max should be lower for the bounded case.
138 uint32_t max_unbound = *test_cases[0].percentiles.GetPercentile(1.0);
139 uint32_t max_bounded = *test_cases[1].percentiles.GetPercentile(1.0);
140 EXPECT_GT(max_unbound, static_cast<uint32_t>(max_bounded * 1.25));
“Michaele0f37042019-06-04 10:04:12 -0500141
142 // With rtt_mult = 1, max should be lower with small rtt_mult add cap value.
143 max_unbound = *test_cases[2].percentiles.GetPercentile(1.0);
144 max_bounded = *test_cases[3].percentiles.GetPercentile(1.0);
145 EXPECT_GT(max_unbound, static_cast<uint32_t>(max_bounded * 1.25));
sprang@webrtc.org70e2d112014-09-24 14:06:56 +0000146}
Erik Språngb1e031a2018-11-01 11:20:49 +0100147
oprypin60c56682017-03-24 03:22:49 -0700148} // namespace webrtc