blob: 23e66c23bfafaf266b66cbb094124806eeb4ab36 [file] [log] [blame]
stefan@webrtc.orge0d6fa42012-03-20 22:10:56 +00001/*
2 * Copyright (c) 2012 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/rtp_rtcp/source/fec_test_helper.h"
stefan@webrtc.orge0d6fa42012-03-20 22:10:56 +000012
brandtrece4aba2016-09-20 23:16:28 -070013#include <memory>
14#include <utility>
15
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020016#include "modules/rtp_rtcp/include/rtp_rtcp_defines.h"
17#include "modules/rtp_rtcp/source/byte_io.h"
Danil Chapovalov04fd2152019-09-20 11:40:12 +020018#include "modules/rtp_rtcp/source/rtp_packet.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020019#include "rtc_base/checks.h"
stefan@webrtc.orge0d6fa42012-03-20 22:10:56 +000020
21namespace webrtc {
brandtrece4aba2016-09-20 23:16:28 -070022namespace test {
23namespace fec {
24
25namespace {
Danil Chapovalov64a59f12021-07-27 22:17:04 +020026
27constexpr uint8_t kRtpMarkerBitMask = 0x80;
28
brandtrece4aba2016-09-20 23:16:28 -070029constexpr uint8_t kFecPayloadType = 96;
30constexpr uint8_t kRedPayloadType = 97;
31constexpr uint8_t kVp8PayloadType = 120;
brandtr8d02ea72016-10-03 23:47:05 -070032
33constexpr int kPacketTimestampIncrement = 3000;
brandtrece4aba2016-09-20 23:16:28 -070034} // namespace
stefan@webrtc.orge0d6fa42012-03-20 22:10:56 +000035
Danil Chapovalovdd7e2842018-03-09 15:37:03 +000036MediaPacketGenerator::MediaPacketGenerator(uint32_t min_packet_size,
37 uint32_t max_packet_size,
38 uint32_t ssrc,
39 Random* random)
40 : min_packet_size_(min_packet_size),
41 max_packet_size_(max_packet_size),
42 ssrc_(ssrc),
43 random_(random) {}
44
45MediaPacketGenerator::~MediaPacketGenerator() = default;
46
brandtrece4aba2016-09-20 23:16:28 -070047ForwardErrorCorrection::PacketList MediaPacketGenerator::ConstructMediaPackets(
48 int num_media_packets,
49 uint16_t start_seq_num) {
50 RTC_DCHECK_GT(num_media_packets, 0);
51 uint16_t seq_num = start_seq_num;
52 int time_stamp = random_->Rand<int>();
53
54 ForwardErrorCorrection::PacketList media_packets;
55
56 for (int i = 0; i < num_media_packets; ++i) {
57 std::unique_ptr<ForwardErrorCorrection::Packet> media_packet(
58 new ForwardErrorCorrection::Packet());
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +020059 media_packet->data.SetSize(
60 random_->Rand(min_packet_size_, max_packet_size_));
brandtrece4aba2016-09-20 23:16:28 -070061
Danil Chapovalove15dc582021-01-07 15:24:05 +010062 uint8_t* data = media_packet->data.MutableData();
brandtrece4aba2016-09-20 23:16:28 -070063 // Generate random values for the first 2 bytes
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +020064 data[0] = random_->Rand<uint8_t>();
65 data[1] = random_->Rand<uint8_t>();
brandtrece4aba2016-09-20 23:16:28 -070066
67 // The first two bits are assumed to be 10 by the FEC encoder.
68 // In fact the FEC decoder will set the two first bits to 10 regardless of
69 // what they actually were. Set the first two bits to 10 so that a memcmp
70 // can be performed for the whole restored packet.
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +020071 data[0] |= 0x80;
72 data[0] &= 0xbf;
brandtrece4aba2016-09-20 23:16:28 -070073
74 // FEC is applied to a whole frame.
75 // A frame is signaled by multiple packets without the marker bit set
76 // followed by the last packet of the frame for which the marker bit is set.
77 // Only push one (fake) frame to the FEC.
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +020078 data[1] &= 0x7f;
brandtrece4aba2016-09-20 23:16:28 -070079
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +020080 webrtc::ByteWriter<uint16_t>::WriteBigEndian(&data[2], seq_num);
81 webrtc::ByteWriter<uint32_t>::WriteBigEndian(&data[4], time_stamp);
82 webrtc::ByteWriter<uint32_t>::WriteBigEndian(&data[8], ssrc_);
brandtrece4aba2016-09-20 23:16:28 -070083
84 // Generate random values for payload.
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +020085 for (size_t j = 12; j < media_packet->data.size(); ++j)
86 data[j] = random_->Rand<uint8_t>();
brandtrece4aba2016-09-20 23:16:28 -070087 seq_num++;
88 media_packets.push_back(std::move(media_packet));
89 }
90 // Last packet, set marker bit.
91 ForwardErrorCorrection::Packet* media_packet = media_packets.back().get();
92 RTC_DCHECK(media_packet);
Danil Chapovalove15dc582021-01-07 15:24:05 +010093 media_packet->data.MutableData()[1] |= 0x80;
brandtrece4aba2016-09-20 23:16:28 -070094
brandtrd726a3f2017-06-29 02:45:35 -070095 next_seq_num_ = seq_num;
brandtrece4aba2016-09-20 23:16:28 -070096
97 return media_packets;
98}
99
100ForwardErrorCorrection::PacketList MediaPacketGenerator::ConstructMediaPackets(
101 int num_media_packets) {
102 return ConstructMediaPackets(num_media_packets, random_->Rand<uint16_t>());
103}
104
brandtrd726a3f2017-06-29 02:45:35 -0700105uint16_t MediaPacketGenerator::GetNextSeqNum() {
106 return next_seq_num_;
brandtrece4aba2016-09-20 23:16:28 -0700107}
108
brandtr0aabdac2016-10-03 06:36:43 -0700109AugmentedPacketGenerator::AugmentedPacketGenerator(uint32_t ssrc)
110 : num_packets_(0), ssrc_(ssrc), seq_num_(0), timestamp_(0) {}
brandtra4545ee2016-10-03 02:58:45 -0700111
brandtr0aabdac2016-10-03 06:36:43 -0700112void AugmentedPacketGenerator::NewFrame(size_t num_packets) {
brandtra4545ee2016-10-03 02:58:45 -0700113 num_packets_ = num_packets;
brandtr8d02ea72016-10-03 23:47:05 -0700114 timestamp_ += kPacketTimestampIncrement;
brandtra4545ee2016-10-03 02:58:45 -0700115}
116
brandtr0aabdac2016-10-03 06:36:43 -0700117uint16_t AugmentedPacketGenerator::NextPacketSeqNum() {
brandtra4545ee2016-10-03 02:58:45 -0700118 return ++seq_num_;
119}
120
brandtr0aabdac2016-10-03 06:36:43 -0700121std::unique_ptr<AugmentedPacket> AugmentedPacketGenerator::NextPacket(
122 size_t offset,
123 size_t length) {
124 std::unique_ptr<AugmentedPacket> packet(new AugmentedPacket());
125
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +0200126 packet->data.SetSize(length + kRtpHeaderSize);
Danil Chapovalove15dc582021-01-07 15:24:05 +0100127 uint8_t* data = packet->data.MutableData();
brandtra4545ee2016-10-03 02:58:45 -0700128 for (size_t i = 0; i < length; ++i)
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +0200129 data[i + kRtpHeaderSize] = offset + i;
130 packet->data.SetSize(length + kRtpHeaderSize);
Niels Möller70a83942019-03-12 09:54:30 +0100131 packet->header.headerLength = kRtpHeaderSize;
132 packet->header.markerBit = (num_packets_ == 1);
133 packet->header.payloadType = kVp8PayloadType;
134 packet->header.sequenceNumber = seq_num_;
135 packet->header.timestamp = timestamp_;
136 packet->header.ssrc = ssrc_;
Danil Chapovalove15dc582021-01-07 15:24:05 +0100137 WriteRtpHeader(packet->header, data);
brandtra4545ee2016-10-03 02:58:45 -0700138 ++seq_num_;
139 --num_packets_;
brandtr0aabdac2016-10-03 06:36:43 -0700140
141 return packet;
brandtra4545ee2016-10-03 02:58:45 -0700142}
143
brandtr0aabdac2016-10-03 06:36:43 -0700144void AugmentedPacketGenerator::WriteRtpHeader(const RTPHeader& header,
145 uint8_t* data) {
146 data[0] = 0x80; // Version 2.
147 data[1] = header.payloadType;
148 data[1] |= (header.markerBit ? kRtpMarkerBitMask : 0);
149 ByteWriter<uint16_t>::WriteBigEndian(data + 2, header.sequenceNumber);
150 ByteWriter<uint32_t>::WriteBigEndian(data + 4, header.timestamp);
151 ByteWriter<uint32_t>::WriteBigEndian(data + 8, header.ssrc);
152}
153
brandtr8d02ea72016-10-03 23:47:05 -0700154FlexfecPacketGenerator::FlexfecPacketGenerator(uint32_t media_ssrc,
155 uint32_t flexfec_ssrc)
156 : AugmentedPacketGenerator(media_ssrc),
157 flexfec_ssrc_(flexfec_ssrc),
158 flexfec_seq_num_(0),
159 flexfec_timestamp_(0) {}
160
161std::unique_ptr<AugmentedPacket> FlexfecPacketGenerator::BuildFlexfecPacket(
162 const ForwardErrorCorrection::Packet& packet) {
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +0200163 RTC_DCHECK_LE(packet.data.size(),
brandtr8d02ea72016-10-03 23:47:05 -0700164 static_cast<size_t>(IP_PACKET_SIZE - kRtpHeaderSize));
165
166 RTPHeader header;
167 header.sequenceNumber = flexfec_seq_num_;
168 ++flexfec_seq_num_;
169 header.timestamp = flexfec_timestamp_;
170 flexfec_timestamp_ += kPacketTimestampIncrement;
171 header.ssrc = flexfec_ssrc_;
172
173 std::unique_ptr<AugmentedPacket> packet_with_rtp_header(
174 new AugmentedPacket());
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +0200175 packet_with_rtp_header->data.SetSize(kRtpHeaderSize + packet.data.size());
Danil Chapovalove15dc582021-01-07 15:24:05 +0100176 WriteRtpHeader(header, packet_with_rtp_header->data.MutableData());
177 memcpy(packet_with_rtp_header->data.MutableData() + kRtpHeaderSize,
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +0200178 packet.data.cdata(), packet.data.size());
brandtr8d02ea72016-10-03 23:47:05 -0700179
180 return packet_with_rtp_header;
181}
182
brandtr0aabdac2016-10-03 06:36:43 -0700183UlpfecPacketGenerator::UlpfecPacketGenerator(uint32_t ssrc)
184 : AugmentedPacketGenerator(ssrc) {}
185
Artem Titov68173942020-03-11 12:59:07 +0100186RtpPacketReceived UlpfecPacketGenerator::BuildMediaRedPacket(
187 const AugmentedPacket& packet,
188 bool is_recovered) {
Erik Språngf19aec82021-03-16 19:24:58 +0100189 // Create a temporary buffer used to wrap the media packet in RED.
190 rtc::CopyOnWriteBuffer red_buffer;
Danil Chapovalov04fd2152019-09-20 11:40:12 +0200191 const size_t kHeaderLength = packet.header.headerLength;
Erik Språngf19aec82021-03-16 19:24:58 +0100192 // Append header.
193 red_buffer.SetData(packet.data.data(), kHeaderLength);
194 // Find payload type and add it as RED header.
195 uint8_t media_payload_type = red_buffer[1] & 0x7F;
196 red_buffer.AppendData({media_payload_type});
197 // Append rest of payload/padding.
198 red_buffer.AppendData(
199 packet.data.Slice(kHeaderLength, packet.data.size() - kHeaderLength));
200
201 RtpPacketReceived red_packet;
202 RTC_CHECK(red_packet.Parse(std::move(red_buffer)));
Danil Chapovalov04fd2152019-09-20 11:40:12 +0200203 red_packet.SetPayloadType(kRedPayloadType);
Artem Titov68173942020-03-11 12:59:07 +0100204 red_packet.set_recovered(is_recovered);
brandtr0aabdac2016-10-03 06:36:43 -0700205
brandtra4545ee2016-10-03 02:58:45 -0700206 return red_packet;
207}
208
Artem Titov68173942020-03-11 12:59:07 +0100209RtpPacketReceived UlpfecPacketGenerator::BuildUlpfecRedPacket(
brandtr0aabdac2016-10-03 06:36:43 -0700210 const ForwardErrorCorrection::Packet& packet) {
brandtra4545ee2016-10-03 02:58:45 -0700211 // Create a fake media packet to get a correct header. 1 byte RED header.
212 ++num_packets_;
Danil Chapovalov04fd2152019-09-20 11:40:12 +0200213 std::unique_ptr<AugmentedPacket> fake_packet =
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +0200214 NextPacket(0, packet.data.size() + 1);
brandtr0aabdac2016-10-03 06:36:43 -0700215
Artem Titov68173942020-03-11 12:59:07 +0100216 RtpPacketReceived red_packet;
Danil Chapovalov04fd2152019-09-20 11:40:12 +0200217 red_packet.Parse(fake_packet->data);
218 red_packet.SetMarker(false);
219 uint8_t* rtp_payload = red_packet.AllocatePayload(packet.data.size() + 1);
220 rtp_payload[0] = kFecPayloadType;
221 red_packet.SetPayloadType(kRedPayloadType);
Danil Chapovalov04fd2152019-09-20 11:40:12 +0200222 memcpy(rtp_payload + 1, packet.data.cdata(), packet.data.size());
Artem Titov68173942020-03-11 12:59:07 +0100223 red_packet.set_recovered(false);
brandtr0aabdac2016-10-03 06:36:43 -0700224
brandtra4545ee2016-10-03 02:58:45 -0700225 return red_packet;
226}
227
brandtrece4aba2016-09-20 23:16:28 -0700228} // namespace fec
229} // namespace test
stefan@webrtc.orge0d6fa42012-03-20 22:10:56 +0000230} // namespace webrtc