/* * Copyright (c) 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 "call/receive_time_calculator.h" #include #include #include #include "rtc_base/experiments/field_trial_parser.h" #include "rtc_base/numerics/safe_minmax.h" namespace webrtc { namespace { const char kBweReceiveTimeCorrection[] = "WebRTC-Bwe-ReceiveTimeFix"; } // namespace ReceiveTimeCalculatorConfig::ReceiveTimeCalculatorConfig( const FieldTrialsView& field_trials) : max_packet_time_repair("maxrep", TimeDelta::Millis(2000)), stall_threshold("stall", TimeDelta::Millis(5)), tolerance("tol", TimeDelta::Millis(1)), max_stall("maxstall", TimeDelta::Seconds(5)) { std::string trial_string = field_trials.Lookup(kBweReceiveTimeCorrection); ParseFieldTrial( {&max_packet_time_repair, &stall_threshold, &tolerance, &max_stall}, trial_string); } ReceiveTimeCalculatorConfig::ReceiveTimeCalculatorConfig( const ReceiveTimeCalculatorConfig&) = default; ReceiveTimeCalculatorConfig::~ReceiveTimeCalculatorConfig() = default; ReceiveTimeCalculator::ReceiveTimeCalculator( const FieldTrialsView& field_trials) : config_(field_trials) {} std::unique_ptr ReceiveTimeCalculator::CreateFromFieldTrial( const FieldTrialsView& field_trials) { if (!field_trials.IsEnabled(kBweReceiveTimeCorrection)) return nullptr; return std::make_unique(field_trials); } int64_t ReceiveTimeCalculator::ReconcileReceiveTimes(int64_t packet_time_us, int64_t system_time_us, int64_t safe_time_us) { int64_t stall_time_us = system_time_us - packet_time_us; if (total_system_time_passed_us_ < config_.stall_threshold->us()) { stall_time_us = rtc::SafeMin(stall_time_us, config_.max_stall->us()); } int64_t corrected_time_us = safe_time_us - stall_time_us; if (last_packet_time_us_ == -1 && stall_time_us < 0) { static_clock_offset_us_ = stall_time_us; corrected_time_us += static_clock_offset_us_; } else if (last_packet_time_us_ > 0) { // All repairs depend on variables being intialized int64_t packet_time_delta_us = packet_time_us - last_packet_time_us_; int64_t system_time_delta_us = system_time_us - last_system_time_us_; int64_t safe_time_delta_us = safe_time_us - last_safe_time_us_; // Repair backwards clock resets during initial stall. In this case, the // reset is observed only in packet time but never in system time. if (system_time_delta_us < 0) total_system_time_passed_us_ += config_.stall_threshold->us(); else total_system_time_passed_us_ += system_time_delta_us; if (packet_time_delta_us < 0 && total_system_time_passed_us_ < config_.stall_threshold->us()) { static_clock_offset_us_ -= packet_time_delta_us; } corrected_time_us += static_clock_offset_us_; // Detect resets inbetween clock readings in socket and app. bool forward_clock_reset = corrected_time_us + config_.tolerance->us() < last_corrected_time_us_; bool obvious_backward_clock_reset = system_time_us < packet_time_us; // Harder case with backward clock reset during stall, the reset being // smaller than the stall. Compensate throughout the stall. bool small_backward_clock_reset = !obvious_backward_clock_reset && safe_time_delta_us > system_time_delta_us + config_.tolerance->us(); bool stall_start = packet_time_delta_us >= 0 && system_time_delta_us > packet_time_delta_us + config_.tolerance->us(); bool stall_is_over = safe_time_delta_us > config_.stall_threshold->us(); bool packet_time_caught_up = packet_time_delta_us < 0 && system_time_delta_us >= 0; if (stall_start && small_backward_clock_reset) small_reset_during_stall_ = true; else if (stall_is_over || packet_time_caught_up) small_reset_during_stall_ = false; // If resets are detected, advance time by (capped) packet time increase. if (forward_clock_reset || obvious_backward_clock_reset || small_reset_during_stall_) { corrected_time_us = last_corrected_time_us_ + rtc::SafeClamp(packet_time_delta_us, 0, config_.max_packet_time_repair->us()); } } last_corrected_time_us_ = corrected_time_us; last_packet_time_us_ = packet_time_us; last_system_time_us_ = system_time_us; last_safe_time_us_ = safe_time_us; return corrected_time_us; } } // namespace webrtc