/* * Copyright (c) 2012 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. */ #include "modules/rtp_rtcp/source/fec_test_helper.h" #include #include #include "modules/rtp_rtcp/include/rtp_rtcp_defines.h" #include "modules/rtp_rtcp/source/byte_io.h" #include "modules/rtp_rtcp/source/rtp_packet.h" #include "rtc_base/checks.h" namespace webrtc { namespace test { namespace fec { namespace { constexpr uint8_t kRtpMarkerBitMask = 0x80; constexpr uint8_t kFecPayloadType = 96; constexpr uint8_t kRedPayloadType = 97; constexpr uint8_t kVp8PayloadType = 120; constexpr int kPacketTimestampIncrement = 3000; } // namespace MediaPacketGenerator::MediaPacketGenerator(uint32_t min_packet_size, uint32_t max_packet_size, uint32_t ssrc, Random* random) : min_packet_size_(min_packet_size), max_packet_size_(max_packet_size), ssrc_(ssrc), random_(random) {} MediaPacketGenerator::~MediaPacketGenerator() = default; ForwardErrorCorrection::PacketList MediaPacketGenerator::ConstructMediaPackets( int num_media_packets, uint16_t start_seq_num) { RTC_DCHECK_GT(num_media_packets, 0); uint16_t seq_num = start_seq_num; int time_stamp = random_->Rand(); ForwardErrorCorrection::PacketList media_packets; for (int i = 0; i < num_media_packets; ++i) { std::unique_ptr media_packet( new ForwardErrorCorrection::Packet()); media_packet->data.SetSize( random_->Rand(min_packet_size_, max_packet_size_)); uint8_t* data = media_packet->data.MutableData(); // Generate random values for the first 2 bytes data[0] = random_->Rand(); data[1] = random_->Rand(); // The first two bits are assumed to be 10 by the FEC encoder. // In fact the FEC decoder will set the two first bits to 10 regardless of // what they actually were. Set the first two bits to 10 so that a memcmp // can be performed for the whole restored packet. data[0] |= 0x80; data[0] &= 0xbf; // FEC is applied to a whole frame. // A frame is signaled by multiple packets without the marker bit set // followed by the last packet of the frame for which the marker bit is set. // Only push one (fake) frame to the FEC. data[1] &= 0x7f; webrtc::ByteWriter::WriteBigEndian(&data[2], seq_num); webrtc::ByteWriter::WriteBigEndian(&data[4], time_stamp); webrtc::ByteWriter::WriteBigEndian(&data[8], ssrc_); // Generate random values for payload. for (size_t j = 12; j < media_packet->data.size(); ++j) data[j] = random_->Rand(); seq_num++; media_packets.push_back(std::move(media_packet)); } // Last packet, set marker bit. ForwardErrorCorrection::Packet* media_packet = media_packets.back().get(); RTC_DCHECK(media_packet); media_packet->data.MutableData()[1] |= 0x80; next_seq_num_ = seq_num; return media_packets; } ForwardErrorCorrection::PacketList MediaPacketGenerator::ConstructMediaPackets( int num_media_packets) { return ConstructMediaPackets(num_media_packets, random_->Rand()); } uint16_t MediaPacketGenerator::GetNextSeqNum() { return next_seq_num_; } AugmentedPacketGenerator::AugmentedPacketGenerator(uint32_t ssrc) : num_packets_(0), ssrc_(ssrc), seq_num_(0), timestamp_(0) {} void AugmentedPacketGenerator::NewFrame(size_t num_packets) { num_packets_ = num_packets; timestamp_ += kPacketTimestampIncrement; } uint16_t AugmentedPacketGenerator::NextPacketSeqNum() { return ++seq_num_; } std::unique_ptr AugmentedPacketGenerator::NextPacket( size_t offset, size_t length) { std::unique_ptr packet(new AugmentedPacket()); packet->data.SetSize(length + kRtpHeaderSize); uint8_t* data = packet->data.MutableData(); for (size_t i = 0; i < length; ++i) data[i + kRtpHeaderSize] = offset + i; packet->data.SetSize(length + kRtpHeaderSize); packet->header.headerLength = kRtpHeaderSize; packet->header.markerBit = (num_packets_ == 1); packet->header.payloadType = kVp8PayloadType; packet->header.sequenceNumber = seq_num_; packet->header.timestamp = timestamp_; packet->header.ssrc = ssrc_; WriteRtpHeader(packet->header, data); ++seq_num_; --num_packets_; return packet; } void AugmentedPacketGenerator::WriteRtpHeader(const RTPHeader& header, uint8_t* data) { data[0] = 0x80; // Version 2. data[1] = header.payloadType; data[1] |= (header.markerBit ? kRtpMarkerBitMask : 0); ByteWriter::WriteBigEndian(data + 2, header.sequenceNumber); ByteWriter::WriteBigEndian(data + 4, header.timestamp); ByteWriter::WriteBigEndian(data + 8, header.ssrc); } FlexfecPacketGenerator::FlexfecPacketGenerator(uint32_t media_ssrc, uint32_t flexfec_ssrc) : AugmentedPacketGenerator(media_ssrc), flexfec_ssrc_(flexfec_ssrc), flexfec_seq_num_(0), flexfec_timestamp_(0) {} std::unique_ptr FlexfecPacketGenerator::BuildFlexfecPacket( const ForwardErrorCorrection::Packet& packet) { RTC_DCHECK_LE(packet.data.size(), static_cast(IP_PACKET_SIZE - kRtpHeaderSize)); RTPHeader header; header.sequenceNumber = flexfec_seq_num_; ++flexfec_seq_num_; header.timestamp = flexfec_timestamp_; flexfec_timestamp_ += kPacketTimestampIncrement; header.ssrc = flexfec_ssrc_; std::unique_ptr packet_with_rtp_header( new AugmentedPacket()); packet_with_rtp_header->data.SetSize(kRtpHeaderSize + packet.data.size()); WriteRtpHeader(header, packet_with_rtp_header->data.MutableData()); memcpy(packet_with_rtp_header->data.MutableData() + kRtpHeaderSize, packet.data.cdata(), packet.data.size()); return packet_with_rtp_header; } UlpfecPacketGenerator::UlpfecPacketGenerator(uint32_t ssrc) : AugmentedPacketGenerator(ssrc) {} RtpPacketReceived UlpfecPacketGenerator::BuildMediaRedPacket( const AugmentedPacket& packet, bool is_recovered) { // Create a temporary buffer used to wrap the media packet in RED. rtc::CopyOnWriteBuffer red_buffer; const size_t kHeaderLength = packet.header.headerLength; // Append header. red_buffer.SetData(packet.data.data(), kHeaderLength); // Find payload type and add it as RED header. uint8_t media_payload_type = red_buffer[1] & 0x7F; red_buffer.AppendData({media_payload_type}); // Append rest of payload/padding. red_buffer.AppendData( packet.data.Slice(kHeaderLength, packet.data.size() - kHeaderLength)); RtpPacketReceived red_packet; RTC_CHECK(red_packet.Parse(std::move(red_buffer))); red_packet.SetPayloadType(kRedPayloadType); red_packet.set_recovered(is_recovered); return red_packet; } RtpPacketReceived UlpfecPacketGenerator::BuildUlpfecRedPacket( const ForwardErrorCorrection::Packet& packet) { // Create a fake media packet to get a correct header. 1 byte RED header. ++num_packets_; std::unique_ptr fake_packet = NextPacket(0, packet.data.size() + 1); RtpPacketReceived red_packet; red_packet.Parse(fake_packet->data); red_packet.SetMarker(false); uint8_t* rtp_payload = red_packet.AllocatePayload(packet.data.size() + 1); rtp_payload[0] = kFecPayloadType; red_packet.SetPayloadType(kRedPayloadType); memcpy(rtp_payload + 1, packet.data.cdata(), packet.data.size()); red_packet.set_recovered(false); return red_packet; } } // namespace fec } // namespace test } // namespace webrtc