blob: 5a78c1198143aa74367977bf65a3733eb4611255 [file] [log] [blame]
niklase@google.com470e71d2011-07-07 08:21:25 +00001/*
leozwang@webrtc.org91b359e2012-02-28 17:26:14 +00002 * Copyright (c) 2012 The WebRTC project authors. All Rights Reserved.
niklase@google.com470e71d2011-07-07 08:21:25 +00003 *
4 * 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.
9 */
10
kjellander@webrtc.org543c3ea2011-11-23 12:20:35 +000011#include "TwoWayCommunication.h"
12
pbos@webrtc.org12dc1a32013-08-05 16:22:53 +000013#include <ctype.h>
niklase@google.com470e71d2011-07-07 08:21:25 +000014#include <stdio.h>
15#include <string.h>
16
kwiberg37478382016-02-14 20:40:57 -080017#include <memory>
18
niklase@google.com470e71d2011-07-07 08:21:25 +000019#ifdef WIN32
20#include <Windows.h>
21#endif
22
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020023#include "api/audio_codecs/builtin_audio_decoder_factory.h"
Mirko Bonadei71207422017-09-15 13:58:09 +020024#include "common_types.h" // NOLINT(build/include)
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020025#include "modules/audio_coding/codecs/audio_format_conversion.h"
26#include "modules/audio_coding/test/PCMFile.h"
27#include "modules/audio_coding/test/utility.h"
28#include "test/gtest.h"
29#include "test/testsupport/fileutils.h"
Mirko Bonadei71207422017-09-15 13:58:09 +020030#include "typedefs.h" // NOLINT(build/include)
niklase@google.com470e71d2011-07-07 08:21:25 +000031
tina.legrand@webrtc.org554ae1a2011-12-16 10:09:04 +000032namespace webrtc {
niklase@google.com470e71d2011-07-07 08:21:25 +000033
34#define MAX_FILE_NAME_LENGTH_BYTE 500
35
henrik.lundin@webrtc.orgadaf8092014-04-17 08:29:10 +000036TwoWayCommunication::TwoWayCommunication(int testMode)
Karl Wiberg5817d3d2018-04-06 10:06:42 +020037 : _acmA(AudioCodingModule::Create(
38 AudioCodingModule::Config(CreateBuiltinAudioDecoderFactory()))),
39 _acmRefA(AudioCodingModule::Create(
40 AudioCodingModule::Config(CreateBuiltinAudioDecoderFactory()))),
henrik.lundin1bd0e032015-09-28 06:12:17 -070041 _testMode(testMode) {
42 AudioCodingModule::Config config;
43 // The clicks will be more obvious in FAX mode. TODO(henrik.lundin) Really?
44 config.neteq_config.playout_mode = kPlayoutFax;
ossue3525782016-05-25 07:37:43 -070045 config.decoder_factory = CreateBuiltinAudioDecoderFactory();
henrik.lundin1bd0e032015-09-28 06:12:17 -070046 _acmB.reset(AudioCodingModule::Create(config));
henrik.lundin1bd0e032015-09-28 06:12:17 -070047 _acmRefB.reset(AudioCodingModule::Create(config));
48}
niklase@google.com470e71d2011-07-07 08:21:25 +000049
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000050TwoWayCommunication::~TwoWayCommunication() {
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000051 delete _channel_A2B;
52 delete _channel_B2A;
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000053 delete _channelRef_A2B;
54 delete _channelRef_B2A;
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000055 _inFileA.Close();
56 _inFileB.Close();
57 _outFileA.Close();
58 _outFileB.Close();
59 _outFileRefA.Close();
60 _outFileRefB.Close();
niklase@google.com470e71d2011-07-07 08:21:25 +000061}
62
Yves Gerey665174f2018-06-19 15:03:05 +020063void TwoWayCommunication::ChooseCodec(uint8_t* codecID_A, uint8_t* codecID_B) {
Karl Wiberg5817d3d2018-04-06 10:06:42 +020064 std::unique_ptr<AudioCodingModule> tmpACM(AudioCodingModule::Create(
65 AudioCodingModule::Config(CreateBuiltinAudioDecoderFactory())));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000066 uint8_t noCodec = tmpACM->NumberOfCodecs();
67 CodecInst codecInst;
68 printf("List of Supported Codecs\n");
69 printf("========================\n");
70 for (uint8_t codecCntr = 0; codecCntr < noCodec; codecCntr++) {
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +000071 EXPECT_EQ(tmpACM->Codec(codecCntr, &codecInst), 0);
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000072 printf("%d- %s\n", codecCntr, codecInst.plname);
73 }
74 printf("\nChoose a send codec for side A [0]: ");
75 char myStr[15] = "";
76 EXPECT_TRUE(fgets(myStr, 10, stdin) != NULL);
Yves Gerey665174f2018-06-19 15:03:05 +020077 *codecID_A = (uint8_t)atoi(myStr);
niklase@google.com470e71d2011-07-07 08:21:25 +000078
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000079 printf("\nChoose a send codec for side B [0]: ");
80 EXPECT_TRUE(fgets(myStr, 10, stdin) != NULL);
Yves Gerey665174f2018-06-19 15:03:05 +020081 *codecID_B = (uint8_t)atoi(myStr);
niklase@google.com470e71d2011-07-07 08:21:25 +000082
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000083 printf("\n");
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000084}
niklase@google.com470e71d2011-07-07 08:21:25 +000085
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +000086void TwoWayCommunication::SetUp() {
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000087 uint8_t codecID_A;
88 uint8_t codecID_B;
89
90 ChooseCodec(&codecID_A, &codecID_B);
91 CodecInst codecInst_A;
92 CodecInst codecInst_B;
93 CodecInst dummyCodec;
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +000094 EXPECT_EQ(0, _acmA->Codec(codecID_A, &codecInst_A));
95 EXPECT_EQ(0, _acmB->Codec(codecID_B, &codecInst_B));
96 EXPECT_EQ(0, _acmA->Codec(6, &dummyCodec));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000097
98 //--- Set A codecs
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +000099 EXPECT_EQ(0, _acmA->RegisterSendCodec(codecInst_A));
kwibergda2bf4e2016-10-24 13:47:09 -0700100 EXPECT_EQ(true, _acmA->RegisterReceiveCodec(codecInst_B.pltype,
101 CodecInstToSdp(codecInst_B)));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000102 //--- Set ref-A codecs
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000103 EXPECT_EQ(0, _acmRefA->RegisterSendCodec(codecInst_A));
kwibergda2bf4e2016-10-24 13:47:09 -0700104 EXPECT_EQ(true, _acmRefA->RegisterReceiveCodec(codecInst_B.pltype,
105 CodecInstToSdp(codecInst_B)));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000106
107 //--- Set B codecs
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000108 EXPECT_EQ(0, _acmB->RegisterSendCodec(codecInst_B));
kwibergda2bf4e2016-10-24 13:47:09 -0700109 EXPECT_EQ(true, _acmB->RegisterReceiveCodec(codecInst_A.pltype,
110 CodecInstToSdp(codecInst_A)));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000111
112 //--- Set ref-B codecs
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000113 EXPECT_EQ(0, _acmRefB->RegisterSendCodec(codecInst_B));
kwibergda2bf4e2016-10-24 13:47:09 -0700114 EXPECT_EQ(true, _acmRefB->RegisterReceiveCodec(codecInst_A.pltype,
115 CodecInstToSdp(codecInst_A)));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000116
117 uint16_t frequencyHz;
118
119 //--- Input A
Yves Gerey665174f2018-06-19 15:03:05 +0200120 std::string in_file_name =
121 webrtc::test::ResourcePath("audio_coding/testfile32kHz", "pcm");
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000122 frequencyHz = 32000;
123 printf("Enter input file at side A [%s]: ", in_file_name.c_str());
124 PCMFile::ChooseFile(&in_file_name, 499, &frequencyHz);
125 _inFileA.Open(in_file_name, frequencyHz, "rb");
126
127 //--- Output A
128 std::string out_file_a = webrtc::test::OutputPath() + "outA.pcm";
129 printf("Output file at side A: %s\n", out_file_a.c_str());
130 printf("Sampling frequency (in Hz) of the above file: %u\n", frequencyHz);
131 _outFileA.Open(out_file_a, frequencyHz, "wb");
132 std::string ref_file_name = webrtc::test::OutputPath() + "ref_outA.pcm";
133 _outFileRefA.Open(ref_file_name, frequencyHz, "wb");
134
135 //--- Input B
Yves Gerey665174f2018-06-19 15:03:05 +0200136 in_file_name =
137 webrtc::test::ResourcePath("audio_coding/testfile32kHz", "pcm");
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000138 frequencyHz = 32000;
139 printf("\n\nEnter input file at side B [%s]: ", in_file_name.c_str());
140 PCMFile::ChooseFile(&in_file_name, 499, &frequencyHz);
141 _inFileB.Open(in_file_name, frequencyHz, "rb");
142
143 //--- Output B
144 std::string out_file_b = webrtc::test::OutputPath() + "outB.pcm";
145 printf("Output file at side B: %s\n", out_file_b.c_str());
146 printf("Sampling frequency (in Hz) of the above file: %u\n", frequencyHz);
147 _outFileB.Open(out_file_b, frequencyHz, "wb");
148 ref_file_name = webrtc::test::OutputPath() + "ref_outB.pcm";
149 _outFileRefB.Open(ref_file_name, frequencyHz, "wb");
150
151 //--- Set A-to-B channel
152 _channel_A2B = new Channel;
153 _acmA->RegisterTransportCallback(_channel_A2B);
andrew@webrtc.org89df0922013-09-12 01:27:43 +0000154 _channel_A2B->RegisterReceiverACM(_acmB.get());
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000155 //--- Do the same for the reference
156 _channelRef_A2B = new Channel;
157 _acmRefA->RegisterTransportCallback(_channelRef_A2B);
andrew@webrtc.org89df0922013-09-12 01:27:43 +0000158 _channelRef_A2B->RegisterReceiverACM(_acmRefB.get());
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000159
160 //--- Set B-to-A channel
161 _channel_B2A = new Channel;
162 _acmB->RegisterTransportCallback(_channel_B2A);
andrew@webrtc.org89df0922013-09-12 01:27:43 +0000163 _channel_B2A->RegisterReceiverACM(_acmA.get());
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000164 //--- Do the same for reference
165 _channelRef_B2A = new Channel;
166 _acmRefB->RegisterTransportCallback(_channelRef_B2A);
andrew@webrtc.org89df0922013-09-12 01:27:43 +0000167 _channelRef_B2A->RegisterReceiverACM(_acmRefA.get());
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000168}
169
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000170void TwoWayCommunication::SetUpAutotest() {
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000171 CodecInst codecInst_A;
172 CodecInst codecInst_B;
173 CodecInst dummyCodec;
174
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000175 EXPECT_EQ(0, _acmA->Codec("ISAC", &codecInst_A, 16000, 1));
176 EXPECT_EQ(0, _acmB->Codec("L16", &codecInst_B, 8000, 1));
177 EXPECT_EQ(0, _acmA->Codec(6, &dummyCodec));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000178
179 //--- Set A codecs
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000180 EXPECT_EQ(0, _acmA->RegisterSendCodec(codecInst_A));
kwibergda2bf4e2016-10-24 13:47:09 -0700181 EXPECT_EQ(true, _acmA->RegisterReceiveCodec(codecInst_B.pltype,
182 CodecInstToSdp(codecInst_B)));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000183
184 //--- Set ref-A codecs
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000185 EXPECT_GT(_acmRefA->RegisterSendCodec(codecInst_A), -1);
kwibergda2bf4e2016-10-24 13:47:09 -0700186 EXPECT_EQ(true, _acmRefA->RegisterReceiveCodec(codecInst_B.pltype,
187 CodecInstToSdp(codecInst_B)));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000188
189 //--- Set B codecs
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000190 EXPECT_GT(_acmB->RegisterSendCodec(codecInst_B), -1);
kwibergda2bf4e2016-10-24 13:47:09 -0700191 EXPECT_EQ(true, _acmB->RegisterReceiveCodec(codecInst_A.pltype,
192 CodecInstToSdp(codecInst_A)));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000193
194 //--- Set ref-B codecs
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000195 EXPECT_EQ(0, _acmRefB->RegisterSendCodec(codecInst_B));
kwibergda2bf4e2016-10-24 13:47:09 -0700196 EXPECT_EQ(true, _acmRefB->RegisterReceiveCodec(codecInst_A.pltype,
197 CodecInstToSdp(codecInst_A)));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000198
199 uint16_t frequencyHz;
200
201 //--- Input A and B
Yves Gerey665174f2018-06-19 15:03:05 +0200202 std::string in_file_name =
203 webrtc::test::ResourcePath("audio_coding/testfile32kHz", "pcm");
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000204 frequencyHz = 16000;
205 _inFileA.Open(in_file_name, frequencyHz, "rb");
206 _inFileB.Open(in_file_name, frequencyHz, "rb");
207
208 //--- Output A
209 std::string output_file_a = webrtc::test::OutputPath() + "outAutotestA.pcm";
210 frequencyHz = 16000;
211 _outFileA.Open(output_file_a, frequencyHz, "wb");
Yves Gerey665174f2018-06-19 15:03:05 +0200212 std::string output_ref_file_a =
213 webrtc::test::OutputPath() + "ref_outAutotestA.pcm";
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000214 _outFileRefA.Open(output_ref_file_a, frequencyHz, "wb");
215
216 //--- Output B
217 std::string output_file_b = webrtc::test::OutputPath() + "outAutotestB.pcm";
218 frequencyHz = 16000;
219 _outFileB.Open(output_file_b, frequencyHz, "wb");
Yves Gerey665174f2018-06-19 15:03:05 +0200220 std::string output_ref_file_b =
221 webrtc::test::OutputPath() + "ref_outAutotestB.pcm";
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000222 _outFileRefB.Open(output_ref_file_b, frequencyHz, "wb");
223
224 //--- Set A-to-B channel
225 _channel_A2B = new Channel;
226 _acmA->RegisterTransportCallback(_channel_A2B);
andrew@webrtc.org89df0922013-09-12 01:27:43 +0000227 _channel_A2B->RegisterReceiverACM(_acmB.get());
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000228 //--- Do the same for the reference
229 _channelRef_A2B = new Channel;
230 _acmRefA->RegisterTransportCallback(_channelRef_A2B);
andrew@webrtc.org89df0922013-09-12 01:27:43 +0000231 _channelRef_A2B->RegisterReceiverACM(_acmRefB.get());
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000232
233 //--- Set B-to-A channel
234 _channel_B2A = new Channel;
235 _acmB->RegisterTransportCallback(_channel_B2A);
andrew@webrtc.org89df0922013-09-12 01:27:43 +0000236 _channel_B2A->RegisterReceiverACM(_acmA.get());
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000237 //--- Do the same for reference
238 _channelRef_B2A = new Channel;
239 _acmRefB->RegisterTransportCallback(_channelRef_B2A);
andrew@webrtc.org89df0922013-09-12 01:27:43 +0000240 _channelRef_B2A->RegisterReceiverACM(_acmRefA.get());
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000241}
242
243void TwoWayCommunication::Perform() {
244 if (_testMode == 0) {
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000245 SetUpAutotest();
246 } else {
247 SetUp();
248 }
249 unsigned int msecPassed = 0;
250 unsigned int secPassed = 0;
251
252 int32_t outFreqHzA = _outFileA.SamplingFrequency();
253 int32_t outFreqHzB = _outFileB.SamplingFrequency();
254
255 AudioFrame audioFrame;
256
kwiberg1fd4a4a2015-11-03 11:20:50 -0800257 auto codecInst_B = _acmB->SendCodec();
258 ASSERT_TRUE(codecInst_B);
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000259
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000260 // In the following loop we tests that the code can handle misuse of the APIs.
261 // In the middle of a session with data flowing between two sides, called A
Karl Wibergdd00f112015-08-25 09:37:04 +0200262 // and B, APIs will be called, and the code should continue to run, and be
263 // able to recover.
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000264 while (!_inFileA.EndOfFile() && !_inFileB.EndOfFile()) {
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000265 msecPassed += 10;
266 EXPECT_GT(_inFileA.Read10MsData(audioFrame), 0);
henrik.lundin@webrtc.orgf56c1622015-03-02 12:29:30 +0000267 EXPECT_GE(_acmA->Add10MsData(audioFrame), 0);
268 EXPECT_GE(_acmRefA->Add10MsData(audioFrame), 0);
niklase@google.com470e71d2011-07-07 08:21:25 +0000269
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000270 EXPECT_GT(_inFileB.Read10MsData(audioFrame), 0);
niklase@google.com470e71d2011-07-07 08:21:25 +0000271
Henrik Lundin45c64492015-03-30 19:00:44 +0200272 EXPECT_GE(_acmB->Add10MsData(audioFrame), 0);
henrik.lundin@webrtc.orgf56c1622015-03-02 12:29:30 +0000273 EXPECT_GE(_acmRefB->Add10MsData(audioFrame), 0);
henrik.lundind4ccb002016-05-17 12:21:55 -0700274 bool muted;
275 EXPECT_EQ(0, _acmA->PlayoutData10Ms(outFreqHzA, &audioFrame, &muted));
276 ASSERT_FALSE(muted);
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000277 _outFileA.Write10MsData(audioFrame);
henrik.lundind4ccb002016-05-17 12:21:55 -0700278 EXPECT_EQ(0, _acmRefA->PlayoutData10Ms(outFreqHzA, &audioFrame, &muted));
279 ASSERT_FALSE(muted);
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000280 _outFileRefA.Write10MsData(audioFrame);
henrik.lundind4ccb002016-05-17 12:21:55 -0700281 EXPECT_EQ(0, _acmB->PlayoutData10Ms(outFreqHzB, &audioFrame, &muted));
282 ASSERT_FALSE(muted);
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000283 _outFileB.Write10MsData(audioFrame);
henrik.lundind4ccb002016-05-17 12:21:55 -0700284 EXPECT_EQ(0, _acmRefB->PlayoutData10Ms(outFreqHzB, &audioFrame, &muted));
285 ASSERT_FALSE(muted);
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000286 _outFileRefB.Write10MsData(audioFrame);
niklase@google.com470e71d2011-07-07 08:21:25 +0000287
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000288 // Update time counters each time a second of data has passed.
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000289 if (msecPassed >= 1000) {
290 msecPassed = 0;
291 secPassed++;
292 }
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000293 // Re-register send codec on side B.
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000294 if (((secPassed % 5) == 4) && (msecPassed >= 990)) {
kwiberg1fd4a4a2015-11-03 11:20:50 -0800295 EXPECT_EQ(0, _acmB->RegisterSendCodec(*codecInst_B));
296 EXPECT_TRUE(_acmB->SendCodec());
niklase@google.com470e71d2011-07-07 08:21:25 +0000297 }
Karl Wibergdd00f112015-08-25 09:37:04 +0200298 // Initialize receiver on side A.
299 if (((secPassed % 7) == 6) && (msecPassed == 0))
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000300 EXPECT_EQ(0, _acmA->InitializeReceiver());
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000301 // Re-register codec on side A.
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000302 if (((secPassed % 7) == 6) && (msecPassed >= 990)) {
kwibergda2bf4e2016-10-24 13:47:09 -0700303 EXPECT_EQ(true, _acmA->RegisterReceiveCodec(
304 codecInst_B->pltype, CodecInstToSdp(*codecInst_B)));
niklase@google.com470e71d2011-07-07 08:21:25 +0000305 }
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000306 }
niklase@google.com470e71d2011-07-07 08:21:25 +0000307}
308
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000309} // namespace webrtc