blob: d73af5a31c8880033861db10741f8ae10d618fb6 [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
brandtrece4aba2016-09-20 23:16:28 -070033ForwardErrorCorrection::PacketList MediaPacketGenerator::ConstructMediaPackets(
34 int num_media_packets,
35 uint16_t start_seq_num) {
36 RTC_DCHECK_GT(num_media_packets, 0);
37 uint16_t seq_num = start_seq_num;
38 int time_stamp = random_->Rand<int>();
39
40 ForwardErrorCorrection::PacketList media_packets;
41
42 for (int i = 0; i < num_media_packets; ++i) {
43 std::unique_ptr<ForwardErrorCorrection::Packet> media_packet(
44 new ForwardErrorCorrection::Packet());
45 media_packet->length = random_->Rand(min_packet_size_, max_packet_size_);
46
47 // Generate random values for the first 2 bytes
48 media_packet->data[0] = random_->Rand<uint8_t>();
49 media_packet->data[1] = random_->Rand<uint8_t>();
50
51 // The first two bits are assumed to be 10 by the FEC encoder.
52 // In fact the FEC decoder will set the two first bits to 10 regardless of
53 // what they actually were. Set the first two bits to 10 so that a memcmp
54 // can be performed for the whole restored packet.
55 media_packet->data[0] |= 0x80;
56 media_packet->data[0] &= 0xbf;
57
58 // FEC is applied to a whole frame.
59 // A frame is signaled by multiple packets without the marker bit set
60 // followed by the last packet of the frame for which the marker bit is set.
61 // Only push one (fake) frame to the FEC.
62 media_packet->data[1] &= 0x7f;
63
64 webrtc::ByteWriter<uint16_t>::WriteBigEndian(&media_packet->data[2],
65 seq_num);
66 webrtc::ByteWriter<uint32_t>::WriteBigEndian(&media_packet->data[4],
67 time_stamp);
68 webrtc::ByteWriter<uint32_t>::WriteBigEndian(&media_packet->data[8], ssrc_);
69
70 // Generate random values for payload.
brandtr0aabdac2016-10-03 06:36:43 -070071 for (size_t j = 12; j < media_packet->length; ++j)
brandtrece4aba2016-09-20 23:16:28 -070072 media_packet->data[j] = random_->Rand<uint8_t>();
brandtrece4aba2016-09-20 23:16:28 -070073 seq_num++;
74 media_packets.push_back(std::move(media_packet));
75 }
76 // Last packet, set marker bit.
77 ForwardErrorCorrection::Packet* media_packet = media_packets.back().get();
78 RTC_DCHECK(media_packet);
79 media_packet->data[1] |= 0x80;
80
brandtrd726a3f2017-06-29 02:45:35 -070081 next_seq_num_ = seq_num;
brandtrece4aba2016-09-20 23:16:28 -070082
83 return media_packets;
84}
85
86ForwardErrorCorrection::PacketList MediaPacketGenerator::ConstructMediaPackets(
87 int num_media_packets) {
88 return ConstructMediaPackets(num_media_packets, random_->Rand<uint16_t>());
89}
90
brandtrd726a3f2017-06-29 02:45:35 -070091uint16_t MediaPacketGenerator::GetNextSeqNum() {
92 return next_seq_num_;
brandtrece4aba2016-09-20 23:16:28 -070093}
94
brandtr0aabdac2016-10-03 06:36:43 -070095AugmentedPacketGenerator::AugmentedPacketGenerator(uint32_t ssrc)
96 : num_packets_(0), ssrc_(ssrc), seq_num_(0), timestamp_(0) {}
brandtra4545ee2016-10-03 02:58:45 -070097
brandtr0aabdac2016-10-03 06:36:43 -070098void AugmentedPacketGenerator::NewFrame(size_t num_packets) {
brandtra4545ee2016-10-03 02:58:45 -070099 num_packets_ = num_packets;
brandtr8d02ea72016-10-03 23:47:05 -0700100 timestamp_ += kPacketTimestampIncrement;
brandtra4545ee2016-10-03 02:58:45 -0700101}
102
brandtr0aabdac2016-10-03 06:36:43 -0700103uint16_t AugmentedPacketGenerator::NextPacketSeqNum() {
brandtra4545ee2016-10-03 02:58:45 -0700104 return ++seq_num_;
105}
106
brandtr0aabdac2016-10-03 06:36:43 -0700107std::unique_ptr<AugmentedPacket> AugmentedPacketGenerator::NextPacket(
108 size_t offset,
109 size_t length) {
110 std::unique_ptr<AugmentedPacket> packet(new AugmentedPacket());
111
brandtra4545ee2016-10-03 02:58:45 -0700112 for (size_t i = 0; i < length; ++i)
brandtr0aabdac2016-10-03 06:36:43 -0700113 packet->data[i + kRtpHeaderSize] = offset + i;
114 packet->length = length + kRtpHeaderSize;
115 memset(&packet->header, 0, sizeof(WebRtcRTPHeader));
116 packet->header.frameType = kVideoFrameDelta;
117 packet->header.header.headerLength = kRtpHeaderSize;
118 packet->header.header.markerBit = (num_packets_ == 1);
119 packet->header.header.payloadType = kVp8PayloadType;
120 packet->header.header.sequenceNumber = seq_num_;
121 packet->header.header.timestamp = timestamp_;
122 packet->header.header.ssrc = ssrc_;
123 WriteRtpHeader(packet->header.header, packet->data);
brandtra4545ee2016-10-03 02:58:45 -0700124 ++seq_num_;
125 --num_packets_;
brandtr0aabdac2016-10-03 06:36:43 -0700126
127 return packet;
brandtra4545ee2016-10-03 02:58:45 -0700128}
129
brandtr0aabdac2016-10-03 06:36:43 -0700130void AugmentedPacketGenerator::WriteRtpHeader(const RTPHeader& header,
131 uint8_t* data) {
132 data[0] = 0x80; // Version 2.
133 data[1] = header.payloadType;
134 data[1] |= (header.markerBit ? kRtpMarkerBitMask : 0);
135 ByteWriter<uint16_t>::WriteBigEndian(data + 2, header.sequenceNumber);
136 ByteWriter<uint32_t>::WriteBigEndian(data + 4, header.timestamp);
137 ByteWriter<uint32_t>::WriteBigEndian(data + 8, header.ssrc);
138}
139
brandtr8d02ea72016-10-03 23:47:05 -0700140FlexfecPacketGenerator::FlexfecPacketGenerator(uint32_t media_ssrc,
141 uint32_t flexfec_ssrc)
142 : AugmentedPacketGenerator(media_ssrc),
143 flexfec_ssrc_(flexfec_ssrc),
144 flexfec_seq_num_(0),
145 flexfec_timestamp_(0) {}
146
147std::unique_ptr<AugmentedPacket> FlexfecPacketGenerator::BuildFlexfecPacket(
148 const ForwardErrorCorrection::Packet& packet) {
149 RTC_DCHECK_LE(packet.length,
150 static_cast<size_t>(IP_PACKET_SIZE - kRtpHeaderSize));
151
152 RTPHeader header;
153 header.sequenceNumber = flexfec_seq_num_;
154 ++flexfec_seq_num_;
155 header.timestamp = flexfec_timestamp_;
156 flexfec_timestamp_ += kPacketTimestampIncrement;
157 header.ssrc = flexfec_ssrc_;
158
159 std::unique_ptr<AugmentedPacket> packet_with_rtp_header(
160 new AugmentedPacket());
161 WriteRtpHeader(header, packet_with_rtp_header->data);
162 memcpy(packet_with_rtp_header->data + kRtpHeaderSize, packet.data,
163 packet.length);
164 packet_with_rtp_header->length = kRtpHeaderSize + packet.length;
165
166 return packet_with_rtp_header;
167}
168
brandtr0aabdac2016-10-03 06:36:43 -0700169UlpfecPacketGenerator::UlpfecPacketGenerator(uint32_t ssrc)
170 : AugmentedPacketGenerator(ssrc) {}
171
172std::unique_ptr<AugmentedPacket> UlpfecPacketGenerator::BuildMediaRedPacket(
173 const AugmentedPacket& packet) {
174 std::unique_ptr<AugmentedPacket> red_packet(new AugmentedPacket());
175
176 const size_t kHeaderLength = packet.header.header.headerLength;
177 red_packet->header = packet.header;
178 red_packet->length = packet.length + 1; // 1 byte RED header.
brandtra4545ee2016-10-03 02:58:45 -0700179 // Copy RTP header.
brandtr0aabdac2016-10-03 06:36:43 -0700180 memcpy(red_packet->data, packet.data, kHeaderLength);
181 SetRedHeader(red_packet->data[1] & 0x7f, kHeaderLength, red_packet.get());
182 memcpy(red_packet->data + kHeaderLength + 1, packet.data + kHeaderLength,
183 packet.length - kHeaderLength);
184
brandtra4545ee2016-10-03 02:58:45 -0700185 return red_packet;
186}
187
brandtr0aabdac2016-10-03 06:36:43 -0700188std::unique_ptr<AugmentedPacket> UlpfecPacketGenerator::BuildUlpfecRedPacket(
189 const ForwardErrorCorrection::Packet& packet) {
brandtra4545ee2016-10-03 02:58:45 -0700190 // Create a fake media packet to get a correct header. 1 byte RED header.
191 ++num_packets_;
brandtr0aabdac2016-10-03 06:36:43 -0700192 std::unique_ptr<AugmentedPacket> red_packet =
193 NextPacket(0, packet.length + 1);
194
brandtra4545ee2016-10-03 02:58:45 -0700195 red_packet->data[1] &= ~0x80; // Clear marker bit.
196 const size_t kHeaderLength = red_packet->header.header.headerLength;
brandtr0aabdac2016-10-03 06:36:43 -0700197 SetRedHeader(kFecPayloadType, kHeaderLength, red_packet.get());
198 memcpy(red_packet->data + kHeaderLength + 1, packet.data, packet.length);
199 red_packet->length = kHeaderLength + 1 + packet.length;
200
brandtra4545ee2016-10-03 02:58:45 -0700201 return red_packet;
202}
203
brandtr0aabdac2016-10-03 06:36:43 -0700204void UlpfecPacketGenerator::SetRedHeader(uint8_t payload_type,
205 size_t header_length,
206 AugmentedPacket* red_packet) {
207 // Replace payload type.
brandtra4545ee2016-10-03 02:58:45 -0700208 red_packet->data[1] &= 0x80; // Reset.
209 red_packet->data[1] += kRedPayloadType; // Replace.
210
211 // Add RED header, f-bit always 0.
212 red_packet->data[header_length] = payload_type;
213}
214
brandtrece4aba2016-09-20 23:16:28 -0700215} // namespace fec
216} // namespace test
stefan@webrtc.orge0d6fa42012-03-20 22:10:56 +0000217} // namespace webrtc