blob: 5f5b3dc0f5fe946ba7258bb4ec5b46bcf463ffa1 [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
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020013#include "modules/rtp_rtcp/include/rtp_cvo.h"
14#include "modules/rtp_rtcp/source/byte_io.h"
15#include "modules/rtp_rtcp/source/rtp_header_extensions.h"
16#include "rtc_base/logging.h"
Sergey Ulanov6acefdb2017-12-11 17:38:13 -080017#include "rtc_base/stringutils.h"
niklase@google.com470e71d2011-07-07 08:21:25 +000018
niklase@google.com470e71d2011-07-07 08:21:25 +000019namespace webrtc {
pwestin@webrtc.org0644b1d2011-12-01 15:42:31 +000020
pbos@webrtc.org62bafae2014-07-08 12:10:51 +000021namespace RtpUtility {
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +000022
solenberg@webrtc.orgd8a6e722013-03-26 14:02:30 +000023enum {
solenberg@webrtc.orga5fd2f12013-06-26 08:36:07 +000024 kRtcpExpectedVersion = 2,
solenberg@webrtc.orgd8a6e722013-03-26 14:02:30 +000025 kRtcpMinHeaderLength = 4,
solenberg@webrtc.orga5fd2f12013-06-26 08:36:07 +000026 kRtcpMinParseLength = 8,
27
28 kRtpExpectedVersion = 2,
29 kRtpMinParseLength = 12
solenberg@webrtc.orgd8a6e722013-03-26 14:02:30 +000030};
31
pwestin@webrtc.org0644b1d2011-12-01 15:42:31 +000032/*
pwestin@webrtc.org0644b1d2011-12-01 15:42:31 +000033 * Misc utility routines
34 */
35
Yves Gerey665174f2018-06-19 15:03:05 +020036bool StringCompare(const char* str1, const char* str2, const uint32_t length) {
danilchapf6975f42015-12-28 10:18:46 -080037 return _strnicmp(str1, str2, length) == 0;
pwestin@webrtc.org0644b1d2011-12-01 15:42:31 +000038}
niklase@google.com470e71d2011-07-07 08:21:25 +000039
sprang@webrtc.org30933902015-03-17 14:33:12 +000040size_t Word32Align(size_t size) {
41 uint32_t remainder = size % 4;
42 if (remainder != 0)
43 return size + 4 - remainder;
44 return size;
45}
46
pbos@webrtc.org62bafae2014-07-08 12:10:51 +000047RtpHeaderParser::RtpHeaderParser(const uint8_t* rtpData,
48 const size_t rtpDataLength)
49 : _ptrRTPDataBegin(rtpData),
Yves Gerey665174f2018-06-19 15:03:05 +020050 _ptrRTPDataEnd(rtpData ? (rtpData + rtpDataLength) : NULL) {}
niklase@google.com470e71d2011-07-07 08:21:25 +000051
Yves Gerey665174f2018-06-19 15:03:05 +020052RtpHeaderParser::~RtpHeaderParser() {}
niklase@google.com470e71d2011-07-07 08:21:25 +000053
pbos@webrtc.org62bafae2014-07-08 12:10:51 +000054bool RtpHeaderParser::RTCP() const {
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +000055 // 72 to 76 is reserved for RTP
56 // 77 to 79 is not reserver but they are not assigned we will block them
57 // for RTCP 200 SR == marker bit + 72
58 // for RTCP 204 APP == marker bit + 76
59 /*
Yves Gerey665174f2018-06-19 15:03:05 +020060 * RTCP
61 *
62 * FIR full INTRA-frame request 192 [RFC2032] supported
63 * NACK negative acknowledgement 193 [RFC2032]
64 * IJ Extended inter-arrival jitter report 195 [RFC-ietf-avt-rtp-toff
65 * set-07.txt] http://tools.ietf.org/html/draft-ietf-avt-rtp-toffset-07
66 * SR sender report 200 [RFC3551] supported
67 * RR receiver report 201 [RFC3551] supported
68 * SDES source description 202 [RFC3551] supported
69 * BYE goodbye 203 [RFC3551] supported
70 * APP application-defined 204 [RFC3551] ignored
71 * RTPFB Transport layer FB message 205 [RFC4585] supported
72 * PSFB Payload-specific FB message 206 [RFC4585] supported
73 * XR extended report 207 [RFC3611] supported
74 */
niklase@google.com470e71d2011-07-07 08:21:25 +000075
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +000076 /* 205 RFC 5104
77 * FMT 1 NACK supported
78 * FMT 2 reserved
79 * FMT 3 TMMBR supported
80 * FMT 4 TMMBN supported
81 */
niklase@google.com470e71d2011-07-07 08:21:25 +000082
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +000083 /* 206 RFC 5104
Yves Gerey665174f2018-06-19 15:03:05 +020084 * FMT 1: Picture Loss Indication (PLI) supported
85 * FMT 2: Slice Lost Indication (SLI)
86 * FMT 3: Reference Picture Selection Indication (RPSI)
87 * FMT 4: Full Intra Request (FIR) Command supported
88 * FMT 5: Temporal-Spatial Trade-off Request (TSTR)
89 * FMT 6: Temporal-Spatial Trade-off Notification (TSTN)
90 * FMT 7: Video Back Channel Message (VBCM)
91 * FMT 15: Application layer FB message
92 */
niklase@google.com470e71d2011-07-07 08:21:25 +000093
solenberg@webrtc.orgd8a6e722013-03-26 14:02:30 +000094 const ptrdiff_t length = _ptrRTPDataEnd - _ptrRTPDataBegin;
95 if (length < kRtcpMinHeaderLength) {
96 return false;
97 }
98
pkasting@chromium.orgd3245462015-02-23 21:28:22 +000099 const uint8_t V = _ptrRTPDataBegin[0] >> 6;
solenberg@webrtc.orgd8a6e722013-03-26 14:02:30 +0000100 if (V != kRtcpExpectedVersion) {
101 return false;
102 }
103
pkasting@chromium.orgd3245462015-02-23 21:28:22 +0000104 const uint8_t payloadType = _ptrRTPDataBegin[1];
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000105 switch (payloadType) {
niklase@google.com470e71d2011-07-07 08:21:25 +0000106 case 192:
pkasting@chromium.orgd3245462015-02-23 21:28:22 +0000107 return true;
niklase@google.com470e71d2011-07-07 08:21:25 +0000108 case 193:
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000109 // not supported
110 // pass through and check for a potential RTP packet
pkasting@chromium.orgd3245462015-02-23 21:28:22 +0000111 return false;
asapersson@webrtc.org5249cc82011-12-16 14:31:37 +0000112 case 195:
niklase@google.com470e71d2011-07-07 08:21:25 +0000113 case 200:
114 case 201:
115 case 202:
116 case 203:
117 case 204:
118 case 205:
119 case 206:
120 case 207:
pkasting@chromium.orgd3245462015-02-23 21:28:22 +0000121 return true;
122 default:
123 return false;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000124 }
niklase@google.com470e71d2011-07-07 08:21:25 +0000125}
126
pbos@webrtc.org62bafae2014-07-08 12:10:51 +0000127bool RtpHeaderParser::ParseRtcp(RTPHeader* header) const {
solenberg@webrtc.orga5fd2f12013-06-26 08:36:07 +0000128 assert(header != NULL);
129
130 const ptrdiff_t length = _ptrRTPDataEnd - _ptrRTPDataBegin;
131 if (length < kRtcpMinParseLength) {
132 return false;
133 }
134
135 const uint8_t V = _ptrRTPDataBegin[0] >> 6;
136 if (V != kRtcpExpectedVersion) {
137 return false;
138 }
139
140 const uint8_t PT = _ptrRTPDataBegin[1];
pkasting@chromium.org4591fbd2014-11-20 22:28:14 +0000141 const size_t len = (_ptrRTPDataBegin[2] << 8) + _ptrRTPDataBegin[3];
solenberg@webrtc.orga5fd2f12013-06-26 08:36:07 +0000142 const uint8_t* ptr = &_ptrRTPDataBegin[4];
143
sprang@webrtc.org779c3d12015-03-17 16:42:49 +0000144 uint32_t SSRC = ByteReader<uint32_t>::ReadBigEndian(ptr);
145 ptr += 4;
solenberg@webrtc.orga5fd2f12013-06-26 08:36:07 +0000146
Yves Gerey665174f2018-06-19 15:03:05 +0200147 header->payloadType = PT;
148 header->ssrc = SSRC;
solenberg@webrtc.orga5fd2f12013-06-26 08:36:07 +0000149 header->headerLength = 4 + (len << 2);
150
151 return true;
152}
153
Bjorn Tereliusb2bfba62018-03-05 20:52:04 +0100154bool RtpHeaderParser::Parse(
155 RTPHeader* header,
156 const RtpHeaderExtensionMap* ptrExtensionMap) const {
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000157 const ptrdiff_t length = _ptrRTPDataEnd - _ptrRTPDataBegin;
solenberg@webrtc.orga5fd2f12013-06-26 08:36:07 +0000158 if (length < kRtpMinParseLength) {
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000159 return false;
160 }
161
162 // Version
Yves Gerey665174f2018-06-19 15:03:05 +0200163 const uint8_t V = _ptrRTPDataBegin[0] >> 6;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000164 // Padding
Yves Gerey665174f2018-06-19 15:03:05 +0200165 const bool P = ((_ptrRTPDataBegin[0] & 0x20) == 0) ? false : true;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000166 // eXtension
Yves Gerey665174f2018-06-19 15:03:05 +0200167 const bool X = ((_ptrRTPDataBegin[0] & 0x10) == 0) ? false : true;
pbos@webrtc.org2f446732013-04-08 11:08:41 +0000168 const uint8_t CC = _ptrRTPDataBegin[0] & 0x0f;
Yves Gerey665174f2018-06-19 15:03:05 +0200169 const bool M = ((_ptrRTPDataBegin[1] & 0x80) == 0) ? false : true;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000170
pbos@webrtc.org2f446732013-04-08 11:08:41 +0000171 const uint8_t PT = _ptrRTPDataBegin[1] & 0x7f;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000172
Yves Gerey665174f2018-06-19 15:03:05 +0200173 const uint16_t sequenceNumber =
174 (_ptrRTPDataBegin[2] << 8) + _ptrRTPDataBegin[3];
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000175
pbos@webrtc.org2f446732013-04-08 11:08:41 +0000176 const uint8_t* ptr = &_ptrRTPDataBegin[4];
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000177
sprang@webrtc.org779c3d12015-03-17 16:42:49 +0000178 uint32_t RTPTimestamp = ByteReader<uint32_t>::ReadBigEndian(ptr);
179 ptr += 4;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000180
sprang@webrtc.org779c3d12015-03-17 16:42:49 +0000181 uint32_t SSRC = ByteReader<uint32_t>::ReadBigEndian(ptr);
182 ptr += 4;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000183
solenberg@webrtc.orga5fd2f12013-06-26 08:36:07 +0000184 if (V != kRtpExpectedVersion) {
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000185 return false;
186 }
187
pkasting@chromium.org4591fbd2014-11-20 22:28:14 +0000188 const size_t CSRCocts = CC * 4;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000189
190 if ((ptr + CSRCocts) > _ptrRTPDataEnd) {
191 return false;
192 }
193
Yves Gerey665174f2018-06-19 15:03:05 +0200194 header->markerBit = M;
195 header->payloadType = PT;
danilchapf6975f42015-12-28 10:18:46 -0800196 header->sequenceNumber = sequenceNumber;
Yves Gerey665174f2018-06-19 15:03:05 +0200197 header->timestamp = RTPTimestamp;
198 header->ssrc = SSRC;
199 header->numCSRCs = CC;
200 header->paddingLength = P ? *(_ptrRTPDataEnd - 1) : 0;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000201
pkasting@chromium.org4591fbd2014-11-20 22:28:14 +0000202 for (uint8_t i = 0; i < CC; ++i) {
sprang@webrtc.org779c3d12015-03-17 16:42:49 +0000203 uint32_t CSRC = ByteReader<uint32_t>::ReadBigEndian(ptr);
204 ptr += 4;
danilchapf6975f42015-12-28 10:18:46 -0800205 header->arrOfCSRCs[i] = CSRC;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000206 }
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000207
Yves Gerey665174f2018-06-19 15:03:05 +0200208 header->headerLength = 12 + CSRCocts;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000209
210 // If in effect, MAY be omitted for those packets for which the offset
211 // is zero.
danilchapf6975f42015-12-28 10:18:46 -0800212 header->extension.hasTransmissionTimeOffset = false;
213 header->extension.transmissionTimeOffset = 0;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000214
solenberg@webrtc.org7ebbea12013-05-16 11:10:31 +0000215 // May not be present in packet.
danilchapf6975f42015-12-28 10:18:46 -0800216 header->extension.hasAbsoluteSendTime = false;
217 header->extension.absoluteSendTime = 0;
solenberg@webrtc.org7ebbea12013-05-16 11:10:31 +0000218
wu@webrtc.orgebdb0e32014-03-06 23:49:08 +0000219 // May not be present in packet.
danilchapf6975f42015-12-28 10:18:46 -0800220 header->extension.hasAudioLevel = false;
221 header->extension.voiceActivity = false;
222 header->extension.audioLevel = 0;
wu@webrtc.orgebdb0e32014-03-06 23:49:08 +0000223
guoweis@webrtc.org45362892015-03-04 22:55:15 +0000224 // May not be present in packet.
danilchapf6975f42015-12-28 10:18:46 -0800225 header->extension.hasVideoRotation = false;
magjed71eb61c2016-09-08 03:24:58 -0700226 header->extension.videoRotation = kVideoRotation_0;
guoweis@webrtc.org45362892015-03-04 22:55:15 +0000227
isheriff6b4b5f32016-06-08 00:24:21 -0700228 // May not be present in packet.
229 header->extension.playout_delay.min_ms = -1;
230 header->extension.playout_delay.max_ms = -1;
231
ilnik00d802b2017-04-11 10:34:31 -0700232 // May not be present in packet.
233 header->extension.hasVideoContentType = false;
234 header->extension.videoContentType = VideoContentType::UNSPECIFIED;
235
ilnik04f4d122017-06-19 07:18:55 -0700236 header->extension.has_video_timing = false;
237 header->extension.video_timing = {0u, 0u, 0u, 0u, 0u, 0u, false};
238
Johnny Leee0c8b232018-09-11 16:50:49 -0400239 header->extension.has_frame_marking = false;
240 header->extension.frame_marking = {false, false, false, false, false,
241 kNoTemporalIdx, 0, 0};
242
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000243 if (X) {
244 /* RTP header extension, RFC 3550.
245 0 1 2 3
246 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
247 +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
248 | defined by profile | length |
249 +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
250 | header extension |
251 | .... |
252 */
253 const ptrdiff_t remain = _ptrRTPDataEnd - ptr;
254 if (remain < 4) {
255 return false;
niklase@google.com470e71d2011-07-07 08:21:25 +0000256 }
257
danilchapf6975f42015-12-28 10:18:46 -0800258 header->headerLength += 4;
niklase@google.com470e71d2011-07-07 08:21:25 +0000259
sprang@webrtc.org779c3d12015-03-17 16:42:49 +0000260 uint16_t definedByProfile = ByteReader<uint16_t>::ReadBigEndian(ptr);
261 ptr += 2;
niklase@google.com470e71d2011-07-07 08:21:25 +0000262
sprang@webrtc.org779c3d12015-03-17 16:42:49 +0000263 // in 32 bit words
264 size_t XLen = ByteReader<uint16_t>::ReadBigEndian(ptr);
265 ptr += 2;
266 XLen *= 4; // in bytes
niklase@google.com470e71d2011-07-07 08:21:25 +0000267
pkasting@chromium.org4591fbd2014-11-20 22:28:14 +0000268 if (static_cast<size_t>(remain) < (4 + XLen)) {
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000269 return false;
niklase@google.com470e71d2011-07-07 08:21:25 +0000270 }
danilchap5c4f24a2017-01-23 11:10:20 -0800271 static constexpr uint16_t kRtpOneByteHeaderExtensionId = 0xBEDE;
pbos@webrtc.org3004c792013-05-07 12:36:21 +0000272 if (definedByProfile == kRtpOneByteHeaderExtensionId) {
pbos@webrtc.org2f446732013-04-08 11:08:41 +0000273 const uint8_t* ptrRTPDataExtensionEnd = ptr + XLen;
Yves Gerey665174f2018-06-19 15:03:05 +0200274 ParseOneByteExtensionHeader(header, ptrExtensionMap,
275 ptrRTPDataExtensionEnd, ptr);
niklase@google.com470e71d2011-07-07 08:21:25 +0000276 }
danilchapf6975f42015-12-28 10:18:46 -0800277 header->headerLength += XLen;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000278 }
danilchapf6975f42015-12-28 10:18:46 -0800279 if (header->headerLength + header->paddingLength >
280 static_cast<size_t>(length))
pbosbd2522a2015-07-01 05:35:53 -0700281 return false;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000282 return true;
niklase@google.com470e71d2011-07-07 08:21:25 +0000283}
284
pbos@webrtc.org62bafae2014-07-08 12:10:51 +0000285void RtpHeaderParser::ParseOneByteExtensionHeader(
danilchapf6975f42015-12-28 10:18:46 -0800286 RTPHeader* header,
asapersson@webrtc.org5249cc82011-12-16 14:31:37 +0000287 const RtpHeaderExtensionMap* ptrExtensionMap,
pbos@webrtc.org2f446732013-04-08 11:08:41 +0000288 const uint8_t* ptrRTPDataExtensionEnd,
289 const uint8_t* ptr) const {
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000290 if (!ptrExtensionMap) {
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000291 return;
292 }
293
294 while (ptrRTPDataExtensionEnd - ptr > 0) {
295 // 0
296 // 0 1 2 3 4 5 6 7
297 // +-+-+-+-+-+-+-+-+
298 // | ID | len |
299 // +-+-+-+-+-+-+-+-+
300
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000301 // Note that 'len' is the header extension element length, which is the
302 // number of bytes - 1.
Stefan Holmer3842c5c2016-01-12 13:55:00 +0100303 const int id = (*ptr & 0xf0) >> 4;
304 const int len = (*ptr & 0x0f);
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000305 ptr++;
306
danilchapb7374db2016-11-24 09:06:05 -0800307 if (id == 0) {
308 // Padding byte, skip ignoring len.
309 continue;
310 }
311
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000312 if (id == 15) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100313 RTC_LOG(LS_VERBOSE)
andresp@webrtc.orgdc80bae2014-04-08 11:06:12 +0000314 << "RTP extension header 15 encountered. Terminate parsing.";
asapersson@webrtc.org5249cc82011-12-16 14:31:37 +0000315 return;
316 }
317
Danil Chapovalov30a3a752016-06-16 15:57:15 +0200318 if (ptrRTPDataExtensionEnd - ptr < (len + 1)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100319 RTC_LOG(LS_WARNING) << "Incorrect one-byte extension len: " << (len + 1)
320 << ", bytes left in buffer: "
321 << (ptrRTPDataExtensionEnd - ptr);
Danil Chapovalov30a3a752016-06-16 15:57:15 +0200322 return;
323 }
324
danilchap14546692016-12-01 08:39:35 -0800325 RTPExtensionType type = ptrExtensionMap->GetType(id);
326 if (type == RtpHeaderExtensionMap::kInvalidType) {
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000327 // If we encounter an unknown extension, just skip over it.
Mirko Bonadei675513b2017-11-09 11:09:25 +0100328 RTC_LOG(LS_WARNING) << "Failed to find extension id: " << id;
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000329 } else {
330 switch (type) {
331 case kRtpExtensionTransmissionTimeOffset: {
332 if (len != 2) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100333 RTC_LOG(LS_WARNING)
334 << "Incorrect transmission time offset len: " << len;
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000335 return;
336 }
337 // 0 1 2 3
338 // 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
339 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
340 // | ID | len=2 | transmission offset |
341 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
asapersson@webrtc.org5249cc82011-12-16 14:31:37 +0000342
danilchapf6975f42015-12-28 10:18:46 -0800343 header->extension.transmissionTimeOffset =
sprang@webrtc.org779c3d12015-03-17 16:42:49 +0000344 ByteReader<int32_t, 3>::ReadBigEndian(ptr);
danilchapf6975f42015-12-28 10:18:46 -0800345 header->extension.hasTransmissionTimeOffset = true;
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000346 break;
347 }
348 case kRtpExtensionAudioLevel: {
349 if (len != 0) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100350 RTC_LOG(LS_WARNING) << "Incorrect audio level len: " << len;
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000351 return;
352 }
sprang@webrtc.org30933902015-03-17 14:33:12 +0000353 // 0 1
354 // 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5
355 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
356 // | ID | len=0 |V| level |
357 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000358 //
danilchapf6975f42015-12-28 10:18:46 -0800359 header->extension.audioLevel = ptr[0] & 0x7f;
360 header->extension.voiceActivity = (ptr[0] & 0x80) != 0;
361 header->extension.hasAudioLevel = true;
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000362 break;
363 }
364 case kRtpExtensionAbsoluteSendTime: {
365 if (len != 2) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100366 RTC_LOG(LS_WARNING) << "Incorrect absolute send time len: " << len;
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000367 return;
368 }
369 // 0 1 2 3
370 // 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
371 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
372 // | ID | len=2 | absolute send time |
373 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
374
danilchapf6975f42015-12-28 10:18:46 -0800375 header->extension.absoluteSendTime =
sprang@webrtc.org779c3d12015-03-17 16:42:49 +0000376 ByteReader<uint32_t, 3>::ReadBigEndian(ptr);
danilchapf6975f42015-12-28 10:18:46 -0800377 header->extension.hasAbsoluteSendTime = true;
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000378 break;
379 }
guoweis@webrtc.org45362892015-03-04 22:55:15 +0000380 case kRtpExtensionVideoRotation: {
381 if (len != 0) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100382 RTC_LOG(LS_WARNING)
guoweis@webrtc.org45362892015-03-04 22:55:15 +0000383 << "Incorrect coordination of video coordination len: " << len;
384 return;
385 }
sprang@webrtc.org779c3d12015-03-17 16:42:49 +0000386 // 0 1
387 // 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5
388 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
389 // | ID | len=0 |0 0 0 0 C F R R|
390 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
danilchapf6975f42015-12-28 10:18:46 -0800391 header->extension.hasVideoRotation = true;
magjed71eb61c2016-09-08 03:24:58 -0700392 header->extension.videoRotation =
393 ConvertCVOByteToVideoRotation(ptr[0]);
guoweis@webrtc.org45362892015-03-04 22:55:15 +0000394 break;
395 }
sprang@webrtc.org30933902015-03-17 14:33:12 +0000396 case kRtpExtensionTransportSequenceNumber: {
397 if (len != 1) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100398 RTC_LOG(LS_WARNING)
399 << "Incorrect transport sequence number len: " << len;
sprang@webrtc.org30933902015-03-17 14:33:12 +0000400 return;
401 }
402 // 0 1 2
403 // 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3
404 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
405 // | ID | L=1 |transport wide sequence number |
406 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
407
408 uint16_t sequence_number = ptr[0] << 8;
409 sequence_number += ptr[1];
danilchapf6975f42015-12-28 10:18:46 -0800410 header->extension.transportSequenceNumber = sequence_number;
411 header->extension.hasTransportSequenceNumber = true;
sprang@webrtc.org30933902015-03-17 14:33:12 +0000412 break;
413 }
isheriff6b4b5f32016-06-08 00:24:21 -0700414 case kRtpExtensionPlayoutDelay: {
415 if (len != 2) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100416 RTC_LOG(LS_WARNING) << "Incorrect playout delay len: " << len;
isheriff6b4b5f32016-06-08 00:24:21 -0700417 return;
418 }
419 // 0 1 2 3
420 // 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
421 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
422 // | ID | len=2 | MIN delay | MAX delay |
423 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
424
425 int min_playout_delay = (ptr[0] << 4) | ((ptr[1] >> 4) & 0xf);
426 int max_playout_delay = ((ptr[1] & 0xf) << 8) | ptr[2];
427 header->extension.playout_delay.min_ms =
danilchap5c4f24a2017-01-23 11:10:20 -0800428 min_playout_delay * PlayoutDelayLimits::kGranularityMs;
isheriff6b4b5f32016-06-08 00:24:21 -0700429 header->extension.playout_delay.max_ms =
danilchap5c4f24a2017-01-23 11:10:20 -0800430 max_playout_delay * PlayoutDelayLimits::kGranularityMs;
isheriff6b4b5f32016-06-08 00:24:21 -0700431 break;
432 }
ilnik00d802b2017-04-11 10:34:31 -0700433 case kRtpExtensionVideoContentType: {
434 if (len != 0) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100435 RTC_LOG(LS_WARNING) << "Incorrect video content type len: " << len;
ilnik00d802b2017-04-11 10:34:31 -0700436 return;
437 }
438 // 0 1
439 // 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5
440 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
441 // | ID | len=0 | Content type |
442 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
443
ilnik6d5b4d62017-08-30 03:32:14 -0700444 if (videocontenttypehelpers::IsValidContentType(ptr[0])) {
ilnik00d802b2017-04-11 10:34:31 -0700445 header->extension.hasVideoContentType = true;
446 header->extension.videoContentType =
447 static_cast<VideoContentType>(ptr[0]);
448 }
449 break;
450 }
ilnik04f4d122017-06-19 07:18:55 -0700451 case kRtpExtensionVideoTiming: {
452 if (len != VideoTimingExtension::kValueSizeBytes - 1) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100453 RTC_LOG(LS_WARNING) << "Incorrect video timing len: " << len;
ilnik04f4d122017-06-19 07:18:55 -0700454 return;
455 }
456 header->extension.has_video_timing = true;
457 VideoTimingExtension::Parse(rtc::MakeArrayView(ptr, len + 1),
458 &header->extension.video_timing);
459 break;
460 }
Johnny Leee0c8b232018-09-11 16:50:49 -0400461 case kRtpExtensionFrameMarking: {
462 if (!FrameMarkingExtension::Parse(rtc::MakeArrayView(ptr, len + 1),
463 &header->extension.frame_marking)) {
464 RTC_LOG(LS_WARNING) << "Incorrect frame marking len: " << len;
465 return;
466 }
467 header->extension.has_frame_marking = true;
468 break;
469 }
danilchapef8d7732017-04-19 02:59:48 -0700470 case kRtpExtensionRtpStreamId: {
471 header->extension.stream_id.Set(rtc::MakeArrayView(ptr, len + 1));
472 break;
473 }
474 case kRtpExtensionRepairedRtpStreamId: {
475 header->extension.repaired_stream_id.Set(
476 rtc::MakeArrayView(ptr, len + 1));
477 break;
478 }
Steve Antona3251dd2017-07-21 09:58:31 -0700479 case kRtpExtensionMid: {
480 header->extension.mid.Set(rtc::MakeArrayView(ptr, len + 1));
481 break;
482 }
Danil Chapovalov916ec7d2018-06-25 18:22:15 +0200483 case kRtpExtensionGenericFrameDescriptor:
484 RTC_LOG(WARNING)
485 << "RtpGenericFrameDescriptor unsupported by rtp header parser.";
486 break;
danilchap14546692016-12-01 08:39:35 -0800487 case kRtpExtensionNone:
488 case kRtpExtensionNumberOfExtensions: {
489 RTC_NOTREACHED() << "Invalid extension type: " << type;
asapersson@webrtc.org5249cc82011-12-16 14:31:37 +0000490 return;
491 }
asapersson@webrtc.org5249cc82011-12-16 14:31:37 +0000492 }
asapersson@webrtc.org5249cc82011-12-16 14:31:37 +0000493 }
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000494 ptr += (len + 1);
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000495 }
asapersson@webrtc.org5249cc82011-12-16 14:31:37 +0000496}
497
pbos@webrtc.org62bafae2014-07-08 12:10:51 +0000498} // namespace RtpUtility
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000499} // namespace webrtc