blob: 82badc10440c442fc9cec0f2b6bd237df4b5e929 [file] [log] [blame]
pbos@webrtc.org289a35c2014-06-03 14:51:34 +00001/*
kjellander1afca732016-02-07 20:46:45 -08002 * Copyright (c) 2014 The WebRTC project authors. All Rights Reserved.
pbos@webrtc.org289a35c2014-06-03 14:51:34 +00003 *
kjellander1afca732016-02-07 20:46:45 -08004 * 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.
pbos@webrtc.org289a35c2014-06-03 14:51:34 +00009 */
10
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020011#include "media/engine/webrtcmediaengine.h"
solenberg7e4e01a2015-12-02 08:05:01 -080012
13#include <algorithm>
Steve Antone78bcb92017-10-31 09:53:08 -070014#include <memory>
15#include <tuple>
16#include <utility>
solenberg7e4e01a2015-12-02 08:05:01 -080017
Magnus Jedvert58b03162017-09-15 19:02:47 +020018#include "api/video_codecs/video_decoder_factory.h"
19#include "api/video_codecs/video_encoder_factory.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020020#include "media/engine/webrtcvoiceengine.h"
kwiberg087bd342017-02-10 08:15:44 -080021
jbauch4cb3e392016-01-26 13:07:54 -080022#ifdef HAVE_WEBRTC_VIDEO
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020023#include "media/engine/webrtcvideoengine.h"
jbauch4cb3e392016-01-26 13:07:54 -080024#else
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020025#include "media/engine/nullwebrtcvideoengine.h"
jbauch4cb3e392016-01-26 13:07:54 -080026#endif
henrike@webrtc.org0481f152014-08-19 14:56:59 +000027
28namespace cricket {
29
Anders Carlssondd8c1652018-01-30 10:32:13 +010030#if defined(USE_BUILTIN_SW_CODECS)
magjed2475ae22017-09-12 04:42:15 -070031namespace {
henrike@webrtc.org0481f152014-08-19 14:56:59 +000032
magjed2475ae22017-09-12 04:42:15 -070033MediaEngineInterface* CreateWebRtcMediaEngine(
pbos@webrtc.org9e65a3b2014-06-11 13:42:37 +000034 webrtc::AudioDeviceModule* adm,
ossueb1fde42017-05-02 06:46:30 -070035 const rtc::scoped_refptr<webrtc::AudioEncoderFactory>&
36 audio_encoder_factory,
ossu29b1a8d2016-06-13 07:34:51 -070037 const rtc::scoped_refptr<webrtc::AudioDecoderFactory>&
38 audio_decoder_factory,
magjed2475ae22017-09-12 04:42:15 -070039 WebRtcVideoEncoderFactory* video_encoder_factory,
40 WebRtcVideoDecoderFactory* video_decoder_factory,
peaha9cc40b2017-06-29 08:32:09 -070041 rtc::scoped_refptr<webrtc::AudioMixer> audio_mixer,
42 rtc::scoped_refptr<webrtc::AudioProcessing> audio_processing) {
magjed2475ae22017-09-12 04:42:15 -070043#ifdef HAVE_WEBRTC_VIDEO
44 typedef WebRtcVideoEngine VideoEngine;
Magnus Jedvert02e7a192017-09-23 17:21:32 +020045 std::tuple<std::unique_ptr<WebRtcVideoEncoderFactory>,
46 std::unique_ptr<WebRtcVideoDecoderFactory>>
47 video_args(
48 (std::unique_ptr<WebRtcVideoEncoderFactory>(video_encoder_factory)),
49 (std::unique_ptr<WebRtcVideoDecoderFactory>(video_decoder_factory)));
magjed2475ae22017-09-12 04:42:15 -070050#else
51 typedef NullWebRtcVideoEngine VideoEngine;
52 std::tuple<> video_args;
53#endif
54 return new CompositeMediaEngine<WebRtcVoiceEngine, VideoEngine>(
55 std::forward_as_tuple(adm, audio_encoder_factory, audio_decoder_factory,
56 audio_mixer, audio_processing),
57 std::move(video_args));
pbos@webrtc.org9e65a3b2014-06-11 13:42:37 +000058}
59
magjed2475ae22017-09-12 04:42:15 -070060} // namespace
henrike@webrtc.org0481f152014-08-19 14:56:59 +000061
ossueb1fde42017-05-02 06:46:30 -070062MediaEngineInterface* WebRtcMediaEngineFactory::Create(
63 webrtc::AudioDeviceModule* adm,
64 const rtc::scoped_refptr<webrtc::AudioEncoderFactory>&
65 audio_encoder_factory,
66 const rtc::scoped_refptr<webrtc::AudioDecoderFactory>&
67 audio_decoder_factory,
68 WebRtcVideoEncoderFactory* video_encoder_factory,
69 WebRtcVideoDecoderFactory* video_decoder_factory) {
Ivo Creusen62337e52018-01-09 14:17:33 +010070 return CreateWebRtcMediaEngine(adm, audio_encoder_factory,
71 audio_decoder_factory, video_encoder_factory,
72 video_decoder_factory, nullptr,
73 webrtc::AudioProcessingBuilder().Create());
ossueb1fde42017-05-02 06:46:30 -070074}
75
76MediaEngineInterface* WebRtcMediaEngineFactory::Create(
77 webrtc::AudioDeviceModule* adm,
78 const rtc::scoped_refptr<webrtc::AudioEncoderFactory>&
79 audio_encoder_factory,
80 const rtc::scoped_refptr<webrtc::AudioDecoderFactory>&
81 audio_decoder_factory,
82 WebRtcVideoEncoderFactory* video_encoder_factory,
83 WebRtcVideoDecoderFactory* video_decoder_factory,
peaha9cc40b2017-06-29 08:32:09 -070084 rtc::scoped_refptr<webrtc::AudioMixer> audio_mixer,
85 rtc::scoped_refptr<webrtc::AudioProcessing> audio_processing) {
86 return CreateWebRtcMediaEngine(
87 adm, audio_encoder_factory, audio_decoder_factory, video_encoder_factory,
88 video_decoder_factory, audio_mixer, audio_processing);
henrike@webrtc.org0481f152014-08-19 14:56:59 +000089}
Anders Carlssondd8c1652018-01-30 10:32:13 +010090#endif
henrike@webrtc.org0481f152014-08-19 14:56:59 +000091
Magnus Jedvert58b03162017-09-15 19:02:47 +020092std::unique_ptr<MediaEngineInterface> WebRtcMediaEngineFactory::Create(
93 rtc::scoped_refptr<webrtc::AudioDeviceModule> adm,
94 rtc::scoped_refptr<webrtc::AudioEncoderFactory> audio_encoder_factory,
95 rtc::scoped_refptr<webrtc::AudioDecoderFactory> audio_decoder_factory,
96 std::unique_ptr<webrtc::VideoEncoderFactory> video_encoder_factory,
97 std::unique_ptr<webrtc::VideoDecoderFactory> video_decoder_factory,
98 rtc::scoped_refptr<webrtc::AudioMixer> audio_mixer,
99 rtc::scoped_refptr<webrtc::AudioProcessing> audio_processing) {
100#ifdef HAVE_WEBRTC_VIDEO
101 typedef WebRtcVideoEngine VideoEngine;
102 std::tuple<std::unique_ptr<webrtc::VideoEncoderFactory>,
103 std::unique_ptr<webrtc::VideoDecoderFactory>>
104 video_args(std::move(video_encoder_factory),
105 std::move(video_decoder_factory));
106#else
107 typedef NullWebRtcVideoEngine VideoEngine;
108 std::tuple<> video_args;
109#endif
110 return std::unique_ptr<MediaEngineInterface>(
111 new CompositeMediaEngine<WebRtcVoiceEngine, VideoEngine>(
112 std::forward_as_tuple(adm, audio_encoder_factory,
113 audio_decoder_factory, audio_mixer,
114 audio_processing),
115 std::move(video_args)));
116}
117
solenberg7e4e01a2015-12-02 08:05:01 -0800118namespace {
119// Remove mutually exclusive extensions with lower priority.
120void DiscardRedundantExtensions(
121 std::vector<webrtc::RtpExtension>* extensions,
agrieve26622d32017-08-08 10:48:15 -0700122 rtc::ArrayView<const char* const> extensions_decreasing_prio) {
solenberg7e4e01a2015-12-02 08:05:01 -0800123 RTC_DCHECK(extensions);
124 bool found = false;
isheriff6f8d6862016-05-26 11:24:55 -0700125 for (const char* uri : extensions_decreasing_prio) {
126 auto it = std::find_if(
127 extensions->begin(), extensions->end(),
128 [uri](const webrtc::RtpExtension& rhs) { return rhs.uri == uri; });
solenberg7e4e01a2015-12-02 08:05:01 -0800129 if (it != extensions->end()) {
130 if (found) {
131 extensions->erase(it);
132 }
133 found = true;
Stefan Holmerbbaf3632015-10-29 18:53:23 +0100134 }
135 }
solenberg7e4e01a2015-12-02 08:05:01 -0800136}
137} // namespace
138
isheriff6f8d6862016-05-26 11:24:55 -0700139bool ValidateRtpExtensions(
140 const std::vector<webrtc::RtpExtension>& extensions) {
solenberg7e4e01a2015-12-02 08:05:01 -0800141 bool id_used[14] = {false};
142 for (const auto& extension : extensions) {
143 if (extension.id <= 0 || extension.id >= 15) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100144 RTC_LOG(LS_ERROR) << "Bad RTP extension ID: " << extension.ToString();
solenberg7e4e01a2015-12-02 08:05:01 -0800145 return false;
146 }
147 if (id_used[extension.id - 1]) {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100148 RTC_LOG(LS_ERROR) << "Duplicate RTP extension ID: "
149 << extension.ToString();
solenberg7e4e01a2015-12-02 08:05:01 -0800150 return false;
151 }
152 id_used[extension.id - 1] = true;
153 }
154 return true;
Stefan Holmerbbaf3632015-10-29 18:53:23 +0100155}
156
solenberg7e4e01a2015-12-02 08:05:01 -0800157std::vector<webrtc::RtpExtension> FilterRtpExtensions(
isheriff6f8d6862016-05-26 11:24:55 -0700158 const std::vector<webrtc::RtpExtension>& extensions,
solenberg7e4e01a2015-12-02 08:05:01 -0800159 bool (*supported)(const std::string&),
160 bool filter_redundant_extensions) {
161 RTC_DCHECK(ValidateRtpExtensions(extensions));
162 RTC_DCHECK(supported);
163 std::vector<webrtc::RtpExtension> result;
164
165 // Ignore any extensions that we don't recognize.
166 for (const auto& extension : extensions) {
167 if (supported(extension.uri)) {
isheriff6f8d6862016-05-26 11:24:55 -0700168 result.push_back(extension);
solenberg7e4e01a2015-12-02 08:05:01 -0800169 } else {
Mirko Bonadei675513b2017-11-09 11:09:25 +0100170 RTC_LOG(LS_WARNING) << "Unsupported RTP extension: "
171 << extension.ToString();
solenberg7e4e01a2015-12-02 08:05:01 -0800172 }
173 }
174
jbauch5869f502017-06-29 12:31:36 -0700175 // Sort by name, ascending (prioritise encryption), so that we don't reset
176 // extensions if they were specified in a different order (also allows us
177 // to use std::unique below).
Yves Gerey665174f2018-06-19 15:03:05 +0200178 std::sort(
179 result.begin(), result.end(),
180 [](const webrtc::RtpExtension& rhs, const webrtc::RtpExtension& lhs) {
181 return rhs.encrypt == lhs.encrypt ? rhs.uri < lhs.uri
182 : rhs.encrypt > lhs.encrypt;
183 });
solenberg7e4e01a2015-12-02 08:05:01 -0800184
185 // Remove unnecessary extensions (used on send side).
186 if (filter_redundant_extensions) {
isheriff6f8d6862016-05-26 11:24:55 -0700187 auto it = std::unique(
188 result.begin(), result.end(),
solenberg7e4e01a2015-12-02 08:05:01 -0800189 [](const webrtc::RtpExtension& rhs, const webrtc::RtpExtension& lhs) {
jbauch5869f502017-06-29 12:31:36 -0700190 return rhs.uri == lhs.uri && rhs.encrypt == lhs.encrypt;
solenberg7e4e01a2015-12-02 08:05:01 -0800191 });
192 result.erase(it, result.end());
193
194 // Keep just the highest priority extension of any in the following list.
agrieve26622d32017-08-08 10:48:15 -0700195 static const char* const kBweExtensionPriorities[] = {
isheriff6f8d6862016-05-26 11:24:55 -0700196 webrtc::RtpExtension::kTransportSequenceNumberUri,
197 webrtc::RtpExtension::kAbsSendTimeUri,
198 webrtc::RtpExtension::kTimestampOffsetUri};
solenberg7e4e01a2015-12-02 08:05:01 -0800199 DiscardRedundantExtensions(&result, kBweExtensionPriorities);
200 }
201
202 return result;
203}
stefan13f1a0a2016-11-30 07:22:58 -0800204
Sebastian Janssonfc8d26b2018-02-21 09:52:06 +0100205webrtc::BitrateConstraints GetBitrateConfigForCodec(const Codec& codec) {
206 webrtc::BitrateConstraints config;
stefan13f1a0a2016-11-30 07:22:58 -0800207 int bitrate_kbps = 0;
208 if (codec.GetParam(kCodecParamMinBitrate, &bitrate_kbps) &&
209 bitrate_kbps > 0) {
210 config.min_bitrate_bps = bitrate_kbps * 1000;
211 } else {
212 config.min_bitrate_bps = 0;
213 }
214 if (codec.GetParam(kCodecParamStartBitrate, &bitrate_kbps) &&
215 bitrate_kbps > 0) {
216 config.start_bitrate_bps = bitrate_kbps * 1000;
217 } else {
218 // Do not reconfigure start bitrate unless it's specified and positive.
219 config.start_bitrate_bps = -1;
220 }
221 if (codec.GetParam(kCodecParamMaxBitrate, &bitrate_kbps) &&
222 bitrate_kbps > 0) {
223 config.max_bitrate_bps = bitrate_kbps * 1000;
224 } else {
225 config.max_bitrate_bps = -1;
226 }
227 return config;
228}
henrike@webrtc.org0481f152014-08-19 14:56:59 +0000229} // namespace cricket