| /* |
| * Copyright (c) 2011 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/video_coding/timing/timing.h" |
| |
| #include <algorithm> |
| #include <cstdint> |
| #include <memory> |
| #include <optional> |
| |
| #include "api/field_trials_view.h" |
| #include "api/units/time_delta.h" |
| #include "api/units/timestamp.h" |
| #include "api/video/video_frame.h" |
| #include "api/video/video_timing.h" |
| #include "modules/video_coding/timing/decode_time_percentile_filter.h" |
| #include "modules/video_coding/timing/timestamp_extrapolator.h" |
| #include "rtc_base/checks.h" |
| #include "rtc_base/experiments/field_trial_parser.h" |
| #include "rtc_base/logging.h" |
| #include "rtc_base/synchronization/mutex.h" |
| #include "system_wrappers/include/clock.h" |
| |
| namespace webrtc { |
| namespace { |
| |
| // Default pacing that is used for the low-latency renderer path. |
| constexpr TimeDelta kZeroPlayoutDelayDefaultMinPacing = TimeDelta::Millis(8); |
| constexpr TimeDelta kLowLatencyStreamMaxPlayoutDelayThreshold = |
| TimeDelta::Millis(500); |
| |
| void CheckDelaysValid(TimeDelta min_delay, TimeDelta max_delay) { |
| if (min_delay > max_delay) { |
| RTC_LOG(LS_ERROR) |
| << "Playout delays set incorrectly: min playout delay (" << min_delay |
| << ") > max playout delay (" << max_delay |
| << "). This is undefined behaviour. Application writers should " |
| "ensure that the min delay is always less than or equals max " |
| "delay. If trying to use the playout delay header extensions " |
| "described in " |
| "https://webrtc.googlesource.com/src/+/refs/heads/main/docs/" |
| "native-code/rtp-hdrext/playout-delay/, be careful that a playout " |
| "delay hint or A/V sync settings may have caused this conflict."; |
| } |
| } |
| |
| } // namespace |
| |
| void VCMTiming::VideoDelayTimings::Reset() { |
| minimum_delay = TimeDelta::Zero(); |
| estimated_max_decode_time = TimeDelta::Zero(); |
| render_delay = kDefaultRenderDelay; |
| min_playout_delay = TimeDelta::Zero(); |
| target_delay = TimeDelta::Zero(); |
| current_delay = TimeDelta::Zero(); |
| } |
| |
| TimeDelta VCMTiming::VideoDelayTimings::TargetDelay() const { |
| return std::max(min_playout_delay, |
| minimum_delay + estimated_max_decode_time + render_delay); |
| } |
| |
| TimeDelta VCMTiming::VideoDelayTimings::StatsTargetDelay() const { |
| TimeDelta stats_target_delay = |
| TargetDelay() - (estimated_max_decode_time + render_delay); |
| return std::max(TimeDelta::Zero(), stats_target_delay); |
| } |
| |
| bool VCMTiming::VideoDelayTimings::UseLowLatencyRendering() const { |
| return min_playout_delay.IsZero() && |
| max_playout_delay <= kLowLatencyStreamMaxPlayoutDelayThreshold; |
| } |
| |
| VCMTiming::VCMTiming(Clock* clock, const FieldTrialsView& field_trials) |
| : clock_(clock), |
| ts_extrapolator_( |
| std::make_unique<TimestampExtrapolator>(clock_->CurrentTime(), |
| field_trials)), |
| decode_time_filter_(std::make_unique<DecodeTimePercentileFilter>()), |
| zero_playout_delay_min_pacing_("min_pacing", |
| kZeroPlayoutDelayDefaultMinPacing), |
| last_decode_scheduled_(Timestamp::Zero()) { |
| ParseFieldTrial({&zero_playout_delay_min_pacing_}, |
| field_trials.Lookup("WebRTC-ZeroPlayoutDelay")); |
| } |
| |
| void VCMTiming::Reset() { |
| MutexLock lock(&mutex_); |
| ts_extrapolator_->Reset(clock_->CurrentTime()); |
| decode_time_filter_ = std::make_unique<DecodeTimePercentileFilter>(); |
| timings_.Reset(); |
| } |
| |
| void VCMTiming::set_render_delay(TimeDelta render_delay) { |
| MutexLock lock(&mutex_); |
| timings_.render_delay = render_delay; |
| } |
| |
| TimeDelta VCMTiming::min_playout_delay() const { |
| MutexLock lock(&mutex_); |
| return timings_.min_playout_delay; |
| } |
| |
| void VCMTiming::set_min_playout_delay(TimeDelta min_playout_delay) { |
| MutexLock lock(&mutex_); |
| if (timings_.min_playout_delay != min_playout_delay) { |
| CheckDelaysValid(min_playout_delay, timings_.max_playout_delay); |
| timings_.min_playout_delay = min_playout_delay; |
| } |
| } |
| |
| void VCMTiming::set_playout_delay(const VideoPlayoutDelay& playout_delay) { |
| MutexLock lock(&mutex_); |
| // No need to call `CheckDelaysValid` as the same invariant (min <= max) |
| // is guaranteed by the `VideoPlayoutDelay` type. |
| timings_.min_playout_delay = playout_delay.min(); |
| timings_.max_playout_delay = playout_delay.max(); |
| } |
| |
| void VCMTiming::SetMinimumDelay(TimeDelta minimum_delay) { |
| MutexLock lock(&mutex_); |
| if (minimum_delay != timings_.minimum_delay) { |
| timings_.minimum_delay = minimum_delay; |
| // When in initial state, set current delay to minimum delay. |
| if (timings_.current_delay.IsZero()) { |
| timings_.current_delay = timings_.minimum_delay; |
| } |
| } |
| } |
| |
| void VCMTiming::UpdateCurrentDelay(Timestamp render_time, |
| Timestamp actual_decode_time) { |
| MutexLock lock(&mutex_); |
| TimeDelta target_delay = timings_.TargetDelay(); |
| TimeDelta delayed = (actual_decode_time - render_time) + |
| timings_.estimated_max_decode_time + |
| timings_.render_delay; |
| |
| // Only consider `delayed` as negative by more than a few microseconds. |
| if (delayed.ms() < 0) { |
| return; |
| } |
| if (timings_.current_delay + delayed <= target_delay) { |
| timings_.current_delay += delayed; |
| } else { |
| timings_.current_delay = target_delay; |
| } |
| } |
| |
| void VCMTiming::StopDecodeTimer(TimeDelta decode_time, Timestamp now) { |
| MutexLock lock(&mutex_); |
| RTC_DCHECK_GE(decode_time, TimeDelta::Zero()); |
| decode_time_filter_->AddSample(decode_time.ms(), now.ms()); |
| ++timings_.num_decoded_frames; |
| timings_.estimated_max_decode_time = |
| TimeDelta::Millis(decode_time_filter_->GetPercentileMs()); |
| } |
| |
| void VCMTiming::OnCompleteTemporalUnit(uint32_t rtp_timestamp, Timestamp now) { |
| MutexLock lock(&mutex_); |
| ts_extrapolator_->Update(now, rtp_timestamp); |
| } |
| |
| Timestamp VCMTiming::RenderTime(uint32_t rtp_timestamp, Timestamp now) const { |
| MutexLock lock(&mutex_); |
| if (timings_.UseLowLatencyRendering()) { |
| // Render as soon as possible or with low-latency renderer algorithm. |
| return Timestamp::Zero(); |
| } |
| std::optional<Timestamp> local_time = |
| ts_extrapolator_->ExtrapolateLocalTime(rtp_timestamp); |
| if (!local_time.has_value()) { |
| return now; |
| } |
| return *local_time + std::clamp(timings_.current_delay, |
| timings_.min_playout_delay, |
| timings_.max_playout_delay); |
| } |
| |
| void VCMTiming::SetLastDecodeScheduledTimestamp( |
| Timestamp last_decode_scheduled) { |
| MutexLock lock(&mutex_); |
| last_decode_scheduled_ = last_decode_scheduled; |
| } |
| |
| TimeDelta VCMTiming::MaxWaitingTime(Timestamp render_time, |
| Timestamp now, |
| bool too_many_frames_queued) const { |
| MutexLock lock(&mutex_); |
| |
| if (render_time.IsZero() && zero_playout_delay_min_pacing_->us() > 0 && |
| timings_.min_playout_delay.IsZero() && |
| timings_.max_playout_delay > TimeDelta::Zero()) { |
| // `render_time` == 0 indicates that the frame should be decoded and |
| // rendered as soon as possible. However, the decoder can be choked if too |
| // many frames are sent at once. Therefore, limit the interframe delay to |
| // `zero_playout_delay_min_pacing_` unless too many frames are queued in |
| // which case the frames are sent to the decoder at once. |
| if (too_many_frames_queued) { |
| return TimeDelta::Zero(); |
| } |
| Timestamp earliest_next_decode_start_time = |
| last_decode_scheduled_ + zero_playout_delay_min_pacing_; |
| TimeDelta max_wait_time = now >= earliest_next_decode_start_time |
| ? TimeDelta::Zero() |
| : earliest_next_decode_start_time - now; |
| return max_wait_time; |
| } |
| return render_time - now - timings_.estimated_max_decode_time - |
| timings_.render_delay; |
| } |
| |
| TimeDelta VCMTiming::TargetVideoDelay() const { |
| MutexLock lock(&mutex_); |
| return timings_.TargetDelay(); |
| } |
| |
| VCMTiming::VideoDelayTimings VCMTiming::GetTimings() const { |
| MutexLock lock(&mutex_); |
| VideoDelayTimings timings = timings_; |
| timings.target_delay = timings_.StatsTargetDelay(); |
| return timings; |
| } |
| |
| VideoFrame::RenderParameters VCMTiming::RenderParameters() const { |
| MutexLock lock(&mutex_); |
| return {.use_low_latency_rendering = timings_.UseLowLatencyRendering(), |
| .max_composition_delay_in_frames = max_composition_delay_in_frames_}; |
| } |
| |
| void VCMTiming::SetMaxCompositionDelayInFrames( |
| std::optional<int> max_composition_delay_in_frames) { |
| MutexLock lock(&mutex_); |
| max_composition_delay_in_frames_ = max_composition_delay_in_frames; |
| } |
| |
| } // namespace webrtc |