/* * Copyright 2018 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/goog_cc/test/goog_cc_printer.h" #include #include #include "absl/types/optional.h" #include "modules/congestion_controller/goog_cc/alr_detector.h" #include "modules/congestion_controller/goog_cc/delay_based_bwe.h" #include "modules/congestion_controller/goog_cc/trendline_estimator.h" #include "modules/remote_bitrate_estimator/aimd_rate_control.h" #include "rtc_base/checks.h" namespace webrtc { namespace { void WriteTypedValue(RtcEventLogOutput* out, int value) { LogWriteFormat(out, "%i", value); } void WriteTypedValue(RtcEventLogOutput* out, double value) { LogWriteFormat(out, "%.6f", value); } void WriteTypedValue(RtcEventLogOutput* out, absl::optional value) { LogWriteFormat(out, "%.0f", value ? value->bytes_per_sec() : NAN); } void WriteTypedValue(RtcEventLogOutput* out, absl::optional value) { LogWriteFormat(out, "%.0f", value ? value->bytes() : NAN); } void WriteTypedValue(RtcEventLogOutput* out, absl::optional value) { LogWriteFormat(out, "%.3f", value ? value->seconds() : NAN); } void WriteTypedValue(RtcEventLogOutput* out, absl::optional value) { LogWriteFormat(out, "%.3f", value ? value->seconds() : NAN); } template class TypedFieldLogger : public FieldLogger { public: TypedFieldLogger(std::string name, F&& getter) : name_(std::move(name)), getter_(std::forward(getter)) {} const std::string& name() const override { return name_; } void WriteValue(RtcEventLogOutput* out) override { WriteTypedValue(out, getter_()); } private: std::string name_; F getter_; }; template FieldLogger* Log(std::string name, F&& getter) { return new TypedFieldLogger(std::move(name), std::forward(getter)); } } // namespace GoogCcStatePrinter::GoogCcStatePrinter() { for (auto* logger : CreateLoggers()) { loggers_.emplace_back(logger); } } std::deque GoogCcStatePrinter::CreateLoggers() { auto stable_estimate = [this] { return DataRate::KilobitsPerSec( controller_->delay_based_bwe_->rate_control_.link_capacity_ .estimate_kbps_.value_or(-INFINITY)); }; auto rate_control_state = [this] { return static_cast( controller_->delay_based_bwe_->rate_control_.rate_control_state_); }; auto trend = [this] { return reinterpret_cast( controller_->delay_based_bwe_->active_delay_detector_); }; auto acknowledged_rate = [this] { return controller_->acknowledged_bitrate_estimator_->bitrate(); }; auto loss_cont = [&] { return &controller_->bandwidth_estimation_ ->loss_based_bandwidth_estimation_; }; std::deque loggers({ Log("time", [=] { return target_.at_time; }), Log("rtt", [=] { return target_.network_estimate.round_trip_time; }), Log("target", [=] { return target_.target_rate; }), Log("stable_target", [=] { return target_.stable_target_rate; }), Log("pacing", [=] { return pacing_.data_rate(); }), Log("padding", [=] { return pacing_.pad_rate(); }), Log("window", [=] { return congestion_window_; }), Log("rate_control_state", [=] { return rate_control_state(); }), Log("stable_estimate", [=] { return stable_estimate(); }), Log("trendline", [=] { return trend()->prev_trend_; }), Log("trendline_modified_offset", [=] { return trend()->prev_modified_trend_; }), Log("trendline_offset_threshold", [=] { return trend()->threshold_; }), Log("acknowledged_rate", [=] { return acknowledged_rate(); }), Log("est_capacity", [=] { return est_.link_capacity; }), Log("est_capacity_dev", [=] { return est_.link_capacity_std_dev; }), Log("est_capacity_min", [=] { return est_.link_capacity_min; }), Log("est_cross_traffic", [=] { return est_.cross_traffic_ratio; }), Log("est_cross_delay", [=] { return est_.cross_delay_rate; }), Log("est_spike_delay", [=] { return est_.spike_delay_rate; }), Log("est_pre_buffer", [=] { return est_.pre_link_buffer_delay; }), Log("est_post_buffer", [=] { return est_.post_link_buffer_delay; }), Log("est_propagation", [=] { return est_.propagation_delay; }), Log("loss_ratio", [=] { return loss_cont()->last_loss_ratio_; }), Log("loss_average", [=] { return loss_cont()->average_loss_; }), Log("loss_average_max", [=] { return loss_cont()->average_loss_max_; }), Log("loss_thres_inc", [=] { return loss_cont()->loss_increase_threshold(); }), Log("loss_thres_dec", [=] { return loss_cont()->loss_decrease_threshold(); }), Log("loss_dec_rate", [=] { return loss_cont()->decreased_bitrate(); }), Log("loss_based_rate", [=] { return loss_cont()->loss_based_bitrate_; }), Log("loss_ack_rate", [=] { return loss_cont()->acknowledged_bitrate_max_; }), Log("data_window", [=] { return controller_->current_data_window_; }), Log("pushback_target", [=] { return controller_->last_pushback_target_rate_; }), }); return loggers; } GoogCcStatePrinter::~GoogCcStatePrinter() = default; void GoogCcStatePrinter::PrintHeaders(RtcEventLogOutput* log) { int ix = 0; for (const auto& logger : loggers_) { if (ix++) log->Write(" "); log->Write(logger->name()); } log->Write("\n"); log->Flush(); } void GoogCcStatePrinter::PrintState(RtcEventLogOutput* log, GoogCcNetworkController* controller, Timestamp at_time) { controller_ = controller; auto state_update = controller_->GetNetworkState(at_time); target_ = state_update.target_rate.value(); pacing_ = state_update.pacer_config.value(); if (state_update.congestion_window) congestion_window_ = *state_update.congestion_window; if (controller_->network_estimator_) { est_ = controller_->network_estimator_->GetCurrentEstimate().value_or( NetworkStateEstimate()); } int ix = 0; for (const auto& logger : loggers_) { if (ix++) log->Write(" "); logger->WriteValue(log); } log->Write("\n"); log->Flush(); } GoogCcDebugFactory::GoogCcDebugFactory() : GoogCcDebugFactory(GoogCcFactoryConfig()) {} GoogCcDebugFactory::GoogCcDebugFactory(GoogCcFactoryConfig config) : GoogCcNetworkControllerFactory(std::move(config)) {} std::unique_ptr GoogCcDebugFactory::Create( NetworkControllerConfig config) { RTC_CHECK(controller_ == nullptr); auto controller = GoogCcNetworkControllerFactory::Create(config); controller_ = static_cast(controller.get()); return controller; } void GoogCcDebugFactory::PrintState(const Timestamp at_time) { if (controller_ && log_writer_) { printer_.PrintState(log_writer_.get(), controller_, at_time); } } void GoogCcDebugFactory::AttachWriter( std::unique_ptr log_writer) { if (log_writer) { log_writer_ = std::move(log_writer); printer_.PrintHeaders(log_writer_.get()); } } } // namespace webrtc