Move JitterEstimator into timing sub-folder
Bug: webrtc:14111 Change-Id: Ic5c66f35e3cedac7a328fbb6613e5cdf13a8005e Reviewed-on: https://webrtc-review.googlesource.com/c/src/+/263582 Commit-Queue: Rasmus Brandt <brandtr@webrtc.org> Reviewed-by: Åsa Persson <asapersson@webrtc.org> Cr-Commit-Position: refs/heads/main@{#37000}
This commit is contained in:
committed by
WebRTC LUCI CQ
parent
4662f53285
commit
10944e61ac
172
modules/video_coding/timing/jitter_estimator_unittest.cc
Normal file
172
modules/video_coding/timing/jitter_estimator_unittest.cc
Normal file
@ -0,0 +1,172 @@
|
||||
/* Copyright (c) 2014 The WebRTC project authors. All Rights Reserved.
|
||||
*
|
||||
* Use of this source code is governed by a BSD-style license
|
||||
* that can be found in the LICENSE file in the root of the source
|
||||
* tree. An additional intellectual property rights grant can be found
|
||||
* in the file PATENTS. All contributing project authors may
|
||||
* be found in the AUTHORS file in the root of the source tree.
|
||||
*/
|
||||
|
||||
#include "modules/video_coding/timing/jitter_estimator.h"
|
||||
|
||||
#include <stdint.h>
|
||||
|
||||
#include <memory>
|
||||
#include <vector>
|
||||
|
||||
#include "absl/types/optional.h"
|
||||
#include "api/array_view.h"
|
||||
#include "api/units/data_size.h"
|
||||
#include "api/units/frequency.h"
|
||||
#include "api/units/time_delta.h"
|
||||
#include "rtc_base/experiments/jitter_upper_bound_experiment.h"
|
||||
#include "rtc_base/numerics/histogram_percentile_counter.h"
|
||||
#include "rtc_base/strings/string_builder.h"
|
||||
#include "rtc_base/time_utils.h"
|
||||
#include "system_wrappers/include/clock.h"
|
||||
#include "test/gtest.h"
|
||||
#include "test/scoped_key_value_config.h"
|
||||
|
||||
namespace webrtc {
|
||||
|
||||
class TestJitterEstimator : public ::testing::Test {
|
||||
protected:
|
||||
TestJitterEstimator() : fake_clock_(0) {}
|
||||
|
||||
virtual void SetUp() {
|
||||
estimator_ = std::make_unique<JitterEstimator>(&fake_clock_, field_trials_);
|
||||
}
|
||||
|
||||
SimulatedClock fake_clock_;
|
||||
test::ScopedKeyValueConfig field_trials_;
|
||||
std::unique_ptr<JitterEstimator> estimator_;
|
||||
};
|
||||
|
||||
// Generates some simple test data in the form of a sawtooth wave.
|
||||
class ValueGenerator {
|
||||
public:
|
||||
explicit ValueGenerator(int32_t amplitude)
|
||||
: amplitude_(amplitude), counter_(0) {}
|
||||
|
||||
virtual ~ValueGenerator() = default;
|
||||
|
||||
TimeDelta Delay() const {
|
||||
return TimeDelta::Millis((counter_ % 11) - 5) * amplitude_;
|
||||
}
|
||||
|
||||
DataSize FrameSize() const {
|
||||
return DataSize::Bytes(1000 + Delay().ms() / 5);
|
||||
}
|
||||
|
||||
void Advance() { ++counter_; }
|
||||
|
||||
private:
|
||||
const int32_t amplitude_;
|
||||
int64_t counter_;
|
||||
};
|
||||
|
||||
// 5 fps, disable jitter delay altogether.
|
||||
TEST_F(TestJitterEstimator, TestLowRate) {
|
||||
ValueGenerator gen(10);
|
||||
TimeDelta time_delta = 1 / Frequency::Hertz(5);
|
||||
for (int i = 0; i < 60; ++i) {
|
||||
estimator_->UpdateEstimate(gen.Delay(), gen.FrameSize());
|
||||
fake_clock_.AdvanceTime(time_delta);
|
||||
if (i > 2)
|
||||
EXPECT_EQ(estimator_->GetJitterEstimate(0, absl::nullopt),
|
||||
TimeDelta::Zero());
|
||||
gen.Advance();
|
||||
}
|
||||
}
|
||||
|
||||
TEST_F(TestJitterEstimator, TestLowRateDisabled) {
|
||||
test::ScopedKeyValueConfig field_trials(
|
||||
field_trials_, "WebRTC-ReducedJitterDelayKillSwitch/Enabled/");
|
||||
SetUp();
|
||||
|
||||
ValueGenerator gen(10);
|
||||
TimeDelta time_delta = 1 / Frequency::Hertz(5);
|
||||
for (int i = 0; i < 60; ++i) {
|
||||
estimator_->UpdateEstimate(gen.Delay(), gen.FrameSize());
|
||||
fake_clock_.AdvanceTime(time_delta);
|
||||
if (i > 2)
|
||||
EXPECT_GT(estimator_->GetJitterEstimate(0, absl::nullopt),
|
||||
TimeDelta::Zero());
|
||||
gen.Advance();
|
||||
}
|
||||
}
|
||||
|
||||
TEST_F(TestJitterEstimator, TestUpperBound) {
|
||||
struct TestContext {
|
||||
TestContext()
|
||||
: upper_bound(0.0),
|
||||
rtt_mult(0),
|
||||
rtt_mult_add_cap_ms(absl::nullopt),
|
||||
percentiles(1000) {}
|
||||
double upper_bound;
|
||||
double rtt_mult;
|
||||
absl::optional<TimeDelta> rtt_mult_add_cap_ms;
|
||||
rtc::HistogramPercentileCounter percentiles;
|
||||
};
|
||||
std::vector<TestContext> test_cases(4);
|
||||
|
||||
// Large upper bound, rtt_mult = 0, and nullopt for rtt_mult addition cap.
|
||||
test_cases[0].upper_bound = 100.0;
|
||||
test_cases[0].rtt_mult = 0;
|
||||
test_cases[0].rtt_mult_add_cap_ms = absl::nullopt;
|
||||
// Small upper bound, rtt_mult = 0, and nullopt for rtt_mult addition cap.
|
||||
test_cases[1].upper_bound = 3.5;
|
||||
test_cases[1].rtt_mult = 0;
|
||||
test_cases[1].rtt_mult_add_cap_ms = absl::nullopt;
|
||||
// Large upper bound, rtt_mult = 1, and large rtt_mult addition cap value.
|
||||
test_cases[2].upper_bound = 1000.0;
|
||||
test_cases[2].rtt_mult = 1.0;
|
||||
test_cases[2].rtt_mult_add_cap_ms = TimeDelta::Millis(200);
|
||||
// Large upper bound, rtt_mult = 1, and small rtt_mult addition cap value.
|
||||
test_cases[3].upper_bound = 1000.0;
|
||||
test_cases[3].rtt_mult = 1.0;
|
||||
test_cases[3].rtt_mult_add_cap_ms = TimeDelta::Millis(10);
|
||||
|
||||
// Test jitter buffer upper_bound and rtt_mult addition cap sizes.
|
||||
for (TestContext& context : test_cases) {
|
||||
// Set up field trial and reset jitter estimator.
|
||||
char string_buf[64];
|
||||
rtc::SimpleStringBuilder ssb(string_buf);
|
||||
ssb << JitterUpperBoundExperiment::kJitterUpperBoundExperimentName
|
||||
<< "/Enabled-" << context.upper_bound << "/";
|
||||
test::ScopedKeyValueConfig field_trials(field_trials_, ssb.str());
|
||||
SetUp();
|
||||
|
||||
ValueGenerator gen(50);
|
||||
TimeDelta time_delta = 1 / Frequency::Hertz(30);
|
||||
constexpr TimeDelta kRtt = TimeDelta::Millis(250);
|
||||
for (int i = 0; i < 100; ++i) {
|
||||
estimator_->UpdateEstimate(gen.Delay(), gen.FrameSize());
|
||||
fake_clock_.AdvanceTime(time_delta);
|
||||
estimator_->FrameNacked(); // To test rtt_mult.
|
||||
estimator_->UpdateRtt(kRtt); // To test rtt_mult.
|
||||
context.percentiles.Add(
|
||||
estimator_
|
||||
->GetJitterEstimate(context.rtt_mult, context.rtt_mult_add_cap_ms)
|
||||
.ms());
|
||||
gen.Advance();
|
||||
}
|
||||
}
|
||||
|
||||
// Median should be similar after three seconds. Allow 5% error margin.
|
||||
uint32_t median_unbound = *test_cases[0].percentiles.GetPercentile(0.5);
|
||||
uint32_t median_bounded = *test_cases[1].percentiles.GetPercentile(0.5);
|
||||
EXPECT_NEAR(median_unbound, median_bounded, (median_unbound * 5) / 100);
|
||||
|
||||
// Max should be lower for the bounded case.
|
||||
uint32_t max_unbound = *test_cases[0].percentiles.GetPercentile(1.0);
|
||||
uint32_t max_bounded = *test_cases[1].percentiles.GetPercentile(1.0);
|
||||
EXPECT_GT(max_unbound, static_cast<uint32_t>(max_bounded * 1.25));
|
||||
|
||||
// With rtt_mult = 1, max should be lower with small rtt_mult add cap value.
|
||||
max_unbound = *test_cases[2].percentiles.GetPercentile(1.0);
|
||||
max_bounded = *test_cases[3].percentiles.GetPercentile(1.0);
|
||||
EXPECT_GT(max_unbound, static_cast<uint32_t>(max_bounded * 1.25));
|
||||
}
|
||||
|
||||
} // namespace webrtc
|
||||
Reference in New Issue
Block a user