wu@webrtc.org | 822fbd8 | 2013-08-15 23:38:54 +0000 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (c) 2013 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 Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 11 | #ifndef MODULES_RTP_RTCP_SOURCE_RECEIVE_STATISTICS_IMPL_H_ |
| 12 | #define MODULES_RTP_RTCP_SOURCE_RECEIVE_STATISTICS_IMPL_H_ |
wu@webrtc.org | 822fbd8 | 2013-08-15 23:38:54 +0000 | [diff] [blame] | 13 | |
wu@webrtc.org | 822fbd8 | 2013-08-15 23:38:54 +0000 | [diff] [blame] | 14 | #include <algorithm> |
danilchap | b8b6fbb | 2015-12-10 05:05:27 -0800 | [diff] [blame] | 15 | #include <map> |
danilchap | 0bc8423 | 2017-08-11 08:12:54 -0700 | [diff] [blame] | 16 | #include <vector> |
wu@webrtc.org | 822fbd8 | 2013-08-15 23:38:54 +0000 | [diff] [blame] | 17 | |
Danil Chapovalov | b438b5a | 2018-12-05 14:55:46 +0000 | [diff] [blame] | 18 | #include "absl/types/optional.h" |
| 19 | #include "modules/include/module_common_types_public.h" |
Jonas Olsson | a4d8737 | 2019-07-05 19:08:33 +0200 | [diff] [blame] | 20 | #include "modules/rtp_rtcp/include/receive_statistics.h" |
Steve Anton | 10542f2 | 2019-01-11 09:11:00 -0800 | [diff] [blame] | 21 | #include "rtc_base/critical_section.h" |
Mirko Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 22 | #include "rtc_base/rate_statistics.h" |
Danil Chapovalov | ebb50c2 | 2018-11-22 14:04:02 +0100 | [diff] [blame] | 23 | #include "rtc_base/thread_annotations.h" |
wu@webrtc.org | 822fbd8 | 2013-08-15 23:38:54 +0000 | [diff] [blame] | 24 | |
| 25 | namespace webrtc { |
| 26 | |
Niels Möller | 1a3859c | 2019-09-04 09:43:15 +0200 | [diff] [blame] | 27 | class StreamStatisticianImpl : public StreamStatistician { |
wu@webrtc.org | 822fbd8 | 2013-08-15 23:38:54 +0000 | [diff] [blame] | 28 | public: |
danilchap | ec86be0 | 2017-08-14 05:51:02 -0700 | [diff] [blame] | 29 | StreamStatisticianImpl(uint32_t ssrc, |
| 30 | Clock* clock, |
Niels Möller | d781965 | 2019-08-13 14:43:02 +0200 | [diff] [blame] | 31 | int max_reordering_threshold); |
Danil Chapovalov | 2a5ce2b | 2018-02-07 09:38:31 +0100 | [diff] [blame] | 32 | ~StreamStatisticianImpl() override; |
wu@webrtc.org | 822fbd8 | 2013-08-15 23:38:54 +0000 | [diff] [blame] | 33 | |
Niels Möller | d77cc24 | 2019-08-22 09:40:25 +0200 | [diff] [blame] | 34 | RtpReceiveStats GetStats() const override; |
| 35 | |
Danil Chapovalov | c5267d2 | 2017-09-18 13:57:19 +0200 | [diff] [blame] | 36 | bool GetActiveStatisticsAndReset(RtcpStatistics* statistics); |
Niels Möller | 9a9f18a | 2019-08-02 13:52:37 +0200 | [diff] [blame] | 37 | absl::optional<int> GetFractionLostInPercent() const override; |
Niels Möller | 58b496b | 2019-08-12 12:16:31 +0200 | [diff] [blame] | 38 | StreamDataCounters GetReceiveStreamDataCounters() const override; |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 +0000 | [diff] [blame] | 39 | uint32_t BitrateReceived() const override; |
stefan@webrtc.org | 286fe0b | 2013-08-21 20:58:21 +0000 | [diff] [blame] | 40 | |
stefan@webrtc.org | 7bb8f02 | 2013-09-06 13:40:11 +0000 | [diff] [blame] | 41 | void SetMaxReorderingThreshold(int max_reordering_threshold); |
Niels Möller | 5304a32 | 2018-08-27 13:27:05 +0200 | [diff] [blame] | 42 | void EnableRetransmitDetection(bool enable); |
wu@webrtc.org | 822fbd8 | 2013-08-15 23:38:54 +0000 | [diff] [blame] | 43 | |
Niels Möller | 1a3859c | 2019-09-04 09:43:15 +0200 | [diff] [blame] | 44 | // Updates StreamStatistician for incoming packets. |
| 45 | void UpdateCounters(const RtpPacketReceived& packet); |
| 46 | |
wu@webrtc.org | 822fbd8 | 2013-08-15 23:38:54 +0000 | [diff] [blame] | 47 | private: |
Danil Chapovalov | 44727b4 | 2018-11-22 11:28:45 +0100 | [diff] [blame] | 48 | bool IsRetransmitOfOldPacket(const RtpPacketReceived& packet, |
| 49 | int64_t now_ms) const |
Niels Möller | b615d1a | 2018-08-27 12:32:21 +0200 | [diff] [blame] | 50 | RTC_EXCLUSIVE_LOCKS_REQUIRED(stream_lock_); |
Qingsi Wang | 2370b08 | 2018-08-21 14:24:26 -0700 | [diff] [blame] | 51 | RtcpStatistics CalculateRtcpStatistics() |
danilchap | 56359be | 2017-09-07 07:53:45 -0700 | [diff] [blame] | 52 | RTC_EXCLUSIVE_LOCKS_REQUIRED(stream_lock_); |
Danil Chapovalov | 856cf22 | 2018-11-26 10:20:01 +0100 | [diff] [blame] | 53 | void UpdateJitter(const RtpPacketReceived& packet, int64_t receive_time_ms) |
Niels Möller | b615d1a | 2018-08-27 12:32:21 +0200 | [diff] [blame] | 54 | RTC_EXCLUSIVE_LOCKS_REQUIRED(stream_lock_); |
Danil Chapovalov | b438b5a | 2018-12-05 14:55:46 +0000 | [diff] [blame] | 55 | // Updates StreamStatistician for out of order packets. |
| 56 | // Returns true if packet considered to be out of order. |
| 57 | bool UpdateOutOfOrder(const RtpPacketReceived& packet, |
| 58 | int64_t sequence_number, |
| 59 | int64_t now_ms) |
| 60 | RTC_EXCLUSIVE_LOCKS_REQUIRED(stream_lock_); |
Danil Chapovalov | b438b5a | 2018-12-05 14:55:46 +0000 | [diff] [blame] | 61 | // Checks if this StreamStatistician received any rtp packets. |
| 62 | bool ReceivedRtpPacket() const RTC_EXCLUSIVE_LOCKS_REQUIRED(stream_lock_) { |
Niels Möller | 1a3859c | 2019-09-04 09:43:15 +0200 | [diff] [blame] | 63 | return received_seq_first_ >= 0; |
Danil Chapovalov | b438b5a | 2018-12-05 14:55:46 +0000 | [diff] [blame] | 64 | } |
stefan@webrtc.org | 7bb8f02 | 2013-09-06 13:40:11 +0000 | [diff] [blame] | 65 | |
danilchap | ec86be0 | 2017-08-14 05:51:02 -0700 | [diff] [blame] | 66 | const uint32_t ssrc_; |
sprang | cd349d9 | 2016-07-13 09:11:28 -0700 | [diff] [blame] | 67 | Clock* const clock_; |
danilchap | 7c9426c | 2016-04-14 03:05:31 -0700 | [diff] [blame] | 68 | rtc::CriticalSection stream_lock_; |
Niels Möller | b615d1a | 2018-08-27 12:32:21 +0200 | [diff] [blame] | 69 | RateStatistics incoming_bitrate_ RTC_GUARDED_BY(&stream_lock_); |
| 70 | // In number of packets or sequence numbers. |
| 71 | int max_reordering_threshold_ RTC_GUARDED_BY(&stream_lock_); |
Niels Möller | 5304a32 | 2018-08-27 13:27:05 +0200 | [diff] [blame] | 72 | bool enable_retransmit_detection_ RTC_GUARDED_BY(&stream_lock_); |
stefan@webrtc.org | 286fe0b | 2013-08-21 20:58:21 +0000 | [diff] [blame] | 73 | |
wu@webrtc.org | 822fbd8 | 2013-08-15 23:38:54 +0000 | [diff] [blame] | 74 | // Stats on received RTP packets. |
Niels Möller | b615d1a | 2018-08-27 12:32:21 +0200 | [diff] [blame] | 75 | uint32_t jitter_q4_ RTC_GUARDED_BY(&stream_lock_); |
Niels Möller | 1a3859c | 2019-09-04 09:43:15 +0200 | [diff] [blame] | 76 | // Cumulative loss according to RFC 3550, which may be negative (and often is, |
| 77 | // if packets are reordered and there are non-RTX retransmissions). |
| 78 | int32_t cumulative_loss_ RTC_GUARDED_BY(&stream_lock_); |
| 79 | // Offset added to outgoing rtcp reports, to make ensure that the reported |
| 80 | // cumulative loss is non-negative. Reports with negative values confuse some |
| 81 | // senders, in particular, our own loss-based bandwidth estimator. |
| 82 | int32_t cumulative_loss_rtcp_offset_ RTC_GUARDED_BY(&stream_lock_); |
wu@webrtc.org | 822fbd8 | 2013-08-15 23:38:54 +0000 | [diff] [blame] | 83 | |
Niels Möller | b615d1a | 2018-08-27 12:32:21 +0200 | [diff] [blame] | 84 | int64_t last_receive_time_ms_ RTC_GUARDED_BY(&stream_lock_); |
Niels Möller | b615d1a | 2018-08-27 12:32:21 +0200 | [diff] [blame] | 85 | uint32_t last_received_timestamp_ RTC_GUARDED_BY(&stream_lock_); |
Danil Chapovalov | b438b5a | 2018-12-05 14:55:46 +0000 | [diff] [blame] | 86 | SequenceNumberUnwrapper seq_unwrapper_ RTC_GUARDED_BY(&stream_lock_); |
| 87 | int64_t received_seq_first_ RTC_GUARDED_BY(&stream_lock_); |
| 88 | int64_t received_seq_max_ RTC_GUARDED_BY(&stream_lock_); |
| 89 | // Assume that the other side restarted when there are two sequential packets |
| 90 | // with large jump from received_seq_max_. |
| 91 | absl::optional<uint16_t> received_seq_out_of_order_ |
| 92 | RTC_GUARDED_BY(&stream_lock_); |
wu@webrtc.org | 822fbd8 | 2013-08-15 23:38:54 +0000 | [diff] [blame] | 93 | |
| 94 | // Current counter values. |
Niels Möller | b615d1a | 2018-08-27 12:32:21 +0200 | [diff] [blame] | 95 | StreamDataCounters receive_counters_ RTC_GUARDED_BY(&stream_lock_); |
wu@webrtc.org | 822fbd8 | 2013-08-15 23:38:54 +0000 | [diff] [blame] | 96 | |
Qingsi Wang | 2370b08 | 2018-08-21 14:24:26 -0700 | [diff] [blame] | 97 | // Counter values when we sent the last report. |
Niels Möller | 1a3859c | 2019-09-04 09:43:15 +0200 | [diff] [blame] | 98 | int32_t last_report_cumulative_loss_ RTC_GUARDED_BY(&stream_lock_); |
Danil Chapovalov | b438b5a | 2018-12-05 14:55:46 +0000 | [diff] [blame] | 99 | int64_t last_report_seq_max_ RTC_GUARDED_BY(&stream_lock_); |
stefan@webrtc.org | 286fe0b | 2013-08-21 20:58:21 +0000 | [diff] [blame] | 100 | }; |
| 101 | |
Danil Chapovalov | 8ce0d2b | 2018-11-23 11:03:25 +0100 | [diff] [blame] | 102 | class ReceiveStatisticsImpl : public ReceiveStatistics { |
stefan@webrtc.org | 286fe0b | 2013-08-21 20:58:21 +0000 | [diff] [blame] | 103 | public: |
Niels Möller | d781965 | 2019-08-13 14:43:02 +0200 | [diff] [blame] | 104 | explicit ReceiveStatisticsImpl(Clock* clock); |
stefan@webrtc.org | 286fe0b | 2013-08-21 20:58:21 +0000 | [diff] [blame] | 105 | |
Danil Chapovalov | 2a5ce2b | 2018-02-07 09:38:31 +0100 | [diff] [blame] | 106 | ~ReceiveStatisticsImpl() override; |
stefan@webrtc.org | 286fe0b | 2013-08-21 20:58:21 +0000 | [diff] [blame] | 107 | |
Niels Möller | dbb988b | 2018-11-15 08:05:16 +0100 | [diff] [blame] | 108 | // Implements ReceiveStatisticsProvider. |
danilchap | 0bc8423 | 2017-08-11 08:12:54 -0700 | [diff] [blame] | 109 | std::vector<rtcp::ReportBlock> RtcpReportBlocks(size_t max_blocks) override; |
| 110 | |
Niels Möller | dbb988b | 2018-11-15 08:05:16 +0100 | [diff] [blame] | 111 | // Implements RtpPacketSinkInterface |
Niels Möller | 1f3206c | 2018-09-14 08:26:32 +0200 | [diff] [blame] | 112 | void OnRtpPacket(const RtpPacketReceived& packet) override; |
| 113 | |
Niels Möller | dbb988b | 2018-11-15 08:05:16 +0100 | [diff] [blame] | 114 | // Implements ReceiveStatistics. |
Niels Möller | 87da109 | 2019-05-24 14:04:28 +0200 | [diff] [blame] | 115 | // Note: More specific return type for use in the implementation. |
| 116 | StreamStatisticianImpl* GetStatistician(uint32_t ssrc) const override; |
kjellander@webrtc.org | 14665ff | 2015-03-04 12:58:35 +0000 | [diff] [blame] | 117 | void SetMaxReorderingThreshold(int max_reordering_threshold) override; |
Niels Möller | 87da109 | 2019-05-24 14:04:28 +0200 | [diff] [blame] | 118 | void SetMaxReorderingThreshold(uint32_t ssrc, |
| 119 | int max_reordering_threshold) override; |
Niels Möller | 5304a32 | 2018-08-27 13:27:05 +0200 | [diff] [blame] | 120 | void EnableRetransmitDetection(uint32_t ssrc, bool enable) override; |
stefan@webrtc.org | 286fe0b | 2013-08-21 20:58:21 +0000 | [diff] [blame] | 121 | |
stefan@webrtc.org | 286fe0b | 2013-08-21 20:58:21 +0000 | [diff] [blame] | 122 | private: |
Niels Möller | 87da109 | 2019-05-24 14:04:28 +0200 | [diff] [blame] | 123 | StreamStatisticianImpl* GetOrCreateStatistician(uint32_t ssrc); |
| 124 | |
sprang | cd349d9 | 2016-07-13 09:11:28 -0700 | [diff] [blame] | 125 | Clock* const clock_; |
danilchap | 7c9426c | 2016-04-14 03:05:31 -0700 | [diff] [blame] | 126 | rtc::CriticalSection receive_statistics_lock_; |
Danil Chapovalov | d1996b7 | 2018-01-16 11:07:18 +0100 | [diff] [blame] | 127 | uint32_t last_returned_ssrc_; |
Danil Chapovalov | ebb50c2 | 2018-11-22 14:04:02 +0100 | [diff] [blame] | 128 | int max_reordering_threshold_ RTC_GUARDED_BY(receive_statistics_lock_); |
| 129 | std::map<uint32_t, StreamStatisticianImpl*> statisticians_ |
| 130 | RTC_GUARDED_BY(receive_statistics_lock_); |
wu@webrtc.org | 822fbd8 | 2013-08-15 23:38:54 +0000 | [diff] [blame] | 131 | }; |
| 132 | } // namespace webrtc |
Mirko Bonadei | 92ea95e | 2017-09-15 06:47:31 +0200 | [diff] [blame] | 133 | #endif // MODULES_RTP_RTCP_SOURCE_RECEIVE_STATISTICS_IMPL_H_ |