blob: 14e1691153b643344d5a50547f597ffe39b81935 [file] [log] [blame]
Zeke Chin71f6f442015-06-29 14:34:58 -07001/*
2 * Copyright (c) 2015 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
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020012#include "modules/video_coding/codecs/h264/include/h264.h"
Zeke Chin71f6f442015-06-29 14:34:58 -070013
Mirko Bonadei317a1f02019-09-17 17:06:18 +020014#include <memory>
Yves Gerey3e707812018-11-28 16:47:49 +010015#include <string>
16
17#include "absl/types/optional.h"
Magnus Jedvert849b3ae2017-09-29 17:54:09 +020018#include "api/video_codecs/sdp_video_format.h"
Steve Anton10542f22019-01-11 09:11:00 -080019#include "media/base/media_constants.h"
Markus Handellfccb0522021-06-02 13:28:10 +020020#include "rtc_base/trace_event.h"
Magnus Jedvert849b3ae2017-09-29 17:54:09 +020021
hbos9dc59282016-02-03 05:09:37 -080022#if defined(WEBRTC_USE_H264)
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020023#include "modules/video_coding/codecs/h264/h264_decoder_impl.h"
24#include "modules/video_coding/codecs/h264/h264_encoder_impl.h"
hbosbab934b2016-01-27 01:36:03 -080025#endif
Zeke Chin71f6f442015-06-29 14:34:58 -070026
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020027#include "rtc_base/checks.h"
28#include "rtc_base/logging.h"
Zeke Chin71f6f442015-06-29 14:34:58 -070029
30namespace webrtc {
31
hbos9dc59282016-02-03 05:09:37 -080032namespace {
33
34#if defined(WEBRTC_USE_H264)
35bool g_rtc_use_h264 = true;
36#endif
37
Magnus Jedvert8deb8182017-10-05 13:13:32 +020038// If H.264 OpenH264/FFmpeg codec is supported.
Magnus Jedvert849b3ae2017-09-29 17:54:09 +020039bool IsH264CodecSupported() {
40#if defined(WEBRTC_USE_H264)
41 return g_rtc_use_h264;
42#else
43 return false;
44#endif
45}
46
Johannes Kron3e983682020-03-29 22:17:00 +020047} // namespace
48
Johannes Kronc3fcee72021-04-19 09:09:26 +020049SdpVideoFormat CreateH264Format(H264Profile profile,
50 H264Level level,
Taylor Brandstetter28deb902018-05-30 14:56:50 -070051 const std::string& packetization_mode) {
Danil Chapovalov0040b662018-06-18 10:48:16 +020052 const absl::optional<std::string> profile_string =
Johannes Kronc3fcee72021-04-19 09:09:26 +020053 H264ProfileLevelIdToString(H264ProfileLevelId(profile, level));
Magnus Jedvert8deb8182017-10-05 13:13:32 +020054 RTC_CHECK(profile_string);
Taylor Brandstetter28deb902018-05-30 14:56:50 -070055 return SdpVideoFormat(
56 cricket::kH264CodecName,
57 {{cricket::kH264FmtpProfileLevelId, *profile_string},
58 {cricket::kH264FmtpLevelAsymmetryAllowed, "1"},
59 {cricket::kH264FmtpPacketizationMode, packetization_mode}});
Magnus Jedvert8deb8182017-10-05 13:13:32 +020060}
61
hbos9dc59282016-02-03 05:09:37 -080062void DisableRtcUseH264() {
63#if defined(WEBRTC_USE_H264)
64 g_rtc_use_h264 = false;
65#endif
66}
67
Magnus Jedvert849b3ae2017-09-29 17:54:09 +020068std::vector<SdpVideoFormat> SupportedH264Codecs() {
Markus Handellfccb0522021-06-02 13:28:10 +020069 TRACE_EVENT0("webrtc", __func__);
Magnus Jedvert849b3ae2017-09-29 17:54:09 +020070 if (!IsH264CodecSupported())
71 return std::vector<SdpVideoFormat>();
Magnus Jedvert8deb8182017-10-05 13:13:32 +020072 // We only support encoding Constrained Baseline Profile (CBP), but the
73 // decoder supports more profiles. We can list all profiles here that are
74 // supported by the decoder and that are also supersets of CBP, i.e. the
75 // decoder for that profile is required to be able to decode CBP. This means
76 // we can encode and send CBP even though we negotiated a potentially
77 // higher profile. See the H264 spec for more information.
Taylor Brandstetter28deb902018-05-30 14:56:50 -070078 //
79 // We support both packetization modes 0 (mandatory) and 1 (optional,
80 // preferred).
Johannes Kronc3fcee72021-04-19 09:09:26 +020081 return {CreateH264Format(H264Profile::kProfileBaseline, H264Level::kLevel3_1,
82 "1"),
83 CreateH264Format(H264Profile::kProfileBaseline, H264Level::kLevel3_1,
84 "0"),
85 CreateH264Format(H264Profile::kProfileConstrainedBaseline,
86 H264Level::kLevel3_1, "1"),
87 CreateH264Format(H264Profile::kProfileConstrainedBaseline,
88 H264Level::kLevel3_1, "0")};
Zeke Chin71f6f442015-06-29 14:34:58 -070089}
90
Magnus Jedvert46a27652017-11-13 14:10:02 +010091std::unique_ptr<H264Encoder> H264Encoder::Create(
92 const cricket::VideoCodec& codec) {
henrikg91d6ede2015-09-17 00:24:34 -070093 RTC_DCHECK(H264Encoder::IsSupported());
hbos9dc59282016-02-03 05:09:37 -080094#if defined(WEBRTC_USE_H264)
95 RTC_CHECK(g_rtc_use_h264);
Mirko Bonadei675513b2017-11-09 11:09:25 +010096 RTC_LOG(LS_INFO) << "Creating H264EncoderImpl.";
Mirko Bonadei317a1f02019-09-17 17:06:18 +020097 return std::make_unique<H264EncoderImpl>(codec);
Zeke Chin71f6f442015-06-29 14:34:58 -070098#else
99 RTC_NOTREACHED();
100 return nullptr;
101#endif
102}
103
104bool H264Encoder::IsSupported() {
105 return IsH264CodecSupported();
106}
107
Magnus Jedvert46a27652017-11-13 14:10:02 +0100108std::unique_ptr<H264Decoder> H264Decoder::Create() {
henrikg91d6ede2015-09-17 00:24:34 -0700109 RTC_DCHECK(H264Decoder::IsSupported());
hbos9dc59282016-02-03 05:09:37 -0800110#if defined(WEBRTC_USE_H264)
111 RTC_CHECK(g_rtc_use_h264);
Mirko Bonadei675513b2017-11-09 11:09:25 +0100112 RTC_LOG(LS_INFO) << "Creating H264DecoderImpl.";
Mirko Bonadei317a1f02019-09-17 17:06:18 +0200113 return std::make_unique<H264DecoderImpl>();
Zeke Chin71f6f442015-06-29 14:34:58 -0700114#else
115 RTC_NOTREACHED();
116 return nullptr;
117#endif
118}
119
120bool H264Decoder::IsSupported() {
121 return IsH264CodecSupported();
122}
123
124} // namespace webrtc