blob: 5d8f79c40cc5b695a78baf310de861086aff99d2 [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"
perkja3ede6c2016-03-08 01:27:48 +010024#include "webrtc/api/videocapturertracksource.h"
Henrik Kjellander15583c12016-02-10 10:53:12 +010025#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(
hbos52913932016-03-07 15:14:40 -080049 const rtc::KeyParams& key_params,
50 const rtc::Optional<uint64_t>& expires_ms,
Henrik Boström5e56c592015-08-11 10:33:13 +020051 const rtc::scoped_refptr<webrtc::DtlsIdentityRequestObserver>&
52 observer) override {
hbos52913932016-03-07 15:14:40 -080053 store_->RequestIdentity(key_params, expires_ms, 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
Taylor Brandstettera8415fe2016-03-23 10:38:07 -070067 RTC_CHECK(rtc::Thread::Current() == pc_factory->signaling_thread());
68 // The signaling thread is the current thread so we can
69 // safely call Initialize directly.
70 if (!pc_factory->Initialize()) {
71 return nullptr;
henrike@webrtc.org28e20752013-07-10 00:45:36 +000072 }
perkj@webrtc.org81134d02015-01-12 08:30:16 +000073 return PeerConnectionFactoryProxy::Create(pc_factory->signaling_thread(),
74 pc_factory);
henrike@webrtc.org28e20752013-07-10 00:45:36 +000075}
76
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000077rtc::scoped_refptr<PeerConnectionFactoryInterface>
henrike@webrtc.org28e20752013-07-10 00:45:36 +000078CreatePeerConnectionFactory(
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000079 rtc::Thread* worker_thread,
80 rtc::Thread* signaling_thread,
henrike@webrtc.org28e20752013-07-10 00:45:36 +000081 AudioDeviceModule* default_adm,
82 cricket::WebRtcVideoEncoderFactory* encoder_factory,
83 cricket::WebRtcVideoDecoderFactory* decoder_factory) {
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +000084 rtc::scoped_refptr<PeerConnectionFactory> pc_factory(
85 new rtc::RefCountedObject<PeerConnectionFactory>(worker_thread,
perkj@webrtc.org81134d02015-01-12 08:30:16 +000086 signaling_thread,
87 default_adm,
88 encoder_factory,
89 decoder_factory));
90
91 // Call Initialize synchronously but make sure its executed on
92 // |signaling_thread|.
93 MethodCall0<PeerConnectionFactory, bool> call(
94 pc_factory.get(),
95 &PeerConnectionFactory::Initialize);
96 bool result = call.Marshal(signaling_thread);
97
98 if (!result) {
Taylor Brandstettera8415fe2016-03-23 10:38:07 -070099 return nullptr;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000100 }
perkj@webrtc.org81134d02015-01-12 08:30:16 +0000101 return PeerConnectionFactoryProxy::Create(signaling_thread, pc_factory);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000102}
103
104PeerConnectionFactory::PeerConnectionFactory()
105 : owns_ptrs_(true),
perkj@webrtc.org81134d02015-01-12 08:30:16 +0000106 wraps_current_thread_(false),
107 signaling_thread_(rtc::ThreadManager::Instance()->CurrentThread()),
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000108 worker_thread_(new rtc::Thread) {
perkj@webrtc.org81134d02015-01-12 08:30:16 +0000109 if (!signaling_thread_) {
110 signaling_thread_ = rtc::ThreadManager::Instance()->WrapCurrentThread();
111 wraps_current_thread_ = true;
112 }
113 worker_thread_->Start();
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000114}
115
116PeerConnectionFactory::PeerConnectionFactory(
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000117 rtc::Thread* worker_thread,
118 rtc::Thread* signaling_thread,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000119 AudioDeviceModule* default_adm,
120 cricket::WebRtcVideoEncoderFactory* video_encoder_factory,
121 cricket::WebRtcVideoDecoderFactory* video_decoder_factory)
122 : owns_ptrs_(false),
perkj@webrtc.org81134d02015-01-12 08:30:16 +0000123 wraps_current_thread_(false),
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000124 signaling_thread_(signaling_thread),
125 worker_thread_(worker_thread),
126 default_adm_(default_adm),
127 video_encoder_factory_(video_encoder_factory),
128 video_decoder_factory_(video_decoder_factory) {
129 ASSERT(worker_thread != NULL);
130 ASSERT(signaling_thread != NULL);
131 // TODO: Currently there is no way creating an external adm in
132 // libjingle source tree. So we can 't currently assert if this is NULL.
133 // ASSERT(default_adm != NULL);
134}
135
136PeerConnectionFactory::~PeerConnectionFactory() {
henrikg91d6ede2015-09-17 00:24:34 -0700137 RTC_DCHECK(signaling_thread_->IsCurrent());
Henrik Boström5e56c592015-08-11 10:33:13 +0200138 channel_manager_.reset(nullptr);
jiayl@webrtc.orgd83f4ef2015-03-13 21:26:12 +0000139
140 // Make sure |worker_thread_| and |signaling_thread_| outlive
deadbeef41b07982015-12-01 15:01:24 -0800141 // |dtls_identity_store_|, |default_socket_factory_| and
142 // |default_network_manager_|.
Henrik Boström5e56c592015-08-11 10:33:13 +0200143 dtls_identity_store_ = nullptr;
deadbeef41b07982015-12-01 15:01:24 -0800144 default_socket_factory_ = nullptr;
145 default_network_manager_ = nullptr;
jiayl@webrtc.orgd83f4ef2015-03-13 21:26:12 +0000146
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000147 if (owns_ptrs_) {
perkj@webrtc.org81134d02015-01-12 08:30:16 +0000148 if (wraps_current_thread_)
149 rtc::ThreadManager::Instance()->UnwrapCurrentThread();
fischman@webrtc.org29540b12014-04-17 22:54:30 +0000150 delete worker_thread_;
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000151 }
152}
153
154bool PeerConnectionFactory::Initialize() {
henrikg91d6ede2015-09-17 00:24:34 -0700155 RTC_DCHECK(signaling_thread_->IsCurrent());
Honghai Zhang82d78622016-05-06 11:29:15 -0700156 rtc::InitRandom(rtc::Time32());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000157
deadbeef41b07982015-12-01 15:01:24 -0800158 default_network_manager_.reset(new rtc::BasicNetworkManager());
159 if (!default_network_manager_) {
160 return false;
161 }
162
163 default_socket_factory_.reset(
164 new rtc::BasicPacketSocketFactory(worker_thread_));
165 if (!default_socket_factory_) {
166 return false;
167 }
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000168
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000169 // TODO: Need to make sure only one VoE is created inside
170 // WebRtcMediaEngine.
henrika@webrtc.org62f6e752015-02-11 08:38:35 +0000171 cricket::MediaEngineInterface* media_engine =
172 worker_thread_->Invoke<cricket::MediaEngineInterface*>(rtc::Bind(
173 &PeerConnectionFactory::CreateMediaEngine_w, this));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000174
Danil Chapovalov33b01f22016-05-11 19:55:27 +0200175 rtc::Thread* const network_thread = worker_thread_;
176 channel_manager_.reset(new cricket::ChannelManager(
177 media_engine, worker_thread_, network_thread));
henrika@webrtc.org62f6e752015-02-11 08:38:35 +0000178
stefan@webrtc.org85d27942014-06-09 12:51:39 +0000179 channel_manager_->SetVideoRtxEnabled(true);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000180 if (!channel_manager_->Init()) {
181 return false;
182 }
jiayl@webrtc.org61e00b02015-03-04 22:17:38 +0000183
Henrik Boström5e56c592015-08-11 10:33:13 +0200184 dtls_identity_store_ = new RefCountedDtlsIdentityStore(
185 signaling_thread_, worker_thread_);
jiayl@webrtc.org61e00b02015-03-04 22:17:38 +0000186
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000187 return true;
188}
189
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000190rtc::scoped_refptr<AudioSourceInterface>
perkj@webrtc.org81134d02015-01-12 08:30:16 +0000191PeerConnectionFactory::CreateAudioSource(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000192 const MediaConstraintsInterface* constraints) {
henrikg91d6ede2015-09-17 00:24:34 -0700193 RTC_DCHECK(signaling_thread_->IsCurrent());
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000194 rtc::scoped_refptr<LocalAudioSource> source(
wu@webrtc.org97077a32013-10-25 21:18:33 +0000195 LocalAudioSource::Create(options_, constraints));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000196 return source;
197}
198
htaa2a49d92016-03-04 02:51:39 -0800199rtc::scoped_refptr<AudioSourceInterface>
200PeerConnectionFactory::CreateAudioSource(const cricket::AudioOptions& options) {
201 RTC_DCHECK(signaling_thread_->IsCurrent());
202 rtc::scoped_refptr<LocalAudioSource> source(
203 LocalAudioSource::Create(options_, &options));
204 return source;
205}
206
perkja3ede6c2016-03-08 01:27:48 +0100207rtc::scoped_refptr<VideoTrackSourceInterface>
perkj@webrtc.org81134d02015-01-12 08:30:16 +0000208PeerConnectionFactory::CreateVideoSource(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000209 cricket::VideoCapturer* capturer,
210 const MediaConstraintsInterface* constraints) {
henrikg91d6ede2015-09-17 00:24:34 -0700211 RTC_DCHECK(signaling_thread_->IsCurrent());
perkja3ede6c2016-03-08 01:27:48 +0100212 rtc::scoped_refptr<VideoTrackSourceInterface> source(
213 VideoCapturerTrackSource::Create(worker_thread_, capturer, constraints,
214 false));
nisse5b68ab52016-04-07 07:45:54 -0700215 return VideoTrackSourceProxy::Create(signaling_thread_, worker_thread_,
216 source);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000217}
218
perkja3ede6c2016-03-08 01:27:48 +0100219rtc::scoped_refptr<VideoTrackSourceInterface>
htaa2a49d92016-03-04 02:51:39 -0800220PeerConnectionFactory::CreateVideoSource(cricket::VideoCapturer* capturer) {
221 RTC_DCHECK(signaling_thread_->IsCurrent());
perkja3ede6c2016-03-08 01:27:48 +0100222 rtc::scoped_refptr<VideoTrackSourceInterface> source(
223 VideoCapturerTrackSource::Create(worker_thread_, capturer, false));
nisse5b68ab52016-04-07 07:45:54 -0700224 return VideoTrackSourceProxy::Create(signaling_thread_, worker_thread_,
225 source);
htaa2a49d92016-03-04 02:51:39 -0800226}
227
ivocd66b44d2016-01-15 03:06:36 -0800228bool PeerConnectionFactory::StartAecDump(rtc::PlatformFile file,
229 int64_t max_size_bytes) {
henrikg91d6ede2015-09-17 00:24:34 -0700230 RTC_DCHECK(signaling_thread_->IsCurrent());
ivocd66b44d2016-01-15 03:06:36 -0800231 return channel_manager_->StartAecDump(file, max_size_bytes);
wu@webrtc.orga9890802013-12-13 00:21:03 +0000232}
233
ivoc797ef122015-10-22 03:25:41 -0700234void PeerConnectionFactory::StopAecDump() {
235 RTC_DCHECK(signaling_thread_->IsCurrent());
236 channel_manager_->StopAecDump();
237}
238
ivoc112a3d82015-10-16 02:22:18 -0700239bool PeerConnectionFactory::StartRtcEventLog(rtc::PlatformFile file) {
240 RTC_DCHECK(signaling_thread_->IsCurrent());
241 return channel_manager_->StartRtcEventLog(file);
242}
243
244void PeerConnectionFactory::StopRtcEventLog() {
245 RTC_DCHECK(signaling_thread_->IsCurrent());
246 channel_manager_->StopRtcEventLog();
247}
248
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000249rtc::scoped_refptr<PeerConnectionInterface>
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000250PeerConnectionFactory::CreatePeerConnection(
htaa2a49d92016-03-04 02:51:39 -0800251 const PeerConnectionInterface::RTCConfiguration& configuration_in,
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000252 const MediaConstraintsInterface* constraints,
kwibergd1fe2812016-04-27 06:47:29 -0700253 std::unique_ptr<cricket::PortAllocator> allocator,
254 std::unique_ptr<DtlsIdentityStoreInterface> dtls_identity_store,
deadbeef41b07982015-12-01 15:01:24 -0800255 PeerConnectionObserver* observer) {
256 RTC_DCHECK(signaling_thread_->IsCurrent());
257
htaa2a49d92016-03-04 02:51:39 -0800258 // We merge constraints and configuration into a single configuration.
259 PeerConnectionInterface::RTCConfiguration configuration = configuration_in;
260 CopyConstraintsIntoRtcConfiguration(constraints, &configuration);
261
262 return CreatePeerConnection(configuration, std::move(allocator),
263 std::move(dtls_identity_store), observer);
264}
265
266rtc::scoped_refptr<PeerConnectionInterface>
267PeerConnectionFactory::CreatePeerConnection(
268 const PeerConnectionInterface::RTCConfiguration& configuration,
kwibergd1fe2812016-04-27 06:47:29 -0700269 std::unique_ptr<cricket::PortAllocator> allocator,
270 std::unique_ptr<DtlsIdentityStoreInterface> dtls_identity_store,
htaa2a49d92016-03-04 02:51:39 -0800271 PeerConnectionObserver* observer) {
272 RTC_DCHECK(signaling_thread_->IsCurrent());
273
deadbeef41b07982015-12-01 15:01:24 -0800274 if (!dtls_identity_store.get()) {
275 // Because |pc|->Initialize takes ownership of the store we need a new
276 // wrapper object that can be deleted without deleting the underlying
277 // |dtls_identity_store_|, protecting it from being deleted multiple times.
278 dtls_identity_store.reset(
279 new DtlsIdentityStoreWrapper(dtls_identity_store_));
280 }
281
282 if (!allocator) {
283 allocator.reset(new cricket::BasicPortAllocator(
284 default_network_manager_.get(), default_socket_factory_.get()));
285 }
Taylor Brandstettera1c30352016-05-13 08:15:11 -0700286 worker_thread_->Invoke<void>(
287 rtc::Bind(&cricket::PortAllocator::SetNetworkIgnoreMask, allocator.get(),
288 options_.network_ignore_mask));
deadbeef41b07982015-12-01 15:01:24 -0800289
290 rtc::scoped_refptr<PeerConnection> pc(
291 new rtc::RefCountedObject<PeerConnection>(this));
htaa2a49d92016-03-04 02:51:39 -0800292
nissec36b31b2016-04-11 23:25:29 -0700293 if (!pc->Initialize(configuration, std::move(allocator),
deadbeef41b07982015-12-01 15:01:24 -0800294 std::move(dtls_identity_store), observer)) {
295 return nullptr;
296 }
297 return PeerConnectionProxy::Create(signaling_thread(), pc);
298}
299
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000300rtc::scoped_refptr<MediaStreamInterface>
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000301PeerConnectionFactory::CreateLocalMediaStream(const std::string& label) {
henrikg91d6ede2015-09-17 00:24:34 -0700302 RTC_DCHECK(signaling_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000303 return MediaStreamProxy::Create(signaling_thread_,
304 MediaStream::Create(label));
305}
306
perkja3ede6c2016-03-08 01:27:48 +0100307rtc::scoped_refptr<VideoTrackInterface> PeerConnectionFactory::CreateVideoTrack(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000308 const std::string& id,
perkja3ede6c2016-03-08 01:27:48 +0100309 VideoTrackSourceInterface* source) {
henrikg91d6ede2015-09-17 00:24:34 -0700310 RTC_DCHECK(signaling_thread_->IsCurrent());
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000311 rtc::scoped_refptr<VideoTrackInterface> track(
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000312 VideoTrack::Create(id, source));
nisse5b68ab52016-04-07 07:45:54 -0700313 return VideoTrackProxy::Create(signaling_thread_, worker_thread_, track);
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000314}
315
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000316rtc::scoped_refptr<AudioTrackInterface>
pbos@webrtc.orgb5a22b12014-05-13 11:07:01 +0000317PeerConnectionFactory::CreateAudioTrack(const std::string& id,
318 AudioSourceInterface* source) {
henrikg91d6ede2015-09-17 00:24:34 -0700319 RTC_DCHECK(signaling_thread_->IsCurrent());
tommi6eca7e32015-12-15 04:27:11 -0800320 rtc::scoped_refptr<AudioTrackInterface> track(AudioTrack::Create(id, source));
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000321 return AudioTrackProxy::Create(signaling_thread_, track);
322}
323
nisse51542be2016-02-12 02:27:06 -0800324webrtc::MediaControllerInterface* PeerConnectionFactory::CreateMediaController(
325 const cricket::MediaConfig& config) const {
henrikg91d6ede2015-09-17 00:24:34 -0700326 RTC_DCHECK(signaling_thread_->IsCurrent());
nisse51542be2016-02-12 02:27:06 -0800327 return MediaControllerInterface::Create(config, worker_thread_,
stefanc1aeaf02015-10-15 07:26:07 -0700328 channel_manager_.get());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000329}
330
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000331rtc::Thread* PeerConnectionFactory::signaling_thread() {
perkj@webrtc.org81134d02015-01-12 08:30:16 +0000332 // This method can be called on a different thread when the factory is
333 // created in CreatePeerConnectionFactory().
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000334 return signaling_thread_;
335}
336
buildbot@webrtc.orgd4e598d2014-07-29 17:36:52 +0000337rtc::Thread* PeerConnectionFactory::worker_thread() {
henrikg91d6ede2015-09-17 00:24:34 -0700338 RTC_DCHECK(signaling_thread_->IsCurrent());
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000339 return worker_thread_;
340}
341
henrika@webrtc.org62f6e752015-02-11 08:38:35 +0000342cricket::MediaEngineInterface* PeerConnectionFactory::CreateMediaEngine_w() {
343 ASSERT(worker_thread_ == rtc::Thread::Current());
344 return cricket::WebRtcMediaEngineFactory::Create(
Fredrik Solenbergccb49e72015-05-19 11:37:56 +0200345 default_adm_.get(), video_encoder_factory_.get(),
henrika@webrtc.org62f6e752015-02-11 08:38:35 +0000346 video_decoder_factory_.get());
347}
348
henrike@webrtc.org28e20752013-07-10 00:45:36 +0000349} // namespace webrtc