blob: e94e9b075ed6e4d6f2a6dce1fdc6fc1271c040fa [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"
18#include "modules/rtp_rtcp/source/rtp_utility.h"
19#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 {
26constexpr uint8_t kFecPayloadType = 96;
27constexpr uint8_t kRedPayloadType = 97;
28constexpr uint8_t kVp8PayloadType = 120;
brandtr8d02ea72016-10-03 23:47:05 -070029
30constexpr int kPacketTimestampIncrement = 3000;
brandtrece4aba2016-09-20 23:16:28 -070031} // namespace
stefan@webrtc.orge0d6fa42012-03-20 22:10:56 +000032
Danil Chapovalovdd7e2842018-03-09 15:37:03 +000033MediaPacketGenerator::MediaPacketGenerator(uint32_t min_packet_size,
34 uint32_t max_packet_size,
35 uint32_t ssrc,
36 Random* random)
37 : min_packet_size_(min_packet_size),
38 max_packet_size_(max_packet_size),
39 ssrc_(ssrc),
40 random_(random) {}
41
42MediaPacketGenerator::~MediaPacketGenerator() = default;
43
brandtrece4aba2016-09-20 23:16:28 -070044ForwardErrorCorrection::PacketList MediaPacketGenerator::ConstructMediaPackets(
45 int num_media_packets,
46 uint16_t start_seq_num) {
47 RTC_DCHECK_GT(num_media_packets, 0);
48 uint16_t seq_num = start_seq_num;
49 int time_stamp = random_->Rand<int>();
50
51 ForwardErrorCorrection::PacketList media_packets;
52
53 for (int i = 0; i < num_media_packets; ++i) {
54 std::unique_ptr<ForwardErrorCorrection::Packet> media_packet(
55 new ForwardErrorCorrection::Packet());
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +020056 media_packet->data.SetSize(
57 random_->Rand(min_packet_size_, max_packet_size_));
brandtrece4aba2016-09-20 23:16:28 -070058
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +020059 uint8_t* data = media_packet->data.data();
brandtrece4aba2016-09-20 23:16:28 -070060 // Generate random values for the first 2 bytes
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +020061 data[0] = random_->Rand<uint8_t>();
62 data[1] = random_->Rand<uint8_t>();
brandtrece4aba2016-09-20 23:16:28 -070063
64 // The first two bits are assumed to be 10 by the FEC encoder.
65 // In fact the FEC decoder will set the two first bits to 10 regardless of
66 // what they actually were. Set the first two bits to 10 so that a memcmp
67 // can be performed for the whole restored packet.
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +020068 data[0] |= 0x80;
69 data[0] &= 0xbf;
brandtrece4aba2016-09-20 23:16:28 -070070
71 // FEC is applied to a whole frame.
72 // A frame is signaled by multiple packets without the marker bit set
73 // followed by the last packet of the frame for which the marker bit is set.
74 // Only push one (fake) frame to the FEC.
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +020075 data[1] &= 0x7f;
brandtrece4aba2016-09-20 23:16:28 -070076
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +020077 webrtc::ByteWriter<uint16_t>::WriteBigEndian(&data[2], seq_num);
78 webrtc::ByteWriter<uint32_t>::WriteBigEndian(&data[4], time_stamp);
79 webrtc::ByteWriter<uint32_t>::WriteBigEndian(&data[8], ssrc_);
brandtrece4aba2016-09-20 23:16:28 -070080
81 // Generate random values for payload.
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +020082 for (size_t j = 12; j < media_packet->data.size(); ++j)
83 data[j] = random_->Rand<uint8_t>();
brandtrece4aba2016-09-20 23:16:28 -070084 seq_num++;
85 media_packets.push_back(std::move(media_packet));
86 }
87 // Last packet, set marker bit.
88 ForwardErrorCorrection::Packet* media_packet = media_packets.back().get();
89 RTC_DCHECK(media_packet);
90 media_packet->data[1] |= 0x80;
91
brandtrd726a3f2017-06-29 02:45:35 -070092 next_seq_num_ = seq_num;
brandtrece4aba2016-09-20 23:16:28 -070093
94 return media_packets;
95}
96
97ForwardErrorCorrection::PacketList MediaPacketGenerator::ConstructMediaPackets(
98 int num_media_packets) {
99 return ConstructMediaPackets(num_media_packets, random_->Rand<uint16_t>());
100}
101
brandtrd726a3f2017-06-29 02:45:35 -0700102uint16_t MediaPacketGenerator::GetNextSeqNum() {
103 return next_seq_num_;
brandtrece4aba2016-09-20 23:16:28 -0700104}
105
brandtr0aabdac2016-10-03 06:36:43 -0700106AugmentedPacketGenerator::AugmentedPacketGenerator(uint32_t ssrc)
107 : num_packets_(0), ssrc_(ssrc), seq_num_(0), timestamp_(0) {}
brandtra4545ee2016-10-03 02:58:45 -0700108
brandtr0aabdac2016-10-03 06:36:43 -0700109void AugmentedPacketGenerator::NewFrame(size_t num_packets) {
brandtra4545ee2016-10-03 02:58:45 -0700110 num_packets_ = num_packets;
brandtr8d02ea72016-10-03 23:47:05 -0700111 timestamp_ += kPacketTimestampIncrement;
brandtra4545ee2016-10-03 02:58:45 -0700112}
113
brandtr0aabdac2016-10-03 06:36:43 -0700114uint16_t AugmentedPacketGenerator::NextPacketSeqNum() {
brandtra4545ee2016-10-03 02:58:45 -0700115 return ++seq_num_;
116}
117
brandtr0aabdac2016-10-03 06:36:43 -0700118std::unique_ptr<AugmentedPacket> AugmentedPacketGenerator::NextPacket(
119 size_t offset,
120 size_t length) {
121 std::unique_ptr<AugmentedPacket> packet(new AugmentedPacket());
122
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +0200123 packet->data.SetSize(length + kRtpHeaderSize);
124 uint8_t* data = packet->data.data();
brandtra4545ee2016-10-03 02:58:45 -0700125 for (size_t i = 0; i < length; ++i)
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +0200126 data[i + kRtpHeaderSize] = offset + i;
127 packet->data.SetSize(length + kRtpHeaderSize);
Niels Möller70a83942019-03-12 09:54:30 +0100128 packet->header.headerLength = kRtpHeaderSize;
129 packet->header.markerBit = (num_packets_ == 1);
130 packet->header.payloadType = kVp8PayloadType;
131 packet->header.sequenceNumber = seq_num_;
132 packet->header.timestamp = timestamp_;
133 packet->header.ssrc = ssrc_;
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +0200134 WriteRtpHeader(packet->header, packet->data.data());
brandtra4545ee2016-10-03 02:58:45 -0700135 ++seq_num_;
136 --num_packets_;
brandtr0aabdac2016-10-03 06:36:43 -0700137
138 return packet;
brandtra4545ee2016-10-03 02:58:45 -0700139}
140
brandtr0aabdac2016-10-03 06:36:43 -0700141void AugmentedPacketGenerator::WriteRtpHeader(const RTPHeader& header,
142 uint8_t* data) {
143 data[0] = 0x80; // Version 2.
144 data[1] = header.payloadType;
145 data[1] |= (header.markerBit ? kRtpMarkerBitMask : 0);
146 ByteWriter<uint16_t>::WriteBigEndian(data + 2, header.sequenceNumber);
147 ByteWriter<uint32_t>::WriteBigEndian(data + 4, header.timestamp);
148 ByteWriter<uint32_t>::WriteBigEndian(data + 8, header.ssrc);
149}
150
brandtr8d02ea72016-10-03 23:47:05 -0700151FlexfecPacketGenerator::FlexfecPacketGenerator(uint32_t media_ssrc,
152 uint32_t flexfec_ssrc)
153 : AugmentedPacketGenerator(media_ssrc),
154 flexfec_ssrc_(flexfec_ssrc),
155 flexfec_seq_num_(0),
156 flexfec_timestamp_(0) {}
157
158std::unique_ptr<AugmentedPacket> FlexfecPacketGenerator::BuildFlexfecPacket(
159 const ForwardErrorCorrection::Packet& packet) {
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +0200160 RTC_DCHECK_LE(packet.data.size(),
brandtr8d02ea72016-10-03 23:47:05 -0700161 static_cast<size_t>(IP_PACKET_SIZE - kRtpHeaderSize));
162
163 RTPHeader header;
164 header.sequenceNumber = flexfec_seq_num_;
165 ++flexfec_seq_num_;
166 header.timestamp = flexfec_timestamp_;
167 flexfec_timestamp_ += kPacketTimestampIncrement;
168 header.ssrc = flexfec_ssrc_;
169
170 std::unique_ptr<AugmentedPacket> packet_with_rtp_header(
171 new AugmentedPacket());
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +0200172 packet_with_rtp_header->data.SetSize(kRtpHeaderSize + packet.data.size());
173 WriteRtpHeader(header, packet_with_rtp_header->data.data());
174 memcpy(packet_with_rtp_header->data.data() + kRtpHeaderSize,
175 packet.data.cdata(), packet.data.size());
brandtr8d02ea72016-10-03 23:47:05 -0700176
177 return packet_with_rtp_header;
178}
179
brandtr0aabdac2016-10-03 06:36:43 -0700180UlpfecPacketGenerator::UlpfecPacketGenerator(uint32_t ssrc)
181 : AugmentedPacketGenerator(ssrc) {}
182
183std::unique_ptr<AugmentedPacket> UlpfecPacketGenerator::BuildMediaRedPacket(
184 const AugmentedPacket& packet) {
185 std::unique_ptr<AugmentedPacket> red_packet(new AugmentedPacket());
186
Niels Möller70a83942019-03-12 09:54:30 +0100187 const size_t kHeaderLength = packet.header.headerLength;
brandtr0aabdac2016-10-03 06:36:43 -0700188 red_packet->header = packet.header;
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +0200189 red_packet->data.SetSize(packet.data.size() + 1);
brandtra4545ee2016-10-03 02:58:45 -0700190 // Copy RTP header.
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +0200191 memcpy(red_packet->data.data(), packet.data.cdata(), kHeaderLength);
brandtr0aabdac2016-10-03 06:36:43 -0700192 SetRedHeader(red_packet->data[1] & 0x7f, kHeaderLength, red_packet.get());
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +0200193 memcpy(red_packet->data.data() + kHeaderLength + 1,
194 packet.data.cdata() + kHeaderLength,
195 packet.data.size() - kHeaderLength);
brandtr0aabdac2016-10-03 06:36:43 -0700196
brandtra4545ee2016-10-03 02:58:45 -0700197 return red_packet;
198}
199
brandtr0aabdac2016-10-03 06:36:43 -0700200std::unique_ptr<AugmentedPacket> UlpfecPacketGenerator::BuildUlpfecRedPacket(
201 const ForwardErrorCorrection::Packet& packet) {
brandtra4545ee2016-10-03 02:58:45 -0700202 // Create a fake media packet to get a correct header. 1 byte RED header.
203 ++num_packets_;
brandtr0aabdac2016-10-03 06:36:43 -0700204 std::unique_ptr<AugmentedPacket> red_packet =
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +0200205 NextPacket(0, packet.data.size() + 1);
brandtr0aabdac2016-10-03 06:36:43 -0700206
brandtra4545ee2016-10-03 02:58:45 -0700207 red_packet->data[1] &= ~0x80; // Clear marker bit.
Niels Möller70a83942019-03-12 09:54:30 +0100208 const size_t kHeaderLength = red_packet->header.headerLength;
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +0200209 red_packet->data.SetSize(kHeaderLength + 1 + packet.data.size());
brandtr0aabdac2016-10-03 06:36:43 -0700210 SetRedHeader(kFecPayloadType, kHeaderLength, red_packet.get());
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +0200211 memcpy(red_packet->data.data() + kHeaderLength + 1, packet.data.cdata(),
212 packet.data.size());
brandtr0aabdac2016-10-03 06:36:43 -0700213
brandtra4545ee2016-10-03 02:58:45 -0700214 return red_packet;
215}
216
brandtr0aabdac2016-10-03 06:36:43 -0700217void UlpfecPacketGenerator::SetRedHeader(uint8_t payload_type,
218 size_t header_length,
219 AugmentedPacket* red_packet) {
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +0200220 uint8_t* data = red_packet->data.data();
brandtr0aabdac2016-10-03 06:36:43 -0700221 // Replace payload type.
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +0200222 data[1] &= 0x80; // Reset.
223 data[1] += kRedPayloadType; // Replace.
brandtra4545ee2016-10-03 02:58:45 -0700224
225 // Add RED header, f-bit always 0.
Ilya Nikolaevskiya5d952f2019-09-03 11:07:37 +0200226 data[header_length] = payload_type;
brandtra4545ee2016-10-03 02:58:45 -0700227}
228
brandtrece4aba2016-09-20 23:16:28 -0700229} // namespace fec
230} // namespace test
stefan@webrtc.orge0d6fa42012-03-20 22:10:56 +0000231} // namespace webrtc