blob: 374cb6f8cd759bbe1b469b8cf47602675ff32813 [file] [log] [blame]
danilchapb7374db2016-11-24 09:06:05 -08001/*
2 * Copyright (c) 2016 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/rtp_utility.h"
Danil Chapovalov07633bd2017-06-01 17:10:51 +020012
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020013#include "modules/rtp_rtcp/include/rtp_header_extension_map.h"
14#include "modules/rtp_rtcp/source/rtp_header_extensions.h"
15#include "test/gmock.h"
16#include "test/gtest.h"
danilchapb7374db2016-11-24 09:06:05 -080017
18namespace webrtc {
19namespace {
20
21using ::testing::ElementsAreArray;
22using ::testing::make_tuple;
23
24const int8_t kPayloadType = 100;
25const uint32_t kSsrc = 0x12345678;
26const uint16_t kSeqNum = 88;
27const uint32_t kTimestamp = 0x65431278;
28
29} // namespace
30
31TEST(RtpHeaderParser, ParseMinimum) {
32 // clang-format off
33 const uint8_t kPacket[] = {
34 0x80, kPayloadType, 0x00, kSeqNum,
35 0x65, 0x43, 0x12, 0x78, // kTimestamp.
36 0x12, 0x34, 0x56, 0x78}; // kSsrc.
37 // clang-format on
38 RtpUtility::RtpHeaderParser parser(kPacket, sizeof(kPacket));
39 RTPHeader header;
40
41 EXPECT_TRUE(parser.Parse(&header, nullptr));
42
43 EXPECT_EQ(kPayloadType, header.payloadType);
44 EXPECT_EQ(kSeqNum, header.sequenceNumber);
45 EXPECT_EQ(kTimestamp, header.timestamp);
46 EXPECT_EQ(kSsrc, header.ssrc);
47 EXPECT_EQ(0u, header.paddingLength);
48 EXPECT_EQ(sizeof(kPacket), header.headerLength);
49}
50
51TEST(RtpHeaderParser, ParseWithExtension) {
52 // clang-format off
53 const uint8_t kPacket[] = {
54 0x90, kPayloadType, 0x00, kSeqNum,
55 0x65, 0x43, 0x12, 0x78, // kTimestamp.
56 0x12, 0x34, 0x56, 0x78, // kSsrc.
57 0xbe, 0xde, 0x00, 0x01, // Extension block of size 1 x 32bit words.
58 0x12, 0x01, 0x56, 0xce};
59 // clang-format on
60 RtpHeaderExtensionMap extensions;
61 extensions.Register<TransmissionOffset>(1);
62 RtpUtility::RtpHeaderParser parser(kPacket, sizeof(kPacket));
63 RTPHeader header;
64
65 EXPECT_TRUE(parser.Parse(&header, &extensions));
66
67 EXPECT_EQ(kPayloadType, header.payloadType);
68 EXPECT_EQ(kSeqNum, header.sequenceNumber);
69 EXPECT_EQ(kTimestamp, header.timestamp);
70 EXPECT_EQ(kSsrc, header.ssrc);
71
72 ASSERT_TRUE(header.extension.hasTransmissionTimeOffset);
73 EXPECT_EQ(0x156ce, header.extension.transmissionTimeOffset);
74}
75
76TEST(RtpHeaderParser, ParseWithInvalidSizedExtension) {
77 const size_t kPayloadSize = 7;
78 // clang-format off
79 const uint8_t kPacket[] = {
80 0x90, kPayloadType, 0x00, kSeqNum,
81 0x65, 0x43, 0x12, 0x78, // kTimestamp.
82 0x12, 0x34, 0x56, 0x78, // kSsrc.
83 0xbe, 0xde, 0x00, 0x02, // Extension block of size 2 x 32bit words.
84 0x16, // (6+1)-bytes, but Transmission Offset expected to be 3-bytes.
85 'e', 'x', 't',
86 'd', 'a', 't', 'a',
87 'p', 'a', 'y', 'l', 'o', 'a', 'd'
88 };
89 // clang-format on
90 RtpHeaderExtensionMap extensions;
91 extensions.Register<TransmissionOffset>(1);
92 RtpUtility::RtpHeaderParser parser(kPacket, sizeof(kPacket));
93 RTPHeader header;
94
95 EXPECT_TRUE(parser.Parse(&header, &extensions));
96
97 // Extension should be ignored.
98 EXPECT_FALSE(header.extension.hasTransmissionTimeOffset);
99 // But shouldn't prevent reading payload.
100 EXPECT_THAT(sizeof(kPacket) - kPayloadSize, header.headerLength);
101}
102
103TEST(RtpHeaderParser, ParseWithExtensionPadding) {
104 // clang-format off
105 const uint8_t kPacket[] = {
106 0x90, kPayloadType, 0x00, kSeqNum,
107 0x65, 0x43, 0x12, 0x78, // kTimestamp.
108 0x12, 0x34, 0x56, 0x78, // kSsrc.
109 0xbe, 0xde, 0x00, 0x02, // Extension of size 1x32bit word.
110 0x02, // A byte of (invalid) padding.
111 0x12, 0x1a, 0xda, 0x03, // TransmissionOffset extension.
112 0x0f, 0x00, 0x03, // More invalid padding bytes: id=0, but len > 0.
113 };
114 // clang-format on
115 RtpHeaderExtensionMap extensions;
116 extensions.Register<TransmissionOffset>(1);
117 RtpUtility::RtpHeaderParser parser(kPacket, sizeof(kPacket));
118 RTPHeader header;
119
120 EXPECT_TRUE(parser.Parse(&header, &extensions));
121
122 // Parse should skip padding and read extension.
123 EXPECT_TRUE(header.extension.hasTransmissionTimeOffset);
124 EXPECT_EQ(0x1ada03, header.extension.transmissionTimeOffset);
125 EXPECT_EQ(sizeof(kPacket), header.headerLength);
126}
127
128TEST(RtpHeaderParser, ParseWithOverSizedExtension) {
129 // clang-format off
130 const uint8_t kPacket[] = {
131 0x90, kPayloadType, 0x00, kSeqNum,
132 0x65, 0x43, 0x12, 0x78, // kTimestamp.
133 0x12, 0x34, 0x56, 0x78, // kSsrc.
134 0xbe, 0xde, 0x00, 0x01, // Extension of size 1x32bit word.
135 0x00, // Add a byte of padding.
136 0x12, // Extension id 1 size (2+1).
137 0xda, 0x1a // Only 2 bytes of extension payload.
138 };
139 // clang-format on
140 RtpHeaderExtensionMap extensions;
141 extensions.Register<TransmissionOffset>(1);
142 RtpUtility::RtpHeaderParser parser(kPacket, sizeof(kPacket));
143 RTPHeader header;
144
145 EXPECT_TRUE(parser.Parse(&header, &extensions));
146
147 // Parse should ignore extension.
148 EXPECT_FALSE(header.extension.hasTransmissionTimeOffset);
149 EXPECT_EQ(sizeof(kPacket), header.headerLength);
150}
151
Chen Xingcd8a6e22019-07-01 10:56:51 +0200152TEST(RtpHeaderParser, ParseAll9Extensions) {
danilchapb7374db2016-11-24 09:06:05 -0800153 const uint8_t kAudioLevel = 0x5a;
154 // clang-format off
155 const uint8_t kPacket[] = {
156 0x90, kPayloadType, 0x00, kSeqNum,
157 0x65, 0x43, 0x12, 0x78, // kTimestamp.
158 0x12, 0x34, 0x56, 0x78, // kSsrc.
Chen Xingcd8a6e22019-07-01 10:56:51 +0200159 0xbe, 0xde, 0x00, 0x0c, // Extension of size 12x32bit words.
danilchapb7374db2016-11-24 09:06:05 -0800160 0x40, 0x80|kAudioLevel, // AudioLevel.
161 0x22, 0x01, 0x56, 0xce, // TransmissionOffset.
162 0x62, 0x12, 0x34, 0x56, // AbsoluteSendTime.
Chen Xingcd8a6e22019-07-01 10:56:51 +0200163 0x7f, 0x12, 0x34, 0x56, 0x78, // AbsoluteCaptureTime.
164 0x90, 0xab, 0xcd, 0xef, // AbsoluteCaptureTime. (cont.)
165 0x11, 0x22, 0x33, 0x44, // AbsoluteCaptureTime. (cont.)
166 0x55, 0x66, 0x77, 0x88, // AbsoluteCaptureTime. (cont.)
danilchapb7374db2016-11-24 09:06:05 -0800167 0x81, 0xce, 0xab, // TransportSequenceNumber.
168 0xa0, 0x03, // VideoRotation.
169 0xb2, 0x12, 0x48, 0x76, // PlayoutDelayLimits.
danilchapef8d7732017-04-19 02:59:48 -0700170 0xc2, 'r', 't', 'x', // RtpStreamId
171 0xd5, 's', 't', 'r', 'e', 'a', 'm', // RepairedRtpStreamId
Chen Xingcd8a6e22019-07-01 10:56:51 +0200172 0x00, // Padding to 32bit boundary.
danilchapb7374db2016-11-24 09:06:05 -0800173 };
174 // clang-format on
175 ASSERT_EQ(sizeof(kPacket) % 4, 0u);
176
177 RtpHeaderExtensionMap extensions;
178 extensions.Register<TransmissionOffset>(2);
179 extensions.Register<AudioLevel>(4);
180 extensions.Register<AbsoluteSendTime>(6);
Chen Xingcd8a6e22019-07-01 10:56:51 +0200181 extensions.Register<AbsoluteCaptureTimeExtension>(7);
danilchapb7374db2016-11-24 09:06:05 -0800182 extensions.Register<TransportSequenceNumber>(8);
183 extensions.Register<VideoOrientation>(0xa);
184 extensions.Register<PlayoutDelayLimits>(0xb);
danilchapef8d7732017-04-19 02:59:48 -0700185 extensions.Register<RtpStreamId>(0xc);
186 extensions.Register<RepairedRtpStreamId>(0xd);
danilchapb7374db2016-11-24 09:06:05 -0800187 RtpUtility::RtpHeaderParser parser(kPacket, sizeof(kPacket));
188 RTPHeader header;
189
190 EXPECT_TRUE(parser.Parse(&header, &extensions));
191
192 EXPECT_TRUE(header.extension.hasTransmissionTimeOffset);
193 EXPECT_EQ(0x156ce, header.extension.transmissionTimeOffset);
194
195 EXPECT_TRUE(header.extension.hasAudioLevel);
196 EXPECT_TRUE(header.extension.voiceActivity);
197 EXPECT_EQ(kAudioLevel, header.extension.audioLevel);
198
199 EXPECT_TRUE(header.extension.hasAbsoluteSendTime);
200 EXPECT_EQ(0x123456U, header.extension.absoluteSendTime);
201
Chen Xingcd8a6e22019-07-01 10:56:51 +0200202 ASSERT_TRUE(header.extension.absolute_capture_time.has_value());
203 EXPECT_EQ(0x1234567890abcdefULL,
204 header.extension.absolute_capture_time->absolute_capture_timestamp);
205 ASSERT_TRUE(header.extension.absolute_capture_time
206 ->estimated_capture_clock_offset.has_value());
207 EXPECT_EQ(0x1122334455667788LL, header.extension.absolute_capture_time
208 ->estimated_capture_clock_offset.value());
209
danilchapb7374db2016-11-24 09:06:05 -0800210 EXPECT_TRUE(header.extension.hasTransportSequenceNumber);
211 EXPECT_EQ(0xceab, header.extension.transportSequenceNumber);
212
213 EXPECT_TRUE(header.extension.hasVideoRotation);
214 EXPECT_EQ(kVideoRotation_270, header.extension.videoRotation);
215
216 EXPECT_EQ(0x124 * PlayoutDelayLimits::kGranularityMs,
217 header.extension.playout_delay.min_ms);
218 EXPECT_EQ(0x876 * PlayoutDelayLimits::kGranularityMs,
219 header.extension.playout_delay.max_ms);
Niels Möllerd57efc12019-03-22 14:02:11 +0100220 EXPECT_EQ(header.extension.stream_id, "rtx");
221 EXPECT_EQ(header.extension.repaired_stream_id, "stream");
danilchapef8d7732017-04-19 02:59:48 -0700222}
223
224TEST(RtpHeaderParser, ParseMalformedRsidExtensions) {
225 // clang-format off
226 const uint8_t kPacket[] = {
227 0x90, kPayloadType, 0x00, kSeqNum,
228 0x65, 0x43, 0x12, 0x78, // kTimestamp.
229 0x12, 0x34, 0x56, 0x78, // kSsrc.
230 0xbe, 0xde, 0x00, 0x03, // Extension of size 3x32bit words.
231 0xc2, '\0', 't', 'x', // empty RtpStreamId
232 0xd5, 's', 't', 'r', '\0', 'a', 'm', // RepairedRtpStreamId
233 0x00, // Padding to 32bit boundary.
234 };
235 // clang-format on
236 ASSERT_EQ(sizeof(kPacket) % 4, 0u);
237
238 RtpHeaderExtensionMap extensions;
239 extensions.Register<RtpStreamId>(0xc);
240 extensions.Register<RepairedRtpStreamId>(0xd);
241 RtpUtility::RtpHeaderParser parser(kPacket, sizeof(kPacket));
242 RTPHeader header;
243
244 EXPECT_TRUE(parser.Parse(&header, &extensions));
245 EXPECT_TRUE(header.extension.stream_id.empty());
Niels Möllerd57efc12019-03-22 14:02:11 +0100246 EXPECT_TRUE(header.extension.repaired_stream_id.empty());
danilchapb7374db2016-11-24 09:06:05 -0800247}
248
249TEST(RtpHeaderParser, ParseWithCsrcsExtensionAndPadding) {
250 const uint8_t kPacketPaddingSize = 8;
251 const uint32_t kCsrcs[] = {0x34567890, 0x32435465};
252 const size_t kPayloadSize = 7;
253 // clang-format off
254 const uint8_t kPacket[] = {
255 0xb2, kPayloadType, 0x00, kSeqNum,
256 0x65, 0x43, 0x12, 0x78, // kTimestamp.
257 0x12, 0x34, 0x56, 0x78, // kSsrc.
258 0x34, 0x56, 0x78, 0x90, // kCsrcs[0].
259 0x32, 0x43, 0x54, 0x65, // kCsrcs[1].
260 0xbe, 0xde, 0x00, 0x01, // Extension.
261 0x12, 0x00, 0x56, 0xce, // TransmissionTimeOffset with id = 1.
262 'p', 'a', 'y', 'l', 'o', 'a', 'd',
263 'p', 'a', 'd', 'd', 'i', 'n', 'g', kPacketPaddingSize};
264 // clang-format on
265 RtpHeaderExtensionMap extensions;
266 extensions.Register<TransmissionOffset>(1);
267 RtpUtility::RtpHeaderParser parser(kPacket, sizeof(kPacket));
268 RTPHeader header;
269
270 EXPECT_TRUE(parser.Parse(&header, &extensions));
271
272 EXPECT_EQ(kPayloadType, header.payloadType);
273 EXPECT_EQ(kSeqNum, header.sequenceNumber);
274 EXPECT_EQ(kTimestamp, header.timestamp);
275 EXPECT_EQ(kSsrc, header.ssrc);
276 EXPECT_THAT(make_tuple(header.arrOfCSRCs, header.numCSRCs),
277 ElementsAreArray(kCsrcs));
278 EXPECT_EQ(kPacketPaddingSize, header.paddingLength);
279 EXPECT_THAT(sizeof(kPacket) - kPayloadSize - kPacketPaddingSize,
280 header.headerLength);
281 EXPECT_TRUE(header.extension.hasTransmissionTimeOffset);
282 EXPECT_EQ(0x56ce, header.extension.transmissionTimeOffset);
283}
284
285} // namespace webrtc