2011-07-07 08:21:25 +00:00
|
|
|
/*
|
2012-01-30 13:05:29 +00:00
|
|
|
* Copyright (c) 2012 The WebRTC project authors. All Rights Reserved.
|
2011-07-07 08:21:25 +00:00
|
|
|
*
|
|
|
|
|
* 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.
|
|
|
|
|
*/
|
|
|
|
|
|
2022-08-09 17:50:45 +02:00
|
|
|
#include "modules/rtp_rtcp/source/ulpfec_receiver.h"
|
2012-02-01 02:40:37 +00:00
|
|
|
|
2016-04-27 01:19:58 -07:00
|
|
|
#include <memory>
|
2016-08-09 01:23:23 -07:00
|
|
|
#include <utility>
|
2016-04-27 01:19:58 -07:00
|
|
|
|
2019-01-25 20:26:48 +01:00
|
|
|
#include "api/scoped_refptr.h"
|
2019-06-26 14:39:36 +02:00
|
|
|
#include "modules/rtp_rtcp/source/rtp_packet_received.h"
|
2017-09-15 06:47:31 +02:00
|
|
|
#include "rtc_base/logging.h"
|
2019-01-23 08:45:57 +01:00
|
|
|
#include "rtc_base/time_utils.h"
|
2022-08-10 07:45:43 +02:00
|
|
|
#include "system_wrappers/include/metrics.h"
|
2011-07-07 08:21:25 +00:00
|
|
|
|
|
|
|
|
namespace webrtc {
|
2013-09-06 13:40:11 +00:00
|
|
|
|
2022-08-09 17:50:45 +02:00
|
|
|
UlpfecReceiver::UlpfecReceiver(uint32_t ssrc,
|
2022-08-10 07:45:43 +02:00
|
|
|
int ulpfec_payload_type,
|
2022-08-09 17:50:45 +02:00
|
|
|
RecoveredPacketReceiver* callback,
|
2022-08-10 07:45:43 +02:00
|
|
|
rtc::ArrayView<const RtpExtension> extensions,
|
|
|
|
|
Clock* clock)
|
2017-06-29 02:45:35 -07:00
|
|
|
: ssrc_(ssrc),
|
2022-08-10 07:45:43 +02:00
|
|
|
ulpfec_payload_type_(ulpfec_payload_type),
|
|
|
|
|
clock_(clock),
|
2019-06-26 14:39:36 +02:00
|
|
|
extensions_(extensions),
|
2017-06-29 02:45:35 -07:00
|
|
|
recovered_packet_callback_(callback),
|
|
|
|
|
fec_(ForwardErrorCorrection::CreateUlpfec(ssrc_)) {}
|
2011-07-07 08:21:25 +00:00
|
|
|
|
2022-08-09 17:50:45 +02:00
|
|
|
UlpfecReceiver::~UlpfecReceiver() {
|
2021-01-19 13:30:23 +01:00
|
|
|
RTC_DCHECK_RUN_ON(&sequence_checker_);
|
2022-08-10 07:45:43 +02:00
|
|
|
|
|
|
|
|
if (packet_counter_.first_packet_time != Timestamp::MinusInfinity()) {
|
|
|
|
|
const Timestamp now = clock_->CurrentTime();
|
|
|
|
|
TimeDelta elapsed = (now - packet_counter_.first_packet_time);
|
|
|
|
|
if (elapsed.seconds() >= metrics::kMinRunTimeInSeconds) {
|
|
|
|
|
if (packet_counter_.num_packets > 0) {
|
|
|
|
|
RTC_HISTOGRAM_PERCENTAGE(
|
|
|
|
|
"WebRTC.Video.ReceivedFecPacketsInPercent",
|
|
|
|
|
static_cast<int>(packet_counter_.num_fec_packets * 100 /
|
|
|
|
|
packet_counter_.num_packets));
|
|
|
|
|
}
|
|
|
|
|
if (packet_counter_.num_fec_packets > 0) {
|
|
|
|
|
RTC_HISTOGRAM_PERCENTAGE(
|
|
|
|
|
"WebRTC.Video.RecoveredMediaPacketsInPercentOfFec",
|
|
|
|
|
static_cast<int>(packet_counter_.num_recovered_packets * 100 /
|
|
|
|
|
packet_counter_.num_fec_packets));
|
|
|
|
|
}
|
|
|
|
|
if (ulpfec_payload_type_ != -1) {
|
|
|
|
|
RTC_HISTOGRAM_COUNTS_10000(
|
|
|
|
|
"WebRTC.Video.FecBitrateReceivedInKbps",
|
|
|
|
|
static_cast<int>(packet_counter_.num_bytes * 8 / elapsed.seconds() /
|
|
|
|
|
1000));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2016-08-10 00:51:50 -07:00
|
|
|
received_packets_.clear();
|
2016-09-21 09:19:34 +02:00
|
|
|
fec_->ResetState(&recovered_packets_);
|
2011-07-07 08:21:25 +00:00
|
|
|
}
|
|
|
|
|
|
2022-08-09 17:50:45 +02:00
|
|
|
FecPacketCounter UlpfecReceiver::GetPacketCounter() const {
|
2021-01-19 13:30:23 +01:00
|
|
|
RTC_DCHECK_RUN_ON(&sequence_checker_);
|
2015-01-15 07:40:20 +00:00
|
|
|
return packet_counter_;
|
|
|
|
|
}
|
|
|
|
|
|
2022-08-09 17:50:45 +02:00
|
|
|
void UlpfecReceiver::SetRtpExtensions(
|
2021-09-08 10:44:50 +02:00
|
|
|
rtc::ArrayView<const RtpExtension> extensions) {
|
|
|
|
|
RTC_DCHECK_RUN_ON(&sequence_checker_);
|
|
|
|
|
extensions_.Reset(extensions);
|
|
|
|
|
}
|
|
|
|
|
|
2013-05-16 15:06:28 +00:00
|
|
|
// 0 1 2 3
|
|
|
|
|
// 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
|
|
|
|
|
// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
|
|
|
|
// |F| block PT | timestamp offset | block length |
|
|
|
|
|
// +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
|
|
|
|
|
//
|
|
|
|
|
//
|
|
|
|
|
// RFC 2198 RTP Payload for Redundant Audio Data September 1997
|
|
|
|
|
//
|
|
|
|
|
// The bits in the header are specified as follows:
|
|
|
|
|
//
|
|
|
|
|
// F: 1 bit First bit in header indicates whether another header block
|
|
|
|
|
// follows. If 1 further header blocks follow, if 0 this is the
|
|
|
|
|
// last header block.
|
|
|
|
|
// If 0 there is only 1 byte RED header
|
|
|
|
|
//
|
|
|
|
|
// block PT: 7 bits RTP payload type for this block.
|
|
|
|
|
//
|
|
|
|
|
// timestamp offset: 14 bits Unsigned offset of timestamp of this block
|
|
|
|
|
// relative to timestamp given in RTP header. The use of an unsigned
|
|
|
|
|
// offset implies that redundant data must be sent after the primary
|
|
|
|
|
// data, and is hence a time to be subtracted from the current
|
|
|
|
|
// timestamp to determine the timestamp of the data for which this
|
|
|
|
|
// block is the redundancy.
|
|
|
|
|
//
|
|
|
|
|
// block length: 10 bits Length in bytes of the corresponding data
|
|
|
|
|
// block excluding header.
|
|
|
|
|
|
2022-08-10 07:45:43 +02:00
|
|
|
bool UlpfecReceiver::AddReceivedRedPacket(const RtpPacketReceived& rtp_packet) {
|
2021-01-19 13:30:23 +01:00
|
|
|
RTC_DCHECK_RUN_ON(&sequence_checker_);
|
|
|
|
|
// TODO(bugs.webrtc.org/11993): We get here via Call::DeliverRtp, so should be
|
|
|
|
|
// moved to the network thread.
|
|
|
|
|
|
2019-09-20 11:40:12 +02:00
|
|
|
if (rtp_packet.Ssrc() != ssrc_) {
|
2017-11-09 11:09:25 +01:00
|
|
|
RTC_LOG(LS_WARNING)
|
2017-06-29 02:45:35 -07:00
|
|
|
<< "Received RED packet with different SSRC than expected; dropping.";
|
2019-09-20 11:40:12 +02:00
|
|
|
return false;
|
2017-06-29 02:45:35 -07:00
|
|
|
}
|
2019-09-20 11:40:12 +02:00
|
|
|
if (rtp_packet.size() > IP_PACKET_SIZE) {
|
2018-05-18 13:48:58 +02:00
|
|
|
RTC_LOG(LS_WARNING) << "Received RED packet with length exceeds maximum IP "
|
|
|
|
|
"packet size; dropping.";
|
2019-09-20 11:40:12 +02:00
|
|
|
return false;
|
2018-05-18 13:48:58 +02:00
|
|
|
}
|
2016-08-10 00:51:50 -07:00
|
|
|
|
2019-09-20 11:40:12 +02:00
|
|
|
static constexpr uint8_t kRedHeaderLength = 1;
|
2012-01-20 06:59:06 +00:00
|
|
|
|
2019-09-20 11:40:12 +02:00
|
|
|
if (rtp_packet.payload_size() == 0) {
|
2017-11-09 11:09:25 +01:00
|
|
|
RTC_LOG(LS_WARNING) << "Corrupt/truncated FEC packet.";
|
2019-09-20 11:40:12 +02:00
|
|
|
return false;
|
2015-06-29 07:22:04 -07:00
|
|
|
}
|
|
|
|
|
|
2016-08-10 00:51:50 -07:00
|
|
|
// Remove RED header of incoming packet and store as a virtual RTP packet.
|
2019-09-20 11:40:12 +02:00
|
|
|
auto received_packet =
|
|
|
|
|
std::make_unique<ForwardErrorCorrection::ReceivedPacket>();
|
2016-07-07 09:40:51 +02:00
|
|
|
received_packet->pkt = new ForwardErrorCorrection::Packet();
|
2012-01-20 06:59:06 +00:00
|
|
|
|
2016-08-10 00:51:50 -07:00
|
|
|
// Get payload type from RED header and sequence number from RTP header.
|
2019-09-20 11:40:12 +02:00
|
|
|
uint8_t payload_type = rtp_packet.payload()[0] & 0x7f;
|
2022-08-10 07:45:43 +02:00
|
|
|
received_packet->is_fec = payload_type == ulpfec_payload_type_;
|
2020-03-11 12:59:07 +01:00
|
|
|
received_packet->is_recovered = rtp_packet.recovered();
|
2019-09-20 11:40:12 +02:00
|
|
|
received_packet->ssrc = rtp_packet.Ssrc();
|
|
|
|
|
received_packet->seq_num = rtp_packet.SequenceNumber();
|
2012-01-20 06:59:06 +00:00
|
|
|
|
2019-09-20 11:40:12 +02:00
|
|
|
if (rtp_packet.payload()[0] & 0x80) {
|
2016-08-10 00:51:50 -07:00
|
|
|
// f bit set in RED header, i.e. there are more than one RED header blocks.
|
2019-06-27 10:08:50 +02:00
|
|
|
// WebRTC never generates multiple blocks in a RED packet for FEC.
|
|
|
|
|
RTC_LOG(LS_WARNING) << "More than 1 block in RED packet is not supported.";
|
2019-09-20 11:40:12 +02:00
|
|
|
return false;
|
2012-01-20 06:59:06 +00:00
|
|
|
}
|
2019-06-27 10:08:50 +02:00
|
|
|
|
2015-01-15 07:40:20 +00:00
|
|
|
++packet_counter_.num_packets;
|
2019-09-20 11:40:12 +02:00
|
|
|
packet_counter_.num_bytes += rtp_packet.size();
|
2022-08-10 07:45:43 +02:00
|
|
|
if (packet_counter_.first_packet_time == Timestamp::MinusInfinity()) {
|
|
|
|
|
packet_counter_.first_packet_time = clock_->CurrentTime();
|
2016-11-30 01:42:26 -08:00
|
|
|
}
|
2011-07-07 08:21:25 +00:00
|
|
|
|
2019-06-27 10:08:50 +02:00
|
|
|
if (received_packet->is_fec) {
|
2015-01-15 07:40:20 +00:00
|
|
|
++packet_counter_.num_fec_packets;
|
2012-01-20 06:59:06 +00:00
|
|
|
// everything behind the RED header
|
2019-09-25 14:37:10 +02:00
|
|
|
received_packet->pkt->data =
|
|
|
|
|
rtp_packet.Buffer().Slice(rtp_packet.headers_size() + kRedHeaderLength,
|
|
|
|
|
rtp_packet.payload_size() - kRedHeaderLength);
|
2012-01-20 06:59:06 +00:00
|
|
|
} else {
|
2021-03-16 19:24:58 +01:00
|
|
|
received_packet->pkt->data.EnsureCapacity(rtp_packet.size() -
|
|
|
|
|
kRedHeaderLength);
|
2016-08-10 00:51:50 -07:00
|
|
|
// Copy RTP header.
|
2019-09-20 11:40:12 +02:00
|
|
|
received_packet->pkt->data.SetData(rtp_packet.data(),
|
|
|
|
|
rtp_packet.headers_size());
|
2016-08-10 00:51:50 -07:00
|
|
|
// Set payload type.
|
2021-01-07 15:24:05 +01:00
|
|
|
uint8_t& payload_type_byte = received_packet->pkt->data.MutableData()[1];
|
|
|
|
|
payload_type_byte &= 0x80; // Reset RED payload type.
|
|
|
|
|
payload_type_byte += payload_type; // Set media payload type.
|
2021-03-16 19:24:58 +01:00
|
|
|
// Copy payload and padding data, after the RED header.
|
|
|
|
|
received_packet->pkt->data.AppendData(
|
|
|
|
|
rtp_packet.data() + rtp_packet.headers_size() + kRedHeaderLength,
|
|
|
|
|
rtp_packet.size() - rtp_packet.headers_size() - kRedHeaderLength);
|
2012-01-20 06:59:06 +00:00
|
|
|
}
|
2011-07-07 08:21:25 +00:00
|
|
|
|
2019-09-20 11:40:12 +02:00
|
|
|
if (received_packet->pkt->data.size() > 0) {
|
|
|
|
|
received_packets_.push_back(std::move(received_packet));
|
2012-01-20 06:59:06 +00:00
|
|
|
}
|
2019-09-20 11:40:12 +02:00
|
|
|
return true;
|
2011-07-07 08:21:25 +00:00
|
|
|
}
|
|
|
|
|
|
2022-08-09 17:50:45 +02:00
|
|
|
void UlpfecReceiver::ProcessReceivedFec() {
|
2021-01-19 13:30:23 +01:00
|
|
|
RTC_DCHECK_RUN_ON(&sequence_checker_);
|
2018-01-19 14:41:41 +01:00
|
|
|
|
2021-07-28 23:57:33 +02:00
|
|
|
// If we iterate over `received_packets_` and it contains a packet that cause
|
2018-01-19 14:41:41 +01:00
|
|
|
// us to recurse back to this function (for example a RED packet encapsulating
|
|
|
|
|
// a RED packet), then we will recurse forever. To avoid this we swap
|
2021-07-28 23:57:33 +02:00
|
|
|
// `received_packets_` with an empty vector so that the next recursive call
|
2018-01-19 14:41:41 +01:00
|
|
|
// wont iterate over the same packet again. This also solves the problem of
|
|
|
|
|
// not modifying the vector we are currently iterating over (packets are added
|
|
|
|
|
// in AddReceivedRedPacket).
|
|
|
|
|
std::vector<std::unique_ptr<ForwardErrorCorrection::ReceivedPacket>>
|
|
|
|
|
received_packets;
|
|
|
|
|
received_packets.swap(received_packets_);
|
|
|
|
|
|
|
|
|
|
for (const auto& received_packet : received_packets) {
|
2012-03-30 16:16:21 +00:00
|
|
|
// Send received media packet to VCM.
|
2017-09-18 07:58:59 -07:00
|
|
|
if (!received_packet->is_fec) {
|
2022-04-21 15:06:35 +02:00
|
|
|
ForwardErrorCorrection::Packet* packet = received_packet->pkt.get();
|
2019-09-03 11:07:37 +02:00
|
|
|
recovered_packet_callback_->OnRecoveredPacket(packet->data.data(),
|
|
|
|
|
packet->data.size());
|
|
|
|
|
// Create a packet with the buffer to modify it.
|
2019-06-26 14:39:36 +02:00
|
|
|
RtpPacketReceived rtp_packet;
|
2019-09-25 14:37:10 +02:00
|
|
|
const uint8_t* const original_data = packet->data.cdata();
|
2019-09-30 11:36:42 +02:00
|
|
|
if (!rtp_packet.Parse(packet->data)) {
|
|
|
|
|
RTC_LOG(LS_WARNING) << "Corrupted media packet";
|
|
|
|
|
} else {
|
|
|
|
|
rtp_packet.IdentifyExtensions(extensions_);
|
|
|
|
|
// Reset buffer reference, so zeroing would work on a buffer with a
|
|
|
|
|
// single reference.
|
|
|
|
|
packet->data = rtc::CopyOnWriteBuffer(0);
|
|
|
|
|
rtp_packet.ZeroMutableExtensions();
|
|
|
|
|
packet->data = rtp_packet.Buffer();
|
|
|
|
|
// Ensure that zeroing of extensions was done in place.
|
|
|
|
|
RTC_DCHECK_EQ(packet->data.cdata(), original_data);
|
|
|
|
|
}
|
2012-03-30 16:16:21 +00:00
|
|
|
}
|
2020-03-11 12:59:07 +01:00
|
|
|
if (!received_packet->is_recovered) {
|
|
|
|
|
// Do not pass recovered packets to FEC. Recovered packet might have
|
|
|
|
|
// different set of the RTP header extensions and thus different byte
|
|
|
|
|
// representation than the original packet, That will corrupt
|
|
|
|
|
// FEC calculation.
|
|
|
|
|
fec_->DecodeFec(*received_packet, &recovered_packets_);
|
|
|
|
|
}
|
2012-01-20 06:59:06 +00:00
|
|
|
}
|
2017-09-18 07:58:59 -07:00
|
|
|
|
2012-03-30 16:16:21 +00:00
|
|
|
// Send any recovered media packets to VCM.
|
2016-08-10 00:51:50 -07:00
|
|
|
for (const auto& recovered_packet : recovered_packets_) {
|
2016-07-07 09:40:51 +02:00
|
|
|
if (recovered_packet->returned) {
|
|
|
|
|
// Already sent to the VCM and the jitter buffer.
|
2012-02-09 12:34:52 +00:00
|
|
|
continue;
|
2016-07-07 09:40:51 +02:00
|
|
|
}
|
2022-04-21 15:06:35 +02:00
|
|
|
ForwardErrorCorrection::Packet* packet = recovered_packet->pkt.get();
|
2015-01-15 07:40:20 +00:00
|
|
|
++packet_counter_.num_recovered_packets;
|
2017-08-25 09:08:44 -07:00
|
|
|
// Set this flag first; in case the recovered packet carries a RED
|
|
|
|
|
// header, OnRecoveredPacket will recurse back here.
|
|
|
|
|
recovered_packet->returned = true;
|
2019-09-03 11:07:37 +02:00
|
|
|
recovered_packet_callback_->OnRecoveredPacket(packet->data.data(),
|
|
|
|
|
packet->data.size());
|
2012-01-20 06:59:06 +00:00
|
|
|
}
|
2011-12-16 17:21:09 +00:00
|
|
|
}
|
2012-02-09 12:34:52 +00:00
|
|
|
|
2013-05-16 15:06:28 +00:00
|
|
|
} // namespace webrtc
|