mirror of
https://github.com/mollyim/webrtc.git
synced 2025-05-13 13:50:40 +01:00

This is a reland of commit 6762fbd988
Can reland now that upstream tests are fixed.
Original change's description:
> [Unwrap] Migrate ReceiveStatisticsImpl to use RtpSequenceNumberUnwrapper
>
> Bug: webrtc:13982
> Change-Id: Ic971371d4295e87380a77ef6aa7986a83d86f615
> Reviewed-on: https://webrtc-review.googlesource.com/c/src/+/288962
> Commit-Queue: Evan Shrubsole <eshr@webrtc.org>
> Auto-Submit: Evan Shrubsole <eshr@webrtc.org>
> Reviewed-by: Danil Chapovalov <danilchap@webrtc.org>
> Cr-Commit-Position: refs/heads/main@{#39046}
Bug: webrtc:13982
Change-Id: I1cb4faf5c6348be00e15d9f499a957a508199df6
Reviewed-on: https://webrtc-review.googlesource.com/c/src/+/290800
Reviewed-by: Danil Chapovalov <danilchap@webrtc.org>
Commit-Queue: Evan Shrubsole <eshr@webrtc.org>
Cr-Commit-Position: refs/heads/main@{#39067}
250 lines
9.6 KiB
C++
250 lines
9.6 KiB
C++
/*
|
|
* Copyright (c) 2013 The WebRTC project authors. All Rights Reserved.
|
|
*
|
|
* Use of this source code is governed by a BSD-style license
|
|
* that can be found in the LICENSE file in the root of the source
|
|
* tree. An additional intellectual property rights grant can be found
|
|
* in the file PATENTS. All contributing project authors may
|
|
* be found in the AUTHORS file in the root of the source tree.
|
|
*/
|
|
|
|
#ifndef MODULES_RTP_RTCP_SOURCE_RECEIVE_STATISTICS_IMPL_H_
|
|
#define MODULES_RTP_RTCP_SOURCE_RECEIVE_STATISTICS_IMPL_H_
|
|
|
|
#include <algorithm>
|
|
#include <functional>
|
|
#include <memory>
|
|
#include <utility>
|
|
#include <vector>
|
|
|
|
#include "absl/types/optional.h"
|
|
#include "modules/rtp_rtcp/include/receive_statistics.h"
|
|
#include "modules/rtp_rtcp/source/rtcp_packet/report_block.h"
|
|
#include "rtc_base/containers/flat_map.h"
|
|
#include "rtc_base/numerics/sequence_number_unwrapper.h"
|
|
#include "rtc_base/rate_statistics.h"
|
|
#include "rtc_base/synchronization/mutex.h"
|
|
#include "rtc_base/thread_annotations.h"
|
|
|
|
namespace webrtc {
|
|
|
|
// Extends StreamStatistician with methods needed by the implementation.
|
|
class StreamStatisticianImplInterface : public StreamStatistician {
|
|
public:
|
|
virtual ~StreamStatisticianImplInterface() = default;
|
|
virtual void MaybeAppendReportBlockAndReset(
|
|
std::vector<rtcp::ReportBlock>& report_blocks) = 0;
|
|
virtual void SetMaxReorderingThreshold(int max_reordering_threshold) = 0;
|
|
virtual void EnableRetransmitDetection(bool enable) = 0;
|
|
virtual void UpdateCounters(const RtpPacketReceived& packet) = 0;
|
|
};
|
|
|
|
// Thread-compatible implementation of StreamStatisticianImplInterface.
|
|
class StreamStatisticianImpl : public StreamStatisticianImplInterface {
|
|
public:
|
|
StreamStatisticianImpl(uint32_t ssrc,
|
|
Clock* clock,
|
|
int max_reordering_threshold);
|
|
~StreamStatisticianImpl() override;
|
|
|
|
// Implements StreamStatistician
|
|
RtpReceiveStats GetStats() const override;
|
|
absl::optional<int> GetFractionLostInPercent() const override;
|
|
StreamDataCounters GetReceiveStreamDataCounters() const override;
|
|
uint32_t BitrateReceived() const override;
|
|
|
|
// Implements StreamStatisticianImplInterface
|
|
void MaybeAppendReportBlockAndReset(
|
|
std::vector<rtcp::ReportBlock>& report_blocks) override;
|
|
void SetMaxReorderingThreshold(int max_reordering_threshold) override;
|
|
void EnableRetransmitDetection(bool enable) override;
|
|
// Updates StreamStatistician for incoming packets.
|
|
void UpdateCounters(const RtpPacketReceived& packet) override;
|
|
|
|
private:
|
|
bool IsRetransmitOfOldPacket(const RtpPacketReceived& packet,
|
|
int64_t now_ms) const;
|
|
void UpdateJitter(const RtpPacketReceived& packet, int64_t receive_time_ms);
|
|
void ReviseFrequencyAndJitter(int payload_type_frequency);
|
|
// Updates StreamStatistician for out of order packets.
|
|
// Returns true if packet considered to be out of order.
|
|
bool UpdateOutOfOrder(const RtpPacketReceived& packet,
|
|
int64_t sequence_number,
|
|
int64_t now_ms);
|
|
// Checks if this StreamStatistician received any rtp packets.
|
|
bool ReceivedRtpPacket() const { return received_seq_first_ >= 0; }
|
|
|
|
const uint32_t ssrc_;
|
|
Clock* const clock_;
|
|
// Delta used to map internal timestamps to Unix epoch ones.
|
|
const int64_t delta_internal_unix_epoch_ms_;
|
|
RateStatistics incoming_bitrate_;
|
|
// In number of packets or sequence numbers.
|
|
int max_reordering_threshold_;
|
|
bool enable_retransmit_detection_;
|
|
bool cumulative_loss_is_capped_;
|
|
|
|
// Stats on received RTP packets.
|
|
uint32_t jitter_q4_;
|
|
// Cumulative loss according to RFC 3550, which may be negative (and often is,
|
|
// if packets are reordered and there are non-RTX retransmissions).
|
|
int32_t cumulative_loss_;
|
|
// Offset added to outgoing rtcp reports, to make ensure that the reported
|
|
// cumulative loss is non-negative. Reports with negative values confuse some
|
|
// senders, in particular, our own loss-based bandwidth estimator.
|
|
int32_t cumulative_loss_rtcp_offset_;
|
|
|
|
int64_t last_receive_time_ms_;
|
|
uint32_t last_received_timestamp_;
|
|
RtpSequenceNumberUnwrapper seq_unwrapper_;
|
|
int64_t received_seq_first_;
|
|
int64_t received_seq_max_;
|
|
// Assume that the other side restarted when there are two sequential packets
|
|
// with large jump from received_seq_max_.
|
|
absl::optional<uint16_t> received_seq_out_of_order_;
|
|
|
|
// Current counter values.
|
|
StreamDataCounters receive_counters_;
|
|
|
|
// Counter values when we sent the last report.
|
|
int32_t last_report_cumulative_loss_;
|
|
int64_t last_report_seq_max_;
|
|
|
|
// The sample frequency of the last received packet.
|
|
int last_payload_type_frequency_;
|
|
};
|
|
|
|
// Thread-safe implementation of StreamStatisticianImplInterface.
|
|
class StreamStatisticianLocked : public StreamStatisticianImplInterface {
|
|
public:
|
|
StreamStatisticianLocked(uint32_t ssrc,
|
|
Clock* clock,
|
|
int max_reordering_threshold)
|
|
: impl_(ssrc, clock, max_reordering_threshold) {}
|
|
~StreamStatisticianLocked() override = default;
|
|
|
|
RtpReceiveStats GetStats() const override {
|
|
MutexLock lock(&stream_lock_);
|
|
return impl_.GetStats();
|
|
}
|
|
absl::optional<int> GetFractionLostInPercent() const override {
|
|
MutexLock lock(&stream_lock_);
|
|
return impl_.GetFractionLostInPercent();
|
|
}
|
|
StreamDataCounters GetReceiveStreamDataCounters() const override {
|
|
MutexLock lock(&stream_lock_);
|
|
return impl_.GetReceiveStreamDataCounters();
|
|
}
|
|
uint32_t BitrateReceived() const override {
|
|
MutexLock lock(&stream_lock_);
|
|
return impl_.BitrateReceived();
|
|
}
|
|
void MaybeAppendReportBlockAndReset(
|
|
std::vector<rtcp::ReportBlock>& report_blocks) override {
|
|
MutexLock lock(&stream_lock_);
|
|
impl_.MaybeAppendReportBlockAndReset(report_blocks);
|
|
}
|
|
void SetMaxReorderingThreshold(int max_reordering_threshold) override {
|
|
MutexLock lock(&stream_lock_);
|
|
return impl_.SetMaxReorderingThreshold(max_reordering_threshold);
|
|
}
|
|
void EnableRetransmitDetection(bool enable) override {
|
|
MutexLock lock(&stream_lock_);
|
|
return impl_.EnableRetransmitDetection(enable);
|
|
}
|
|
void UpdateCounters(const RtpPacketReceived& packet) override {
|
|
MutexLock lock(&stream_lock_);
|
|
return impl_.UpdateCounters(packet);
|
|
}
|
|
|
|
private:
|
|
mutable Mutex stream_lock_;
|
|
StreamStatisticianImpl impl_ RTC_GUARDED_BY(&stream_lock_);
|
|
};
|
|
|
|
// Thread-compatible implementation.
|
|
class ReceiveStatisticsImpl : public ReceiveStatistics {
|
|
public:
|
|
ReceiveStatisticsImpl(
|
|
Clock* clock,
|
|
std::function<std::unique_ptr<StreamStatisticianImplInterface>(
|
|
uint32_t ssrc,
|
|
Clock* clock,
|
|
int max_reordering_threshold)> stream_statistician_factory);
|
|
~ReceiveStatisticsImpl() override = default;
|
|
|
|
// Implements ReceiveStatisticsProvider.
|
|
std::vector<rtcp::ReportBlock> RtcpReportBlocks(size_t max_blocks) override;
|
|
|
|
// Implements RtpPacketSinkInterface
|
|
void OnRtpPacket(const RtpPacketReceived& packet) override;
|
|
|
|
// Implements ReceiveStatistics.
|
|
StreamStatistician* GetStatistician(uint32_t ssrc) const override;
|
|
void SetMaxReorderingThreshold(int max_reordering_threshold) override;
|
|
void SetMaxReorderingThreshold(uint32_t ssrc,
|
|
int max_reordering_threshold) override;
|
|
void EnableRetransmitDetection(uint32_t ssrc, bool enable) override;
|
|
|
|
private:
|
|
StreamStatisticianImplInterface* GetOrCreateStatistician(uint32_t ssrc);
|
|
|
|
Clock* const clock_;
|
|
std::function<std::unique_ptr<StreamStatisticianImplInterface>(
|
|
uint32_t ssrc,
|
|
Clock* clock,
|
|
int max_reordering_threshold)>
|
|
stream_statistician_factory_;
|
|
// The index within `all_ssrcs_` that was last returned.
|
|
size_t last_returned_ssrc_idx_;
|
|
std::vector<uint32_t> all_ssrcs_;
|
|
int max_reordering_threshold_;
|
|
flat_map<uint32_t /*ssrc*/, std::unique_ptr<StreamStatisticianImplInterface>>
|
|
statisticians_;
|
|
};
|
|
|
|
// Thread-safe implementation wrapping access to ReceiveStatisticsImpl with a
|
|
// mutex.
|
|
class ReceiveStatisticsLocked : public ReceiveStatistics {
|
|
public:
|
|
explicit ReceiveStatisticsLocked(
|
|
Clock* clock,
|
|
std::function<std::unique_ptr<StreamStatisticianImplInterface>(
|
|
uint32_t ssrc,
|
|
Clock* clock,
|
|
int max_reordering_threshold)> stream_statitician_factory)
|
|
: impl_(clock, std::move(stream_statitician_factory)) {}
|
|
~ReceiveStatisticsLocked() override = default;
|
|
std::vector<rtcp::ReportBlock> RtcpReportBlocks(size_t max_blocks) override {
|
|
MutexLock lock(&receive_statistics_lock_);
|
|
return impl_.RtcpReportBlocks(max_blocks);
|
|
}
|
|
void OnRtpPacket(const RtpPacketReceived& packet) override {
|
|
MutexLock lock(&receive_statistics_lock_);
|
|
return impl_.OnRtpPacket(packet);
|
|
}
|
|
StreamStatistician* GetStatistician(uint32_t ssrc) const override {
|
|
MutexLock lock(&receive_statistics_lock_);
|
|
return impl_.GetStatistician(ssrc);
|
|
}
|
|
void SetMaxReorderingThreshold(int max_reordering_threshold) override {
|
|
MutexLock lock(&receive_statistics_lock_);
|
|
return impl_.SetMaxReorderingThreshold(max_reordering_threshold);
|
|
}
|
|
void SetMaxReorderingThreshold(uint32_t ssrc,
|
|
int max_reordering_threshold) override {
|
|
MutexLock lock(&receive_statistics_lock_);
|
|
return impl_.SetMaxReorderingThreshold(ssrc, max_reordering_threshold);
|
|
}
|
|
void EnableRetransmitDetection(uint32_t ssrc, bool enable) override {
|
|
MutexLock lock(&receive_statistics_lock_);
|
|
return impl_.EnableRetransmitDetection(ssrc, enable);
|
|
}
|
|
|
|
private:
|
|
mutable Mutex receive_statistics_lock_;
|
|
ReceiveStatisticsImpl impl_ RTC_GUARDED_BY(&receive_statistics_lock_);
|
|
};
|
|
|
|
} // namespace webrtc
|
|
#endif // MODULES_RTP_RTCP_SOURCE_RECEIVE_STATISTICS_IMPL_H_
|