blob: 551eb5f75b401ffb2f95828b282d8fdd5afca9c2 [file] [log] [blame]
henrik.lundin@webrtc.orgd94659d2013-01-29 12:09:21 +00001/*
2 * Copyright (c) 2011 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
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020011#include "modules/audio_coding/neteq/rtcp.h"
henrik.lundin@webrtc.orgd94659d2013-01-29 12:09:21 +000012
ossu6b6c88f2016-10-31 08:59:26 -070013#include <stdlib.h>
henrik.lundin@webrtc.orgd94659d2013-01-29 12:09:21 +000014#include <string.h>
15
henrik.lundin@webrtc.orge7ce4372014-01-09 14:01:55 +000016#include <algorithm>
17
henrik.lundin@webrtc.orgd94659d2013-01-29 12:09:21 +000018namespace webrtc {
19
20void Rtcp::Init(uint16_t start_sequence_number) {
21 cycles_ = 0;
22 max_seq_no_ = start_sequence_number;
23 base_seq_no_ = start_sequence_number;
24 received_packets_ = 0;
25 received_packets_prior_ = 0;
26 expected_prior_ = 0;
27 jitter_ = 0;
28 transit_ = 0;
29}
30
31void Rtcp::Update(const RTPHeader& rtp_header, uint32_t receive_timestamp) {
32 // Update number of received packets, and largest packet number received.
33 received_packets_++;
34 int16_t sn_diff = rtp_header.sequenceNumber - max_seq_no_;
35 if (sn_diff >= 0) {
36 if (rtp_header.sequenceNumber < max_seq_no_) {
37 // Wrap-around detected.
38 cycles_++;
39 }
40 max_seq_no_ = rtp_header.sequenceNumber;
41 }
42
43 // Calculate jitter according to RFC 3550, and update previous timestamps.
44 // Note that the value in |jitter_| is in Q4.
45 if (received_packets_ > 1) {
46 int32_t ts_diff = receive_timestamp - (rtp_header.timestamp - transit_);
ossu6b6c88f2016-10-31 08:59:26 -070047 int64_t jitter_diff = (std::abs(int64_t{ts_diff}) << 4) - jitter_;
henrik.lundin@webrtc.orgd94659d2013-01-29 12:09:21 +000048 // Calculate 15 * jitter_ / 16 + jitter_diff / 16 (with proper rounding).
49 jitter_ = jitter_ + ((jitter_diff + 8) >> 4);
ossu6b6c88f2016-10-31 08:59:26 -070050 RTC_DCHECK_GE(jitter_, 0);
henrik.lundin@webrtc.orgd94659d2013-01-29 12:09:21 +000051 }
52 transit_ = rtp_header.timestamp - receive_timestamp;
53}
54
55void Rtcp::GetStatistics(bool no_reset, RtcpStatistics* stats) {
56 // Extended highest sequence number received.
srte186d9c32017-08-04 05:03:53 -070057 stats->extended_highest_sequence_number =
sprang@webrtc.orgfe5d36b2013-10-28 09:21:07 +000058 (static_cast<int>(cycles_) << 16) + max_seq_no_;
henrik.lundin@webrtc.orgd94659d2013-01-29 12:09:21 +000059
60 // Calculate expected number of packets and compare it with the number of
61 // packets that were actually received. The cumulative number of lost packets
62 // can be extracted.
sprang@webrtc.orgfe5d36b2013-10-28 09:21:07 +000063 uint32_t expected_packets =
srte186d9c32017-08-04 05:03:53 -070064 stats->extended_highest_sequence_number - base_seq_no_ + 1;
henrik.lundin@webrtc.orgd94659d2013-01-29 12:09:21 +000065 if (received_packets_ == 0) {
66 // No packets received, assume none lost.
srte186d9c32017-08-04 05:03:53 -070067 stats->packets_lost = 0;
henrik.lundin@webrtc.orgd94659d2013-01-29 12:09:21 +000068 } else if (expected_packets > received_packets_) {
srte186d9c32017-08-04 05:03:53 -070069 stats->packets_lost = expected_packets - received_packets_;
70 if (stats->packets_lost > 0xFFFFFF) {
71 stats->packets_lost = 0xFFFFFF;
henrik.lundin@webrtc.orgd94659d2013-01-29 12:09:21 +000072 }
73 } else {
srte186d9c32017-08-04 05:03:53 -070074 stats->packets_lost = 0;
henrik.lundin@webrtc.orgd94659d2013-01-29 12:09:21 +000075 }
76
77 // Fraction lost since last report.
78 uint32_t expected_since_last = expected_packets - expected_prior_;
79 uint32_t received_since_last = received_packets_ - received_packets_prior_;
80 if (!no_reset) {
81 expected_prior_ = expected_packets;
82 received_packets_prior_ = received_packets_;
83 }
84 int32_t lost = expected_since_last - received_since_last;
85 if (expected_since_last == 0 || lost <= 0 || received_packets_ == 0) {
86 stats->fraction_lost = 0;
87 } else {
sprang@webrtc.orgfe5d36b2013-10-28 09:21:07 +000088 stats->fraction_lost = std::min(0xFFU, (lost << 8) / expected_since_last);
henrik.lundin@webrtc.orgd94659d2013-01-29 12:09:21 +000089 }
90
91 stats->jitter = jitter_ >> 4; // Scaling from Q4.
92}
93
94} // namespace webrtc