blob: e696f0e02f14a42da8f92732c12759d86f12f1fa [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>
11#include <memory>
12#include <vector>
sprang@webrtc.org70e2d112014-09-24 14:06:56 +000013
Yves Gerey3e707812018-11-28 16:47:49 +010014#include "absl/memory/memory.h"
15#include "absl/types/optional.h"
16#include "api/array_view.h"
17#include "modules/video_coding/jitter_estimator.h"
Erik Språngb1e031a2018-11-01 11:20:49 +010018#include "rtc_base/experiments/jitter_upper_bound_experiment.h"
Erik Språngb1e031a2018-11-01 11:20:49 +010019#include "rtc_base/numerics/histogram_percentile_counter.h"
Yves Gerey3e707812018-11-28 16:47:49 +010020#include "rtc_base/strings/string_builder.h"
Steve Anton10542f22019-01-11 09:11:00 -080021#include "rtc_base/time_utils.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020022#include "system_wrappers/include/clock.h"
Erik Språngb1e031a2018-11-01 11:20:49 +010023#include "test/field_trial.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020024#include "test/gtest.h"
sprang@webrtc.org70e2d112014-09-24 14:06:56 +000025
26namespace webrtc {
27
Erik Språngb1e031a2018-11-01 11:20:49 +010028class TestVCMJitterEstimator : public ::testing::Test {
29 protected:
30 TestVCMJitterEstimator() : fake_clock_(0) {}
sprang@webrtc.org70e2d112014-09-24 14:06:56 +000031
Erik Språngb1e031a2018-11-01 11:20:49 +010032 virtual void SetUp() {
33 estimator_ = absl::make_unique<VCMJitterEstimator>(&fake_clock_, 0, 0);
34 }
sprang@webrtc.org70e2d112014-09-24 14:06:56 +000035
36 void AdvanceClock(int64_t microseconds) {
37 fake_clock_.AdvanceTimeMicroseconds(microseconds);
38 }
39
sprang@webrtc.org70e2d112014-09-24 14:06:56 +000040 SimulatedClock fake_clock_;
Erik Språngb1e031a2018-11-01 11:20:49 +010041 std::unique_ptr<VCMJitterEstimator> estimator_;
sprang@webrtc.org70e2d112014-09-24 14:06:56 +000042};
43
44// Generates some simple test data in the form of a sawtooth wave.
45class ValueGenerator {
46 public:
oprypin60c56682017-03-24 03:22:49 -070047 explicit ValueGenerator(int32_t amplitude)
48 : amplitude_(amplitude), counter_(0) {}
sprang@webrtc.org70e2d112014-09-24 14:06:56 +000049 virtual ~ValueGenerator() {}
50
51 int64_t Delay() { return ((counter_ % 11) - 5) * amplitude_; }
52
53 uint32_t FrameSize() { return 1000 + Delay(); }
54
55 void Advance() { ++counter_; }
56
57 private:
58 const int32_t amplitude_;
59 int64_t counter_;
60};
61
62// 5 fps, disable jitter delay altogether.
63TEST_F(TestVCMJitterEstimator, TestLowRate) {
64 ValueGenerator gen(10);
Erik Språngb1e031a2018-11-01 11:20:49 +010065 uint64_t time_delta_us = rtc::kNumMicrosecsPerSec / 5;
sprang@webrtc.org70e2d112014-09-24 14:06:56 +000066 for (int i = 0; i < 60; ++i) {
Erik Språngb1e031a2018-11-01 11:20:49 +010067 estimator_->UpdateEstimate(gen.Delay(), gen.FrameSize());
68 AdvanceClock(time_delta_us);
sprang@webrtc.org70e2d112014-09-24 14:06:56 +000069 if (i > 2)
Erik Språngb1e031a2018-11-01 11:20:49 +010070 EXPECT_EQ(estimator_->GetJitterEstimate(0), 0);
sprang@webrtc.org70e2d112014-09-24 14:06:56 +000071 gen.Advance();
72 }
73}
74
Erik Språngb1e031a2018-11-01 11:20:49 +010075TEST_F(TestVCMJitterEstimator, TestUpperBound) {
76 struct TestContext {
77 TestContext() : upper_bound(0.0), percentiles(1000) {}
78 double upper_bound;
79 rtc::HistogramPercentileCounter percentiles;
80 };
81 std::vector<TestContext> test_cases(2);
sprang@webrtc.org70e2d112014-09-24 14:06:56 +000082
Erik Språngb1e031a2018-11-01 11:20:49 +010083 test_cases[0].upper_bound = 100.0; // First use essentially no cap.
84 test_cases[1].upper_bound = 3.5; // Second, reasonably small cap.
sprang@webrtc.org70e2d112014-09-24 14:06:56 +000085
Erik Språngb1e031a2018-11-01 11:20:49 +010086 for (TestContext& context : test_cases) {
87 // Set up field trial and reset jitter estimator.
88 char string_buf[64];
89 rtc::SimpleStringBuilder ssb(string_buf);
90 ssb << JitterUpperBoundExperiment::kJitterUpperBoundExperimentName
91 << "/Enabled-" << context.upper_bound << "/";
92 test::ScopedFieldTrials field_trials(ssb.str());
93 SetUp();
sprang@webrtc.org70e2d112014-09-24 14:06:56 +000094
Erik Språngb1e031a2018-11-01 11:20:49 +010095 ValueGenerator gen(50);
96 uint64_t time_delta_us = rtc::kNumMicrosecsPerSec / 30;
97 for (int i = 0; i < 100; ++i) {
98 estimator_->UpdateEstimate(gen.Delay(), gen.FrameSize());
99 AdvanceClock(time_delta_us);
100 context.percentiles.Add(
101 static_cast<uint32_t>(estimator_->GetJitterEstimate(0)));
102 gen.Advance();
sprang@webrtc.org70e2d112014-09-24 14:06:56 +0000103 }
sprang@webrtc.org70e2d112014-09-24 14:06:56 +0000104 }
105
Erik Språngb1e031a2018-11-01 11:20:49 +0100106 // Median should be similar after three seconds. Allow 5% error margin.
107 uint32_t median_unbound = *test_cases[0].percentiles.GetPercentile(0.5);
108 uint32_t median_bounded = *test_cases[1].percentiles.GetPercentile(0.5);
109 EXPECT_NEAR(median_unbound, median_bounded, (median_unbound * 5) / 100);
110
111 // Max should be lower for the bounded case.
112 uint32_t max_unbound = *test_cases[0].percentiles.GetPercentile(1.0);
113 uint32_t max_bounded = *test_cases[1].percentiles.GetPercentile(1.0);
114 EXPECT_GT(max_unbound, static_cast<uint32_t>(max_bounded * 1.25));
sprang@webrtc.org70e2d112014-09-24 14:06:56 +0000115}
Erik Språngb1e031a2018-11-01 11:20:49 +0100116
oprypin60c56682017-03-24 03:22:49 -0700117} // namespace webrtc