blob: ae7fe6ebe402f486387c39482954899083a0759a [file] [log] [blame]
Chen Xing05f8f1d2019-08-22 08:05:011/*
2 * Copyright (c) 2019 The WebRTC project authors. All Rights Reserved.
3 *
4 * Use of this source code is governed by a BSD-style license
5 * that can be found in the LICENSE file in the root of the source
6 * tree. An additional intellectual property rights grant can be found
7 * in the file PATENTS. All contributing project authors may
8 * be found in the AUTHORS file in the root of the source tree.
9 */
10
11#include "modules/rtp_rtcp/source/absolute_capture_time_sender.h"
12
13#include <limits>
14
Minyue Li6e65f6a2021-06-21 13:13:1415#include "modules/rtp_rtcp/source/absolute_capture_time_interpolator.h"
Chen Xing05f8f1d2019-08-22 08:05:0116#include "system_wrappers/include/ntp_time.h"
17
18namespace webrtc {
Chen Xing05f8f1d2019-08-22 08:05:0119
20static_assert(
Minyue Li6e65f6a2021-06-21 13:13:1421 AbsoluteCaptureTimeInterpolator::kInterpolationMaxInterval >=
Chen Xing05f8f1d2019-08-22 08:05:0122 AbsoluteCaptureTimeSender::kInterpolationMaxInterval,
23 "Receivers should be as willing to interpolate timestamps as senders.");
24
25AbsoluteCaptureTimeSender::AbsoluteCaptureTimeSender(Clock* clock)
Danil Chapovalov6634c912023-10-26 09:46:0226 : clock_(clock) {}
Chen Xing05f8f1d2019-08-22 08:05:0127
28uint32_t AbsoluteCaptureTimeSender::GetSource(
29 uint32_t ssrc,
30 rtc::ArrayView<const uint32_t> csrcs) {
Minyue Li6e65f6a2021-06-21 13:13:1431 return AbsoluteCaptureTimeInterpolator::GetSource(ssrc, csrcs);
Chen Xing05f8f1d2019-08-22 08:05:0132}
33
Florent Castelli8037fc62024-08-29 13:00:4034std::optional<AbsoluteCaptureTime> AbsoluteCaptureTimeSender::OnSendPacket(
Chen Xing05f8f1d2019-08-22 08:05:0135 uint32_t source,
36 uint32_t rtp_timestamp,
37 uint32_t rtp_clock_frequency,
38 uint64_t absolute_capture_timestamp,
Florent Castelli8037fc62024-08-29 13:00:4039 std::optional<int64_t> estimated_capture_clock_offset) {
Danil Chapovalov6634c912023-10-26 09:46:0240 return OnSendPacket(source, rtp_timestamp, rtp_clock_frequency,
41 NtpTime(absolute_capture_timestamp),
42 estimated_capture_clock_offset, /*force=*/false);
43}
Chen Xing05f8f1d2019-08-22 08:05:0144
Florent Castelli8037fc62024-08-29 13:00:4045std::optional<AbsoluteCaptureTime> AbsoluteCaptureTimeSender::OnSendPacket(
Danil Chapovalov6634c912023-10-26 09:46:0246 uint32_t source,
47 uint32_t rtp_timestamp,
48 int rtp_clock_frequency_hz,
49 NtpTime absolute_capture_time,
Florent Castelli8037fc62024-08-29 13:00:4050 std::optional<int64_t> estimated_capture_clock_offset,
Danil Chapovalov6634c912023-10-26 09:46:0251 bool force) {
52 Timestamp send_time = clock_->CurrentTime();
53 if (!(force || ShouldSendExtension(
54 send_time, source, rtp_timestamp, rtp_clock_frequency_hz,
55 absolute_capture_time, estimated_capture_clock_offset))) {
Florent Castelli8037fc62024-08-29 13:00:4056 return std::nullopt;
Chen Xing05f8f1d2019-08-22 08:05:0157 }
58
59 last_source_ = source;
60 last_rtp_timestamp_ = rtp_timestamp;
Danil Chapovalov6634c912023-10-26 09:46:0261 last_rtp_clock_frequency_hz_ = rtp_clock_frequency_hz;
62 last_absolute_capture_time_ = absolute_capture_time;
Chen Xing05f8f1d2019-08-22 08:05:0163 last_estimated_capture_clock_offset_ = estimated_capture_clock_offset;
Chen Xing05f8f1d2019-08-22 08:05:0164 last_send_time_ = send_time;
65
Danil Chapovalov6634c912023-10-26 09:46:0266 return AbsoluteCaptureTime{
67 .absolute_capture_timestamp = uint64_t{absolute_capture_time},
68 .estimated_capture_clock_offset = estimated_capture_clock_offset,
69 };
Chen Xing05f8f1d2019-08-22 08:05:0170}
71
72bool AbsoluteCaptureTimeSender::ShouldSendExtension(
73 Timestamp send_time,
74 uint32_t source,
75 uint32_t rtp_timestamp,
Danil Chapovalov6634c912023-10-26 09:46:0276 int rtp_clock_frequency_hz,
77 NtpTime absolute_capture_time,
Florent Castelli8037fc62024-08-29 13:00:4078 std::optional<int64_t> estimated_capture_clock_offset) const {
Danil Chapovalov6634c912023-10-26 09:46:0279 // Should if the last sent extension is too old, in particular if we've never
80 // sent anything before.
81 if (send_time - last_send_time_ > kInterpolationMaxInterval) {
Chen Xing05f8f1d2019-08-22 08:05:0182 return true;
83 }
84
85 // Should if the source has changed.
86 if (last_source_ != source) {
87 return true;
88 }
89
90 // Should if the RTP clock frequency has changed.
Danil Chapovalov6634c912023-10-26 09:46:0291 if (last_rtp_clock_frequency_hz_ != rtp_clock_frequency_hz) {
Chen Xing05f8f1d2019-08-22 08:05:0192 return true;
93 }
94
95 // Should if the RTP clock frequency is invalid.
Danil Chapovalov6634c912023-10-26 09:46:0296 if (rtp_clock_frequency_hz <= 0) {
Chen Xing05f8f1d2019-08-22 08:05:0197 return true;
98 }
99
100 // Should if the estimated capture clock offset has changed.
101 if (last_estimated_capture_clock_offset_ != estimated_capture_clock_offset) {
102 return true;
103 }
104
105 // Should if interpolation would introduce too much error.
106 const uint64_t interpolated_absolute_capture_timestamp =
Minyue Li6e65f6a2021-06-21 13:13:14107 AbsoluteCaptureTimeInterpolator::InterpolateAbsoluteCaptureTimestamp(
Danil Chapovalov6634c912023-10-26 09:46:02108 rtp_timestamp, rtp_clock_frequency_hz, last_rtp_timestamp_,
109 uint64_t{last_absolute_capture_time_});
110 const uint64_t absolute_capture_timestamp = uint64_t{absolute_capture_time};
Chen Xing05f8f1d2019-08-22 08:05:01111 const int64_t interpolation_error_ms = UQ32x32ToInt64Ms(std::min(
112 interpolated_absolute_capture_timestamp - absolute_capture_timestamp,
113 absolute_capture_timestamp - interpolated_absolute_capture_timestamp));
114 if (interpolation_error_ms > kInterpolationMaxError.ms()) {
115 return true;
116 }
117
118 return false;
119}
120
121} // namespace webrtc