Source code

Revision control

Copy as Markdown

Other Tools

/*
* Copyright (c) 2025 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/congestion_controller/scream/scream_network_controller.h"
#include <algorithm>
#include <optional>
#include <utility>
#include "api/transport/network_control.h"
#include "api/transport/network_types.h"
#include "api/units/data_rate.h"
#include "api/units/time_delta.h"
#include "api/units/timestamp.h"
#include "modules/congestion_controller/scream/scream_v2.h"
#include "rtc_base/logging.h"
namespace webrtc {
static constexpr DataRate kDefaultStartRate = DataRate::KilobitsPerSec(300);
ScreamNetworkController::ScreamNetworkController(NetworkControllerConfig config)
: env_(config.env),
params_(env_.field_trials()),
default_pacing_window_(config.default_pacing_time_window),
current_pacing_window_(config.default_pacing_time_window),
scream_(std::in_place, env_),
target_rate_constraints_(config.constraints),
last_padding_interval_started_(Timestamp::Zero()) {
if (config.constraints.min_data_rate.has_value() ||
config.constraints.max_data_rate.has_value()) {
scream_->SetTargetBitrateConstraints(
config.constraints.min_data_rate.value_or(DataRate::Zero()),
config.constraints.max_data_rate.value_or(DataRate::PlusInfinity()));
}
}
NetworkControlUpdate ScreamNetworkController::OnNetworkAvailability(
NetworkAvailability msg) {
RTC_LOG(LS_INFO) << " OnNetworkAvailability network_available:"
<< msg.network_available;
if (msg.network_available) {
// TODO: bugs.webrtc.org/447037083 - rtt must currently be set on every
// update. But here it is not yet known.
return CreateUpdate(
msg.at_time,
target_rate_constraints_.starting_rate.value_or(kDefaultStartRate),
/*rtt=*/TimeDelta::Zero());
}
return NetworkControlUpdate();
}
NetworkControlUpdate ScreamNetworkController::OnNetworkRouteChange(
NetworkRouteChange msg) {
RTC_LOG(LS_INFO) << " OnNetworkRouteChange, resetting ScreamV2.";
target_rate_constraints_ = msg.constraints;
scream_.emplace(env_);
// TODO: bugs.webrtc.org/447037083 - We should use the minimum rate from
// constraints, REMB and remote network state estimates.
scream_->SetTargetBitrateConstraints(
target_rate_constraints_.min_data_rate.value_or(DataRate::Zero()),
target_rate_constraints_.max_data_rate.value_or(
DataRate::PlusInfinity()));
// TODO: bugs.webrtc.org/447037083 - rtt must currently be set on every
// update. But here it is not yet known.
return CreateUpdate(
msg.at_time,
target_rate_constraints_.starting_rate.value_or(kDefaultStartRate),
/*rtt=*/TimeDelta::Zero());
}
NetworkControlUpdate ScreamNetworkController::OnProcessInterval(
ProcessInterval msg) {
// Scream currently has no need for periodic processing.
return NetworkControlUpdate();
}
NetworkControlUpdate ScreamNetworkController::OnRemoteBitrateReport(
RemoteBitrateReport msg) {
// TODO: bugs.webrtc.org/447037083 - Implement;
return NetworkControlUpdate();
}
NetworkControlUpdate ScreamNetworkController::OnRoundTripTimeUpdate(
RoundTripTimeUpdate msg) {
// Scream uses Smoothed RTT from TransportFeedback.
return NetworkControlUpdate();
}
NetworkControlUpdate ScreamNetworkController::OnSentPacket(SentPacket msg) {
// Scream does not have to know about sent packets.
return NetworkControlUpdate();
}
NetworkControlUpdate ScreamNetworkController::OnReceivedPacket(
ReceivedPacket msg) {
// Scream does not have to know about received packets.
return NetworkControlUpdate();
}
NetworkControlUpdate ScreamNetworkController::OnStreamsConfig(
StreamsConfig msg) {
streams_config_ = msg;
return NetworkControlUpdate();
}
NetworkControlUpdate ScreamNetworkController::OnTargetRateConstraints(
TargetRateConstraints msg) {
target_rate_constraints_ = msg;
// TODO: bugs.webrtc.org/447037083 - We should use the minimum rate from
// constraints, REMB and remote network state estimates.
scream_->SetTargetBitrateConstraints(
target_rate_constraints_.min_data_rate.value_or(DataRate::Zero()),
target_rate_constraints_.max_data_rate.value_or(
DataRate::PlusInfinity()));
// No need to change target rate immediately. Wait until next feedback.
return NetworkControlUpdate();
}
NetworkControlUpdate ScreamNetworkController::OnTransportLossReport(
TransportLossReport msg) {
return NetworkControlUpdate();
}
NetworkControlUpdate ScreamNetworkController::OnNetworkStateEstimate(
NetworkStateEstimate msg) {
// TODO: bugs.webrtc.org/447037083 - Implement;
RTC_LOG_F(LS_INFO) << "Not implemented";
return NetworkControlUpdate();
}
NetworkControlUpdate ScreamNetworkController::OnTransportPacketsFeedback(
TransportPacketsFeedback msg) {
DataRate target_rate = scream_->OnTransportPacketsFeedback(msg);
return CreateUpdate(msg.feedback_time, target_rate, msg.smoothed_rtt);
}
NetworkControlUpdate ScreamNetworkController::CreateUpdate(Timestamp now,
DataRate target_rate,
TimeDelta rtt) {
TargetTransferRate target_rate_msg;
target_rate_msg.at_time = now;
target_rate_msg.target_rate = target_rate;
target_rate_msg.network_estimate.at_time = now;
target_rate_msg.network_estimate.round_trip_time = rtt;
// TODO: bugs.webrtc.org/447037083 - bwe_period must currently be set but
// it seems like it is not used for anything sensible. Try to remove it.
target_rate_msg.network_estimate.bwe_period = TimeDelta::Millis(25);
NetworkControlUpdate update;
update.target_rate = target_rate_msg;
update.pacer_config = CreatePacerConfig(target_rate);
return update;
}
PacerConfig ScreamNetworkController::CreatePacerConfig(DataRate target_rate) {
constexpr double kPacingRateFactor = 1.5;
// Time window used for calculating pacing window if target rate is
// constrained by CE markings.
constexpr TimeDelta kReducedPacingWindow = TimeDelta::Millis(10);
// Threshold used for guessing if target rate is constrained due to CE
// marking.
constexpr double kL4sAlphaThreshold = 0.01;
DataRate max_needed_rate =
streams_config_.max_total_allocated_bitrate.value_or(DataRate::Zero());
DataRate padding_rate = DataRate::Zero();
Timestamp now = env_.clock().CurrentTime();
if (target_rate < max_needed_rate * kPacingRateFactor &&
target_rate < target_rate_constraints_.max_data_rate.value_or(
DataRate::PlusInfinity())) {
// Periodically allow padding to be used to reach a target rate close to
// kPacingRateFactor*max_needed_rate.
if (params_.periodic_padding_interval->IsFinite() &&
(now - last_padding_interval_started_ >
params_.periodic_padding_interval.Get())) {
last_padding_interval_started_ = now;
}
if (now - last_padding_interval_started_ <
params_.periodic_padding_duration.Get()) {
padding_rate = target_rate;
}
}
if (current_pacing_window_ == default_pacing_window_ &&
target_rate < max_needed_rate &&
scream_->l4s_alpha() > kL4sAlphaThreshold) {
// Do stricter pacing if target rate is lower than what is needed and it
// seems like L4S is enabled. Note that once stricter pacing is enabled, it
// is not stopped.
current_pacing_window_ =
std::min(default_pacing_window_, kReducedPacingWindow);
}
return PacerConfig::Create(now, target_rate * kPacingRateFactor, padding_rate,
current_pacing_window_);
}
} // namespace webrtc