blob: d3c624d1c070a2ecf4d6be9fcc77eb5ace387f4b [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
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020011#include "pc/peerconnectionfactory.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000012
kwiberg0eb15ed2015-12-17 03:04:15 -080013#include <utility>
14
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020015#include "api/mediaconstraintsinterface.h"
16#include "api/mediastreamproxy.h"
17#include "api/mediastreamtrackproxy.h"
18#include "api/peerconnectionfactoryproxy.h"
19#include "api/peerconnectionproxy.h"
20#include "api/videosourceproxy.h"
21#include "logging/rtc_event_log/rtc_event_log.h"
22#include "rtc_base/bind.h"
23#include "rtc_base/checks.h"
24#include "rtc_base/ptr_util.h"
zhihuang38ede132017-06-15 12:52:32 -070025// Adding 'nogncheck' to disable the gn include headers check to support modular
26// WebRTC build targets.
27// TODO(zhihuang): This wouldn't be necessary if the interface and
28// implementation of the media engine were in separate build targets.
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020029#include "media/engine/webrtcmediaengine.h" // nogncheck
30#include "media/engine/webrtcvideodecoderfactory.h" // nogncheck
31#include "media/engine/webrtcvideoencoderfactory.h" // nogncheck
32#include "modules/audio_device/include/audio_device.h" // nogncheck
33#include "p2p/base/basicpacketsocketfactory.h"
34#include "p2p/client/basicportallocator.h"
35#include "pc/audiotrack.h"
36#include "pc/localaudiosource.h"
37#include "pc/mediastream.h"
38#include "pc/peerconnection.h"
39#include "pc/videocapturertracksource.h"
40#include "pc/videotrack.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000041
henrike@webrtc.org28e20752013-07-10 00:45:36 +000042namespace webrtc {
43
kwiberg1e4e8cb2017-01-31 01:48:08 -080044rtc::scoped_refptr<PeerConnectionFactoryInterface>
zhihuang38ede132017-06-15 12:52:32 -070045CreateModularPeerConnectionFactory(
gyzhou95aa9642016-12-13 14:06:26 -080046 rtc::Thread* network_thread,
47 rtc::Thread* worker_thread,
48 rtc::Thread* signaling_thread,
zhihuang38ede132017-06-15 12:52:32 -070049 std::unique_ptr<cricket::MediaEngineInterface> media_engine,
50 std::unique_ptr<CallFactoryInterface> call_factory,
51 std::unique_ptr<RtcEventLogFactoryInterface> event_log_factory) {
gyzhou95aa9642016-12-13 14:06:26 -080052 rtc::scoped_refptr<PeerConnectionFactory> pc_factory(
53 new rtc::RefCountedObject<PeerConnectionFactory>(
Magnus Jedvert835cc0c2017-09-23 16:14:25 +020054 network_thread, worker_thread, signaling_thread,
Magnus Jedvert02e7a192017-09-23 17:21:32 +020055 std::move(media_engine), std::move(call_factory),
56 std::move(event_log_factory)));
gyzhou95aa9642016-12-13 14:06:26 -080057
58 // Call Initialize synchronously but make sure it is executed on
59 // |signaling_thread|.
60 MethodCall0<PeerConnectionFactory, bool> call(
61 pc_factory.get(), &PeerConnectionFactory::Initialize);
zhihuang38ede132017-06-15 12:52:32 -070062 bool result = call.Marshal(RTC_FROM_HERE, pc_factory->signaling_thread());
gyzhou95aa9642016-12-13 14:06:26 -080063
64 if (!result) {
65 return nullptr;
66 }
zhihuang38ede132017-06-15 12:52:32 -070067 return PeerConnectionFactoryProxy::Create(pc_factory->signaling_thread(),
68 pc_factory);
kwiberg1e4e8cb2017-01-31 01:48:08 -080069}
70
henrike@webrtc.org28e20752013-07-10 00:45:36 +000071PeerConnectionFactory::PeerConnectionFactory(
danilchape9021a32016-05-17 01:52:02 -070072 rtc::Thread* network_thread,
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000073 rtc::Thread* worker_thread,
74 rtc::Thread* signaling_thread,
zhihuang38ede132017-06-15 12:52:32 -070075 std::unique_ptr<cricket::MediaEngineInterface> media_engine,
76 std::unique_ptr<webrtc::CallFactoryInterface> call_factory,
77 std::unique_ptr<RtcEventLogFactoryInterface> event_log_factory)
78 : wraps_current_thread_(false),
danilchape9021a32016-05-17 01:52:02 -070079 network_thread_(network_thread),
henrike@webrtc.org28e20752013-07-10 00:45:36 +000080 worker_thread_(worker_thread),
danilchape9021a32016-05-17 01:52:02 -070081 signaling_thread_(signaling_thread),
zhihuang38ede132017-06-15 12:52:32 -070082 media_engine_(std::move(media_engine)),
83 call_factory_(std::move(call_factory)),
84 event_log_factory_(std::move(event_log_factory)) {
85 if (!network_thread_) {
86 owned_network_thread_ = rtc::Thread::CreateWithSocketServer();
87 owned_network_thread_->Start();
88 network_thread_ = owned_network_thread_.get();
89 }
90
91 if (!worker_thread_) {
92 owned_worker_thread_ = rtc::Thread::Create();
93 owned_worker_thread_->Start();
94 worker_thread_ = owned_worker_thread_.get();
95 }
96
97 if (!signaling_thread_) {
98 signaling_thread_ = rtc::Thread::Current();
99 if (!signaling_thread_) {
100 // If this thread isn't already wrapped by an rtc::Thread, create a
101 // wrapper and own it in this class.
102 signaling_thread_ = rtc::ThreadManager::Instance()->WrapCurrentThread();
103 wraps_current_thread_ = true;
104 }
105 }
106
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000107 // TODO: Currently there is no way creating an external adm in
108 // libjingle source tree. So we can 't currently assert if this is NULL.
nisseede5da42017-01-12 05:15:36 -0800109 // RTC_DCHECK(default_adm != NULL);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000110}
111
112PeerConnectionFactory::~PeerConnectionFactory() {
henrikg91d6ede2015-09-17 00:24:34 -0700113 RTC_DCHECK(signaling_thread_->IsCurrent());
Henrik Boström5e56c592015-08-11 10:33:13 +0200114 channel_manager_.reset(nullptr);
jiayl@webrtc.orgd83f4ef2015-03-13 21:26:12 +0000115
116 // Make sure |worker_thread_| and |signaling_thread_| outlive
Henrik Boströmcebf0a22016-06-01 15:45:30 +0200117 // |default_socket_factory_| and |default_network_manager_|.
deadbeef41b07982015-12-01 15:01:24 -0800118 default_socket_factory_ = nullptr;
119 default_network_manager_ = nullptr;
jiayl@webrtc.orgd83f4ef2015-03-13 21:26:12 +0000120
zhihuang38ede132017-06-15 12:52:32 -0700121 if (wraps_current_thread_)
122 rtc::ThreadManager::Instance()->UnwrapCurrentThread();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000123}
124
125bool PeerConnectionFactory::Initialize() {
henrikg91d6ede2015-09-17 00:24:34 -0700126 RTC_DCHECK(signaling_thread_->IsCurrent());
Honghai Zhang82d78622016-05-06 11:29:15 -0700127 rtc::InitRandom(rtc::Time32());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000128
deadbeef41b07982015-12-01 15:01:24 -0800129 default_network_manager_.reset(new rtc::BasicNetworkManager());
130 if (!default_network_manager_) {
131 return false;
132 }
133
134 default_socket_factory_.reset(
danilchape9021a32016-05-17 01:52:02 -0700135 new rtc::BasicPacketSocketFactory(network_thread_));
deadbeef41b07982015-12-01 15:01:24 -0800136 if (!default_socket_factory_) {
137 return false;
138 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000139
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200140 channel_manager_.reset(new cricket::ChannelManager(
zhihuang38ede132017-06-15 12:52:32 -0700141 std::move(media_engine_), worker_thread_, network_thread_));
henrika@webrtc.org62f6e752015-02-11 08:38:35 +0000142
stefan@webrtc.org85d27942014-06-09 12:51:39 +0000143 channel_manager_->SetVideoRtxEnabled(true);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000144 if (!channel_manager_->Init()) {
145 return false;
146 }
jiayl@webrtc.org61e00b02015-03-04 22:17:38 +0000147
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000148 return true;
149}
150
jbauchcb560652016-08-04 05:20:32 -0700151void PeerConnectionFactory::SetOptions(const Options& options) {
152 options_ = options;
jbauchcb560652016-08-04 05:20:32 -0700153}
154
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000155rtc::scoped_refptr<AudioSourceInterface>
perkj@webrtc.org81134d02015-01-12 08:30:16 +0000156PeerConnectionFactory::CreateAudioSource(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000157 const MediaConstraintsInterface* constraints) {
henrikg91d6ede2015-09-17 00:24:34 -0700158 RTC_DCHECK(signaling_thread_->IsCurrent());
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000159 rtc::scoped_refptr<LocalAudioSource> source(
deadbeef757146b2017-02-10 21:26:48 -0800160 LocalAudioSource::Create(constraints));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000161 return source;
162}
163
htaa2a49d92016-03-04 02:51:39 -0800164rtc::scoped_refptr<AudioSourceInterface>
165PeerConnectionFactory::CreateAudioSource(const cricket::AudioOptions& options) {
166 RTC_DCHECK(signaling_thread_->IsCurrent());
167 rtc::scoped_refptr<LocalAudioSource> source(
deadbeef757146b2017-02-10 21:26:48 -0800168 LocalAudioSource::Create(&options));
htaa2a49d92016-03-04 02:51:39 -0800169 return source;
170}
171
perkja3ede6c2016-03-08 01:27:48 +0100172rtc::scoped_refptr<VideoTrackSourceInterface>
perkj@webrtc.org81134d02015-01-12 08:30:16 +0000173PeerConnectionFactory::CreateVideoSource(
deadbeef112b2e92017-02-10 20:13:37 -0800174 std::unique_ptr<cricket::VideoCapturer> capturer,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000175 const MediaConstraintsInterface* constraints) {
henrikg91d6ede2015-09-17 00:24:34 -0700176 RTC_DCHECK(signaling_thread_->IsCurrent());
perkja3ede6c2016-03-08 01:27:48 +0100177 rtc::scoped_refptr<VideoTrackSourceInterface> source(
deadbeef112b2e92017-02-10 20:13:37 -0800178 VideoCapturerTrackSource::Create(worker_thread_, std::move(capturer),
179 constraints, false));
nisse5b68ab52016-04-07 07:45:54 -0700180 return VideoTrackSourceProxy::Create(signaling_thread_, worker_thread_,
181 source);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000182}
183
perkja3ede6c2016-03-08 01:27:48 +0100184rtc::scoped_refptr<VideoTrackSourceInterface>
deadbeef112b2e92017-02-10 20:13:37 -0800185PeerConnectionFactory::CreateVideoSource(
186 std::unique_ptr<cricket::VideoCapturer> capturer) {
htaa2a49d92016-03-04 02:51:39 -0800187 RTC_DCHECK(signaling_thread_->IsCurrent());
perkja3ede6c2016-03-08 01:27:48 +0100188 rtc::scoped_refptr<VideoTrackSourceInterface> source(
deadbeef112b2e92017-02-10 20:13:37 -0800189 VideoCapturerTrackSource::Create(worker_thread_, std::move(capturer),
190 false));
nisse5b68ab52016-04-07 07:45:54 -0700191 return VideoTrackSourceProxy::Create(signaling_thread_, worker_thread_,
192 source);
htaa2a49d92016-03-04 02:51:39 -0800193}
194
ivocd66b44d2016-01-15 03:06:36 -0800195bool PeerConnectionFactory::StartAecDump(rtc::PlatformFile file,
196 int64_t max_size_bytes) {
henrikg91d6ede2015-09-17 00:24:34 -0700197 RTC_DCHECK(signaling_thread_->IsCurrent());
ivocd66b44d2016-01-15 03:06:36 -0800198 return channel_manager_->StartAecDump(file, max_size_bytes);
wu@webrtc.orga9890802013-12-13 00:21:03 +0000199}
200
ivoc797ef122015-10-22 03:25:41 -0700201void PeerConnectionFactory::StopAecDump() {
202 RTC_DCHECK(signaling_thread_->IsCurrent());
203 channel_manager_->StopAecDump();
204}
205
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000206rtc::scoped_refptr<PeerConnectionInterface>
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000207PeerConnectionFactory::CreatePeerConnection(
htaa2a49d92016-03-04 02:51:39 -0800208 const PeerConnectionInterface::RTCConfiguration& configuration_in,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000209 const MediaConstraintsInterface* constraints,
kwibergd1fe2812016-04-27 06:47:29 -0700210 std::unique_ptr<cricket::PortAllocator> allocator,
Henrik Boströmd03c23b2016-06-01 11:44:18 +0200211 std::unique_ptr<rtc::RTCCertificateGeneratorInterface> cert_generator,
deadbeef41b07982015-12-01 15:01:24 -0800212 PeerConnectionObserver* observer) {
213 RTC_DCHECK(signaling_thread_->IsCurrent());
214
htaa2a49d92016-03-04 02:51:39 -0800215 // We merge constraints and configuration into a single configuration.
216 PeerConnectionInterface::RTCConfiguration configuration = configuration_in;
217 CopyConstraintsIntoRtcConfiguration(constraints, &configuration);
218
219 return CreatePeerConnection(configuration, std::move(allocator),
Henrik Boströmd03c23b2016-06-01 11:44:18 +0200220 std::move(cert_generator), observer);
htaa2a49d92016-03-04 02:51:39 -0800221}
222
223rtc::scoped_refptr<PeerConnectionInterface>
224PeerConnectionFactory::CreatePeerConnection(
225 const PeerConnectionInterface::RTCConfiguration& configuration,
kwibergd1fe2812016-04-27 06:47:29 -0700226 std::unique_ptr<cricket::PortAllocator> allocator,
Henrik Boströmd03c23b2016-06-01 11:44:18 +0200227 std::unique_ptr<rtc::RTCCertificateGeneratorInterface> cert_generator,
htaa2a49d92016-03-04 02:51:39 -0800228 PeerConnectionObserver* observer) {
229 RTC_DCHECK(signaling_thread_->IsCurrent());
230
Henrik Boströmd03c23b2016-06-01 11:44:18 +0200231 if (!cert_generator.get()) {
Henrik Boströmcebf0a22016-06-01 15:45:30 +0200232 // No certificate generator specified, use the default one.
Henrik Boströmd03c23b2016-06-01 11:44:18 +0200233 cert_generator.reset(
Henrik Boströmcebf0a22016-06-01 15:45:30 +0200234 new rtc::RTCCertificateGenerator(signaling_thread_, network_thread_));
deadbeef41b07982015-12-01 15:01:24 -0800235 }
236
jonasoc251cb12017-08-29 03:20:58 -0700237 if (!allocator) {
238 allocator.reset(new cricket::BasicPortAllocator(
maxmorine9ef9072017-08-29 04:49:00 -0700239 default_network_manager_.get(), default_socket_factory_.get()));
jonasoc251cb12017-08-29 03:20:58 -0700240 }
241 network_thread_->Invoke<void>(
242 RTC_FROM_HERE, rtc::Bind(&cricket::PortAllocator::SetNetworkIgnoreMask,
243 allocator.get(), options_.network_ignore_mask));
244
eladalon393a9f62017-09-05 04:30:30 -0700245 std::unique_ptr<RtcEventLog> event_log =
eladalon248fd4f2017-09-06 05:18:15 -0700246 worker_thread_->Invoke<std::unique_ptr<RtcEventLog>>(
247 RTC_FROM_HERE,
248 rtc::Bind(&PeerConnectionFactory::CreateRtcEventLog_w, this));
maxmorine9ef9072017-08-29 04:49:00 -0700249
zhihuang38ede132017-06-15 12:52:32 -0700250 std::unique_ptr<Call> call = worker_thread_->Invoke<std::unique_ptr<Call>>(
251 RTC_FROM_HERE,
252 rtc::Bind(&PeerConnectionFactory::CreateCall_w, this, event_log.get()));
253
deadbeef41b07982015-12-01 15:01:24 -0800254 rtc::scoped_refptr<PeerConnection> pc(
zhihuang38ede132017-06-15 12:52:32 -0700255 new rtc::RefCountedObject<PeerConnection>(this, std::move(event_log),
256 std::move(call)));
htaa2a49d92016-03-04 02:51:39 -0800257
nissec36b31b2016-04-11 23:25:29 -0700258 if (!pc->Initialize(configuration, std::move(allocator),
Henrik Boströmd03c23b2016-06-01 11:44:18 +0200259 std::move(cert_generator), observer)) {
deadbeef41b07982015-12-01 15:01:24 -0800260 return nullptr;
261 }
262 return PeerConnectionProxy::Create(signaling_thread(), pc);
263}
264
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000265rtc::scoped_refptr<MediaStreamInterface>
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000266PeerConnectionFactory::CreateLocalMediaStream(const std::string& label) {
henrikg91d6ede2015-09-17 00:24:34 -0700267 RTC_DCHECK(signaling_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000268 return MediaStreamProxy::Create(signaling_thread_,
269 MediaStream::Create(label));
270}
271
perkja3ede6c2016-03-08 01:27:48 +0100272rtc::scoped_refptr<VideoTrackInterface> PeerConnectionFactory::CreateVideoTrack(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000273 const std::string& id,
perkja3ede6c2016-03-08 01:27:48 +0100274 VideoTrackSourceInterface* source) {
henrikg91d6ede2015-09-17 00:24:34 -0700275 RTC_DCHECK(signaling_thread_->IsCurrent());
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000276 rtc::scoped_refptr<VideoTrackInterface> track(
perkj773be362017-07-31 23:22:01 -0700277 VideoTrack::Create(id, source, worker_thread_));
nisse5b68ab52016-04-07 07:45:54 -0700278 return VideoTrackProxy::Create(signaling_thread_, worker_thread_, track);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000279}
280
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000281rtc::scoped_refptr<AudioTrackInterface>
pbos@webrtc.orgb5a22b12014-05-13 11:07:01 +0000282PeerConnectionFactory::CreateAudioTrack(const std::string& id,
283 AudioSourceInterface* source) {
henrikg91d6ede2015-09-17 00:24:34 -0700284 RTC_DCHECK(signaling_thread_->IsCurrent());
tommi6eca7e32015-12-15 04:27:11 -0800285 rtc::scoped_refptr<AudioTrackInterface> track(AudioTrack::Create(id, source));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000286 return AudioTrackProxy::Create(signaling_thread_, track);
287}
288
zhihuang29ff8442016-07-27 11:07:25 -0700289cricket::TransportController* PeerConnectionFactory::CreateTransportController(
Honghai Zhangbfd398c2016-08-30 22:07:42 -0700290 cricket::PortAllocator* port_allocator,
291 bool redetermine_role_on_ice_restart) {
zhihuang29ff8442016-07-27 11:07:25 -0700292 RTC_DCHECK(signaling_thread_->IsCurrent());
deadbeef7914b8c2017-04-21 03:23:33 -0700293 return new cricket::TransportController(
294 signaling_thread_, network_thread_, port_allocator,
295 redetermine_role_on_ice_restart, options_.crypto_options);
zhihuang29ff8442016-07-27 11:07:25 -0700296}
297
nisseeaabdf62017-05-05 02:23:02 -0700298cricket::ChannelManager* PeerConnectionFactory::channel_manager() {
299 return channel_manager_.get();
300}
301
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000302rtc::Thread* PeerConnectionFactory::signaling_thread() {
perkj@webrtc.org81134d02015-01-12 08:30:16 +0000303 // This method can be called on a different thread when the factory is
304 // created in CreatePeerConnectionFactory().
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000305 return signaling_thread_;
306}
307
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000308rtc::Thread* PeerConnectionFactory::worker_thread() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000309 return worker_thread_;
310}
311
danilchape9021a32016-05-17 01:52:02 -0700312rtc::Thread* PeerConnectionFactory::network_thread() {
313 return network_thread_;
314}
315
eladalon248fd4f2017-09-06 05:18:15 -0700316std::unique_ptr<RtcEventLog> PeerConnectionFactory::CreateRtcEventLog_w() {
eladalon591753b2017-09-06 12:33:43 -0700317 RTC_DCHECK_RUN_ON(worker_thread_);
Elad Alon4a87e1c2017-10-03 16:11:34 +0200318 const auto encoding_type = RtcEventLog::EncodingType::Legacy;
319 return event_log_factory_
320 ? event_log_factory_->CreateRtcEventLog(encoding_type)
321 : rtc::MakeUnique<RtcEventLogNullImpl>();
eladalon248fd4f2017-09-06 05:18:15 -0700322}
323
zhihuang38ede132017-06-15 12:52:32 -0700324std::unique_ptr<Call> PeerConnectionFactory::CreateCall_w(
325 RtcEventLog* event_log) {
eladalon591753b2017-09-06 12:33:43 -0700326 RTC_DCHECK_RUN_ON(worker_thread_);
327
zhihuang38ede132017-06-15 12:52:32 -0700328 const int kMinBandwidthBps = 30000;
329 const int kStartBandwidthBps = 300000;
330 const int kMaxBandwidthBps = 2000000;
331
332 webrtc::Call::Config call_config(event_log);
333 if (!channel_manager_->media_engine() || !call_factory_) {
334 return nullptr;
335 }
336 call_config.audio_state = channel_manager_->media_engine()->GetAudioState();
337 call_config.bitrate_config.min_bitrate_bps = kMinBandwidthBps;
338 call_config.bitrate_config.start_bitrate_bps = kStartBandwidthBps;
339 call_config.bitrate_config.max_bitrate_bps = kMaxBandwidthBps;
340
341 return std::unique_ptr<Call>(call_factory_->CreateCall(call_config));
342}
343
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000344} // namespace webrtc