blob: 159de27b4e72d383ea01bdba947e76e6cddf8672 [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 <stddef.h>
Jonas Olssona4d87372019-07-05 19:08:33 +020014
Niels Möllerd57efc12019-03-22 14:02:11 +010015#include <string>
Yves Gerey988cc082018-10-23 12:03:01 +020016
17#include "api/array_view.h"
18#include "api/video/video_content_type.h"
Yves Gerey988cc082018-10-23 12:03:01 +020019#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"
niklase@google.com470e71d2011-07-07 08:21:25 +000027
niklase@google.com470e71d2011-07-07 08:21:25 +000028namespace webrtc {
pwestin@webrtc.org0644b1d2011-12-01 15:42:31 +000029
pbos@webrtc.org62bafae2014-07-08 12:10:51 +000030namespace RtpUtility {
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +000031
solenberg@webrtc.orgd8a6e722013-03-26 14:02:30 +000032enum {
solenberg@webrtc.orga5fd2f12013-06-26 08:36:07 +000033 kRtcpExpectedVersion = 2,
solenberg@webrtc.orgd8a6e722013-03-26 14:02:30 +000034 kRtcpMinHeaderLength = 4,
solenberg@webrtc.orga5fd2f12013-06-26 08:36:07 +000035 kRtcpMinParseLength = 8,
36
37 kRtpExpectedVersion = 2,
38 kRtpMinParseLength = 12
solenberg@webrtc.orgd8a6e722013-03-26 14:02:30 +000039};
40
pbos@webrtc.org62bafae2014-07-08 12:10:51 +000041RtpHeaderParser::RtpHeaderParser(const uint8_t* rtpData,
42 const size_t rtpDataLength)
43 : _ptrRTPDataBegin(rtpData),
Yves Gerey665174f2018-06-19 15:03:05 +020044 _ptrRTPDataEnd(rtpData ? (rtpData + rtpDataLength) : NULL) {}
niklase@google.com470e71d2011-07-07 08:21:25 +000045
Yves Gerey665174f2018-06-19 15:03:05 +020046RtpHeaderParser::~RtpHeaderParser() {}
niklase@google.com470e71d2011-07-07 08:21:25 +000047
pbos@webrtc.org62bafae2014-07-08 12:10:51 +000048bool RtpHeaderParser::RTCP() const {
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +000049 // 72 to 76 is reserved for RTP
50 // 77 to 79 is not reserver but they are not assigned we will block them
51 // for RTCP 200 SR == marker bit + 72
52 // for RTCP 204 APP == marker bit + 76
53 /*
Yves Gerey665174f2018-06-19 15:03:05 +020054 * RTCP
55 *
56 * FIR full INTRA-frame request 192 [RFC2032] supported
57 * NACK negative acknowledgement 193 [RFC2032]
58 * IJ Extended inter-arrival jitter report 195 [RFC-ietf-avt-rtp-toff
59 * set-07.txt] http://tools.ietf.org/html/draft-ietf-avt-rtp-toffset-07
60 * SR sender report 200 [RFC3551] supported
61 * RR receiver report 201 [RFC3551] supported
62 * SDES source description 202 [RFC3551] supported
63 * BYE goodbye 203 [RFC3551] supported
64 * APP application-defined 204 [RFC3551] ignored
65 * RTPFB Transport layer FB message 205 [RFC4585] supported
66 * PSFB Payload-specific FB message 206 [RFC4585] supported
67 * XR extended report 207 [RFC3611] supported
68 */
niklase@google.com470e71d2011-07-07 08:21:25 +000069
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +000070 /* 205 RFC 5104
71 * FMT 1 NACK supported
72 * FMT 2 reserved
73 * FMT 3 TMMBR supported
74 * FMT 4 TMMBN supported
75 */
niklase@google.com470e71d2011-07-07 08:21:25 +000076
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +000077 /* 206 RFC 5104
Yves Gerey665174f2018-06-19 15:03:05 +020078 * FMT 1: Picture Loss Indication (PLI) supported
79 * FMT 2: Slice Lost Indication (SLI)
80 * FMT 3: Reference Picture Selection Indication (RPSI)
81 * FMT 4: Full Intra Request (FIR) Command supported
82 * FMT 5: Temporal-Spatial Trade-off Request (TSTR)
83 * FMT 6: Temporal-Spatial Trade-off Notification (TSTN)
84 * FMT 7: Video Back Channel Message (VBCM)
85 * FMT 15: Application layer FB message
86 */
niklase@google.com470e71d2011-07-07 08:21:25 +000087
solenberg@webrtc.orgd8a6e722013-03-26 14:02:30 +000088 const ptrdiff_t length = _ptrRTPDataEnd - _ptrRTPDataBegin;
89 if (length < kRtcpMinHeaderLength) {
90 return false;
91 }
92
pkasting@chromium.orgd3245462015-02-23 21:28:22 +000093 const uint8_t V = _ptrRTPDataBegin[0] >> 6;
solenberg@webrtc.orgd8a6e722013-03-26 14:02:30 +000094 if (V != kRtcpExpectedVersion) {
95 return false;
96 }
97
pkasting@chromium.orgd3245462015-02-23 21:28:22 +000098 const uint8_t payloadType = _ptrRTPDataBegin[1];
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +000099 switch (payloadType) {
niklase@google.com470e71d2011-07-07 08:21:25 +0000100 case 192:
pkasting@chromium.orgd3245462015-02-23 21:28:22 +0000101 return true;
niklase@google.com470e71d2011-07-07 08:21:25 +0000102 case 193:
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000103 // not supported
104 // pass through and check for a potential RTP packet
pkasting@chromium.orgd3245462015-02-23 21:28:22 +0000105 return false;
asapersson@webrtc.org5249cc82011-12-16 14:31:37 +0000106 case 195:
niklase@google.com470e71d2011-07-07 08:21:25 +0000107 case 200:
108 case 201:
109 case 202:
110 case 203:
111 case 204:
112 case 205:
113 case 206:
114 case 207:
pkasting@chromium.orgd3245462015-02-23 21:28:22 +0000115 return true;
116 default:
117 return false;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000118 }
niklase@google.com470e71d2011-07-07 08:21:25 +0000119}
120
Sebastian Jansson62c7b392019-03-08 13:45:29 +0100121bool RtpHeaderParser::Parse(RTPHeader* header,
122 const RtpHeaderExtensionMap* ptrExtensionMap,
123 bool header_only) const {
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000124 const ptrdiff_t length = _ptrRTPDataEnd - _ptrRTPDataBegin;
solenberg@webrtc.orga5fd2f12013-06-26 08:36:07 +0000125 if (length < kRtpMinParseLength) {
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000126 return false;
127 }
128
129 // Version
Yves Gerey665174f2018-06-19 15:03:05 +0200130 const uint8_t V = _ptrRTPDataBegin[0] >> 6;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000131 // Padding
Yves Gerey665174f2018-06-19 15:03:05 +0200132 const bool P = ((_ptrRTPDataBegin[0] & 0x20) == 0) ? false : true;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000133 // eXtension
Yves Gerey665174f2018-06-19 15:03:05 +0200134 const bool X = ((_ptrRTPDataBegin[0] & 0x10) == 0) ? false : true;
pbos@webrtc.org2f446732013-04-08 11:08:41 +0000135 const uint8_t CC = _ptrRTPDataBegin[0] & 0x0f;
Yves Gerey665174f2018-06-19 15:03:05 +0200136 const bool M = ((_ptrRTPDataBegin[1] & 0x80) == 0) ? false : true;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000137
pbos@webrtc.org2f446732013-04-08 11:08:41 +0000138 const uint8_t PT = _ptrRTPDataBegin[1] & 0x7f;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000139
Yves Gerey665174f2018-06-19 15:03:05 +0200140 const uint16_t sequenceNumber =
141 (_ptrRTPDataBegin[2] << 8) + _ptrRTPDataBegin[3];
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000142
pbos@webrtc.org2f446732013-04-08 11:08:41 +0000143 const uint8_t* ptr = &_ptrRTPDataBegin[4];
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000144
sprang@webrtc.org779c3d12015-03-17 16:42:49 +0000145 uint32_t RTPTimestamp = ByteReader<uint32_t>::ReadBigEndian(ptr);
146 ptr += 4;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000147
sprang@webrtc.org779c3d12015-03-17 16:42:49 +0000148 uint32_t SSRC = ByteReader<uint32_t>::ReadBigEndian(ptr);
149 ptr += 4;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000150
solenberg@webrtc.orga5fd2f12013-06-26 08:36:07 +0000151 if (V != kRtpExpectedVersion) {
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000152 return false;
153 }
154
pkasting@chromium.org4591fbd2014-11-20 22:28:14 +0000155 const size_t CSRCocts = CC * 4;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000156
157 if ((ptr + CSRCocts) > _ptrRTPDataEnd) {
158 return false;
159 }
160
Yves Gerey665174f2018-06-19 15:03:05 +0200161 header->markerBit = M;
162 header->payloadType = PT;
danilchapf6975f42015-12-28 10:18:46 -0800163 header->sequenceNumber = sequenceNumber;
Yves Gerey665174f2018-06-19 15:03:05 +0200164 header->timestamp = RTPTimestamp;
165 header->ssrc = SSRC;
166 header->numCSRCs = CC;
Sebastian Jansson62c7b392019-03-08 13:45:29 +0100167 if (!P || header_only) {
Minyue Li1a800182018-09-12 12:52:48 +0200168 header->paddingLength = 0;
169 }
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000170
pkasting@chromium.org4591fbd2014-11-20 22:28:14 +0000171 for (uint8_t i = 0; i < CC; ++i) {
sprang@webrtc.org779c3d12015-03-17 16:42:49 +0000172 uint32_t CSRC = ByteReader<uint32_t>::ReadBigEndian(ptr);
173 ptr += 4;
danilchapf6975f42015-12-28 10:18:46 -0800174 header->arrOfCSRCs[i] = CSRC;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000175 }
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000176
Yves Gerey665174f2018-06-19 15:03:05 +0200177 header->headerLength = 12 + CSRCocts;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000178
179 // If in effect, MAY be omitted for those packets for which the offset
180 // is zero.
danilchapf6975f42015-12-28 10:18:46 -0800181 header->extension.hasTransmissionTimeOffset = false;
182 header->extension.transmissionTimeOffset = 0;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000183
solenberg@webrtc.org7ebbea12013-05-16 11:10:31 +0000184 // May not be present in packet.
danilchapf6975f42015-12-28 10:18:46 -0800185 header->extension.hasAbsoluteSendTime = false;
186 header->extension.absoluteSendTime = 0;
solenberg@webrtc.org7ebbea12013-05-16 11:10:31 +0000187
wu@webrtc.orgebdb0e32014-03-06 23:49:08 +0000188 // May not be present in packet.
danilchapf6975f42015-12-28 10:18:46 -0800189 header->extension.hasAudioLevel = false;
190 header->extension.voiceActivity = false;
191 header->extension.audioLevel = 0;
wu@webrtc.orgebdb0e32014-03-06 23:49:08 +0000192
guoweis@webrtc.org45362892015-03-04 22:55:15 +0000193 // May not be present in packet.
danilchapf6975f42015-12-28 10:18:46 -0800194 header->extension.hasVideoRotation = false;
magjed71eb61c2016-09-08 03:24:58 -0700195 header->extension.videoRotation = kVideoRotation_0;
guoweis@webrtc.org45362892015-03-04 22:55:15 +0000196
isheriff6b4b5f32016-06-08 00:24:21 -0700197 // May not be present in packet.
198 header->extension.playout_delay.min_ms = -1;
199 header->extension.playout_delay.max_ms = -1;
200
ilnik00d802b2017-04-11 10:34:31 -0700201 // May not be present in packet.
202 header->extension.hasVideoContentType = false;
203 header->extension.videoContentType = VideoContentType::UNSPECIFIED;
204
ilnik04f4d122017-06-19 07:18:55 -0700205 header->extension.has_video_timing = false;
206 header->extension.video_timing = {0u, 0u, 0u, 0u, 0u, 0u, false};
207
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000208 if (X) {
209 /* RTP header extension, RFC 3550.
210 0 1 2 3
211 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
212 +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
213 | defined by profile | length |
214 +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
215 | header extension |
216 | .... |
217 */
218 const ptrdiff_t remain = _ptrRTPDataEnd - ptr;
219 if (remain < 4) {
220 return false;
niklase@google.com470e71d2011-07-07 08:21:25 +0000221 }
222
danilchapf6975f42015-12-28 10:18:46 -0800223 header->headerLength += 4;
niklase@google.com470e71d2011-07-07 08:21:25 +0000224
sprang@webrtc.org779c3d12015-03-17 16:42:49 +0000225 uint16_t definedByProfile = ByteReader<uint16_t>::ReadBigEndian(ptr);
226 ptr += 2;
niklase@google.com470e71d2011-07-07 08:21:25 +0000227
sprang@webrtc.org779c3d12015-03-17 16:42:49 +0000228 // in 32 bit words
229 size_t XLen = ByteReader<uint16_t>::ReadBigEndian(ptr);
230 ptr += 2;
231 XLen *= 4; // in bytes
niklase@google.com470e71d2011-07-07 08:21:25 +0000232
pkasting@chromium.org4591fbd2014-11-20 22:28:14 +0000233 if (static_cast<size_t>(remain) < (4 + XLen)) {
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000234 return false;
niklase@google.com470e71d2011-07-07 08:21:25 +0000235 }
danilchap5c4f24a2017-01-23 11:10:20 -0800236 static constexpr uint16_t kRtpOneByteHeaderExtensionId = 0xBEDE;
pbos@webrtc.org3004c792013-05-07 12:36:21 +0000237 if (definedByProfile == kRtpOneByteHeaderExtensionId) {
pbos@webrtc.org2f446732013-04-08 11:08:41 +0000238 const uint8_t* ptrRTPDataExtensionEnd = ptr + XLen;
Yves Gerey665174f2018-06-19 15:03:05 +0200239 ParseOneByteExtensionHeader(header, ptrExtensionMap,
240 ptrRTPDataExtensionEnd, ptr);
niklase@google.com470e71d2011-07-07 08:21:25 +0000241 }
danilchapf6975f42015-12-28 10:18:46 -0800242 header->headerLength += XLen;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000243 }
Minyue Li1a800182018-09-12 12:52:48 +0200244 if (header->headerLength > static_cast<size_t>(length))
245 return false;
246
Sebastian Jansson62c7b392019-03-08 13:45:29 +0100247 if (P && !header_only) {
Minyue Li1a800182018-09-12 12:52:48 +0200248 // Packet has padding.
249 if (header->headerLength != static_cast<size_t>(length)) {
250 // Packet is not header only. We can parse padding length now.
251 header->paddingLength = *(_ptrRTPDataEnd - 1);
252 } else {
253 RTC_LOG(LS_WARNING) << "Cannot parse padding length.";
254 // Packet is header only. We have no clue of the padding length.
255 return false;
256 }
257 }
258
danilchapf6975f42015-12-28 10:18:46 -0800259 if (header->headerLength + header->paddingLength >
260 static_cast<size_t>(length))
pbosbd2522a2015-07-01 05:35:53 -0700261 return false;
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000262 return true;
niklase@google.com470e71d2011-07-07 08:21:25 +0000263}
264
pbos@webrtc.org62bafae2014-07-08 12:10:51 +0000265void RtpHeaderParser::ParseOneByteExtensionHeader(
danilchapf6975f42015-12-28 10:18:46 -0800266 RTPHeader* header,
asapersson@webrtc.org5249cc82011-12-16 14:31:37 +0000267 const RtpHeaderExtensionMap* ptrExtensionMap,
pbos@webrtc.org2f446732013-04-08 11:08:41 +0000268 const uint8_t* ptrRTPDataExtensionEnd,
269 const uint8_t* ptr) const {
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000270 if (!ptrExtensionMap) {
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000271 return;
272 }
273
274 while (ptrRTPDataExtensionEnd - ptr > 0) {
275 // 0
276 // 0 1 2 3 4 5 6 7
277 // +-+-+-+-+-+-+-+-+
278 // | ID | len |
279 // +-+-+-+-+-+-+-+-+
280
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000281 // Note that 'len' is the header extension element length, which is the
282 // number of bytes - 1.
Stefan Holmer3842c5c2016-01-12 13:55:00 +0100283 const int id = (*ptr & 0xf0) >> 4;
284 const int len = (*ptr & 0x0f);
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000285 ptr++;
286
danilchapb7374db2016-11-24 09:06:05 -0800287 if (id == 0) {
288 // Padding byte, skip ignoring len.
289 continue;
290 }
291
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000292 if (id == 15) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100293 RTC_LOG(LS_VERBOSE)
andresp@webrtc.orgdc80bae2014-04-08 11:06:12 +0000294 << "RTP extension header 15 encountered. Terminate parsing.";
asapersson@webrtc.org5249cc82011-12-16 14:31:37 +0000295 return;
296 }
297
Danil Chapovalov30a3a752016-06-16 15:57:15 +0200298 if (ptrRTPDataExtensionEnd - ptr < (len + 1)) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100299 RTC_LOG(LS_WARNING) << "Incorrect one-byte extension len: " << (len + 1)
300 << ", bytes left in buffer: "
301 << (ptrRTPDataExtensionEnd - ptr);
Danil Chapovalov30a3a752016-06-16 15:57:15 +0200302 return;
303 }
304
danilchap14546692016-12-01 08:39:35 -0800305 RTPExtensionType type = ptrExtensionMap->GetType(id);
306 if (type == RtpHeaderExtensionMap::kInvalidType) {
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000307 // If we encounter an unknown extension, just skip over it.
Mirko Bonadei675513b2017-11-09 11:09:25 +0100308 RTC_LOG(LS_WARNING) << "Failed to find extension id: " << id;
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000309 } else {
310 switch (type) {
311 case kRtpExtensionTransmissionTimeOffset: {
312 if (len != 2) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100313 RTC_LOG(LS_WARNING)
314 << "Incorrect transmission time offset len: " << len;
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000315 return;
316 }
317 // 0 1 2 3
318 // 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
319 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
320 // | ID | len=2 | transmission offset |
321 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
asapersson@webrtc.org5249cc82011-12-16 14:31:37 +0000322
danilchapf6975f42015-12-28 10:18:46 -0800323 header->extension.transmissionTimeOffset =
sprang@webrtc.org779c3d12015-03-17 16:42:49 +0000324 ByteReader<int32_t, 3>::ReadBigEndian(ptr);
danilchapf6975f42015-12-28 10:18:46 -0800325 header->extension.hasTransmissionTimeOffset = true;
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000326 break;
327 }
Doudou Kisabakaae0d1172021-05-24 13:04:45 +0200328 case kRtpExtensionCsrcAudioLevel: {
329 RTC_LOG(LS_WARNING) << "Csrc audio level extension not supported";
330 return;
331 }
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000332 case kRtpExtensionAudioLevel: {
333 if (len != 0) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100334 RTC_LOG(LS_WARNING) << "Incorrect audio level len: " << len;
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000335 return;
336 }
sprang@webrtc.org30933902015-03-17 14:33:12 +0000337 // 0 1
338 // 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5
339 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
340 // | ID | len=0 |V| level |
341 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000342 //
danilchapf6975f42015-12-28 10:18:46 -0800343 header->extension.audioLevel = ptr[0] & 0x7f;
344 header->extension.voiceActivity = (ptr[0] & 0x80) != 0;
345 header->extension.hasAudioLevel = true;
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000346 break;
347 }
348 case kRtpExtensionAbsoluteSendTime: {
349 if (len != 2) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100350 RTC_LOG(LS_WARNING) << "Incorrect absolute send time len: " << len;
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000351 return;
352 }
353 // 0 1 2 3
354 // 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
355 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
356 // | ID | len=2 | absolute send time |
357 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
358
danilchapf6975f42015-12-28 10:18:46 -0800359 header->extension.absoluteSendTime =
sprang@webrtc.org779c3d12015-03-17 16:42:49 +0000360 ByteReader<uint32_t, 3>::ReadBigEndian(ptr);
danilchapf6975f42015-12-28 10:18:46 -0800361 header->extension.hasAbsoluteSendTime = true;
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000362 break;
363 }
Chen Xingcd8a6e22019-07-01 10:56:51 +0200364 case kRtpExtensionAbsoluteCaptureTime: {
365 AbsoluteCaptureTime extension;
366 if (!AbsoluteCaptureTimeExtension::Parse(
367 rtc::MakeArrayView(ptr, len + 1), &extension)) {
368 RTC_LOG(LS_WARNING)
369 << "Incorrect absolute capture time len: " << len;
370 return;
371 }
372 header->extension.absolute_capture_time = extension;
373 break;
374 }
guoweis@webrtc.org45362892015-03-04 22:55:15 +0000375 case kRtpExtensionVideoRotation: {
376 if (len != 0) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100377 RTC_LOG(LS_WARNING)
guoweis@webrtc.org45362892015-03-04 22:55:15 +0000378 << "Incorrect coordination of video coordination len: " << len;
379 return;
380 }
sprang@webrtc.org779c3d12015-03-17 16:42:49 +0000381 // 0 1
382 // 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5
383 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
384 // | ID | len=0 |0 0 0 0 C F R R|
385 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
danilchapf6975f42015-12-28 10:18:46 -0800386 header->extension.hasVideoRotation = true;
magjed71eb61c2016-09-08 03:24:58 -0700387 header->extension.videoRotation =
388 ConvertCVOByteToVideoRotation(ptr[0]);
guoweis@webrtc.org45362892015-03-04 22:55:15 +0000389 break;
390 }
sprang@webrtc.org30933902015-03-17 14:33:12 +0000391 case kRtpExtensionTransportSequenceNumber: {
392 if (len != 1) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100393 RTC_LOG(LS_WARNING)
394 << "Incorrect transport sequence number len: " << len;
sprang@webrtc.org30933902015-03-17 14:33:12 +0000395 return;
396 }
397 // 0 1 2
398 // 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3
399 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
400 // | ID | L=1 |transport wide sequence number |
401 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
402
403 uint16_t sequence_number = ptr[0] << 8;
404 sequence_number += ptr[1];
danilchapf6975f42015-12-28 10:18:46 -0800405 header->extension.transportSequenceNumber = sequence_number;
406 header->extension.hasTransportSequenceNumber = true;
sprang@webrtc.org30933902015-03-17 14:33:12 +0000407 break;
408 }
Johannes Kron54047be2019-02-21 14:09:20 +0000409 case kRtpExtensionTransportSequenceNumber02:
410 RTC_LOG(WARNING) << "TransportSequenceNumberV2 unsupported by rtp "
411 "header parser.";
412 break;
isheriff6b4b5f32016-06-08 00:24:21 -0700413 case kRtpExtensionPlayoutDelay: {
414 if (len != 2) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100415 RTC_LOG(LS_WARNING) << "Incorrect playout delay len: " << len;
isheriff6b4b5f32016-06-08 00:24:21 -0700416 return;
417 }
418 // 0 1 2 3
419 // 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
420 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
421 // | ID | len=2 | MIN delay | MAX delay |
422 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
423
424 int min_playout_delay = (ptr[0] << 4) | ((ptr[1] >> 4) & 0xf);
425 int max_playout_delay = ((ptr[1] & 0xf) << 8) | ptr[2];
426 header->extension.playout_delay.min_ms =
danilchap5c4f24a2017-01-23 11:10:20 -0800427 min_playout_delay * PlayoutDelayLimits::kGranularityMs;
isheriff6b4b5f32016-06-08 00:24:21 -0700428 header->extension.playout_delay.max_ms =
danilchap5c4f24a2017-01-23 11:10:20 -0800429 max_playout_delay * PlayoutDelayLimits::kGranularityMs;
isheriff6b4b5f32016-06-08 00:24:21 -0700430 break;
431 }
ilnik00d802b2017-04-11 10:34:31 -0700432 case kRtpExtensionVideoContentType: {
433 if (len != 0) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100434 RTC_LOG(LS_WARNING) << "Incorrect video content type len: " << len;
ilnik00d802b2017-04-11 10:34:31 -0700435 return;
436 }
437 // 0 1
438 // 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5
439 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
440 // | ID | len=0 | Content type |
441 // +-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+-+
442
ilnik6d5b4d62017-08-30 03:32:14 -0700443 if (videocontenttypehelpers::IsValidContentType(ptr[0])) {
ilnik00d802b2017-04-11 10:34:31 -0700444 header->extension.hasVideoContentType = true;
445 header->extension.videoContentType =
446 static_cast<VideoContentType>(ptr[0]);
447 }
448 break;
449 }
ilnik04f4d122017-06-19 07:18:55 -0700450 case kRtpExtensionVideoTiming: {
451 if (len != VideoTimingExtension::kValueSizeBytes - 1) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100452 RTC_LOG(LS_WARNING) << "Incorrect video timing len: " << len;
ilnik04f4d122017-06-19 07:18:55 -0700453 return;
454 }
455 header->extension.has_video_timing = true;
456 VideoTimingExtension::Parse(rtc::MakeArrayView(ptr, len + 1),
457 &header->extension.video_timing);
458 break;
459 }
Per Kjellander6556ed22020-10-05 13:51:47 +0200460 case kRtpExtensionVideoLayersAllocation:
461 RTC_LOG(WARNING) << "VideoLayersAllocation extension unsupported by "
462 "rtp header parser.";
463 break;
danilchapef8d7732017-04-19 02:59:48 -0700464 case kRtpExtensionRtpStreamId: {
Niels Möllerd57efc12019-03-22 14:02:11 +0100465 std::string name(reinterpret_cast<const char*>(ptr), len + 1);
466 if (IsLegalRsidName(name)) {
467 header->extension.stream_id = name;
468 } else {
469 RTC_LOG(LS_WARNING) << "Incorrect RtpStreamId";
470 }
danilchapef8d7732017-04-19 02:59:48 -0700471 break;
472 }
473 case kRtpExtensionRepairedRtpStreamId: {
Niels Möllerd57efc12019-03-22 14:02:11 +0100474 std::string name(reinterpret_cast<const char*>(ptr), len + 1);
475 if (IsLegalRsidName(name)) {
476 header->extension.repaired_stream_id = name;
477 } else {
478 RTC_LOG(LS_WARNING) << "Incorrect RepairedRtpStreamId";
479 }
danilchapef8d7732017-04-19 02:59:48 -0700480 break;
481 }
Steve Antona3251dd2017-07-21 09:58:31 -0700482 case kRtpExtensionMid: {
Niels Möllerd57efc12019-03-22 14:02:11 +0100483 std::string name(reinterpret_cast<const char*>(ptr), len + 1);
484 if (IsLegalMidName(name)) {
485 header->extension.mid = name;
486 } else {
487 RTC_LOG(LS_WARNING) << "Incorrect Mid";
488 }
Steve Antona3251dd2017-07-21 09:58:31 -0700489 break;
490 }
Elad Alonccb9b752019-02-19 13:01:31 +0100491 case kRtpExtensionGenericFrameDescriptor00:
Danil Chapovalov52e52422019-06-27 16:45:40 +0200492 case kRtpExtensionGenericFrameDescriptor02:
Danil Chapovalov916ec7d2018-06-25 18:22:15 +0200493 RTC_LOG(WARNING)
494 << "RtpGenericFrameDescriptor unsupported by rtp header parser.";
495 break;
Johannes Kron09d65882018-11-27 14:36:41 +0100496 case kRtpExtensionColorSpace:
Johannes Kronad1d9f02018-11-09 11:12:36 +0100497 RTC_LOG(WARNING)
Johannes Kron09d65882018-11-27 14:36:41 +0100498 << "RtpExtensionColorSpace unsupported by rtp header parser.";
Johannes Kronad1d9f02018-11-09 11:12:36 +0100499 break;
Minyue Licae27792019-11-29 16:18:59 +0100500 case kRtpExtensionInbandComfortNoise:
501 RTC_LOG(WARNING) << "Inband comfort noise extension unsupported by "
502 "rtp header parser.";
503 break;
Jeremy Leconte4f88a9d2021-03-17 17:01:31 +0100504 case kRtpExtensionVideoFrameTrackingId:
505 RTC_LOG(WARNING)
506 << "VideoFrameTrackingId unsupported by rtp header parser.";
507 break;
danilchap14546692016-12-01 08:39:35 -0800508 case kRtpExtensionNone:
509 case kRtpExtensionNumberOfExtensions: {
510 RTC_NOTREACHED() << "Invalid extension type: " << type;
asapersson@webrtc.org5249cc82011-12-16 14:31:37 +0000511 return;
512 }
asapersson@webrtc.org5249cc82011-12-16 14:31:37 +0000513 }
asapersson@webrtc.org5249cc82011-12-16 14:31:37 +0000514 }
solenberg@webrtc.org440fa232014-03-25 19:57:07 +0000515 ptr += (len + 1);
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000516 }
asapersson@webrtc.org5249cc82011-12-16 14:31:37 +0000517}
518
pbos@webrtc.org62bafae2014-07-08 12:10:51 +0000519} // namespace RtpUtility
henrike@webrtc.orgd5657c22012-02-08 23:41:49 +0000520} // namespace webrtc