/* * Copyright (c) 2016 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. */ #ifndef MODULES_CONGESTION_CONTROLLER_GOOG_CC_PROBE_CONTROLLER_H_ #define MODULES_CONGESTION_CONTROLLER_GOOG_CC_PROBE_CONTROLLER_H_ #include #include #include #include "absl/base/attributes.h" #include "absl/types/optional.h" #include "api/field_trials_view.h" #include "api/rtc_event_log/rtc_event_log.h" #include "api/transport/network_control.h" #include "api/transport/network_types.h" #include "api/units/data_rate.h" #include "api/units/timestamp.h" #include "rtc_base/experiments/field_trial_parser.h" namespace webrtc { struct ProbeControllerConfig { explicit ProbeControllerConfig(const FieldTrialsView* key_value_config); ProbeControllerConfig(const ProbeControllerConfig&); ProbeControllerConfig& operator=(const ProbeControllerConfig&) = default; ~ProbeControllerConfig(); // These parameters configure the initial probes. First we send one or two // probes of sizes p1 * start_bitrate_ and p2 * start_bitrate_. // Then whenever we get a bitrate estimate of at least further_probe_threshold // times the size of the last sent probe we'll send another one of size // step_size times the new estimate. FieldTrialParameter first_exponential_probe_scale; FieldTrialOptional second_exponential_probe_scale; FieldTrialParameter further_exponential_probe_scale; FieldTrialParameter further_probe_threshold; // Configures how often we send ALR probes and how big they are. FieldTrialParameter alr_probing_interval; FieldTrialParameter alr_probe_scale; // Configures how often we send probes if NetworkStateEstimate is available. FieldTrialParameter network_state_estimate_probing_interval; // Periodically probe as long as the the ratio beteeen current estimate and // NetworkStateEstimate is lower then this. FieldTrialParameter probe_if_estimate_lower_than_network_state_estimate_ratio; FieldTrialParameter estimate_lower_than_network_state_estimate_probing_interval; FieldTrialParameter network_state_probe_scale; // Overrides min_probe_duration if network_state_estimate_probing_interval // is set and a network state estimate is known. FieldTrialParameter network_state_probe_duration; // Configures the probes emitted by changed to the allocated bitrate. FieldTrialParameter probe_on_max_allocated_bitrate_change; FieldTrialOptional first_allocation_probe_scale; FieldTrialOptional second_allocation_probe_scale; FieldTrialFlag allocation_allow_further_probing; FieldTrialParameter allocation_probe_max; // The minimum number probing packets used. FieldTrialParameter min_probe_packets_sent; // The minimum probing duration. FieldTrialParameter min_probe_duration; // Periodically probe when bandwidth estimate is loss limited. FieldTrialParameter limit_probe_target_rate_to_loss_bwe; FieldTrialParameter loss_limited_probe_scale; // Dont send a probe if min(estimate, network state estimate) is larger than // this fraction of the set max bitrate. FieldTrialParameter skip_if_estimate_larger_than_fraction_of_max; // Do not send probes if network is either overusing or underusing. FieldTrialParameter not_probe_if_delay_increased; }; // Reason that bandwidth estimate is limited. Bandwidth estimate can be limited // by either delay based bwe, or loss based bwe when it increases/decreases the // estimate. enum class BandwidthLimitedCause { kLossLimitedBweIncreasing = 0, kLossLimitedBweDecreasing = 1, kDelayBasedLimited = 2, kDelayBasedLimitedDelayIncreased = 3, }; // This class controls initiation of probing to estimate initial channel // capacity. There is also support for probing during a session when max // bitrate is adjusted by an application. class ProbeController { public: explicit ProbeController(const FieldTrialsView* key_value_config, RtcEventLog* event_log); ~ProbeController(); ProbeController(const ProbeController&) = delete; ProbeController& operator=(const ProbeController&) = delete; ABSL_MUST_USE_RESULT std::vector SetBitrates( DataRate min_bitrate, DataRate start_bitrate, DataRate max_bitrate, Timestamp at_time); // The total bitrate, as opposed to the max bitrate, is the sum of the // configured bitrates for all active streams. ABSL_MUST_USE_RESULT std::vector OnMaxTotalAllocatedBitrate(DataRate max_total_allocated_bitrate, Timestamp at_time); ABSL_MUST_USE_RESULT std::vector OnNetworkAvailability( NetworkAvailability msg); ABSL_MUST_USE_RESULT std::vector SetEstimatedBitrate( DataRate bitrate, BandwidthLimitedCause bandwidth_limited_cause, Timestamp at_time); void EnablePeriodicAlrProbing(bool enable); void SetAlrStartTimeMs(absl::optional alr_start_time); void SetAlrEndedTimeMs(int64_t alr_end_time); ABSL_MUST_USE_RESULT std::vector RequestProbe( Timestamp at_time); void SetNetworkStateEstimate(webrtc::NetworkStateEstimate estimate); // Resets the ProbeController to a state equivalent to as if it was just // created EXCEPT for `enable_periodic_alr_probing_`. void Reset(Timestamp at_time); ABSL_MUST_USE_RESULT std::vector Process( Timestamp at_time); // Gets the value of field trial not_probe_if_delay_increased. bool DontProbeIfDelayIncreased() { return config_.not_probe_if_delay_increased; } private: enum class State { // Initial state where no probing has been triggered yet. kInit, // Waiting for probing results to continue further probing. kWaitingForProbingResult, // Probing is complete. kProbingComplete, }; ABSL_MUST_USE_RESULT std::vector InitiateExponentialProbing(Timestamp at_time); ABSL_MUST_USE_RESULT std::vector InitiateProbing( Timestamp now, std::vector bitrates_to_probe, bool probe_further); bool TimeForAlrProbe(Timestamp at_time) const; bool TimeForNetworkStateProbe(Timestamp at_time) const; bool network_available_; BandwidthLimitedCause bandwidth_limited_cause_ = BandwidthLimitedCause::kDelayBasedLimited; State state_; DataRate min_bitrate_to_probe_further_ = DataRate::PlusInfinity(); Timestamp time_last_probing_initiated_ = Timestamp::MinusInfinity(); DataRate estimated_bitrate_ = DataRate::Zero(); absl::optional network_estimate_; DataRate start_bitrate_ = DataRate::Zero(); DataRate max_bitrate_ = DataRate::PlusInfinity(); Timestamp last_bwe_drop_probing_time_ = Timestamp::Zero(); absl::optional alr_start_time_; absl::optional alr_end_time_; bool enable_periodic_alr_probing_; Timestamp time_of_last_large_drop_ = Timestamp::MinusInfinity(); DataRate bitrate_before_last_large_drop_ = DataRate::Zero(); DataRate max_total_allocated_bitrate_ = DataRate::Zero(); const bool in_rapid_recovery_experiment_; RtcEventLog* event_log_; int32_t next_probe_cluster_id_ = 1; ProbeControllerConfig config_; }; } // namespace webrtc #endif // MODULES_CONGESTION_CONTROLLER_GOOG_CC_PROBE_CONTROLLER_H_