Chromium Code Reviews| 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 = |