blob: a75c8cffb61927bbf62510e30c974f7c6fb04890 [file] [log] [blame]
niklase@google.com470e71d2011-07-07 08:21:25 +00001/*
pwestin@webrtc.orgf6bb77a2012-01-24 17:16:59 +00002 * Copyright (c) 2012 The WebRTC project authors. All Rights Reserved.
niklase@google.com470e71d2011-07-07 08:21:25 +00003 *
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"
niklase@google.com470e71d2011-07-07 08:21:25 +000012
Yves Gerey988cc082018-10-23 12:03:01 +020013#include <assert.h>
14#include <stddef.h>
15
16#include "api/array_view.h"
17#include "api/video/video_content_type.h"
18#include "api/video/video_frame_marking.h"
19#include "api/video/video_rotation.h"
20#include "api/video/video_timing.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020021#include "modules/rtp_rtcp/include/rtp_cvo.h"
22#include "modules/rtp_rtcp/source/byte_io.h"
23#include "modules/rtp_rtcp/source/rtp_header_extensions.h"
Yves Gerey988cc082018-10-23 12:03:01 +020024#include "modules/video_coding/codecs/interface/common_constants.h"
25#include "rtc_base/checks.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020026#include "rtc_base/logging.h"
Sergey Ulanov6acefdb2017-12-11 17:38:13 -080027#include "rtc_base/stringutils.h"
niklase@google.com470e71d2011-07-07 08:21:25 +000028
niklase@google.com470e71d2011-07-07 08:21:25 +000029namespace webrtc {
pwestin@webrtc.org0644b1d2011-12-01 15:42:31 +000030
pbos@webrtc.org62bafae2014-07-08 12:10:51 +000031namespace RtpUtility {
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +000032
solenberg@webrtc.orgd8a6e722013-03-26 14:02:30 +000033enum {
solenberg@webrtc.orga5fd2f12013-06-26 08:36:07 +000034 kRtcpExpectedVersion = 2,
solenberg@webrtc.orgd8a6e722013-03-26 14:02:30 +000035 kRtcpMinHeaderLength = 4,
solenberg@webrtc.orga5fd2f12013-06-26 08:36:07 +000036 kRtcpMinParseLength = 8,
37
38 kRtpExpectedVersion = 2,
39 kRtpMinParseLength = 12
solenberg@webrtc.orgd8a6e722013-03-26 14:02:30 +000040};
41
pwestin@webrtc.org0644b1d2011-12-01 15:42:31 +000042/*
pwestin@webrtc.org0644b1d2011-12-01 15:42:31 +000043 * Misc utility routines
44 */
45
Yves Gerey665174f2018-06-19 15:03:05 +020046bool StringCompare(const char* str1, const char* str2, const uint32_t length) {
danilchapf6975f42015-12-28 10:18:46 -080047 return _strnicmp(str1, str2, length) == 0;
pwestin@webrtc.org0644b1d2011-12-01 15:42:31 +000048}
niklase@google.com470e71d2011-07-07 08:21:25 +000049
sprang@webrtc.org30933902015-03-17 14:33:12 +000050size_t Word32Align(size_t size) {
51 uint32_t remainder = size % 4;
52 if (remainder != 0)
53 return size + 4 - remainder;
54 return size;
55}
56
pbos@webrtc.org62bafae2014-07-08 12:10:51 +000057RtpHeaderParser::RtpHeaderParser(const uint8_t* rtpData,
58 const size_t rtpDataLength)
59 : _ptrRTPDataBegin(rtpData),
Yves Gerey665174f2018-06-19 15:03:05 +020060 _ptrRTPDataEnd(rtpData ? (rtpData + rtpDataLength) : NULL) {}
niklase@google.com470e71d2011-07-07 08:21:25 +000061
Yves Gerey665174f2018-06-19 15:03:05 +020062RtpHeaderParser::~RtpHeaderParser() {}
niklase@google.com470e71d2011-07-07 08:21:25 +000063
pbos@webrtc.org62bafae2014-07-08 12:10:51 +000064bool RtpHeaderParser::RTCP() const {
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +000065 // 72 to 76 is reserved for RTP
66 // 77 to 79 is not reserver but they are not assigned we will block them
67 // for RTCP 200 SR == marker bit + 72
68 // for RTCP 204 APP == marker bit + 76
69 /*
Yves Gerey665174f2018-06-19 15:03:05 +020070 * RTCP
71 *
72 * FIR full INTRA-frame request 192 [RFC2032] supported
73 * NACK negative acknowledgement 193 [RFC2032]
74 * IJ Extended inter-arrival jitter report 195 [RFC-ietf-avt-rtp-toff
75 * set-07.txt] http://tools.ietf.org/html/draft-ietf-avt-rtp-toffset-07
76 * SR sender report 200 [RFC3551] supported
77 * RR receiver report 201 [RFC3551] supported
78 * SDES source description 202 [RFC3551] supported
79 * BYE goodbye 203 [RFC3551] supported
80 * APP application-defined 204 [RFC3551] ignored
81 * RTPFB Transport layer FB message 205 [RFC4585] supported
82 * PSFB Payload-specific FB message 206 [RFC4585] supported
83 * XR extended report 207 [RFC3611] supported
84 */
niklase@google.com470e71d2011-07-07 08:21:25 +000085
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +000086 /* 205 RFC 5104
87 * FMT 1 NACK supported
88 * FMT 2 reserved
89 * FMT 3 TMMBR supported
90 * FMT 4 TMMBN supported
91 */
niklase@google.com470e71d2011-07-07 08:21:25 +000092
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +000093 /* 206 RFC 5104
Yves Gerey665174f2018-06-19 15:03:05 +020094 * FMT 1: Picture Loss Indication (PLI) supported
95 * FMT 2: Slice Lost Indication (SLI)
96 * FMT 3: Reference Picture Selection Indication (RPSI)
97 * FMT 4: Full Intra Request (FIR) Command supported
98 * FMT 5: Temporal-Spatial Trade-off Request (TSTR)
99 * FMT 6: Temporal-Spatial Trade-off Notification (TSTN)
100 * FMT 7: Video Back Channel Message (VBCM)
101 * FMT 15: Application layer FB message
102 */
niklase@google.com470e71d2011-07-07 08:21:25 +0000103
solenberg@webrtc.orgd8a6e722013-03-26 14:02:30 +0000104 const ptrdiff_t length = _ptrRTPDataEnd - _ptrRTPDataBegin;
105 if (length < kRtcpMinHeaderLength) {
106 return false;
107 }
108
pkasting@chromium.orgd3245462015-02-23 21:28:22 +0000109 const uint8_t V = _ptrRTPDataBegin[0] >> 6;
solenberg@webrtc.orgd8a6e722013-03-26 14:02:30 +0000110 if (V != kRtcpExpectedVersion) {
111 return false;
112 }
113
pkasting@chromium.orgd3245462015-02-23 21:28:22 +0000114 const uint8_t payloadType = _ptrRTPDataBegin[1];
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000115 switch (payloadType) {
niklase@google.com470e71d2011-07-07 08:21:25 +0000116 case 192:
pkasting@chromium.orgd3245462015-02-23 21:28:22 +0000117 return true;
niklase@google.com470e71d2011-07-07 08:21:25 +0000118 case 193:
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000119 // not supported
120 // pass through and check for a potential RTP packet
pkasting@chromium.orgd3245462015-02-23 21:28:22 +0000121 return false;
asapersson@webrtc.org5249cc82011-12-16 14:31:37 +0000122 case 195:
niklase@google.com470e71d2011-07-07 08:21:25 +0000123 case 200:
124 case 201:
125 case 202:
126 case 203:
127 case 204:
128 case 205:
129 case 206:
130 case 207:
pkasting@chromium.orgd3245462015-02-23 21:28:22 +0000131 return true;
132 default:
133 return false;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000134 }
niklase@google.com470e71d2011-07-07 08:21:25 +0000135}
136
pbos@webrtc.org62bafae2014-07-08 12:10:51 +0000137bool RtpHeaderParser::ParseRtcp(RTPHeader* header) const {
solenberg@webrtc.orga5fd2f12013-06-26 08:36:07 +0000138 assert(header != NULL);
139
140 const ptrdiff_t length = _ptrRTPDataEnd - _ptrRTPDataBegin;
141 if (length < kRtcpMinParseLength) {
142 return false;
143 }
144
145 const uint8_t V = _ptrRTPDataBegin[0] >> 6;
146 if (V != kRtcpExpectedVersion) {
147 return false;
148 }
149
150 const uint8_t PT = _ptrRTPDataBegin[1];
pkasting@chromium.org4591fbd2014-11-20 22:28:14 +0000151 const size_t len = (_ptrRTPDataBegin[2] << 8) + _ptrRTPDataBegin[3];
solenberg@webrtc.orga5fd2f12013-06-26 08:36:07 +0000152 const uint8_t* ptr = &_ptrRTPDataBegin[4];
153
sprang@webrtc.org779c3d12015-03-17 16:42:49 +0000154 uint32_t SSRC = ByteReader<uint32_t>::ReadBigEndian(ptr);
155 ptr += 4;
solenberg@webrtc.orga5fd2f12013-06-26 08:36:07 +0000156
Yves Gerey665174f2018-06-19 15:03:05 +0200157 header->payloadType = PT;
158 header->ssrc = SSRC;
solenberg@webrtc.orga5fd2f12013-06-26 08:36:07 +0000159 header->headerLength = 4 + (len << 2);
160
161 return true;
162}
163
Bjorn Tereliusb2bfba62018-03-05 20:52:04 +0100164bool RtpHeaderParser::Parse(
165 RTPHeader* header,
166 const RtpHeaderExtensionMap* ptrExtensionMap) const {
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000167 const ptrdiff_t length = _ptrRTPDataEnd - _ptrRTPDataBegin;
solenberg@webrtc.orga5fd2f12013-06-26 08:36:07 +0000168 if (length < kRtpMinParseLength) {
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000169 return false;
170 }
171
172 // Version
Yves Gerey665174f2018-06-19 15:03:05 +0200173 const uint8_t V = _ptrRTPDataBegin[0] >> 6;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000174 // Padding
Yves Gerey665174f2018-06-19 15:03:05 +0200175 const bool P = ((_ptrRTPDataBegin[0] & 0x20) == 0) ? false : true;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000176 // eXtension
Yves Gerey665174f2018-06-19 15:03:05 +0200177 const bool X = ((_ptrRTPDataBegin[0] & 0x10) == 0) ? false : true;
pbos@webrtc.org2f446732013-04-08 11:08:41 +0000178 const uint8_t CC = _ptrRTPDataBegin[0] & 0x0f;
Yves Gerey665174f2018-06-19 15:03:05 +0200179 const bool M = ((_ptrRTPDataBegin[1] & 0x80) == 0) ? false : true;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000180
pbos@webrtc.org2f446732013-04-08 11:08:41 +0000181 const uint8_t PT = _ptrRTPDataBegin[1] & 0x7f;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000182
Yves Gerey665174f2018-06-19 15:03:05 +0200183 const uint16_t sequenceNumber =
184 (_ptrRTPDataBegin[2] << 8) + _ptrRTPDataBegin[3];
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000185
pbos@webrtc.org2f446732013-04-08 11:08:41 +0000186 const uint8_t* ptr = &_ptrRTPDataBegin[4];
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000187
sprang@webrtc.org779c3d12015-03-17 16:42:49 +0000188 uint32_t RTPTimestamp = ByteReader<uint32_t>::ReadBigEndian(ptr);
189 ptr += 4;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000190
sprang@webrtc.org779c3d12015-03-17 16:42:49 +0000191 uint32_t SSRC = ByteReader<uint32_t>::ReadBigEndian(ptr);
192 ptr += 4;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000193
solenberg@webrtc.orga5fd2f12013-06-26 08:36:07 +0000194 if (V != kRtpExpectedVersion) {
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000195 return false;
196 }
197
pkasting@chromium.org4591fbd2014-11-20 22:28:14 +0000198 const size_t CSRCocts = CC * 4;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000199
200 if ((ptr + CSRCocts) > _ptrRTPDataEnd) {
201 return false;
202 }
203
Yves Gerey665174f2018-06-19 15:03:05 +0200204 header->markerBit = M;
205 header->payloadType = PT;
danilchapf6975f42015-12-28 10:18:46 -0800206 header->sequenceNumber = sequenceNumber;
Yves Gerey665174f2018-06-19 15:03:05 +0200207 header->timestamp = RTPTimestamp;
208 header->ssrc = SSRC;
209 header->numCSRCs = CC;
Minyue Li1a800182018-09-12 12:52:48 +0200210 if (!P) {
211 header->paddingLength = 0;
212 }
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000213
pkasting@chromium.org4591fbd2014-11-20 22:28:14 +0000214 for (uint8_t i = 0; i < CC; ++i) {
sprang@webrtc.org779c3d12015-03-17 16:42:49 +0000215 uint32_t CSRC = ByteReader<uint32_t>::ReadBigEndian(ptr);
216 ptr += 4;
danilchapf6975f42015-12-28 10:18:46 -0800217 header->arrOfCSRCs[i] = CSRC;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000218 }
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000219
Yves Gerey665174f2018-06-19 15:03:05 +0200220 header->headerLength = 12 + CSRCocts;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000221
222 // If in effect, MAY be omitted for those packets for which the offset
223 // is zero.
danilchapf6975f42015-12-28 10:18:46 -0800224 header->extension.hasTransmissionTimeOffset = false;
225 header->extension.transmissionTimeOffset = 0;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000226
solenberg@webrtc.org7ebbea12013-05-16 11:10:31 +0000227 // May not be present in packet.
danilchapf6975f42015-12-28 10:18:46 -0800228 header->extension.hasAbsoluteSendTime = false;
229 header->extension.absoluteSendTime = 0;
solenberg@webrtc.org7ebbea12013-05-16 11:10:31 +0000230
wu@webrtc.orgebdb0e32014-03-06 23:49:08 +0000231 // May not be present in packet.
danilchapf6975f42015-12-28 10:18:46 -0800232 header->extension.hasAudioLevel = false;
233 header->extension.voiceActivity = false;
234 header->extension.audioLevel = 0;
wu@webrtc.orgebdb0e32014-03-06 23:49:08 +0000235
guoweis@webrtc.org45362892015-03-04 22:55:15 +0000236 // May not be present in packet.
danilchapf6975f42015-12-28 10:18:46 -0800237 header->extension.hasVideoRotation = false;
magjed71eb61c2016-09-08 03:24:58 -0700238 header->extension.videoRotation = kVideoRotation_0;
guoweis@webrtc.org45362892015-03-04 22:55:15 +0000239
isheriff6b4b5f32016-06-08 00:24:21 -0700240 // May not be present in packet.
241 header->extension.playout_delay.min_ms = -1;
242 header->extension.playout_delay.max_ms = -1;
243
ilnik00d802b2017-04-11 10:34:31 -0700244 // May not be present in packet.
245 header->extension.hasVideoContentType = false;
246 header->extension.videoContentType = VideoContentType::UNSPECIFIED;
247
ilnik04f4d122017-06-19 07:18:55 -0700248 header->extension.has_video_timing = false;
249 header->extension.video_timing = {0u, 0u, 0u, 0u, 0u, 0u, false};
250
Johnny Leee0c8b232018-09-11 16:50:49 -0400251 header->extension.has_frame_marking = false;
252 header->extension.frame_marking = {false, false, false, false, false,
253 kNoTemporalIdx, 0, 0};
254
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000255 if (X) {
256 /* RTP header extension, RFC 3550.
257 0 1 2 3
258 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
259 +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
260 | defined by profile | length |
261 +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
262 | header extension |
263 | .... |
264 */
265 const ptrdiff_t remain = _ptrRTPDataEnd - ptr;
266 if (remain < 4) {
267 return false;
niklase@google.com470e71d2011-07-07 08:21:25 +0000268 }
269
danilchapf6975f42015-12-28 10:18:46 -0800270 header->headerLength += 4;
niklase@google.com470e71d2011-07-07 08:21:25 +0000271
sprang@webrtc.org779c3d12015-03-17 16:42:49 +0000272 uint16_t definedByProfile = ByteReader<uint16_t>::ReadBigEndian(ptr);
273 ptr += 2;
niklase@google.com470e71d2011-07-07 08:21:25 +0000274
sprang@webrtc.org779c3d12015-03-17 16:42:49 +0000275 // in 32 bit words
276 size_t XLen = ByteReader<uint16_t>::ReadBigEndian(ptr);
277 ptr += 2;
278 XLen *= 4; // in bytes
niklase@google.com470e71d2011-07-07 08:21:25 +0000279
pkasting@chromium.org4591fbd2014-11-20 22:28:14 +0000280 if (static_cast<size_t>(remain) < (4 + XLen)) {
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000281 return false;
niklase@google.com470e71d2011-07-07 08:21:25 +0000282 }
danilchap5c4f24a2017-01-23 11:10:20 -0800283 static constexpr uint16_t kRtpOneByteHeaderExtensionId = 0xBEDE;
pbos@webrtc.org3004c792013-05-07 12:36:21 +0000284 if (definedByProfile == kRtpOneByteHeaderExtensionId) {
pbos@webrtc.org2f446732013-04-08 11:08:41 +0000285 const uint8_t* ptrRTPDataExtensionEnd = ptr + XLen;
Yves Gerey665174f2018-06-19 15:03:05 +0200286 ParseOneByteExtensionHeader(header, ptrExtensionMap,
287 ptrRTPDataExtensionEnd, ptr);
niklase@google.com470e71d2011-07-07 08:21:25 +0000288 }
danilchapf6975f42015-12-28 10:18:46 -0800289 header->headerLength += XLen;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000290 }
Minyue Li1a800182018-09-12 12:52:48 +0200291 if (header->headerLength > static_cast<size_t>(length))
292 return false;
293
294 if (P) {
295 // Packet has padding.
296 if (header->headerLength != static_cast<size_t>(length)) {
297 // Packet is not header only. We can parse padding length now.
298 header->paddingLength = *(_ptrRTPDataEnd - 1);
299 } else {
300 RTC_LOG(LS_WARNING) << "Cannot parse padding length.";
301 // Packet is header only. We have no clue of the padding length.
302 return false;
303 }
304 }
305
danilchapf6975f42015-12-28 10:18:46 -0800306 if (header->headerLength + header->paddingLength >
307 static_cast<size_t>(length))
pbosbd2522a2015-07-01 05:35:53 -0700308 return false;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000309 return true;
niklase@google.com470e71d2011-07-07 08:21:25 +0000310}
311
pbos@webrtc.org62bafae2014-07-08 12:10:51 +0000312void RtpHeaderParser::ParseOneByteExtensionHeader(
danilchapf6975f42015-12-28 10:18:46 -0800313 RTPHeader* header,
asapersson@webrtc.org5249cc82011-12-16 14:31:37 +0000314 const RtpHeaderExtensionMap* ptrExtensionMap,
pbos@webrtc.org2f446732013-04-08 11:08:41 +0000315 const uint8_t* ptrRTPDataExtensionEnd,
316 const uint8_t* ptr) const {
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000317 if (!ptrExtensionMap) {
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000318 return;
319 }
320
321 while (ptrRTPDataExtensionEnd - ptr > 0) {
322 // 0
323 // 0 1 2 3 4 5 6 7
324 // +-+-+-+-+-+-+-+-+
325 // | ID | len |
326 // +-+-+-+-+-+-+-+-+
327
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000328 // Note that 'len' is the header extension element length, which is the
329 // number of bytes - 1.
Stefan Holmer3842c5c2016-01-12 13:55:00 +0100330 const int id = (*ptr & 0xf0) >> 4;
331 const int len = (*ptr & 0x0f);
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000332 ptr++;
333
danilchapb7374db2016-11-24 09:06:05 -0800334 if (id == 0) {
335 // Padding byte, skip ignoring len.
336 continue;
337 }
338
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000339 if (id == 15) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100340 RTC_LOG(LS_VERBOSE)
andresp@webrtc.orgdc80bae2014-04-08 11:06:12 +0000341 << "RTP extension header 15 encountered. Terminate parsing.";
asapersson@webrtc.org5249cc82011-12-16 14:31:37 +0000342 return;
343 }
344
Danil Chapovalov30a3a752016-06-16 15:57:15 +0200345 if (ptrRTPDataExtensionEnd - ptr < (len + 1)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100346 RTC_LOG(LS_WARNING) << "Incorrect one-byte extension len: " << (len + 1)
347 << ", bytes left in buffer: "
348 << (ptrRTPDataExtensionEnd - ptr);
Danil Chapovalov30a3a752016-06-16 15:57:15 +0200349 return;
350 }
351
danilchap14546692016-12-01 08:39:35 -0800352 RTPExtensionType type = ptrExtensionMap->GetType(id);
353 if (type == RtpHeaderExtensionMap::kInvalidType) {
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000354 // If we encounter an unknown extension, just skip over it.
Mirko Bonadei675513b2017-11-09 11:09:25 +0100355 RTC_LOG(LS_WARNING) << "Failed to find extension id: " << id;
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000356 } else {
357 switch (type) {
358 case kRtpExtensionTransmissionTimeOffset: {
359 if (len != 2) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100360 RTC_LOG(LS_WARNING)
361 << "Incorrect transmission time offset len: " << len;
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000362 return;
363 }
364 // 0 1 2 3
365 // 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
366 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
367 // | ID | len=2 | transmission offset |
368 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
asapersson@webrtc.org5249cc82011-12-16 14:31:37 +0000369
danilchapf6975f42015-12-28 10:18:46 -0800370 header->extension.transmissionTimeOffset =
sprang@webrtc.org779c3d12015-03-17 16:42:49 +0000371 ByteReader<int32_t, 3>::ReadBigEndian(ptr);
danilchapf6975f42015-12-28 10:18:46 -0800372 header->extension.hasTransmissionTimeOffset = true;
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000373 break;
374 }
375 case kRtpExtensionAudioLevel: {
376 if (len != 0) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100377 RTC_LOG(LS_WARNING) << "Incorrect audio level len: " << len;
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000378 return;
379 }
sprang@webrtc.org30933902015-03-17 14:33:12 +0000380 // 0 1
381 // 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5
382 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
383 // | ID | len=0 |V| level |
384 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000385 //
danilchapf6975f42015-12-28 10:18:46 -0800386 header->extension.audioLevel = ptr[0] & 0x7f;
387 header->extension.voiceActivity = (ptr[0] & 0x80) != 0;
388 header->extension.hasAudioLevel = true;
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000389 break;
390 }
391 case kRtpExtensionAbsoluteSendTime: {
392 if (len != 2) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100393 RTC_LOG(LS_WARNING) << "Incorrect absolute send time len: " << len;
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000394 return;
395 }
396 // 0 1 2 3
397 // 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
398 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
399 // | ID | len=2 | absolute send time |
400 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
401
danilchapf6975f42015-12-28 10:18:46 -0800402 header->extension.absoluteSendTime =
sprang@webrtc.org779c3d12015-03-17 16:42:49 +0000403 ByteReader<uint32_t, 3>::ReadBigEndian(ptr);
danilchapf6975f42015-12-28 10:18:46 -0800404 header->extension.hasAbsoluteSendTime = true;
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000405 break;
406 }
guoweis@webrtc.org45362892015-03-04 22:55:15 +0000407 case kRtpExtensionVideoRotation: {
408 if (len != 0) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100409 RTC_LOG(LS_WARNING)
guoweis@webrtc.org45362892015-03-04 22:55:15 +0000410 << "Incorrect coordination of video coordination len: " << len;
411 return;
412 }
sprang@webrtc.org779c3d12015-03-17 16:42:49 +0000413 // 0 1
414 // 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5
415 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
416 // | ID | len=0 |0 0 0 0 C F R R|
417 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
danilchapf6975f42015-12-28 10:18:46 -0800418 header->extension.hasVideoRotation = true;
magjed71eb61c2016-09-08 03:24:58 -0700419 header->extension.videoRotation =
420 ConvertCVOByteToVideoRotation(ptr[0]);
guoweis@webrtc.org45362892015-03-04 22:55:15 +0000421 break;
422 }
sprang@webrtc.org30933902015-03-17 14:33:12 +0000423 case kRtpExtensionTransportSequenceNumber: {
424 if (len != 1) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100425 RTC_LOG(LS_WARNING)
426 << "Incorrect transport sequence number len: " << len;
sprang@webrtc.org30933902015-03-17 14:33:12 +0000427 return;
428 }
429 // 0 1 2
430 // 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3
431 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
432 // | ID | L=1 |transport wide sequence number |
433 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
434
435 uint16_t sequence_number = ptr[0] << 8;
436 sequence_number += ptr[1];
danilchapf6975f42015-12-28 10:18:46 -0800437 header->extension.transportSequenceNumber = sequence_number;
438 header->extension.hasTransportSequenceNumber = true;
sprang@webrtc.org30933902015-03-17 14:33:12 +0000439 break;
440 }
isheriff6b4b5f32016-06-08 00:24:21 -0700441 case kRtpExtensionPlayoutDelay: {
442 if (len != 2) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100443 RTC_LOG(LS_WARNING) << "Incorrect playout delay len: " << len;
isheriff6b4b5f32016-06-08 00:24:21 -0700444 return;
445 }
446 // 0 1 2 3
447 // 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1
448 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
449 // | ID | len=2 | MIN delay | MAX delay |
450 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
451
452 int min_playout_delay = (ptr[0] << 4) | ((ptr[1] >> 4) & 0xf);
453 int max_playout_delay = ((ptr[1] & 0xf) << 8) | ptr[2];
454 header->extension.playout_delay.min_ms =
danilchap5c4f24a2017-01-23 11:10:20 -0800455 min_playout_delay * PlayoutDelayLimits::kGranularityMs;
isheriff6b4b5f32016-06-08 00:24:21 -0700456 header->extension.playout_delay.max_ms =
danilchap5c4f24a2017-01-23 11:10:20 -0800457 max_playout_delay * PlayoutDelayLimits::kGranularityMs;
isheriff6b4b5f32016-06-08 00:24:21 -0700458 break;
459 }
ilnik00d802b2017-04-11 10:34:31 -0700460 case kRtpExtensionVideoContentType: {
461 if (len != 0) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100462 RTC_LOG(LS_WARNING) << "Incorrect video content type len: " << len;
ilnik00d802b2017-04-11 10:34:31 -0700463 return;
464 }
465 // 0 1
466 // 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5
467 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
468 // | ID | len=0 | Content type |
469 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
470
ilnik6d5b4d62017-08-30 03:32:14 -0700471 if (videocontenttypehelpers::IsValidContentType(ptr[0])) {
ilnik00d802b2017-04-11 10:34:31 -0700472 header->extension.hasVideoContentType = true;
473 header->extension.videoContentType =
474 static_cast<VideoContentType>(ptr[0]);
475 }
476 break;
477 }
ilnik04f4d122017-06-19 07:18:55 -0700478 case kRtpExtensionVideoTiming: {
479 if (len != VideoTimingExtension::kValueSizeBytes - 1) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100480 RTC_LOG(LS_WARNING) << "Incorrect video timing len: " << len;
ilnik04f4d122017-06-19 07:18:55 -0700481 return;
482 }
483 header->extension.has_video_timing = true;
484 VideoTimingExtension::Parse(rtc::MakeArrayView(ptr, len + 1),
485 &header->extension.video_timing);
486 break;
487 }
Johnny Leee0c8b232018-09-11 16:50:49 -0400488 case kRtpExtensionFrameMarking: {
489 if (!FrameMarkingExtension::Parse(rtc::MakeArrayView(ptr, len + 1),
490 &header->extension.frame_marking)) {
491 RTC_LOG(LS_WARNING) << "Incorrect frame marking len: " << len;
492 return;
493 }
494 header->extension.has_frame_marking = true;
495 break;
496 }
danilchapef8d7732017-04-19 02:59:48 -0700497 case kRtpExtensionRtpStreamId: {
498 header->extension.stream_id.Set(rtc::MakeArrayView(ptr, len + 1));
499 break;
500 }
501 case kRtpExtensionRepairedRtpStreamId: {
502 header->extension.repaired_stream_id.Set(
503 rtc::MakeArrayView(ptr, len + 1));
504 break;
505 }
Steve Antona3251dd2017-07-21 09:58:31 -0700506 case kRtpExtensionMid: {
507 header->extension.mid.Set(rtc::MakeArrayView(ptr, len + 1));
508 break;
509 }
Danil Chapovalov916ec7d2018-06-25 18:22:15 +0200510 case kRtpExtensionGenericFrameDescriptor:
511 RTC_LOG(WARNING)
512 << "RtpGenericFrameDescriptor unsupported by rtp header parser.";
513 break;
danilchap14546692016-12-01 08:39:35 -0800514 case kRtpExtensionNone:
515 case kRtpExtensionNumberOfExtensions: {
516 RTC_NOTREACHED() << "Invalid extension type: " << type;
asapersson@webrtc.org5249cc82011-12-16 14:31:37 +0000517 return;
518 }
asapersson@webrtc.org5249cc82011-12-16 14:31:37 +0000519 }
asapersson@webrtc.org5249cc82011-12-16 14:31:37 +0000520 }
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000521 ptr += (len + 1);
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000522 }
asapersson@webrtc.org5249cc82011-12-16 14:31:37 +0000523}
524
pbos@webrtc.org62bafae2014-07-08 12:10:51 +0000525} // namespace RtpUtility
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000526} // namespace webrtc