1 /*
2 * Copyright (c) 2004 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
11 #include "media/base/codec.h"
12
13 #include "absl/algorithm/container.h"
14 #include "absl/strings/match.h"
15 #include "api/video_codecs/av1_profile.h"
16 #include "api/video_codecs/h264_profile_level_id.h"
17 #include "api/video_codecs/vp9_profile.h"
18 #include "rtc_base/checks.h"
19 #include "rtc_base/logging.h"
20 #include "rtc_base/string_encode.h"
21 #include "rtc_base/strings/string_builder.h"
22
23 namespace cricket {
24 namespace {
25
GetH264PacketizationModeOrDefault(const CodecParameterMap & params)26 std::string GetH264PacketizationModeOrDefault(const CodecParameterMap& params) {
27 auto it = params.find(kH264FmtpPacketizationMode);
28 if (it != params.end()) {
29 return it->second;
30 }
31 // If packetization-mode is not present, default to "0".
32 // https://tools.ietf.org/html/rfc6184#section-6.2
33 return "0";
34 }
35
IsSameH264PacketizationMode(const CodecParameterMap & left,const CodecParameterMap & right)36 bool IsSameH264PacketizationMode(const CodecParameterMap& left,
37 const CodecParameterMap& right) {
38 return GetH264PacketizationModeOrDefault(left) ==
39 GetH264PacketizationModeOrDefault(right);
40 }
41
42 // Some (video) codecs are actually families of codecs and rely on parameters
43 // to distinguish different incompatible family members.
IsSameCodecSpecific(const std::string & name1,const CodecParameterMap & params1,const std::string & name2,const CodecParameterMap & params2)44 bool IsSameCodecSpecific(const std::string& name1,
45 const CodecParameterMap& params1,
46 const std::string& name2,
47 const CodecParameterMap& params2) {
48 // The names might not necessarily match, so check both.
49 auto either_name_matches = [&](const std::string name) {
50 return absl::EqualsIgnoreCase(name, name1) ||
51 absl::EqualsIgnoreCase(name, name2);
52 };
53 if (either_name_matches(kH264CodecName))
54 return webrtc::H264IsSameProfile(params1, params2) &&
55 IsSameH264PacketizationMode(params1, params2);
56 if (either_name_matches(kVp9CodecName))
57 return webrtc::VP9IsSameProfile(params1, params2);
58 if (either_name_matches(kAv1CodecName))
59 return webrtc::AV1IsSameProfile(params1, params2);
60 return true;
61 }
62
63 } // namespace
64
65 FeedbackParams::FeedbackParams() = default;
66 FeedbackParams::~FeedbackParams() = default;
67
operator ==(const FeedbackParam & other) const68 bool FeedbackParam::operator==(const FeedbackParam& other) const {
69 return absl::EqualsIgnoreCase(other.id(), id()) &&
70 absl::EqualsIgnoreCase(other.param(), param());
71 }
72
operator ==(const FeedbackParams & other) const73 bool FeedbackParams::operator==(const FeedbackParams& other) const {
74 return params_ == other.params_;
75 }
76
Has(const FeedbackParam & param) const77 bool FeedbackParams::Has(const FeedbackParam& param) const {
78 return absl::c_linear_search(params_, param);
79 }
80
Add(const FeedbackParam & param)81 void FeedbackParams::Add(const FeedbackParam& param) {
82 if (param.id().empty()) {
83 return;
84 }
85 if (Has(param)) {
86 // Param already in `this`.
87 return;
88 }
89 params_.push_back(param);
90 RTC_CHECK(!HasDuplicateEntries());
91 }
92
Intersect(const FeedbackParams & from)93 void FeedbackParams::Intersect(const FeedbackParams& from) {
94 std::vector<FeedbackParam>::iterator iter_to = params_.begin();
95 while (iter_to != params_.end()) {
96 if (!from.Has(*iter_to)) {
97 iter_to = params_.erase(iter_to);
98 } else {
99 ++iter_to;
100 }
101 }
102 }
103
HasDuplicateEntries() const104 bool FeedbackParams::HasDuplicateEntries() const {
105 for (std::vector<FeedbackParam>::const_iterator iter = params_.begin();
106 iter != params_.end(); ++iter) {
107 for (std::vector<FeedbackParam>::const_iterator found = iter + 1;
108 found != params_.end(); ++found) {
109 if (*found == *iter) {
110 return true;
111 }
112 }
113 }
114 return false;
115 }
116
Codec(int id,const std::string & name,int clockrate)117 Codec::Codec(int id, const std::string& name, int clockrate)
118 : id(id), name(name), clockrate(clockrate) {}
119
Codec()120 Codec::Codec() : id(0), clockrate(0) {}
121
122 Codec::Codec(const Codec& c) = default;
123 Codec::Codec(Codec&& c) = default;
124 Codec::~Codec() = default;
125 Codec& Codec::operator=(const Codec& c) = default;
126 Codec& Codec::operator=(Codec&& c) = default;
127
operator ==(const Codec & c) const128 bool Codec::operator==(const Codec& c) const {
129 return this->id == c.id && // id is reserved in objective-c
130 name == c.name && clockrate == c.clockrate && params == c.params &&
131 feedback_params == c.feedback_params;
132 }
133
Matches(const Codec & codec,const webrtc::FieldTrialsView * field_trials) const134 bool Codec::Matches(const Codec& codec,
135 const webrtc::FieldTrialsView* field_trials) const {
136 // Match the codec id/name based on the typical static/dynamic name rules.
137 // Matching is case-insensitive.
138
139 // Legacy behaviour with killswitch.
140 if (field_trials &&
141 field_trials->IsDisabled("WebRTC-PayloadTypes-Lower-Dynamic-Range")) {
142 const int kMaxStaticPayloadId = 95;
143 return (id <= kMaxStaticPayloadId || codec.id <= kMaxStaticPayloadId)
144 ? (id == codec.id)
145 : (absl::EqualsIgnoreCase(name, codec.name));
146 }
147 // We support the ranges [96, 127] and more recently [35, 65].
148 // https://www.iana.org/assignments/rtp-parameters/rtp-parameters.xhtml#rtp-parameters-1
149 // Within those ranges we match by codec name, outside by codec id.
150 const int kLowerDynamicRangeMin = 35;
151 const int kLowerDynamicRangeMax = 65;
152 const int kUpperDynamicRangeMin = 96;
153 const int kUpperDynamicRangeMax = 127;
154 const bool is_id_in_dynamic_range =
155 (id >= kLowerDynamicRangeMin && id <= kLowerDynamicRangeMax) ||
156 (id >= kUpperDynamicRangeMin && id <= kUpperDynamicRangeMax);
157 const bool is_codec_id_in_dynamic_range =
158 (codec.id >= kLowerDynamicRangeMin &&
159 codec.id <= kLowerDynamicRangeMax) ||
160 (codec.id >= kUpperDynamicRangeMin && codec.id <= kUpperDynamicRangeMax);
161 return is_id_in_dynamic_range && is_codec_id_in_dynamic_range
162 ? (absl::EqualsIgnoreCase(name, codec.name))
163 : (id == codec.id);
164 }
165
MatchesCapability(const webrtc::RtpCodecCapability & codec_capability) const166 bool Codec::MatchesCapability(
167 const webrtc::RtpCodecCapability& codec_capability) const {
168 webrtc::RtpCodecParameters codec_parameters = ToCodecParameters();
169
170 return codec_parameters.name == codec_capability.name &&
171 codec_parameters.kind == codec_capability.kind &&
172 (codec_parameters.name == cricket::kRtxCodecName ||
173 (codec_parameters.num_channels == codec_capability.num_channels &&
174 codec_parameters.clock_rate == codec_capability.clock_rate &&
175 codec_parameters.parameters == codec_capability.parameters));
176 }
177
GetParam(const std::string & name,std::string * out) const178 bool Codec::GetParam(const std::string& name, std::string* out) const {
179 CodecParameterMap::const_iterator iter = params.find(name);
180 if (iter == params.end())
181 return false;
182 *out = iter->second;
183 return true;
184 }
185
GetParam(const std::string & name,int * out) const186 bool Codec::GetParam(const std::string& name, int* out) const {
187 CodecParameterMap::const_iterator iter = params.find(name);
188 if (iter == params.end())
189 return false;
190 return rtc::FromString(iter->second, out);
191 }
192
SetParam(const std::string & name,const std::string & value)193 void Codec::SetParam(const std::string& name, const std::string& value) {
194 params[name] = value;
195 }
196
SetParam(const std::string & name,int value)197 void Codec::SetParam(const std::string& name, int value) {
198 params[name] = rtc::ToString(value);
199 }
200
RemoveParam(const std::string & name)201 bool Codec::RemoveParam(const std::string& name) {
202 return params.erase(name) == 1;
203 }
204
AddFeedbackParam(const FeedbackParam & param)205 void Codec::AddFeedbackParam(const FeedbackParam& param) {
206 feedback_params.Add(param);
207 }
208
HasFeedbackParam(const FeedbackParam & param) const209 bool Codec::HasFeedbackParam(const FeedbackParam& param) const {
210 return feedback_params.Has(param);
211 }
212
IntersectFeedbackParams(const Codec & other)213 void Codec::IntersectFeedbackParams(const Codec& other) {
214 feedback_params.Intersect(other.feedback_params);
215 }
216
ToCodecParameters() const217 webrtc::RtpCodecParameters Codec::ToCodecParameters() const {
218 webrtc::RtpCodecParameters codec_params;
219 codec_params.payload_type = id;
220 codec_params.name = name;
221 codec_params.clock_rate = clockrate;
222 codec_params.parameters.insert(params.begin(), params.end());
223 return codec_params;
224 }
225
AudioCodec(int id,const std::string & name,int clockrate,int bitrate,size_t channels)226 AudioCodec::AudioCodec(int id,
227 const std::string& name,
228 int clockrate,
229 int bitrate,
230 size_t channels)
231 : Codec(id, name, clockrate), bitrate(bitrate), channels(channels) {}
232
AudioCodec()233 AudioCodec::AudioCodec() : Codec(), bitrate(0), channels(0) {}
234
235 AudioCodec::AudioCodec(const AudioCodec& c) = default;
236 AudioCodec::AudioCodec(AudioCodec&& c) = default;
237 AudioCodec& AudioCodec::operator=(const AudioCodec& c) = default;
238 AudioCodec& AudioCodec::operator=(AudioCodec&& c) = default;
239
operator ==(const AudioCodec & c) const240 bool AudioCodec::operator==(const AudioCodec& c) const {
241 return bitrate == c.bitrate && channels == c.channels && Codec::operator==(c);
242 }
243
Matches(const AudioCodec & codec,const webrtc::FieldTrialsView * field_trials) const244 bool AudioCodec::Matches(const AudioCodec& codec,
245 const webrtc::FieldTrialsView* field_trials) const {
246 // If a nonzero clockrate is specified, it must match the actual clockrate.
247 // If a nonzero bitrate is specified, it must match the actual bitrate,
248 // unless the codec is VBR (0), where we just force the supplied value.
249 // The number of channels must match exactly, with the exception
250 // that channels=0 is treated synonymously as channels=1, per RFC
251 // 4566 section 6: " [The channels] parameter is OPTIONAL and may be
252 // omitted if the number of channels is one."
253 // Preference is ignored.
254 // TODO(juberti): Treat a zero clockrate as 8000Hz, the RTP default clockrate.
255 return Codec::Matches(codec, field_trials) &&
256 ((codec.clockrate == 0 /*&& clockrate == 8000*/) ||
257 clockrate == codec.clockrate) &&
258 (codec.bitrate == 0 || bitrate <= 0 || bitrate == codec.bitrate) &&
259 ((codec.channels < 2 && channels < 2) || channels == codec.channels);
260 }
261
ToString() const262 std::string AudioCodec::ToString() const {
263 char buf[256];
264 rtc::SimpleStringBuilder sb(buf);
265 sb << "AudioCodec[" << id << ":" << name << ":" << clockrate << ":" << bitrate
266 << ":" << channels << "]";
267 return sb.str();
268 }
269
ToCodecParameters() const270 webrtc::RtpCodecParameters AudioCodec::ToCodecParameters() const {
271 webrtc::RtpCodecParameters codec_params = Codec::ToCodecParameters();
272 codec_params.num_channels = static_cast<int>(channels);
273 codec_params.kind = MEDIA_TYPE_AUDIO;
274 return codec_params;
275 }
276
ToString() const277 std::string VideoCodec::ToString() const {
278 char buf[256];
279 rtc::SimpleStringBuilder sb(buf);
280 sb << "VideoCodec[" << id << ":" << name;
281 if (packetization.has_value()) {
282 sb << ":" << *packetization;
283 }
284 sb << "]";
285 return sb.str();
286 }
287
ToCodecParameters() const288 webrtc::RtpCodecParameters VideoCodec::ToCodecParameters() const {
289 webrtc::RtpCodecParameters codec_params = Codec::ToCodecParameters();
290 codec_params.kind = MEDIA_TYPE_VIDEO;
291 return codec_params;
292 }
293
VideoCodec(int id,const std::string & name)294 VideoCodec::VideoCodec(int id, const std::string& name)
295 : Codec(id, name, kVideoCodecClockrate) {
296 SetDefaultParameters();
297 }
298
VideoCodec(const std::string & name)299 VideoCodec::VideoCodec(const std::string& name) : VideoCodec(0 /* id */, name) {
300 SetDefaultParameters();
301 }
302
VideoCodec()303 VideoCodec::VideoCodec() : Codec() {
304 clockrate = kVideoCodecClockrate;
305 }
306
VideoCodec(const webrtc::SdpVideoFormat & c)307 VideoCodec::VideoCodec(const webrtc::SdpVideoFormat& c)
308 : Codec(0 /* id */, c.name, kVideoCodecClockrate) {
309 params = c.parameters;
310 scalability_modes = c.scalability_modes;
311 }
312
313 VideoCodec::VideoCodec(const VideoCodec& c) = default;
314 VideoCodec::VideoCodec(VideoCodec&& c) = default;
315 VideoCodec& VideoCodec::operator=(const VideoCodec& c) = default;
316 VideoCodec& VideoCodec::operator=(VideoCodec&& c) = default;
317
SetDefaultParameters()318 void VideoCodec::SetDefaultParameters() {
319 if (absl::EqualsIgnoreCase(kH264CodecName, name)) {
320 // This default is set for all H.264 codecs created because
321 // that was the default before packetization mode support was added.
322 // TODO(hta): Move this to the places that create VideoCodecs from
323 // SDP or from knowledge of implementation capabilities.
324 SetParam(kH264FmtpPacketizationMode, "1");
325 }
326 }
327
operator ==(const VideoCodec & c) const328 bool VideoCodec::operator==(const VideoCodec& c) const {
329 return Codec::operator==(c) && packetization == c.packetization;
330 }
331
Matches(const VideoCodec & other,const webrtc::FieldTrialsView * field_trials) const332 bool VideoCodec::Matches(const VideoCodec& other,
333 const webrtc::FieldTrialsView* field_trials) const {
334 return Codec::Matches(other, field_trials) &&
335 IsSameCodecSpecific(name, params, other.name, other.params);
336 }
337
IntersectPacketization(const VideoCodec & local_codec,const VideoCodec & remote_codec)338 absl::optional<std::string> VideoCodec::IntersectPacketization(
339 const VideoCodec& local_codec,
340 const VideoCodec& remote_codec) {
341 if (local_codec.packetization == remote_codec.packetization) {
342 return local_codec.packetization;
343 }
344 return absl::nullopt;
345 }
346
CreateRtxCodec(int rtx_payload_type,int associated_payload_type)347 VideoCodec VideoCodec::CreateRtxCodec(int rtx_payload_type,
348 int associated_payload_type) {
349 VideoCodec rtx_codec(rtx_payload_type, kRtxCodecName);
350 rtx_codec.SetParam(kCodecParamAssociatedPayloadType, associated_payload_type);
351 return rtx_codec;
352 }
353
GetCodecType() const354 VideoCodec::CodecType VideoCodec::GetCodecType() const {
355 if (absl::EqualsIgnoreCase(name, kRedCodecName)) {
356 return CODEC_RED;
357 }
358 if (absl::EqualsIgnoreCase(name, kUlpfecCodecName)) {
359 return CODEC_ULPFEC;
360 }
361 if (absl::EqualsIgnoreCase(name, kFlexfecCodecName)) {
362 return CODEC_FLEXFEC;
363 }
364 if (absl::EqualsIgnoreCase(name, kRtxCodecName)) {
365 return CODEC_RTX;
366 }
367
368 return CODEC_VIDEO;
369 }
370
ValidateCodecFormat() const371 bool VideoCodec::ValidateCodecFormat() const {
372 if (id < 0 || id > 127) {
373 RTC_LOG(LS_ERROR) << "Codec with invalid payload type: " << ToString();
374 return false;
375 }
376 if (GetCodecType() != CODEC_VIDEO) {
377 return true;
378 }
379
380 // Video validation from here on.
381 int min_bitrate = -1;
382 int max_bitrate = -1;
383 if (GetParam(kCodecParamMinBitrate, &min_bitrate) &&
384 GetParam(kCodecParamMaxBitrate, &max_bitrate)) {
385 if (max_bitrate < min_bitrate) {
386 RTC_LOG(LS_ERROR) << "Codec with max < min bitrate: " << ToString();
387 return false;
388 }
389 }
390 return true;
391 }
392
HasLntf(const Codec & codec)393 bool HasLntf(const Codec& codec) {
394 return codec.HasFeedbackParam(
395 FeedbackParam(kRtcpFbParamLntf, kParamValueEmpty));
396 }
397
HasNack(const Codec & codec)398 bool HasNack(const Codec& codec) {
399 return codec.HasFeedbackParam(
400 FeedbackParam(kRtcpFbParamNack, kParamValueEmpty));
401 }
402
HasRemb(const Codec & codec)403 bool HasRemb(const Codec& codec) {
404 return codec.HasFeedbackParam(
405 FeedbackParam(kRtcpFbParamRemb, kParamValueEmpty));
406 }
407
HasRrtr(const Codec & codec)408 bool HasRrtr(const Codec& codec) {
409 return codec.HasFeedbackParam(
410 FeedbackParam(kRtcpFbParamRrtr, kParamValueEmpty));
411 }
412
HasTransportCc(const Codec & codec)413 bool HasTransportCc(const Codec& codec) {
414 return codec.HasFeedbackParam(
415 FeedbackParam(kRtcpFbParamTransportCc, kParamValueEmpty));
416 }
417
FindMatchingCodec(const std::vector<VideoCodec> & supported_codecs,const VideoCodec & codec)418 const VideoCodec* FindMatchingCodec(
419 const std::vector<VideoCodec>& supported_codecs,
420 const VideoCodec& codec) {
421 webrtc::SdpVideoFormat sdp_video_format{codec.name, codec.params};
422 for (const VideoCodec& supported_codec : supported_codecs) {
423 if (sdp_video_format.IsSameCodec(
424 {supported_codec.name, supported_codec.params})) {
425 return &supported_codec;
426 }
427 }
428 return nullptr;
429 }
430
431 // If a decoder supports any H264 profile, it is implicitly assumed to also
432 // support constrained base line even though it's not explicitly listed.
AddH264ConstrainedBaselineProfileToSupportedFormats(std::vector<webrtc::SdpVideoFormat> * supported_formats)433 void AddH264ConstrainedBaselineProfileToSupportedFormats(
434 std::vector<webrtc::SdpVideoFormat>* supported_formats) {
435 std::vector<webrtc::SdpVideoFormat> cbr_supported_formats;
436
437 // For any H264 supported profile, add the corresponding constrained baseline
438 // profile.
439 for (auto it = supported_formats->cbegin(); it != supported_formats->cend();
440 ++it) {
441 if (it->name == cricket::kH264CodecName) {
442 const absl::optional<webrtc::H264ProfileLevelId> profile_level_id =
443 webrtc::ParseSdpForH264ProfileLevelId(it->parameters);
444 if (profile_level_id &&
445 profile_level_id->profile !=
446 webrtc::H264Profile::kProfileConstrainedBaseline) {
447 webrtc::SdpVideoFormat cbp_format = *it;
448 webrtc::H264ProfileLevelId cbp_profile = *profile_level_id;
449 cbp_profile.profile = webrtc::H264Profile::kProfileConstrainedBaseline;
450 cbp_format.parameters[cricket::kH264FmtpProfileLevelId] =
451 *webrtc::H264ProfileLevelIdToString(cbp_profile);
452 cbr_supported_formats.push_back(cbp_format);
453 }
454 }
455 }
456
457 size_t original_size = supported_formats->size();
458 // ...if it's not already in the list.
459 std::copy_if(cbr_supported_formats.begin(), cbr_supported_formats.end(),
460 std::back_inserter(*supported_formats),
461 [supported_formats](const webrtc::SdpVideoFormat& format) {
462 return !format.IsCodecInList(*supported_formats);
463 });
464
465 if (supported_formats->size() > original_size) {
466 RTC_LOG(LS_WARNING) << "Explicitly added H264 constrained baseline to list "
467 "of supported formats.";
468 }
469 }
470
471 } // namespace cricket
472