blob: e73e5ff86f654dab8f24bb9eda32a989e4fda4e6 [file] [log] [blame]
/*
* 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 = 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