Back off relative to current estimate rather than ack rate when in ALR.

If we're in ALR, the acked rate is going to be significantly lower than
the current estimate for the link capacity. If we need to back off in
this situation (usually caused by latency spikes), this CL makes us back
off relative to current estimate if. We then immediately send a new
probe just in case the network did actually change.

All of this is behind experiment flags for now.

Bug: webrtc:10144
Change-Id: I062a259c36417eea2211d44592ef7fc979aa22b7
Reviewed-on: https://webrtc-review.googlesource.com/c/113880
Commit-Queue: Erik Språng <sprang@webrtc.org>
Reviewed-by: Sebastian Jansson <srte@webrtc.org>
Cr-Commit-Position: refs/heads/master@{#26045}
This commit is contained in:
Erik Språng 2018-12-18 12:23:12 +01:00 committed by Commit Bot
parent a1bec23f6c
commit b3564c1cb2
8 changed files with 55 additions and 16 deletions

View File

@ -70,13 +70,15 @@ DelayBasedBwe::Result::Result()
: updated(false), : updated(false),
probe(false), probe(false),
target_bitrate(DataRate::Zero()), target_bitrate(DataRate::Zero()),
recovered_from_overuse(false) {} recovered_from_overuse(false),
backoff_in_alr(false) {}
DelayBasedBwe::Result::Result(bool probe, DataRate target_bitrate) DelayBasedBwe::Result::Result(bool probe, DataRate target_bitrate)
: updated(true), : updated(true),
probe(probe), probe(probe),
target_bitrate(target_bitrate), target_bitrate(target_bitrate),
recovered_from_overuse(false) {} recovered_from_overuse(false),
backoff_in_alr(false) {}
DelayBasedBwe::Result::~Result() {} DelayBasedBwe::Result::~Result() {}
@ -93,7 +95,9 @@ DelayBasedBwe::DelayBasedBwe(RtcEventLog* event_log)
trendline_smoothing_coeff_(kDefaultTrendlineSmoothingCoeff), trendline_smoothing_coeff_(kDefaultTrendlineSmoothingCoeff),
trendline_threshold_gain_(kDefaultTrendlineThresholdGain), trendline_threshold_gain_(kDefaultTrendlineThresholdGain),
prev_bitrate_(DataRate::Zero()), prev_bitrate_(DataRate::Zero()),
prev_state_(BandwidthUsage::kBwNormal) { prev_state_(BandwidthUsage::kBwNormal),
alr_limited_backoff_enabled_(
field_trial::IsEnabled("WebRTC-Bwe-AlrLimitedBackoff")) {
RTC_LOG(LS_INFO) RTC_LOG(LS_INFO)
<< "Using Trendline filter for delay change estimation with window size " << "Using Trendline filter for delay change estimation with window size "
<< trendline_window_size_; << trendline_window_size_;
@ -108,6 +112,7 @@ DelayBasedBwe::Result DelayBasedBwe::IncomingPacketFeedbackVector(
const std::vector<PacketFeedback>& packet_feedback_vector, const std::vector<PacketFeedback>& packet_feedback_vector,
absl::optional<DataRate> acked_bitrate, absl::optional<DataRate> acked_bitrate,
absl::optional<DataRate> probe_bitrate, absl::optional<DataRate> probe_bitrate,
bool in_alr,
Timestamp at_time) { Timestamp at_time) {
RTC_DCHECK(std::is_sorted(packet_feedback_vector.begin(), RTC_DCHECK(std::is_sorted(packet_feedback_vector.begin(),
packet_feedback_vector.end(), packet_feedback_vector.end(),
@ -149,7 +154,7 @@ DelayBasedBwe::Result DelayBasedBwe::IncomingPacketFeedbackVector(
return Result(); return Result();
} }
return MaybeUpdateEstimate(acked_bitrate, probe_bitrate, return MaybeUpdateEstimate(acked_bitrate, probe_bitrate,
recovered_from_overuse, at_time); recovered_from_overuse, in_alr, at_time);
} }
void DelayBasedBwe::IncomingPacketFeedback( void DelayBasedBwe::IncomingPacketFeedback(
@ -194,13 +199,19 @@ DelayBasedBwe::Result DelayBasedBwe::MaybeUpdateEstimate(
absl::optional<DataRate> acked_bitrate, absl::optional<DataRate> acked_bitrate,
absl::optional<DataRate> probe_bitrate, absl::optional<DataRate> probe_bitrate,
bool recovered_from_overuse, bool recovered_from_overuse,
bool in_alr,
Timestamp at_time) { Timestamp at_time) {
Result result; Result result;
// Currently overusing the bandwidth. // Currently overusing the bandwidth.
if (delay_detector_->State() == BandwidthUsage::kBwOverusing) { if (delay_detector_->State() == BandwidthUsage::kBwOverusing) {
if (acked_bitrate && if (in_alr && alr_limited_backoff_enabled_ &&
rate_control_.TimeToReduceFurther(at_time, *acked_bitrate)) { rate_control_.TimeToReduceFurther(at_time, prev_bitrate_)) {
result.updated =
UpdateEstimate(at_time, prev_bitrate_, &result.target_bitrate);
result.backoff_in_alr = true;
} else if (acked_bitrate &&
rate_control_.TimeToReduceFurther(at_time, *acked_bitrate)) {
result.updated = result.updated =
UpdateEstimate(at_time, acked_bitrate, &result.target_bitrate); UpdateEstimate(at_time, acked_bitrate, &result.target_bitrate);
} else if (!acked_bitrate && rate_control_.ValidEstimate() && } else if (!acked_bitrate && rate_control_.ValidEstimate() &&
@ -287,4 +298,9 @@ void DelayBasedBwe::SetMinBitrate(DataRate min_bitrate) {
TimeDelta DelayBasedBwe::GetExpectedBwePeriod() const { TimeDelta DelayBasedBwe::GetExpectedBwePeriod() const {
return rate_control_.GetExpectedBandwidthPeriod(); return rate_control_.GetExpectedBandwidthPeriod();
} }
void DelayBasedBwe::SetAlrLimitedBackoffExperiment(bool enabled) {
alr_limited_backoff_enabled_ = enabled;
}
} // namespace webrtc } // namespace webrtc

View File

@ -39,6 +39,7 @@ class DelayBasedBwe {
bool probe; bool probe;
DataRate target_bitrate = DataRate::Zero(); DataRate target_bitrate = DataRate::Zero();
bool recovered_from_overuse; bool recovered_from_overuse;
bool backoff_in_alr;
}; };
explicit DelayBasedBwe(RtcEventLog* event_log); explicit DelayBasedBwe(RtcEventLog* event_log);
@ -48,12 +49,14 @@ class DelayBasedBwe {
const std::vector<PacketFeedback>& packet_feedback_vector, const std::vector<PacketFeedback>& packet_feedback_vector,
absl::optional<DataRate> acked_bitrate, absl::optional<DataRate> acked_bitrate,
absl::optional<DataRate> probe_bitrate, absl::optional<DataRate> probe_bitrate,
bool in_alr,
Timestamp at_time); Timestamp at_time);
void OnRttUpdate(TimeDelta avg_rtt); void OnRttUpdate(TimeDelta avg_rtt);
bool LatestEstimate(std::vector<uint32_t>* ssrcs, DataRate* bitrate) const; bool LatestEstimate(std::vector<uint32_t>* ssrcs, DataRate* bitrate) const;
void SetStartBitrate(DataRate start_bitrate); void SetStartBitrate(DataRate start_bitrate);
void SetMinBitrate(DataRate min_bitrate); void SetMinBitrate(DataRate min_bitrate);
TimeDelta GetExpectedBwePeriod() const; TimeDelta GetExpectedBwePeriod() const;
void SetAlrLimitedBackoffExperiment(bool enabled);
private: private:
friend class GoogCcStatePrinter; friend class GoogCcStatePrinter;
@ -61,7 +64,8 @@ class DelayBasedBwe {
Timestamp at_time); Timestamp at_time);
Result MaybeUpdateEstimate(absl::optional<DataRate> acked_bitrate, Result MaybeUpdateEstimate(absl::optional<DataRate> acked_bitrate,
absl::optional<DataRate> probe_bitrate, absl::optional<DataRate> probe_bitrate,
bool request_probe, bool recovered_from_overuse,
bool in_alr,
Timestamp at_time); Timestamp at_time);
// Updates the current remote rate estimate and returns true if a valid // Updates the current remote rate estimate and returns true if a valid
// estimate exists. // estimate exists.
@ -81,6 +85,7 @@ class DelayBasedBwe {
double trendline_threshold_gain_; double trendline_threshold_gain_;
DataRate prev_bitrate_; DataRate prev_bitrate_;
BandwidthUsage prev_state_; BandwidthUsage prev_state_;
bool alr_limited_backoff_enabled_;
RTC_DISALLOW_IMPLICIT_CONSTRUCTORS(DelayBasedBwe); RTC_DISALLOW_IMPLICIT_CONSTRUCTORS(DelayBasedBwe);
}; };

View File

@ -29,8 +29,9 @@ constexpr Timestamp kDummyTimestamp = Timestamp::Seconds<1000>();
TEST_F(DelayBasedBweTest, NoCrashEmptyFeedback) { TEST_F(DelayBasedBweTest, NoCrashEmptyFeedback) {
std::vector<PacketFeedback> packet_feedback_vector; std::vector<PacketFeedback> packet_feedback_vector;
bitrate_estimator_->IncomingPacketFeedbackVector( bitrate_estimator_->IncomingPacketFeedbackVector(packet_feedback_vector,
packet_feedback_vector, absl::nullopt, absl::nullopt, kDummyTimestamp); absl::nullopt, absl::nullopt,
false, kDummyTimestamp);
} }
TEST_F(DelayBasedBweTest, NoCrashOnlyLostFeedback) { TEST_F(DelayBasedBweTest, NoCrashOnlyLostFeedback) {
@ -41,8 +42,9 @@ TEST_F(DelayBasedBweTest, NoCrashOnlyLostFeedback) {
packet_feedback_vector.push_back(PacketFeedback(PacketFeedback::kNotReceived, packet_feedback_vector.push_back(PacketFeedback(PacketFeedback::kNotReceived,
PacketFeedback::kNoSendTime, PacketFeedback::kNoSendTime,
1, 1500, PacedPacketInfo())); 1, 1500, PacedPacketInfo()));
bitrate_estimator_->IncomingPacketFeedbackVector( bitrate_estimator_->IncomingPacketFeedbackVector(packet_feedback_vector,
packet_feedback_vector, absl::nullopt, absl::nullopt, kDummyTimestamp); absl::nullopt, absl::nullopt,
false, kDummyTimestamp);
} }
TEST_F(DelayBasedBweTest, ProbeDetection) { TEST_F(DelayBasedBweTest, ProbeDetection) {

View File

@ -207,7 +207,7 @@ void DelayBasedBweTest::IncomingFeedback(int64_t arrival_time_ms,
DelayBasedBwe::Result result = DelayBasedBwe::Result result =
bitrate_estimator_->IncomingPacketFeedbackVector( bitrate_estimator_->IncomingPacketFeedbackVector(
packets, acknowledged_bitrate_estimator_->bitrate(), packets, acknowledged_bitrate_estimator_->bitrate(),
probe_bitrate_estimator_->FetchAndResetLastEstimatedBitrate(), probe_bitrate_estimator_->FetchAndResetLastEstimatedBitrate(), false,
Timestamp::ms(clock_.TimeInMilliseconds())); Timestamp::ms(clock_.TimeInMilliseconds()));
const uint32_t kDummySsrc = 0; const uint32_t kDummySsrc = 0;
if (result.updated) { if (result.updated) {
@ -248,7 +248,7 @@ bool DelayBasedBweTest::GenerateAndProcessFrame(uint32_t ssrc,
DelayBasedBwe::Result result = DelayBasedBwe::Result result =
bitrate_estimator_->IncomingPacketFeedbackVector( bitrate_estimator_->IncomingPacketFeedbackVector(
packets, acknowledged_bitrate_estimator_->bitrate(), packets, acknowledged_bitrate_estimator_->bitrate(),
probe_bitrate_estimator_->FetchAndResetLastEstimatedBitrate(), probe_bitrate_estimator_->FetchAndResetLastEstimatedBitrate(), false,
Timestamp::ms(clock_.TimeInMilliseconds())); Timestamp::ms(clock_.TimeInMilliseconds()));
const uint32_t kDummySsrc = 0; const uint32_t kDummySsrc = 0;
if (result.updated) { if (result.updated) {

View File

@ -498,7 +498,7 @@ NetworkControlUpdate GoogCcNetworkController::OnTransportPacketsFeedback(
DelayBasedBwe::Result result; DelayBasedBwe::Result result;
result = delay_based_bwe_->IncomingPacketFeedbackVector( result = delay_based_bwe_->IncomingPacketFeedbackVector(
received_feedback_vector, acknowledged_bitrate, probe_bitrate, received_feedback_vector, acknowledged_bitrate, probe_bitrate,
report.feedback_time); alr_start_time.has_value(), report.feedback_time);
NetworkControlUpdate update; NetworkControlUpdate update;
if (result.updated) { if (result.updated) {
@ -518,6 +518,11 @@ NetworkControlUpdate GoogCcNetworkController::OnTransportPacketsFeedback(
auto probes = probe_controller_->RequestProbe(report.feedback_time.ms()); auto probes = probe_controller_->RequestProbe(report.feedback_time.ms());
update.probe_cluster_configs.insert(update.probe_cluster_configs.end(), update.probe_cluster_configs.insert(update.probe_cluster_configs.end(),
probes.begin(), probes.end()); probes.begin(), probes.end());
} else if (result.backoff_in_alr) {
// If we just backed off during ALR, request a new probe.
auto probes = probe_controller_->RequestProbe(report.feedback_time.ms());
update.probe_cluster_configs.insert(update.probe_cluster_configs.end(),
probes.begin(), probes.end());
} }
// No valid RTT could be because send-side BWE isn't used, in which case // No valid RTT could be because send-side BWE isn't used, in which case

View File

@ -119,6 +119,8 @@ class DEPRECATED_SendSideCongestionController
void EnableCongestionWindowPushback(int64_t accepted_queue_ms, void EnableCongestionWindowPushback(int64_t accepted_queue_ms,
uint32_t min_pushback_target_bitrate_bps); uint32_t min_pushback_target_bitrate_bps);
void SetAlrLimitedBackoffExperiment(bool enable);
private: private:
void MaybeTriggerOnNetworkChanged(); void MaybeTriggerOnNetworkChanged();

View File

@ -186,6 +186,12 @@ void DEPRECATED_SendSideCongestionController::EnableCongestionWindowPushback(
min_pushback_target_bitrate_bps); min_pushback_target_bitrate_bps);
} }
void DEPRECATED_SendSideCongestionController::SetAlrLimitedBackoffExperiment(
bool enable) {
rtc::CritScope cs(&bwe_lock_);
delay_based_bwe_->SetAlrLimitedBackoffExperiment(enable);
}
void DEPRECATED_SendSideCongestionController::RegisterPacketFeedbackObserver( void DEPRECATED_SendSideCongestionController::RegisterPacketFeedbackObserver(
PacketFeedbackObserver* observer) { PacketFeedbackObserver* observer) {
transport_feedback_adapter_.RegisterPacketFeedbackObserver(observer); transport_feedback_adapter_.RegisterPacketFeedbackObserver(observer);
@ -437,7 +443,7 @@ void DEPRECATED_SendSideCongestionController::OnTransportFeedback(
result = delay_based_bwe_->IncomingPacketFeedbackVector( result = delay_based_bwe_->IncomingPacketFeedbackVector(
feedback_vector, acknowledged_bitrate_estimator_->bitrate(), feedback_vector, acknowledged_bitrate_estimator_->bitrate(),
probe_bitrate_estimator_->FetchAndResetLastEstimatedBitrate(), probe_bitrate_estimator_->FetchAndResetLastEstimatedBitrate(),
Timestamp::ms(clock_->TimeInMilliseconds())); currently_in_alr, Timestamp::ms(clock_->TimeInMilliseconds()));
} }
if (result.updated) { if (result.updated) {
bitrate_controller_->OnDelayBasedBweResult(result); bitrate_controller_->OnDelayBasedBweResult(result);
@ -449,6 +455,9 @@ void DEPRECATED_SendSideCongestionController::OnTransportFeedback(
probe_controller_->SetAlrStartTimeMs( probe_controller_->SetAlrStartTimeMs(
pacer_->GetApplicationLimitedRegionStartTime()); pacer_->GetApplicationLimitedRegionStartTime());
SendProbes(probe_controller_->RequestProbe(clock_->TimeInMilliseconds())); SendProbes(probe_controller_->RequestProbe(clock_->TimeInMilliseconds()));
} else if (result.backoff_in_alr) {
rtc::CritScope cs(&probe_lock_);
SendProbes(probe_controller_->RequestProbe(clock_->TimeInMilliseconds()));
} }
if (in_cwnd_experiment_) { if (in_cwnd_experiment_) {
LimitOutstandingBytes(transport_feedback_adapter_.GetOutstandingBytes()); LimitOutstandingBytes(transport_feedback_adapter_.GetOutstandingBytes());

View File

@ -91,7 +91,7 @@ void SendSideBweSender::GiveFeedback(const FeedbackPacket& feedback) {
} }
DelayBasedBwe::Result result = bwe_->IncomingPacketFeedbackVector( DelayBasedBwe::Result result = bwe_->IncomingPacketFeedbackVector(
packet_feedback_vector, acknowledged_bitrate_estimator_->bitrate(), packet_feedback_vector, acknowledged_bitrate_estimator_->bitrate(),
probe_bitrate_estimator_->FetchAndResetLastEstimatedBitrate(), probe_bitrate_estimator_->FetchAndResetLastEstimatedBitrate(), false,
Timestamp::ms(clock_->TimeInMilliseconds())); Timestamp::ms(clock_->TimeInMilliseconds()));
if (result.updated) if (result.updated)
bitrate_controller_->OnDelayBasedBweResult(result); bitrate_controller_->OnDelayBasedBweResult(result);