| /* | 
 |  *  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 <cstdint> | 
 | #include <memory> | 
 | #include <string> | 
 |  | 
 | #include "api/field_trials_view.h" | 
 | #include "api/units/time_delta.h" | 
 | #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> | 
 | ReceiveTimeCalculator::CreateFromFieldTrial( | 
 |     const FieldTrialsView& field_trials) { | 
 |   if (!field_trials.IsEnabled(kBweReceiveTimeCorrection)) | 
 |     return nullptr; | 
 |   return std::make_unique<ReceiveTimeCalculator>(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 = 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_ + | 
 |                           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 |