blob: 147e964b54f9f26121887c407778eae3078d1eda [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"
Magnus Jedvert8deb8182017-10-05 13:13:32 +020019#include "media/base/h264_profile_level_id.h"
Steve Anton10542f22019-01-11 09:11:00 -080020#include "media/base/media_constants.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
Taylor Brandstetter28deb902018-05-30 14:56:50 -070047SdpVideoFormat CreateH264Format(H264::Profile profile,
48 H264::Level level,
49 const std::string& packetization_mode) {
Danil Chapovalov0040b662018-06-18 10:48:16 +020050 const absl::optional<std::string> profile_string =
Magnus Jedvert8deb8182017-10-05 13:13:32 +020051 H264::ProfileLevelIdToString(H264::ProfileLevelId(profile, level));
52 RTC_CHECK(profile_string);
Taylor Brandstetter28deb902018-05-30 14:56:50 -070053 return SdpVideoFormat(
54 cricket::kH264CodecName,
55 {{cricket::kH264FmtpProfileLevelId, *profile_string},
56 {cricket::kH264FmtpLevelAsymmetryAllowed, "1"},
57 {cricket::kH264FmtpPacketizationMode, packetization_mode}});
Magnus Jedvert8deb8182017-10-05 13:13:32 +020058}
59
hbos9dc59282016-02-03 05:09:37 -080060} // namespace
61
62void 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() {
69 if (!IsH264CodecSupported())
70 return std::vector<SdpVideoFormat>();
Magnus Jedvert8deb8182017-10-05 13:13:32 +020071 // We only support encoding Constrained Baseline Profile (CBP), but the
72 // decoder supports more profiles. We can list all profiles here that are
73 // supported by the decoder and that are also supersets of CBP, i.e. the
74 // decoder for that profile is required to be able to decode CBP. This means
75 // we can encode and send CBP even though we negotiated a potentially
76 // higher profile. See the H264 spec for more information.
Taylor Brandstetter28deb902018-05-30 14:56:50 -070077 //
78 // We support both packetization modes 0 (mandatory) and 1 (optional,
79 // preferred).
80 return {
81 CreateH264Format(H264::kProfileBaseline, H264::kLevel3_1, "1"),
82 CreateH264Format(H264::kProfileBaseline, H264::kLevel3_1, "0"),
83 CreateH264Format(H264::kProfileConstrainedBaseline, H264::kLevel3_1, "1"),
84 CreateH264Format(H264::kProfileConstrainedBaseline, H264::kLevel3_1,
85 "0")};
Zeke Chin71f6f442015-06-29 14:34:58 -070086}
87
Magnus Jedvert46a27652017-11-13 14:10:02 +010088std::unique_ptr<H264Encoder> H264Encoder::Create(
89 const cricket::VideoCodec& codec) {
henrikg91d6ede2015-09-17 00:24:34 -070090 RTC_DCHECK(H264Encoder::IsSupported());
hbos9dc59282016-02-03 05:09:37 -080091#if defined(WEBRTC_USE_H264)
92 RTC_CHECK(g_rtc_use_h264);
Mirko Bonadei675513b2017-11-09 11:09:25 +010093 RTC_LOG(LS_INFO) << "Creating H264EncoderImpl.";
Mirko Bonadei317a1f02019-09-17 17:06:18 +020094 return std::make_unique<H264EncoderImpl>(codec);
Zeke Chin71f6f442015-06-29 14:34:58 -070095#else
96 RTC_NOTREACHED();
97 return nullptr;
98#endif
99}
100
101bool H264Encoder::IsSupported() {
102 return IsH264CodecSupported();
103}
104
Magnus Jedvert46a27652017-11-13 14:10:02 +0100105std::unique_ptr<H264Decoder> H264Decoder::Create() {
henrikg91d6ede2015-09-17 00:24:34 -0700106 RTC_DCHECK(H264Decoder::IsSupported());
hbos9dc59282016-02-03 05:09:37 -0800107#if defined(WEBRTC_USE_H264)
108 RTC_CHECK(g_rtc_use_h264);
Mirko Bonadei675513b2017-11-09 11:09:25 +0100109 RTC_LOG(LS_INFO) << "Creating H264DecoderImpl.";
Mirko Bonadei317a1f02019-09-17 17:06:18 +0200110 return std::make_unique<H264DecoderImpl>();
Zeke Chin71f6f442015-06-29 14:34:58 -0700111#else
112 RTC_NOTREACHED();
113 return nullptr;
114#endif
115}
116
117bool H264Decoder::IsSupported() {
118 return IsH264CodecSupported();
119}
120
121} // namespace webrtc