blob: 4d8125fe8ba3cd8174d9a6e808f076ec411b2499 [file] [log] [blame]
henrike@webrtc.org28e20752013-07-10 00:45:36 +00001/*
kjellanderb24317b2016-02-10 07:54:43 -08002 * Copyright 2004 The WebRTC project authors. All Rights Reserved.
henrike@webrtc.org28e20752013-07-10 00:45:36 +00003 *
kjellanderb24317b2016-02-10 07:54:43 -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.
henrike@webrtc.org28e20752013-07-10 00:45:36 +00009 */
10
Henrik Kjellander15583c12016-02-10 10:53:12 +010011#include "webrtc/api/peerconnectionfactory.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000012
kwiberg0eb15ed2015-12-17 03:04:15 -080013#include <utility>
14
Henrik Kjellander15583c12016-02-10 10:53:12 +010015#include "webrtc/api/audiotrack.h"
16#include "webrtc/api/localaudiosource.h"
htaa2a49d92016-03-04 02:51:39 -080017#include "webrtc/api/mediaconstraintsinterface.h"
Henrik Kjellander15583c12016-02-10 10:53:12 +010018#include "webrtc/api/mediastream.h"
19#include "webrtc/api/mediastreamproxy.h"
20#include "webrtc/api/mediastreamtrackproxy.h"
21#include "webrtc/api/peerconnection.h"
22#include "webrtc/api/peerconnectionfactoryproxy.h"
23#include "webrtc/api/peerconnectionproxy.h"
24#include "webrtc/api/videosource.h"
25#include "webrtc/api/videosourceproxy.h"
26#include "webrtc/api/videotrack.h"
jiayl@webrtc.org3987b6d2014-09-24 17:14:05 +000027#include "webrtc/base/bind.h"
kjellander@webrtc.org5ad12972016-02-12 06:39:40 +010028#include "webrtc/media/engine/webrtcmediaengine.h"
29#include "webrtc/media/engine/webrtcvideodecoderfactory.h"
30#include "webrtc/media/engine/webrtcvideoencoderfactory.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000031#include "webrtc/modules/audio_device/include/audio_device.h"
deadbeef41b07982015-12-01 15:01:24 -080032#include "webrtc/p2p/base/basicpacketsocketfactory.h"
33#include "webrtc/p2p/client/basicportallocator.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000034
henrike@webrtc.org28e20752013-07-10 00:45:36 +000035namespace webrtc {
36
Henrik Boström5e56c592015-08-11 10:33:13 +020037namespace {
38
39// Passes down the calls to |store_|. See usage in CreatePeerConnection.
40class DtlsIdentityStoreWrapper : public DtlsIdentityStoreInterface {
41 public:
42 DtlsIdentityStoreWrapper(
43 const rtc::scoped_refptr<RefCountedDtlsIdentityStore>& store)
44 : store_(store) {
henrikg91d6ede2015-09-17 00:24:34 -070045 RTC_DCHECK(store_);
Henrik Boström5e56c592015-08-11 10:33:13 +020046 }
47
48 void RequestIdentity(
hbos25359e02016-03-02 07:55:53 -080049 rtc::KeyParams key_params,
50 rtc::Optional<uint64_t> expires,
Henrik Boström5e56c592015-08-11 10:33:13 +020051 const rtc::scoped_refptr<webrtc::DtlsIdentityRequestObserver>&
52 observer) override {
hbos25359e02016-03-02 07:55:53 -080053 store_->RequestIdentity(key_params, expires, observer);
Henrik Boström5e56c592015-08-11 10:33:13 +020054 }
55
56 private:
57 rtc::scoped_refptr<RefCountedDtlsIdentityStore> store_;
58};
59
60} // anonymous namespace
61
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000062rtc::scoped_refptr<PeerConnectionFactoryInterface>
henrike@webrtc.org28e20752013-07-10 00:45:36 +000063CreatePeerConnectionFactory() {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000064 rtc::scoped_refptr<PeerConnectionFactory> pc_factory(
65 new rtc::RefCountedObject<PeerConnectionFactory>());
henrike@webrtc.org28e20752013-07-10 00:45:36 +000066
perkj@webrtc.org81134d02015-01-12 08:30:16 +000067
68 // Call Initialize synchronously but make sure its executed on
69 // |signaling_thread|.
70 MethodCall0<PeerConnectionFactory, bool> call(
71 pc_factory.get(),
72 &PeerConnectionFactory::Initialize);
73 bool result = call.Marshal(pc_factory->signaling_thread());
74
75 if (!result) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +000076 return NULL;
77 }
perkj@webrtc.org81134d02015-01-12 08:30:16 +000078 return PeerConnectionFactoryProxy::Create(pc_factory->signaling_thread(),
79 pc_factory);
henrike@webrtc.org28e20752013-07-10 00:45:36 +000080}
81
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000082rtc::scoped_refptr<PeerConnectionFactoryInterface>
henrike@webrtc.org28e20752013-07-10 00:45:36 +000083CreatePeerConnectionFactory(
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000084 rtc::Thread* worker_thread,
85 rtc::Thread* signaling_thread,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000086 AudioDeviceModule* default_adm,
87 cricket::WebRtcVideoEncoderFactory* encoder_factory,
88 cricket::WebRtcVideoDecoderFactory* decoder_factory) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000089 rtc::scoped_refptr<PeerConnectionFactory> pc_factory(
90 new rtc::RefCountedObject<PeerConnectionFactory>(worker_thread,
perkj@webrtc.org81134d02015-01-12 08:30:16 +000091 signaling_thread,
92 default_adm,
93 encoder_factory,
94 decoder_factory));
95
96 // Call Initialize synchronously but make sure its executed on
97 // |signaling_thread|.
98 MethodCall0<PeerConnectionFactory, bool> call(
99 pc_factory.get(),
100 &PeerConnectionFactory::Initialize);
101 bool result = call.Marshal(signaling_thread);
102
103 if (!result) {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000104 return NULL;
105 }
perkj@webrtc.org81134d02015-01-12 08:30:16 +0000106 return PeerConnectionFactoryProxy::Create(signaling_thread, pc_factory);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000107}
108
109PeerConnectionFactory::PeerConnectionFactory()
110 : owns_ptrs_(true),
perkj@webrtc.org81134d02015-01-12 08:30:16 +0000111 wraps_current_thread_(false),
112 signaling_thread_(rtc::ThreadManager::Instance()->CurrentThread()),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000113 worker_thread_(new rtc::Thread) {
perkj@webrtc.org81134d02015-01-12 08:30:16 +0000114 if (!signaling_thread_) {
115 signaling_thread_ = rtc::ThreadManager::Instance()->WrapCurrentThread();
116 wraps_current_thread_ = true;
117 }
118 worker_thread_->Start();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000119}
120
121PeerConnectionFactory::PeerConnectionFactory(
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000122 rtc::Thread* worker_thread,
123 rtc::Thread* signaling_thread,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000124 AudioDeviceModule* default_adm,
125 cricket::WebRtcVideoEncoderFactory* video_encoder_factory,
126 cricket::WebRtcVideoDecoderFactory* video_decoder_factory)
127 : owns_ptrs_(false),
perkj@webrtc.org81134d02015-01-12 08:30:16 +0000128 wraps_current_thread_(false),
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000129 signaling_thread_(signaling_thread),
130 worker_thread_(worker_thread),
131 default_adm_(default_adm),
132 video_encoder_factory_(video_encoder_factory),
133 video_decoder_factory_(video_decoder_factory) {
134 ASSERT(worker_thread != NULL);
135 ASSERT(signaling_thread != NULL);
136 // TODO: Currently there is no way creating an external adm in
137 // libjingle source tree. So we can 't currently assert if this is NULL.
138 // ASSERT(default_adm != NULL);
139}
140
141PeerConnectionFactory::~PeerConnectionFactory() {
henrikg91d6ede2015-09-17 00:24:34 -0700142 RTC_DCHECK(signaling_thread_->IsCurrent());
Henrik Boström5e56c592015-08-11 10:33:13 +0200143 channel_manager_.reset(nullptr);
jiayl@webrtc.orgd83f4ef2015-03-13 21:26:12 +0000144
145 // Make sure |worker_thread_| and |signaling_thread_| outlive
deadbeef41b07982015-12-01 15:01:24 -0800146 // |dtls_identity_store_|, |default_socket_factory_| and
147 // |default_network_manager_|.
Henrik Boström5e56c592015-08-11 10:33:13 +0200148 dtls_identity_store_ = nullptr;
deadbeef41b07982015-12-01 15:01:24 -0800149 default_socket_factory_ = nullptr;
150 default_network_manager_ = nullptr;
jiayl@webrtc.orgd83f4ef2015-03-13 21:26:12 +0000151
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000152 if (owns_ptrs_) {
perkj@webrtc.org81134d02015-01-12 08:30:16 +0000153 if (wraps_current_thread_)
154 rtc::ThreadManager::Instance()->UnwrapCurrentThread();
fischman@webrtc.org29540b12014-04-17 22:54:30 +0000155 delete worker_thread_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000156 }
157}
158
159bool PeerConnectionFactory::Initialize() {
henrikg91d6ede2015-09-17 00:24:34 -0700160 RTC_DCHECK(signaling_thread_->IsCurrent());
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000161 rtc::InitRandom(rtc::Time());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000162
deadbeef41b07982015-12-01 15:01:24 -0800163 default_network_manager_.reset(new rtc::BasicNetworkManager());
164 if (!default_network_manager_) {
165 return false;
166 }
167
168 default_socket_factory_.reset(
169 new rtc::BasicPacketSocketFactory(worker_thread_));
170 if (!default_socket_factory_) {
171 return false;
172 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000173
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000174 // TODO: Need to make sure only one VoE is created inside
175 // WebRtcMediaEngine.
henrika@webrtc.org62f6e752015-02-11 08:38:35 +0000176 cricket::MediaEngineInterface* media_engine =
177 worker_thread_->Invoke<cricket::MediaEngineInterface*>(rtc::Bind(
178 &PeerConnectionFactory::CreateMediaEngine_w, this));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000179
solenbergfacbbec2015-09-24 00:41:50 -0700180 channel_manager_.reset(
181 new cricket::ChannelManager(media_engine, worker_thread_));
henrika@webrtc.org62f6e752015-02-11 08:38:35 +0000182
stefan@webrtc.org85d27942014-06-09 12:51:39 +0000183 channel_manager_->SetVideoRtxEnabled(true);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000184 if (!channel_manager_->Init()) {
185 return false;
186 }
jiayl@webrtc.org61e00b02015-03-04 22:17:38 +0000187
Henrik Boström5e56c592015-08-11 10:33:13 +0200188 dtls_identity_store_ = new RefCountedDtlsIdentityStore(
189 signaling_thread_, worker_thread_);
jiayl@webrtc.org61e00b02015-03-04 22:17:38 +0000190
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000191 return true;
192}
193
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000194rtc::scoped_refptr<AudioSourceInterface>
perkj@webrtc.org81134d02015-01-12 08:30:16 +0000195PeerConnectionFactory::CreateAudioSource(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000196 const MediaConstraintsInterface* constraints) {
henrikg91d6ede2015-09-17 00:24:34 -0700197 RTC_DCHECK(signaling_thread_->IsCurrent());
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000198 rtc::scoped_refptr<LocalAudioSource> source(
wu@webrtc.org97077a32013-10-25 21:18:33 +0000199 LocalAudioSource::Create(options_, constraints));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000200 return source;
201}
202
htaa2a49d92016-03-04 02:51:39 -0800203rtc::scoped_refptr<AudioSourceInterface>
204PeerConnectionFactory::CreateAudioSource(const cricket::AudioOptions& options) {
205 RTC_DCHECK(signaling_thread_->IsCurrent());
206 rtc::scoped_refptr<LocalAudioSource> source(
207 LocalAudioSource::Create(options_, &options));
208 return source;
209}
210
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000211rtc::scoped_refptr<VideoSourceInterface>
perkj@webrtc.org81134d02015-01-12 08:30:16 +0000212PeerConnectionFactory::CreateVideoSource(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000213 cricket::VideoCapturer* capturer,
214 const MediaConstraintsInterface* constraints) {
henrikg91d6ede2015-09-17 00:24:34 -0700215 RTC_DCHECK(signaling_thread_->IsCurrent());
tommi6eca7e32015-12-15 04:27:11 -0800216 rtc::scoped_refptr<VideoSource> source(VideoSource::Create(
Per0f13ec12016-03-03 09:22:28 +0100217 worker_thread_, capturer, constraints, false));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000218 return VideoSourceProxy::Create(signaling_thread_, source);
219}
220
htaa2a49d92016-03-04 02:51:39 -0800221rtc::scoped_refptr<VideoSourceInterface>
222PeerConnectionFactory::CreateVideoSource(cricket::VideoCapturer* capturer) {
223 RTC_DCHECK(signaling_thread_->IsCurrent());
224 rtc::scoped_refptr<VideoSource> source(
225 VideoSource::Create(worker_thread_, capturer, false));
226 return VideoSourceProxy::Create(signaling_thread_, source);
227}
228
ivocd66b44d2016-01-15 03:06:36 -0800229bool PeerConnectionFactory::StartAecDump(rtc::PlatformFile file,
230 int64_t max_size_bytes) {
henrikg91d6ede2015-09-17 00:24:34 -0700231 RTC_DCHECK(signaling_thread_->IsCurrent());
ivocd66b44d2016-01-15 03:06:36 -0800232 return channel_manager_->StartAecDump(file, max_size_bytes);
wu@webrtc.orga9890802013-12-13 00:21:03 +0000233}
234
ivoc797ef122015-10-22 03:25:41 -0700235void PeerConnectionFactory::StopAecDump() {
236 RTC_DCHECK(signaling_thread_->IsCurrent());
237 channel_manager_->StopAecDump();
238}
239
ivoc112a3d82015-10-16 02:22:18 -0700240bool PeerConnectionFactory::StartRtcEventLog(rtc::PlatformFile file) {
241 RTC_DCHECK(signaling_thread_->IsCurrent());
242 return channel_manager_->StartRtcEventLog(file);
243}
244
245void PeerConnectionFactory::StopRtcEventLog() {
246 RTC_DCHECK(signaling_thread_->IsCurrent());
247 channel_manager_->StopRtcEventLog();
248}
249
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000250rtc::scoped_refptr<PeerConnectionInterface>
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000251PeerConnectionFactory::CreatePeerConnection(
htaa2a49d92016-03-04 02:51:39 -0800252 const PeerConnectionInterface::RTCConfiguration& configuration_in,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000253 const MediaConstraintsInterface* constraints,
deadbeef41b07982015-12-01 15:01:24 -0800254 rtc::scoped_ptr<cricket::PortAllocator> allocator,
255 rtc::scoped_ptr<DtlsIdentityStoreInterface> dtls_identity_store,
256 PeerConnectionObserver* observer) {
257 RTC_DCHECK(signaling_thread_->IsCurrent());
258
htaa2a49d92016-03-04 02:51:39 -0800259 // We merge constraints and configuration into a single configuration.
260 PeerConnectionInterface::RTCConfiguration configuration = configuration_in;
261 CopyConstraintsIntoRtcConfiguration(constraints, &configuration);
262
263 return CreatePeerConnection(configuration, std::move(allocator),
264 std::move(dtls_identity_store), observer);
265}
266
267rtc::scoped_refptr<PeerConnectionInterface>
268PeerConnectionFactory::CreatePeerConnection(
269 const PeerConnectionInterface::RTCConfiguration& configuration,
270 rtc::scoped_ptr<cricket::PortAllocator> allocator,
271 rtc::scoped_ptr<DtlsIdentityStoreInterface> dtls_identity_store,
272 PeerConnectionObserver* observer) {
273 RTC_DCHECK(signaling_thread_->IsCurrent());
274
deadbeef41b07982015-12-01 15:01:24 -0800275 if (!dtls_identity_store.get()) {
276 // Because |pc|->Initialize takes ownership of the store we need a new
277 // wrapper object that can be deleted without deleting the underlying
278 // |dtls_identity_store_|, protecting it from being deleted multiple times.
279 dtls_identity_store.reset(
280 new DtlsIdentityStoreWrapper(dtls_identity_store_));
281 }
282
283 if (!allocator) {
284 allocator.reset(new cricket::BasicPortAllocator(
285 default_network_manager_.get(), default_socket_factory_.get()));
286 }
Taylor Brandstetter0c7e9f52015-12-29 14:14:52 -0800287 allocator->SetNetworkIgnoreMask(options_.network_ignore_mask);
deadbeef41b07982015-12-01 15:01:24 -0800288
289 rtc::scoped_refptr<PeerConnection> pc(
290 new rtc::RefCountedObject<PeerConnection>(this));
htaa2a49d92016-03-04 02:51:39 -0800291 // We rely on default values when constraints aren't found.
292 cricket::MediaConfig media_config;
293
294 media_config.video.disable_prerenderer_smoothing =
295 configuration.disable_prerenderer_smoothing;
296 if (configuration.enable_dscp) {
297 media_config.enable_dscp = *(configuration.enable_dscp);
298 }
299 if (configuration.cpu_overuse_detection) {
300 media_config.video.enable_cpu_overuse_detection =
301 *(configuration.cpu_overuse_detection);
302 }
303 if (configuration.suspend_below_min_bitrate) {
304 media_config.video.suspend_below_min_bitrate =
305 *(configuration.suspend_below_min_bitrate);
306 }
307
308 if (!pc->Initialize(media_config, configuration, std::move(allocator),
deadbeef41b07982015-12-01 15:01:24 -0800309 std::move(dtls_identity_store), observer)) {
310 return nullptr;
311 }
312 return PeerConnectionProxy::Create(signaling_thread(), pc);
313}
314
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000315rtc::scoped_refptr<MediaStreamInterface>
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000316PeerConnectionFactory::CreateLocalMediaStream(const std::string& label) {
henrikg91d6ede2015-09-17 00:24:34 -0700317 RTC_DCHECK(signaling_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000318 return MediaStreamProxy::Create(signaling_thread_,
319 MediaStream::Create(label));
320}
321
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000322rtc::scoped_refptr<VideoTrackInterface>
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000323PeerConnectionFactory::CreateVideoTrack(
324 const std::string& id,
325 VideoSourceInterface* source) {
henrikg91d6ede2015-09-17 00:24:34 -0700326 RTC_DCHECK(signaling_thread_->IsCurrent());
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000327 rtc::scoped_refptr<VideoTrackInterface> track(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000328 VideoTrack::Create(id, source));
329 return VideoTrackProxy::Create(signaling_thread_, track);
330}
331
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000332rtc::scoped_refptr<AudioTrackInterface>
pbos@webrtc.orgb5a22b12014-05-13 11:07:01 +0000333PeerConnectionFactory::CreateAudioTrack(const std::string& id,
334 AudioSourceInterface* source) {
henrikg91d6ede2015-09-17 00:24:34 -0700335 RTC_DCHECK(signaling_thread_->IsCurrent());
tommi6eca7e32015-12-15 04:27:11 -0800336 rtc::scoped_refptr<AudioTrackInterface> track(AudioTrack::Create(id, source));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000337 return AudioTrackProxy::Create(signaling_thread_, track);
338}
339
nisse51542be2016-02-12 02:27:06 -0800340webrtc::MediaControllerInterface* PeerConnectionFactory::CreateMediaController(
341 const cricket::MediaConfig& config) const {
henrikg91d6ede2015-09-17 00:24:34 -0700342 RTC_DCHECK(signaling_thread_->IsCurrent());
nisse51542be2016-02-12 02:27:06 -0800343 return MediaControllerInterface::Create(config, worker_thread_,
stefanc1aeaf02015-10-15 07:26:07 -0700344 channel_manager_.get());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000345}
346
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000347rtc::Thread* PeerConnectionFactory::signaling_thread() {
perkj@webrtc.org81134d02015-01-12 08:30:16 +0000348 // This method can be called on a different thread when the factory is
349 // created in CreatePeerConnectionFactory().
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000350 return signaling_thread_;
351}
352
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000353rtc::Thread* PeerConnectionFactory::worker_thread() {
henrikg91d6ede2015-09-17 00:24:34 -0700354 RTC_DCHECK(signaling_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000355 return worker_thread_;
356}
357
henrika@webrtc.org62f6e752015-02-11 08:38:35 +0000358cricket::MediaEngineInterface* PeerConnectionFactory::CreateMediaEngine_w() {
359 ASSERT(worker_thread_ == rtc::Thread::Current());
360 return cricket::WebRtcMediaEngineFactory::Create(
Fredrik Solenbergccb49e72015-05-19 11:37:56 +0200361 default_adm_.get(), video_encoder_factory_.get(),
henrika@webrtc.org62f6e752015-02-11 08:38:35 +0000362 video_decoder_factory_.get());
363}
364
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000365} // namespace webrtc