blob: 20edaa6af11754704b53a66cd3fac094a564b7bf [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"
Jonas Orelandbdcee282017-10-10 14:01:40 +020020#include "api/turncustomizer.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020021#include "api/videosourceproxy.h"
22#include "logging/rtc_event_log/rtc_event_log.h"
23#include "rtc_base/bind.h"
24#include "rtc_base/checks.h"
25#include "rtc_base/ptr_util.h"
zhihuang38ede132017-06-15 12:52:32 -070026// Adding 'nogncheck' to disable the gn include headers check to support modular
27// WebRTC build targets.
28// TODO(zhihuang): This wouldn't be necessary if the interface and
29// implementation of the media engine were in separate build targets.
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020030#include "media/engine/webrtcmediaengine.h" // nogncheck
31#include "media/engine/webrtcvideodecoderfactory.h" // nogncheck
32#include "media/engine/webrtcvideoencoderfactory.h" // nogncheck
33#include "modules/audio_device/include/audio_device.h" // nogncheck
34#include "p2p/base/basicpacketsocketfactory.h"
35#include "p2p/client/basicportallocator.h"
36#include "pc/audiotrack.h"
37#include "pc/localaudiosource.h"
38#include "pc/mediastream.h"
39#include "pc/peerconnection.h"
40#include "pc/videocapturertracksource.h"
41#include "pc/videotrack.h"
henrike@webrtc.org28e20752013-07-10 00:45:36 +000042
henrike@webrtc.org28e20752013-07-10 00:45:36 +000043namespace webrtc {
44
kwiberg1e4e8cb2017-01-31 01:48:08 -080045rtc::scoped_refptr<PeerConnectionFactoryInterface>
zhihuang38ede132017-06-15 12:52:32 -070046CreateModularPeerConnectionFactory(
gyzhou95aa9642016-12-13 14:06:26 -080047 rtc::Thread* network_thread,
48 rtc::Thread* worker_thread,
49 rtc::Thread* signaling_thread,
zhihuang38ede132017-06-15 12:52:32 -070050 std::unique_ptr<cricket::MediaEngineInterface> media_engine,
51 std::unique_ptr<CallFactoryInterface> call_factory,
52 std::unique_ptr<RtcEventLogFactoryInterface> event_log_factory) {
gyzhou95aa9642016-12-13 14:06:26 -080053 rtc::scoped_refptr<PeerConnectionFactory> pc_factory(
54 new rtc::RefCountedObject<PeerConnectionFactory>(
Magnus Jedvert835cc0c2017-09-23 16:14:25 +020055 network_thread, worker_thread, signaling_thread,
Magnus Jedvert02e7a192017-09-23 17:21:32 +020056 std::move(media_engine), std::move(call_factory),
57 std::move(event_log_factory)));
gyzhou95aa9642016-12-13 14:06:26 -080058
59 // Call Initialize synchronously but make sure it is executed on
60 // |signaling_thread|.
61 MethodCall0<PeerConnectionFactory, bool> call(
62 pc_factory.get(), &PeerConnectionFactory::Initialize);
zhihuang38ede132017-06-15 12:52:32 -070063 bool result = call.Marshal(RTC_FROM_HERE, pc_factory->signaling_thread());
gyzhou95aa9642016-12-13 14:06:26 -080064
65 if (!result) {
66 return nullptr;
67 }
zhihuang38ede132017-06-15 12:52:32 -070068 return PeerConnectionFactoryProxy::Create(pc_factory->signaling_thread(),
69 pc_factory);
kwiberg1e4e8cb2017-01-31 01:48:08 -080070}
71
henrike@webrtc.org28e20752013-07-10 00:45:36 +000072PeerConnectionFactory::PeerConnectionFactory(
danilchape9021a32016-05-17 01:52:02 -070073 rtc::Thread* network_thread,
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000074 rtc::Thread* worker_thread,
75 rtc::Thread* signaling_thread,
zhihuang38ede132017-06-15 12:52:32 -070076 std::unique_ptr<cricket::MediaEngineInterface> media_engine,
77 std::unique_ptr<webrtc::CallFactoryInterface> call_factory,
78 std::unique_ptr<RtcEventLogFactoryInterface> event_log_factory)
79 : wraps_current_thread_(false),
danilchape9021a32016-05-17 01:52:02 -070080 network_thread_(network_thread),
henrike@webrtc.org28e20752013-07-10 00:45:36 +000081 worker_thread_(worker_thread),
danilchape9021a32016-05-17 01:52:02 -070082 signaling_thread_(signaling_thread),
zhihuang38ede132017-06-15 12:52:32 -070083 media_engine_(std::move(media_engine)),
84 call_factory_(std::move(call_factory)),
85 event_log_factory_(std::move(event_log_factory)) {
86 if (!network_thread_) {
87 owned_network_thread_ = rtc::Thread::CreateWithSocketServer();
88 owned_network_thread_->Start();
89 network_thread_ = owned_network_thread_.get();
90 }
91
92 if (!worker_thread_) {
93 owned_worker_thread_ = rtc::Thread::Create();
94 owned_worker_thread_->Start();
95 worker_thread_ = owned_worker_thread_.get();
96 }
97
98 if (!signaling_thread_) {
99 signaling_thread_ = rtc::Thread::Current();
100 if (!signaling_thread_) {
101 // If this thread isn't already wrapped by an rtc::Thread, create a
102 // wrapper and own it in this class.
103 signaling_thread_ = rtc::ThreadManager::Instance()->WrapCurrentThread();
104 wraps_current_thread_ = true;
105 }
106 }
107
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000108 // TODO: Currently there is no way creating an external adm in
109 // libjingle source tree. So we can 't currently assert if this is NULL.
nisseede5da42017-01-12 05:15:36 -0800110 // RTC_DCHECK(default_adm != NULL);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000111}
112
113PeerConnectionFactory::~PeerConnectionFactory() {
henrikg91d6ede2015-09-17 00:24:34 -0700114 RTC_DCHECK(signaling_thread_->IsCurrent());
Henrik Boström5e56c592015-08-11 10:33:13 +0200115 channel_manager_.reset(nullptr);
jiayl@webrtc.orgd83f4ef2015-03-13 21:26:12 +0000116
117 // Make sure |worker_thread_| and |signaling_thread_| outlive
Henrik Boströmcebf0a22016-06-01 15:45:30 +0200118 // |default_socket_factory_| and |default_network_manager_|.
deadbeef41b07982015-12-01 15:01:24 -0800119 default_socket_factory_ = nullptr;
120 default_network_manager_ = nullptr;
jiayl@webrtc.orgd83f4ef2015-03-13 21:26:12 +0000121
zhihuang38ede132017-06-15 12:52:32 -0700122 if (wraps_current_thread_)
123 rtc::ThreadManager::Instance()->UnwrapCurrentThread();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000124}
125
126bool PeerConnectionFactory::Initialize() {
henrikg91d6ede2015-09-17 00:24:34 -0700127 RTC_DCHECK(signaling_thread_->IsCurrent());
Honghai Zhang82d78622016-05-06 11:29:15 -0700128 rtc::InitRandom(rtc::Time32());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000129
deadbeef41b07982015-12-01 15:01:24 -0800130 default_network_manager_.reset(new rtc::BasicNetworkManager());
131 if (!default_network_manager_) {
132 return false;
133 }
134
135 default_socket_factory_.reset(
danilchape9021a32016-05-17 01:52:02 -0700136 new rtc::BasicPacketSocketFactory(network_thread_));
deadbeef41b07982015-12-01 15:01:24 -0800137 if (!default_socket_factory_) {
138 return false;
139 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000140
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200141 channel_manager_.reset(new cricket::ChannelManager(
zhihuang38ede132017-06-15 12:52:32 -0700142 std::move(media_engine_), worker_thread_, network_thread_));
henrika@webrtc.org62f6e752015-02-11 08:38:35 +0000143
stefan@webrtc.org85d27942014-06-09 12:51:39 +0000144 channel_manager_->SetVideoRtxEnabled(true);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000145 if (!channel_manager_->Init()) {
146 return false;
147 }
jiayl@webrtc.org61e00b02015-03-04 22:17:38 +0000148
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000149 return true;
150}
151
jbauchcb560652016-08-04 05:20:32 -0700152void PeerConnectionFactory::SetOptions(const Options& options) {
153 options_ = options;
jbauchcb560652016-08-04 05:20:32 -0700154}
155
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000156rtc::scoped_refptr<AudioSourceInterface>
perkj@webrtc.org81134d02015-01-12 08:30:16 +0000157PeerConnectionFactory::CreateAudioSource(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000158 const MediaConstraintsInterface* constraints) {
henrikg91d6ede2015-09-17 00:24:34 -0700159 RTC_DCHECK(signaling_thread_->IsCurrent());
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000160 rtc::scoped_refptr<LocalAudioSource> source(
deadbeef757146b2017-02-10 21:26:48 -0800161 LocalAudioSource::Create(constraints));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000162 return source;
163}
164
htaa2a49d92016-03-04 02:51:39 -0800165rtc::scoped_refptr<AudioSourceInterface>
166PeerConnectionFactory::CreateAudioSource(const cricket::AudioOptions& options) {
167 RTC_DCHECK(signaling_thread_->IsCurrent());
168 rtc::scoped_refptr<LocalAudioSource> source(
deadbeef757146b2017-02-10 21:26:48 -0800169 LocalAudioSource::Create(&options));
htaa2a49d92016-03-04 02:51:39 -0800170 return source;
171}
172
perkja3ede6c2016-03-08 01:27:48 +0100173rtc::scoped_refptr<VideoTrackSourceInterface>
perkj@webrtc.org81134d02015-01-12 08:30:16 +0000174PeerConnectionFactory::CreateVideoSource(
deadbeef112b2e92017-02-10 20:13:37 -0800175 std::unique_ptr<cricket::VideoCapturer> capturer,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000176 const MediaConstraintsInterface* constraints) {
henrikg91d6ede2015-09-17 00:24:34 -0700177 RTC_DCHECK(signaling_thread_->IsCurrent());
perkja3ede6c2016-03-08 01:27:48 +0100178 rtc::scoped_refptr<VideoTrackSourceInterface> source(
deadbeef112b2e92017-02-10 20:13:37 -0800179 VideoCapturerTrackSource::Create(worker_thread_, std::move(capturer),
180 constraints, false));
nisse5b68ab52016-04-07 07:45:54 -0700181 return VideoTrackSourceProxy::Create(signaling_thread_, worker_thread_,
182 source);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000183}
184
perkja3ede6c2016-03-08 01:27:48 +0100185rtc::scoped_refptr<VideoTrackSourceInterface>
deadbeef112b2e92017-02-10 20:13:37 -0800186PeerConnectionFactory::CreateVideoSource(
187 std::unique_ptr<cricket::VideoCapturer> capturer) {
htaa2a49d92016-03-04 02:51:39 -0800188 RTC_DCHECK(signaling_thread_->IsCurrent());
perkja3ede6c2016-03-08 01:27:48 +0100189 rtc::scoped_refptr<VideoTrackSourceInterface> source(
deadbeef112b2e92017-02-10 20:13:37 -0800190 VideoCapturerTrackSource::Create(worker_thread_, std::move(capturer),
191 false));
nisse5b68ab52016-04-07 07:45:54 -0700192 return VideoTrackSourceProxy::Create(signaling_thread_, worker_thread_,
193 source);
htaa2a49d92016-03-04 02:51:39 -0800194}
195
ivocd66b44d2016-01-15 03:06:36 -0800196bool PeerConnectionFactory::StartAecDump(rtc::PlatformFile file,
197 int64_t max_size_bytes) {
henrikg91d6ede2015-09-17 00:24:34 -0700198 RTC_DCHECK(signaling_thread_->IsCurrent());
ivocd66b44d2016-01-15 03:06:36 -0800199 return channel_manager_->StartAecDump(file, max_size_bytes);
wu@webrtc.orga9890802013-12-13 00:21:03 +0000200}
201
ivoc797ef122015-10-22 03:25:41 -0700202void PeerConnectionFactory::StopAecDump() {
203 RTC_DCHECK(signaling_thread_->IsCurrent());
204 channel_manager_->StopAecDump();
205}
206
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000207rtc::scoped_refptr<PeerConnectionInterface>
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000208PeerConnectionFactory::CreatePeerConnection(
htaa2a49d92016-03-04 02:51:39 -0800209 const PeerConnectionInterface::RTCConfiguration& configuration_in,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000210 const MediaConstraintsInterface* constraints,
kwibergd1fe2812016-04-27 06:47:29 -0700211 std::unique_ptr<cricket::PortAllocator> allocator,
Henrik Boströmd03c23b2016-06-01 11:44:18 +0200212 std::unique_ptr<rtc::RTCCertificateGeneratorInterface> cert_generator,
deadbeef41b07982015-12-01 15:01:24 -0800213 PeerConnectionObserver* observer) {
214 RTC_DCHECK(signaling_thread_->IsCurrent());
215
htaa2a49d92016-03-04 02:51:39 -0800216 // We merge constraints and configuration into a single configuration.
217 PeerConnectionInterface::RTCConfiguration configuration = configuration_in;
218 CopyConstraintsIntoRtcConfiguration(constraints, &configuration);
219
220 return CreatePeerConnection(configuration, std::move(allocator),
Henrik Boströmd03c23b2016-06-01 11:44:18 +0200221 std::move(cert_generator), observer);
htaa2a49d92016-03-04 02:51:39 -0800222}
223
224rtc::scoped_refptr<PeerConnectionInterface>
225PeerConnectionFactory::CreatePeerConnection(
226 const PeerConnectionInterface::RTCConfiguration& configuration,
kwibergd1fe2812016-04-27 06:47:29 -0700227 std::unique_ptr<cricket::PortAllocator> allocator,
Henrik Boströmd03c23b2016-06-01 11:44:18 +0200228 std::unique_ptr<rtc::RTCCertificateGeneratorInterface> cert_generator,
htaa2a49d92016-03-04 02:51:39 -0800229 PeerConnectionObserver* observer) {
230 RTC_DCHECK(signaling_thread_->IsCurrent());
231
Henrik Boströmd03c23b2016-06-01 11:44:18 +0200232 if (!cert_generator.get()) {
Henrik Boströmcebf0a22016-06-01 15:45:30 +0200233 // No certificate generator specified, use the default one.
Henrik Boströmd03c23b2016-06-01 11:44:18 +0200234 cert_generator.reset(
Henrik Boströmcebf0a22016-06-01 15:45:30 +0200235 new rtc::RTCCertificateGenerator(signaling_thread_, network_thread_));
deadbeef41b07982015-12-01 15:01:24 -0800236 }
237
jonasoc251cb12017-08-29 03:20:58 -0700238 if (!allocator) {
239 allocator.reset(new cricket::BasicPortAllocator(
Jonas Orelandbdcee282017-10-10 14:01:40 +0200240 default_network_manager_.get(), default_socket_factory_.get(),
241 configuration.turn_customizer));
jonasoc251cb12017-08-29 03:20:58 -0700242 }
243 network_thread_->Invoke<void>(
244 RTC_FROM_HERE, rtc::Bind(&cricket::PortAllocator::SetNetworkIgnoreMask,
245 allocator.get(), options_.network_ignore_mask));
246
eladalon393a9f62017-09-05 04:30:30 -0700247 std::unique_ptr<RtcEventLog> event_log =
eladalon248fd4f2017-09-06 05:18:15 -0700248 worker_thread_->Invoke<std::unique_ptr<RtcEventLog>>(
249 RTC_FROM_HERE,
250 rtc::Bind(&PeerConnectionFactory::CreateRtcEventLog_w, this));
maxmorine9ef9072017-08-29 04:49:00 -0700251
zhihuang38ede132017-06-15 12:52:32 -0700252 std::unique_ptr<Call> call = worker_thread_->Invoke<std::unique_ptr<Call>>(
253 RTC_FROM_HERE,
254 rtc::Bind(&PeerConnectionFactory::CreateCall_w, this, event_log.get()));
255
deadbeef41b07982015-12-01 15:01:24 -0800256 rtc::scoped_refptr<PeerConnection> pc(
zhihuang38ede132017-06-15 12:52:32 -0700257 new rtc::RefCountedObject<PeerConnection>(this, std::move(event_log),
258 std::move(call)));
htaa2a49d92016-03-04 02:51:39 -0800259
nissec36b31b2016-04-11 23:25:29 -0700260 if (!pc->Initialize(configuration, std::move(allocator),
Henrik Boströmd03c23b2016-06-01 11:44:18 +0200261 std::move(cert_generator), observer)) {
deadbeef41b07982015-12-01 15:01:24 -0800262 return nullptr;
263 }
264 return PeerConnectionProxy::Create(signaling_thread(), pc);
265}
266
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000267rtc::scoped_refptr<MediaStreamInterface>
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000268PeerConnectionFactory::CreateLocalMediaStream(const std::string& label) {
henrikg91d6ede2015-09-17 00:24:34 -0700269 RTC_DCHECK(signaling_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000270 return MediaStreamProxy::Create(signaling_thread_,
271 MediaStream::Create(label));
272}
273
perkja3ede6c2016-03-08 01:27:48 +0100274rtc::scoped_refptr<VideoTrackInterface> PeerConnectionFactory::CreateVideoTrack(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000275 const std::string& id,
perkja3ede6c2016-03-08 01:27:48 +0100276 VideoTrackSourceInterface* source) {
henrikg91d6ede2015-09-17 00:24:34 -0700277 RTC_DCHECK(signaling_thread_->IsCurrent());
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000278 rtc::scoped_refptr<VideoTrackInterface> track(
perkj773be362017-07-31 23:22:01 -0700279 VideoTrack::Create(id, source, worker_thread_));
nisse5b68ab52016-04-07 07:45:54 -0700280 return VideoTrackProxy::Create(signaling_thread_, worker_thread_, track);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000281}
282
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000283rtc::scoped_refptr<AudioTrackInterface>
pbos@webrtc.orgb5a22b12014-05-13 11:07:01 +0000284PeerConnectionFactory::CreateAudioTrack(const std::string& id,
285 AudioSourceInterface* source) {
henrikg91d6ede2015-09-17 00:24:34 -0700286 RTC_DCHECK(signaling_thread_->IsCurrent());
tommi6eca7e32015-12-15 04:27:11 -0800287 rtc::scoped_refptr<AudioTrackInterface> track(AudioTrack::Create(id, source));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000288 return AudioTrackProxy::Create(signaling_thread_, track);
289}
290
zhihuang29ff8442016-07-27 11:07:25 -0700291cricket::TransportController* PeerConnectionFactory::CreateTransportController(
Honghai Zhangbfd398c2016-08-30 22:07:42 -0700292 cricket::PortAllocator* port_allocator,
293 bool redetermine_role_on_ice_restart) {
zhihuang29ff8442016-07-27 11:07:25 -0700294 RTC_DCHECK(signaling_thread_->IsCurrent());
deadbeef7914b8c2017-04-21 03:23:33 -0700295 return new cricket::TransportController(
296 signaling_thread_, network_thread_, port_allocator,
297 redetermine_role_on_ice_restart, options_.crypto_options);
zhihuang29ff8442016-07-27 11:07:25 -0700298}
299
nisseeaabdf62017-05-05 02:23:02 -0700300cricket::ChannelManager* PeerConnectionFactory::channel_manager() {
301 return channel_manager_.get();
302}
303
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000304rtc::Thread* PeerConnectionFactory::signaling_thread() {
perkj@webrtc.org81134d02015-01-12 08:30:16 +0000305 // This method can be called on a different thread when the factory is
306 // created in CreatePeerConnectionFactory().
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000307 return signaling_thread_;
308}
309
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000310rtc::Thread* PeerConnectionFactory::worker_thread() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000311 return worker_thread_;
312}
313
danilchape9021a32016-05-17 01:52:02 -0700314rtc::Thread* PeerConnectionFactory::network_thread() {
315 return network_thread_;
316}
317
eladalon248fd4f2017-09-06 05:18:15 -0700318std::unique_ptr<RtcEventLog> PeerConnectionFactory::CreateRtcEventLog_w() {
eladalon591753b2017-09-06 12:33:43 -0700319 RTC_DCHECK_RUN_ON(worker_thread_);
Elad Alon4a87e1c2017-10-03 16:11:34 +0200320 const auto encoding_type = RtcEventLog::EncodingType::Legacy;
321 return event_log_factory_
322 ? event_log_factory_->CreateRtcEventLog(encoding_type)
323 : rtc::MakeUnique<RtcEventLogNullImpl>();
eladalon248fd4f2017-09-06 05:18:15 -0700324}
325
zhihuang38ede132017-06-15 12:52:32 -0700326std::unique_ptr<Call> PeerConnectionFactory::CreateCall_w(
327 RtcEventLog* event_log) {
eladalon591753b2017-09-06 12:33:43 -0700328 RTC_DCHECK_RUN_ON(worker_thread_);
329
zhihuang38ede132017-06-15 12:52:32 -0700330 const int kMinBandwidthBps = 30000;
331 const int kStartBandwidthBps = 300000;
332 const int kMaxBandwidthBps = 2000000;
333
334 webrtc::Call::Config call_config(event_log);
335 if (!channel_manager_->media_engine() || !call_factory_) {
336 return nullptr;
337 }
338 call_config.audio_state = channel_manager_->media_engine()->GetAudioState();
339 call_config.bitrate_config.min_bitrate_bps = kMinBandwidthBps;
340 call_config.bitrate_config.start_bitrate_bps = kStartBandwidthBps;
341 call_config.bitrate_config.max_bitrate_bps = kMaxBandwidthBps;
342
343 return std::unique_ptr<Call>(call_factory_->CreateCall(call_config));
344}
345
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000346} // namespace webrtc