blob: c90c259345d5eca490c246f8f42661349d00444d [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,
kwiberg1e4e8cb2017-01-31 01:48:08 -080049 cricket::WebRtcVideoEncoderFactory* video_encoder_factory,
50 cricket::WebRtcVideoDecoderFactory* video_decoder_factory,
zhihuang38ede132017-06-15 12:52:32 -070051 std::unique_ptr<cricket::MediaEngineInterface> media_engine,
52 std::unique_ptr<CallFactoryInterface> call_factory,
53 std::unique_ptr<RtcEventLogFactoryInterface> event_log_factory) {
gyzhou95aa9642016-12-13 14:06:26 -080054 rtc::scoped_refptr<PeerConnectionFactory> pc_factory(
55 new rtc::RefCountedObject<PeerConnectionFactory>(
Magnus Jedvert835cc0c2017-09-23 16:14:25 +020056 network_thread, worker_thread, signaling_thread,
57 video_encoder_factory, video_decoder_factory, std::move(media_engine),
zhihuang38ede132017-06-15 12:52:32 -070058 std::move(call_factory), std::move(event_log_factory)));
gyzhou95aa9642016-12-13 14:06:26 -080059
60 // Call Initialize synchronously but make sure it is executed on
61 // |signaling_thread|.
62 MethodCall0<PeerConnectionFactory, bool> call(
63 pc_factory.get(), &PeerConnectionFactory::Initialize);
zhihuang38ede132017-06-15 12:52:32 -070064 bool result = call.Marshal(RTC_FROM_HERE, pc_factory->signaling_thread());
gyzhou95aa9642016-12-13 14:06:26 -080065
66 if (!result) {
67 return nullptr;
68 }
zhihuang38ede132017-06-15 12:52:32 -070069 return PeerConnectionFactoryProxy::Create(pc_factory->signaling_thread(),
70 pc_factory);
kwiberg1e4e8cb2017-01-31 01:48:08 -080071}
72
henrike@webrtc.org28e20752013-07-10 00:45:36 +000073PeerConnectionFactory::PeerConnectionFactory(
danilchape9021a32016-05-17 01:52:02 -070074 rtc::Thread* network_thread,
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000075 rtc::Thread* worker_thread,
76 rtc::Thread* signaling_thread,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000077 cricket::WebRtcVideoEncoderFactory* video_encoder_factory,
gyzhou95aa9642016-12-13 14:06:26 -080078 cricket::WebRtcVideoDecoderFactory* video_decoder_factory,
zhihuang38ede132017-06-15 12:52:32 -070079 std::unique_ptr<cricket::MediaEngineInterface> media_engine,
80 std::unique_ptr<webrtc::CallFactoryInterface> call_factory,
81 std::unique_ptr<RtcEventLogFactoryInterface> event_log_factory)
82 : wraps_current_thread_(false),
danilchape9021a32016-05-17 01:52:02 -070083 network_thread_(network_thread),
henrike@webrtc.org28e20752013-07-10 00:45:36 +000084 worker_thread_(worker_thread),
danilchape9021a32016-05-17 01:52:02 -070085 signaling_thread_(signaling_thread),
henrike@webrtc.org28e20752013-07-10 00:45:36 +000086 video_encoder_factory_(video_encoder_factory),
gyzhou95aa9642016-12-13 14:06:26 -080087 video_decoder_factory_(video_decoder_factory),
zhihuang38ede132017-06-15 12:52:32 -070088 media_engine_(std::move(media_engine)),
89 call_factory_(std::move(call_factory)),
90 event_log_factory_(std::move(event_log_factory)) {
91 if (!network_thread_) {
92 owned_network_thread_ = rtc::Thread::CreateWithSocketServer();
93 owned_network_thread_->Start();
94 network_thread_ = owned_network_thread_.get();
95 }
96
97 if (!worker_thread_) {
98 owned_worker_thread_ = rtc::Thread::Create();
99 owned_worker_thread_->Start();
100 worker_thread_ = owned_worker_thread_.get();
101 }
102
103 if (!signaling_thread_) {
104 signaling_thread_ = rtc::Thread::Current();
105 if (!signaling_thread_) {
106 // If this thread isn't already wrapped by an rtc::Thread, create a
107 // wrapper and own it in this class.
108 signaling_thread_ = rtc::ThreadManager::Instance()->WrapCurrentThread();
109 wraps_current_thread_ = true;
110 }
111 }
112
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000113 // TODO: Currently there is no way creating an external adm in
114 // libjingle source tree. So we can 't currently assert if this is NULL.
nisseede5da42017-01-12 05:15:36 -0800115 // RTC_DCHECK(default_adm != NULL);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000116}
117
118PeerConnectionFactory::~PeerConnectionFactory() {
henrikg91d6ede2015-09-17 00:24:34 -0700119 RTC_DCHECK(signaling_thread_->IsCurrent());
Henrik Boström5e56c592015-08-11 10:33:13 +0200120 channel_manager_.reset(nullptr);
jiayl@webrtc.orgd83f4ef2015-03-13 21:26:12 +0000121
122 // Make sure |worker_thread_| and |signaling_thread_| outlive
Henrik Boströmcebf0a22016-06-01 15:45:30 +0200123 // |default_socket_factory_| and |default_network_manager_|.
deadbeef41b07982015-12-01 15:01:24 -0800124 default_socket_factory_ = nullptr;
125 default_network_manager_ = nullptr;
jiayl@webrtc.orgd83f4ef2015-03-13 21:26:12 +0000126
zhihuang38ede132017-06-15 12:52:32 -0700127 if (wraps_current_thread_)
128 rtc::ThreadManager::Instance()->UnwrapCurrentThread();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000129}
130
131bool PeerConnectionFactory::Initialize() {
henrikg91d6ede2015-09-17 00:24:34 -0700132 RTC_DCHECK(signaling_thread_->IsCurrent());
Honghai Zhang82d78622016-05-06 11:29:15 -0700133 rtc::InitRandom(rtc::Time32());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000134
deadbeef41b07982015-12-01 15:01:24 -0800135 default_network_manager_.reset(new rtc::BasicNetworkManager());
136 if (!default_network_manager_) {
137 return false;
138 }
139
140 default_socket_factory_.reset(
danilchape9021a32016-05-17 01:52:02 -0700141 new rtc::BasicPacketSocketFactory(network_thread_));
deadbeef41b07982015-12-01 15:01:24 -0800142 if (!default_socket_factory_) {
143 return false;
144 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000145
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200146 channel_manager_.reset(new cricket::ChannelManager(
zhihuang38ede132017-06-15 12:52:32 -0700147 std::move(media_engine_), worker_thread_, network_thread_));
henrika@webrtc.org62f6e752015-02-11 08:38:35 +0000148
stefan@webrtc.org85d27942014-06-09 12:51:39 +0000149 channel_manager_->SetVideoRtxEnabled(true);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000150 if (!channel_manager_->Init()) {
151 return false;
152 }
jiayl@webrtc.org61e00b02015-03-04 22:17:38 +0000153
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000154 return true;
155}
156
jbauchcb560652016-08-04 05:20:32 -0700157void PeerConnectionFactory::SetOptions(const Options& options) {
158 options_ = options;
jbauchcb560652016-08-04 05:20:32 -0700159}
160
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000161rtc::scoped_refptr<AudioSourceInterface>
perkj@webrtc.org81134d02015-01-12 08:30:16 +0000162PeerConnectionFactory::CreateAudioSource(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000163 const MediaConstraintsInterface* constraints) {
henrikg91d6ede2015-09-17 00:24:34 -0700164 RTC_DCHECK(signaling_thread_->IsCurrent());
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000165 rtc::scoped_refptr<LocalAudioSource> source(
deadbeef757146b2017-02-10 21:26:48 -0800166 LocalAudioSource::Create(constraints));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000167 return source;
168}
169
htaa2a49d92016-03-04 02:51:39 -0800170rtc::scoped_refptr<AudioSourceInterface>
171PeerConnectionFactory::CreateAudioSource(const cricket::AudioOptions& options) {
172 RTC_DCHECK(signaling_thread_->IsCurrent());
173 rtc::scoped_refptr<LocalAudioSource> source(
deadbeef757146b2017-02-10 21:26:48 -0800174 LocalAudioSource::Create(&options));
htaa2a49d92016-03-04 02:51:39 -0800175 return source;
176}
177
perkja3ede6c2016-03-08 01:27:48 +0100178rtc::scoped_refptr<VideoTrackSourceInterface>
perkj@webrtc.org81134d02015-01-12 08:30:16 +0000179PeerConnectionFactory::CreateVideoSource(
deadbeef112b2e92017-02-10 20:13:37 -0800180 std::unique_ptr<cricket::VideoCapturer> capturer,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000181 const MediaConstraintsInterface* constraints) {
henrikg91d6ede2015-09-17 00:24:34 -0700182 RTC_DCHECK(signaling_thread_->IsCurrent());
perkja3ede6c2016-03-08 01:27:48 +0100183 rtc::scoped_refptr<VideoTrackSourceInterface> source(
deadbeef112b2e92017-02-10 20:13:37 -0800184 VideoCapturerTrackSource::Create(worker_thread_, std::move(capturer),
185 constraints, false));
nisse5b68ab52016-04-07 07:45:54 -0700186 return VideoTrackSourceProxy::Create(signaling_thread_, worker_thread_,
187 source);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000188}
189
perkja3ede6c2016-03-08 01:27:48 +0100190rtc::scoped_refptr<VideoTrackSourceInterface>
deadbeef112b2e92017-02-10 20:13:37 -0800191PeerConnectionFactory::CreateVideoSource(
192 std::unique_ptr<cricket::VideoCapturer> capturer) {
htaa2a49d92016-03-04 02:51:39 -0800193 RTC_DCHECK(signaling_thread_->IsCurrent());
perkja3ede6c2016-03-08 01:27:48 +0100194 rtc::scoped_refptr<VideoTrackSourceInterface> source(
deadbeef112b2e92017-02-10 20:13:37 -0800195 VideoCapturerTrackSource::Create(worker_thread_, std::move(capturer),
196 false));
nisse5b68ab52016-04-07 07:45:54 -0700197 return VideoTrackSourceProxy::Create(signaling_thread_, worker_thread_,
198 source);
htaa2a49d92016-03-04 02:51:39 -0800199}
200
ivocd66b44d2016-01-15 03:06:36 -0800201bool PeerConnectionFactory::StartAecDump(rtc::PlatformFile file,
202 int64_t max_size_bytes) {
henrikg91d6ede2015-09-17 00:24:34 -0700203 RTC_DCHECK(signaling_thread_->IsCurrent());
ivocd66b44d2016-01-15 03:06:36 -0800204 return channel_manager_->StartAecDump(file, max_size_bytes);
wu@webrtc.orga9890802013-12-13 00:21:03 +0000205}
206
ivoc797ef122015-10-22 03:25:41 -0700207void PeerConnectionFactory::StopAecDump() {
208 RTC_DCHECK(signaling_thread_->IsCurrent());
209 channel_manager_->StopAecDump();
210}
211
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000212rtc::scoped_refptr<PeerConnectionInterface>
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000213PeerConnectionFactory::CreatePeerConnection(
htaa2a49d92016-03-04 02:51:39 -0800214 const PeerConnectionInterface::RTCConfiguration& configuration_in,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000215 const MediaConstraintsInterface* constraints,
kwibergd1fe2812016-04-27 06:47:29 -0700216 std::unique_ptr<cricket::PortAllocator> allocator,
Henrik Boströmd03c23b2016-06-01 11:44:18 +0200217 std::unique_ptr<rtc::RTCCertificateGeneratorInterface> cert_generator,
deadbeef41b07982015-12-01 15:01:24 -0800218 PeerConnectionObserver* observer) {
219 RTC_DCHECK(signaling_thread_->IsCurrent());
220
htaa2a49d92016-03-04 02:51:39 -0800221 // We merge constraints and configuration into a single configuration.
222 PeerConnectionInterface::RTCConfiguration configuration = configuration_in;
223 CopyConstraintsIntoRtcConfiguration(constraints, &configuration);
224
225 return CreatePeerConnection(configuration, std::move(allocator),
Henrik Boströmd03c23b2016-06-01 11:44:18 +0200226 std::move(cert_generator), observer);
htaa2a49d92016-03-04 02:51:39 -0800227}
228
229rtc::scoped_refptr<PeerConnectionInterface>
230PeerConnectionFactory::CreatePeerConnection(
231 const PeerConnectionInterface::RTCConfiguration& configuration,
kwibergd1fe2812016-04-27 06:47:29 -0700232 std::unique_ptr<cricket::PortAllocator> allocator,
Henrik Boströmd03c23b2016-06-01 11:44:18 +0200233 std::unique_ptr<rtc::RTCCertificateGeneratorInterface> cert_generator,
htaa2a49d92016-03-04 02:51:39 -0800234 PeerConnectionObserver* observer) {
235 RTC_DCHECK(signaling_thread_->IsCurrent());
236
Henrik Boströmd03c23b2016-06-01 11:44:18 +0200237 if (!cert_generator.get()) {
Henrik Boströmcebf0a22016-06-01 15:45:30 +0200238 // No certificate generator specified, use the default one.
Henrik Boströmd03c23b2016-06-01 11:44:18 +0200239 cert_generator.reset(
Henrik Boströmcebf0a22016-06-01 15:45:30 +0200240 new rtc::RTCCertificateGenerator(signaling_thread_, network_thread_));
deadbeef41b07982015-12-01 15:01:24 -0800241 }
242
jonasoc251cb12017-08-29 03:20:58 -0700243 if (!allocator) {
244 allocator.reset(new cricket::BasicPortAllocator(
maxmorine9ef9072017-08-29 04:49:00 -0700245 default_network_manager_.get(), default_socket_factory_.get()));
jonasoc251cb12017-08-29 03:20:58 -0700246 }
247 network_thread_->Invoke<void>(
248 RTC_FROM_HERE, rtc::Bind(&cricket::PortAllocator::SetNetworkIgnoreMask,
249 allocator.get(), options_.network_ignore_mask));
250
eladalon393a9f62017-09-05 04:30:30 -0700251 std::unique_ptr<RtcEventLog> event_log =
eladalon248fd4f2017-09-06 05:18:15 -0700252 worker_thread_->Invoke<std::unique_ptr<RtcEventLog>>(
253 RTC_FROM_HERE,
254 rtc::Bind(&PeerConnectionFactory::CreateRtcEventLog_w, this));
maxmorine9ef9072017-08-29 04:49:00 -0700255
zhihuang38ede132017-06-15 12:52:32 -0700256 std::unique_ptr<Call> call = worker_thread_->Invoke<std::unique_ptr<Call>>(
257 RTC_FROM_HERE,
258 rtc::Bind(&PeerConnectionFactory::CreateCall_w, this, event_log.get()));
259
deadbeef41b07982015-12-01 15:01:24 -0800260 rtc::scoped_refptr<PeerConnection> pc(
zhihuang38ede132017-06-15 12:52:32 -0700261 new rtc::RefCountedObject<PeerConnection>(this, std::move(event_log),
262 std::move(call)));
htaa2a49d92016-03-04 02:51:39 -0800263
nissec36b31b2016-04-11 23:25:29 -0700264 if (!pc->Initialize(configuration, std::move(allocator),
Henrik Boströmd03c23b2016-06-01 11:44:18 +0200265 std::move(cert_generator), observer)) {
deadbeef41b07982015-12-01 15:01:24 -0800266 return nullptr;
267 }
268 return PeerConnectionProxy::Create(signaling_thread(), pc);
269}
270
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000271rtc::scoped_refptr<MediaStreamInterface>
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000272PeerConnectionFactory::CreateLocalMediaStream(const std::string& label) {
henrikg91d6ede2015-09-17 00:24:34 -0700273 RTC_DCHECK(signaling_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000274 return MediaStreamProxy::Create(signaling_thread_,
275 MediaStream::Create(label));
276}
277
perkja3ede6c2016-03-08 01:27:48 +0100278rtc::scoped_refptr<VideoTrackInterface> PeerConnectionFactory::CreateVideoTrack(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000279 const std::string& id,
perkja3ede6c2016-03-08 01:27:48 +0100280 VideoTrackSourceInterface* source) {
henrikg91d6ede2015-09-17 00:24:34 -0700281 RTC_DCHECK(signaling_thread_->IsCurrent());
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000282 rtc::scoped_refptr<VideoTrackInterface> track(
perkj773be362017-07-31 23:22:01 -0700283 VideoTrack::Create(id, source, worker_thread_));
nisse5b68ab52016-04-07 07:45:54 -0700284 return VideoTrackProxy::Create(signaling_thread_, worker_thread_, track);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000285}
286
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000287rtc::scoped_refptr<AudioTrackInterface>
pbos@webrtc.orgb5a22b12014-05-13 11:07:01 +0000288PeerConnectionFactory::CreateAudioTrack(const std::string& id,
289 AudioSourceInterface* source) {
henrikg91d6ede2015-09-17 00:24:34 -0700290 RTC_DCHECK(signaling_thread_->IsCurrent());
tommi6eca7e32015-12-15 04:27:11 -0800291 rtc::scoped_refptr<AudioTrackInterface> track(AudioTrack::Create(id, source));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000292 return AudioTrackProxy::Create(signaling_thread_, track);
293}
294
zhihuang29ff8442016-07-27 11:07:25 -0700295cricket::TransportController* PeerConnectionFactory::CreateTransportController(
Honghai Zhangbfd398c2016-08-30 22:07:42 -0700296 cricket::PortAllocator* port_allocator,
297 bool redetermine_role_on_ice_restart) {
zhihuang29ff8442016-07-27 11:07:25 -0700298 RTC_DCHECK(signaling_thread_->IsCurrent());
deadbeef7914b8c2017-04-21 03:23:33 -0700299 return new cricket::TransportController(
300 signaling_thread_, network_thread_, port_allocator,
301 redetermine_role_on_ice_restart, options_.crypto_options);
zhihuang29ff8442016-07-27 11:07:25 -0700302}
303
nisseeaabdf62017-05-05 02:23:02 -0700304cricket::ChannelManager* PeerConnectionFactory::channel_manager() {
305 return channel_manager_.get();
306}
307
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000308rtc::Thread* PeerConnectionFactory::signaling_thread() {
perkj@webrtc.org81134d02015-01-12 08:30:16 +0000309 // This method can be called on a different thread when the factory is
310 // created in CreatePeerConnectionFactory().
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000311 return signaling_thread_;
312}
313
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000314rtc::Thread* PeerConnectionFactory::worker_thread() {
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000315 return worker_thread_;
316}
317
danilchape9021a32016-05-17 01:52:02 -0700318rtc::Thread* PeerConnectionFactory::network_thread() {
319 return network_thread_;
320}
321
eladalon248fd4f2017-09-06 05:18:15 -0700322std::unique_ptr<RtcEventLog> PeerConnectionFactory::CreateRtcEventLog_w() {
eladalon591753b2017-09-06 12:33:43 -0700323 RTC_DCHECK_RUN_ON(worker_thread_);
eladalon248fd4f2017-09-06 05:18:15 -0700324 return event_log_factory_ ? event_log_factory_->CreateRtcEventLog()
325 : rtc::MakeUnique<RtcEventLogNullImpl>();
326}
327
zhihuang38ede132017-06-15 12:52:32 -0700328std::unique_ptr<Call> PeerConnectionFactory::CreateCall_w(
329 RtcEventLog* event_log) {
eladalon591753b2017-09-06 12:33:43 -0700330 RTC_DCHECK_RUN_ON(worker_thread_);
331
zhihuang38ede132017-06-15 12:52:32 -0700332 const int kMinBandwidthBps = 30000;
333 const int kStartBandwidthBps = 300000;
334 const int kMaxBandwidthBps = 2000000;
335
336 webrtc::Call::Config call_config(event_log);
337 if (!channel_manager_->media_engine() || !call_factory_) {
338 return nullptr;
339 }
340 call_config.audio_state = channel_manager_->media_engine()->GetAudioState();
341 call_config.bitrate_config.min_bitrate_bps = kMinBandwidthBps;
342 call_config.bitrate_config.start_bitrate_bps = kStartBandwidthBps;
343 call_config.bitrate_config.max_bitrate_bps = kMaxBandwidthBps;
344
345 return std::unique_ptr<Call>(call_factory_->CreateCall(call_config));
346}
347
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000348} // namespace webrtc