Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(425)

Unified Diff: media/cast/rtcp/rtcp.cc

Issue 280993002: [Cast] Repair receiver playout time calculations and frame skip logic. (Closed) Base URL: svn://svn.chromium.org/chrome/trunk/src
Patch Set: Created 6 years, 7 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View side-by-side diff with in-line comments
Download patch
Index: media/cast/rtcp/rtcp.cc
diff --git a/media/cast/rtcp/rtcp.cc b/media/cast/rtcp/rtcp.cc
index 0944594dceb0902590fedd92f67fee1c84bc575f..badfe131f07926e5f534febd2ff2d9ef1b49e47c 100644
--- a/media/cast/rtcp/rtcp.cc
+++ b/media/cast/rtcp/rtcp.cc
@@ -19,10 +19,7 @@ namespace media {
namespace cast {
static const int kMaxRttMs = 10000; // 10 seconds.
-static const uint16 kMaxDelay = 2000;
-
-// Time limit for received RTCP messages when we stop using it for lip-sync.
-static const int64 kMaxDiffSinceReceivedRtcpMs = 100000; // 100 seconds.
+static const int kMaxDelay = 2000;
class LocalRtcpRttFeedback : public RtcpRttFeedback {
public:
@@ -130,10 +127,8 @@ Rtcp::Rtcp(scoped_refptr<CastEnvironment> cast_environment,
receiver_feedback_(new LocalRtcpReceiverFeedback(this, cast_environment)),
rtcp_sender_(new RtcpSender(cast_environment, paced_packet_sender,
local_ssrc, c_name)),
- last_report_received_(0),
- last_received_rtp_timestamp_(0),
- last_received_ntp_seconds_(0),
- last_received_ntp_fraction_(0),
+ last_report_ntp_time_(0),
+ lip_sync_rtp_timestamp_(0),
min_rtt_(base::TimeDelta::FromMilliseconds(kMaxRttMs)),
number_of_rtt_in_avg_(0),
is_audio_(is_audio) {
@@ -219,7 +214,7 @@ void Rtcp::SendRtcpFromRtpReceiver(
&report_block.extended_high_sequence_number, &report_block.jitter);
}
- report_block.last_sr = last_report_received_;
+ report_block.last_sr = GetTruncatedLastNtpReportTime();
if (!time_last_report_received_.is_null()) {
uint32 delay_seconds = 0;
uint32 delay_fraction = 0;
@@ -257,7 +252,7 @@ void Rtcp::SendRtcpFromRtpSender(
transport::RtcpDlrrReportBlock dlrr;
if (!time_last_report_received_.is_null()) {
packet_type_flags |= transport::kRtcpDlrr;
- dlrr.last_rr = last_report_received_;
+ dlrr.last_rr = GetTruncatedLastNtpReportTime();
uint32 delay_seconds = 0;
uint32 delay_fraction = 0;
base::TimeDelta delta = now - time_last_report_received_;
@@ -274,17 +269,15 @@ void Rtcp::SendRtcpFromRtpSender(
}
void Rtcp::OnReceivedNtp(uint32 ntp_seconds, uint32 ntp_fraction) {
- last_report_received_ = (ntp_seconds << 16) + (ntp_fraction >> 16);
-
- base::TimeTicks now = cast_environment_->Clock()->NowTicks();
- time_last_report_received_ = now;
+ time_last_report_received_ = cast_environment_->Clock()->NowTicks();
+ last_report_ntp_time_ =
+ (static_cast<uint64>(ntp_seconds) << 32) | ntp_fraction;
}
void Rtcp::OnReceivedLipSyncInfo(uint32 rtp_timestamp, uint32 ntp_seconds,
uint32 ntp_fraction) {
- last_received_rtp_timestamp_ = rtp_timestamp;
- last_received_ntp_seconds_ = ntp_seconds;
- last_received_ntp_fraction_ = ntp_fraction;
+ lip_sync_rtp_timestamp_ = rtp_timestamp;
+ lip_sync_capture_time_ = ToApproximateLocalTime(ntp_seconds, ntp_fraction);
}
void Rtcp::OnReceivedSendReportRequest() {
@@ -294,38 +287,41 @@ void Rtcp::OnReceivedSendReportRequest() {
next_time_to_send_rtcp_ = now;
}
-bool Rtcp::RtpTimestampInSenderTime(int frequency, uint32 rtp_timestamp,
- base::TimeTicks* rtp_timestamp_in_ticks)
- const {
- if (last_received_ntp_seconds_ == 0)
- return false;
-
- int wrap = CheckForWrapAround(rtp_timestamp, last_received_rtp_timestamp_);
- int64 rtp_timestamp_int64 = rtp_timestamp;
- int64 last_received_rtp_timestamp_int64 = last_received_rtp_timestamp_;
+base::TimeTicks Rtcp::ToApproximateLocalTime(uint32 remote_ntp_seconds,
+ uint32 remote_ntp_fraction) const {
+ // Determine the clock difference between the local clock and the remote
+ // clock. |last_report_ntp_time_| is assumed to be an accurate timestamp from
+ // the remote clock that was snapshotted "just before" the
+ // |time_last_report_received_| was snapshotted from the local clock.
+ //
+ // TODO(miu): This does not account for packet transmit latency across the
+ // network, which may or may not be siginifcant for our use cases.
+ if (time_last_report_received_.is_null())
+ return base::TimeTicks();
+ const base::TimeDelta local_clock_ahead_by = time_last_report_received_ -
+ ConvertNtpToTimeTicks(static_cast<uint32>(last_report_ntp_time_ >> 32),
+ static_cast<uint32>(last_report_ntp_time_));
+ return ConvertNtpToTimeTicks(remote_ntp_seconds, remote_ntp_fraction) +
+ local_clock_ahead_by;
+}
- if (wrap == 1) {
- rtp_timestamp_int64 += (1LL << 32);
- } else if (wrap == -1) {
- last_received_rtp_timestamp_int64 += (1LL << 32);
+base::TimeTicks Rtcp::ToApproximateCaptureTime(uint32 rtp_timestamp,
+ int rtp_timebase) const {
+ // If the lip sync info has not been received yet, no conversion is possible.
+ if (lip_sync_capture_time_.is_null()) {
+ DVLOG(2) << "Requested capture time mapping before first lip sync update.";
+ return base::TimeTicks();
hubbe 2014/05/14 23:12:23 I think this should be last_rtp_received_ticks + (
miu 2014/05/16 22:45:47 As discussed, this isn't base::TimeTicks::Now(); i
+ } else {
+ // Sanity-check: Getting regular lip sync updates?
+ DCHECK((cast_environment_->Clock()->NowTicks() - lip_sync_capture_time_) <
+ base::TimeDelta::FromMinutes(1));
}
- // Time since the last RTCP message.
- // Note that this can be negative since we can compare a rtp timestamp from
- // a frame older than the last received RTCP message.
- int64 rtp_timestamp_diff =
- rtp_timestamp_int64 - last_received_rtp_timestamp_int64;
-
- int frequency_khz = frequency / 1000;
- int64 rtp_time_diff_ms = rtp_timestamp_diff / frequency_khz;
-
- // Sanity check.
- if (std::abs(rtp_time_diff_ms) > kMaxDiffSinceReceivedRtcpMs)
- return false;
-
- *rtp_timestamp_in_ticks = ConvertNtpToTimeTicks(last_received_ntp_seconds_,
- last_received_ntp_fraction_) +
- base::TimeDelta::FromMilliseconds(rtp_time_diff_ms);
- return true;
+
+ const int32 rtp_since_lip_sync =
+ static_cast<int32>(rtp_timestamp - lip_sync_rtp_timestamp_);
+ const base::TimeDelta time_since_lip_sync =
+ rtp_since_lip_sync * base::TimeDelta::FromSeconds(1) / rtp_timebase;
+ return lip_sync_capture_time_ + time_since_lip_sync;
}
void Rtcp::SetCastReceiverEventHistorySize(size_t size) {
@@ -333,8 +329,8 @@ void Rtcp::SetCastReceiverEventHistorySize(size_t size) {
}
void Rtcp::SetTargetDelay(base::TimeDelta target_delay) {
+ DCHECK(target_delay.InMilliseconds() < kMaxDelay);
target_delay_ms_ = static_cast<uint16>(target_delay.InMilliseconds());
- DCHECK(target_delay_ms_ < kMaxDelay);
}
void Rtcp::OnReceivedDelaySinceLastReport(uint32 receivers_ssrc,
@@ -376,20 +372,28 @@ void Rtcp::SaveLastSentNtpTime(const base::TimeTicks& now,
}
}
+uint32 Rtcp::GetTruncatedLastNtpReportTime() const {
+ return static_cast<uint32>(last_report_ntp_time_ >> 16);
+}
+
void Rtcp::UpdateRtt(const base::TimeDelta& sender_delay,
const base::TimeDelta& receiver_delay) {
base::TimeDelta rtt = sender_delay - receiver_delay;
+ // TODO(miu): Find out why this is capped at 1 ms, and remove the cap if it's
+ // bogus.
rtt = std::max(rtt, base::TimeDelta::FromMilliseconds(1));
rtt_ = rtt;
min_rtt_ = std::min(min_rtt_, rtt);
max_rtt_ = std::max(max_rtt_, rtt);
+ // TODO(miu): Replace "average for all time" with an EWMA, or suitable
+ // "average over recent past" mechanism.
if (number_of_rtt_in_avg_ != 0) {
- float ac = static_cast<float>(number_of_rtt_in_avg_);
+ const double ac = static_cast<double>(number_of_rtt_in_avg_);
avg_rtt_ms_ = ((ac / (ac + 1.0)) * avg_rtt_ms_) +
- ((1.0 / (ac + 1.0)) * rtt.InMilliseconds());
+ ((1.0 / (ac + 1.0)) * rtt.InMillisecondsF());
} else {
- avg_rtt_ms_ = rtt.InMilliseconds();
+ avg_rtt_ms_ = rtt.InMillisecondsF();
}
number_of_rtt_in_avg_++;
}
@@ -404,28 +408,12 @@ bool Rtcp::Rtt(base::TimeDelta* rtt, base::TimeDelta* avg_rtt,
if (number_of_rtt_in_avg_ == 0) return false;
*rtt = rtt_;
- *avg_rtt = base::TimeDelta::FromMilliseconds(avg_rtt_ms_);
+ *avg_rtt = base::TimeDelta::FromMillisecondsD(avg_rtt_ms_);
*min_rtt = min_rtt_;
*max_rtt = max_rtt_;
return true;
}
-int Rtcp::CheckForWrapAround(uint32 new_timestamp, uint32 old_timestamp) const {
hubbe 2014/05/14 23:12:23 Where did this logic go?
miu 2014/05/16 22:45:47 It was meaningless. Subtraction when zero, one, o
- if (new_timestamp < old_timestamp) {
- // This difference should be less than -2^31 if we have had a wrap around
- // (e.g. |new_timestamp| = 1, |rtcp_rtp_timestamp| = 2^32 - 1). Since it is
- // cast to a int32_t, it should be positive.
- if (static_cast<int32>(new_timestamp - old_timestamp) > 0) {
- return 1; // Forward wrap around.
- }
- } else if (static_cast<int32>(old_timestamp - new_timestamp) > 0) {
- // This difference should be less than -2^31 if we have had a backward wrap
- // around. Since it is cast to a int32, it should be positive.
- return -1;
- }
- return 0;
-}
-
void Rtcp::UpdateNextTimeToSendRtcp() {
int random = base::RandInt(0, 999);
base::TimeDelta time_to_next =

Powered by Google App Engine
This is Rietveld 408576698