Upstream version 7.36.149.0
[platform/framework/web/crosswalk.git] / src / third_party / webrtc / modules / audio_coding / main / test / TwoWayCommunication.cc
1 /*
2  *  Copyright (c) 2012 The WebRTC project authors. All Rights Reserved.
3  *
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
11 #include "TwoWayCommunication.h"
12
13 #include <ctype.h>
14 #include <stdio.h>
15 #include <string.h>
16
17 #ifdef WIN32
18 #include <Windows.h>
19 #endif
20
21 #include "gtest/gtest.h"
22 #include "webrtc/engine_configurations.h"
23 #include "webrtc/common_types.h"
24 #include "webrtc/modules/audio_coding/main/test/PCMFile.h"
25 #include "webrtc/modules/audio_coding/main/test/utility.h"
26 #include "webrtc/system_wrappers/interface/trace.h"
27 #include "webrtc/test/testsupport/fileutils.h"
28
29 namespace webrtc {
30
31 #define MAX_FILE_NAME_LENGTH_BYTE 500
32
33 TwoWayCommunication::TwoWayCommunication(int testMode)
34     : _acmA(AudioCodingModule::Create(1)),
35       _acmB(AudioCodingModule::Create(2)),
36       _acmRefA(AudioCodingModule::Create(3)),
37       _acmRefB(AudioCodingModule::Create(4)),
38       _testMode(testMode) {}
39
40 TwoWayCommunication::~TwoWayCommunication() {
41   delete _channel_A2B;
42   delete _channel_B2A;
43   delete _channelRef_A2B;
44   delete _channelRef_B2A;
45 #ifdef WEBRTC_DTMF_DETECTION
46   if (_dtmfDetectorA != NULL) {
47     delete _dtmfDetectorA;
48   }
49   if (_dtmfDetectorB != NULL) {
50     delete _dtmfDetectorB;
51   }
52 #endif
53   _inFileA.Close();
54   _inFileB.Close();
55   _outFileA.Close();
56   _outFileB.Close();
57   _outFileRefA.Close();
58   _outFileRefB.Close();
59 }
60
61 void TwoWayCommunication::ChooseCodec(uint8_t* codecID_A,
62                                       uint8_t* codecID_B) {
63   scoped_ptr<AudioCodingModule> tmpACM(AudioCodingModule::Create(0));
64   uint8_t noCodec = tmpACM->NumberOfCodecs();
65   CodecInst codecInst;
66   printf("List of Supported Codecs\n");
67   printf("========================\n");
68   for (uint8_t codecCntr = 0; codecCntr < noCodec; codecCntr++) {
69     EXPECT_EQ(tmpACM->Codec(codecCntr, &codecInst), 0);
70     printf("%d- %s\n", codecCntr, codecInst.plname);
71   }
72   printf("\nChoose a send codec for side A [0]: ");
73   char myStr[15] = "";
74   EXPECT_TRUE(fgets(myStr, 10, stdin) != NULL);
75   *codecID_A = (uint8_t) atoi(myStr);
76
77   printf("\nChoose a send codec for side B [0]: ");
78   EXPECT_TRUE(fgets(myStr, 10, stdin) != NULL);
79   *codecID_B = (uint8_t) atoi(myStr);
80
81   printf("\n");
82 }
83
84 void TwoWayCommunication::SetUp() {
85   uint8_t codecID_A;
86   uint8_t codecID_B;
87
88   ChooseCodec(&codecID_A, &codecID_B);
89   CodecInst codecInst_A;
90   CodecInst codecInst_B;
91   CodecInst dummyCodec;
92   EXPECT_EQ(0, _acmA->Codec(codecID_A, &codecInst_A));
93   EXPECT_EQ(0, _acmB->Codec(codecID_B, &codecInst_B));
94   EXPECT_EQ(0, _acmA->Codec(6, &dummyCodec));
95
96   //--- Set A codecs
97   EXPECT_EQ(0, _acmA->RegisterSendCodec(codecInst_A));
98   EXPECT_EQ(0, _acmA->RegisterReceiveCodec(codecInst_B));
99 #ifdef WEBRTC_DTMF_DETECTION
100   _dtmfDetectorA = new(DTMFDetector);
101   EXPECT_GT(_acmA->RegisterIncomingMessagesCallback(_dtmfDetectorA, ACMUSA),
102             -1);
103 #endif
104   //--- Set ref-A codecs
105   EXPECT_EQ(0, _acmRefA->RegisterSendCodec(codecInst_A));
106   EXPECT_EQ(0, _acmRefA->RegisterReceiveCodec(codecInst_B));
107
108   //--- Set B codecs
109   EXPECT_EQ(0, _acmB->RegisterSendCodec(codecInst_B));
110   EXPECT_EQ(0, _acmB->RegisterReceiveCodec(codecInst_A));
111 #ifdef WEBRTC_DTMF_DETECTION
112   _dtmfDetectorB = new(DTMFDetector);
113   EXPECT_GT(_acmB->RegisterIncomingMessagesCallback(_dtmfDetectorB, ACMUSA),
114             -1);
115 #endif
116
117   //--- Set ref-B codecs
118   EXPECT_EQ(0, _acmRefB->RegisterSendCodec(codecInst_B));
119   EXPECT_EQ(0, _acmRefB->RegisterReceiveCodec(codecInst_A));
120
121   uint16_t frequencyHz;
122
123   //--- Input A
124   std::string in_file_name = webrtc::test::ResourcePath(
125       "audio_coding/testfile32kHz", "pcm");
126   frequencyHz = 32000;
127   printf("Enter input file at side A [%s]: ", in_file_name.c_str());
128   PCMFile::ChooseFile(&in_file_name, 499, &frequencyHz);
129   _inFileA.Open(in_file_name, frequencyHz, "rb");
130
131   //--- Output A
132   std::string out_file_a = webrtc::test::OutputPath() + "outA.pcm";
133   printf("Output file at side A: %s\n", out_file_a.c_str());
134   printf("Sampling frequency (in Hz) of the above file: %u\n", frequencyHz);
135   _outFileA.Open(out_file_a, frequencyHz, "wb");
136   std::string ref_file_name = webrtc::test::OutputPath() + "ref_outA.pcm";
137   _outFileRefA.Open(ref_file_name, frequencyHz, "wb");
138
139   //--- Input B
140   in_file_name = webrtc::test::ResourcePath("audio_coding/testfile32kHz",
141                                             "pcm");
142   frequencyHz = 32000;
143   printf("\n\nEnter input file at side B [%s]: ", in_file_name.c_str());
144   PCMFile::ChooseFile(&in_file_name, 499, &frequencyHz);
145   _inFileB.Open(in_file_name, frequencyHz, "rb");
146
147   //--- Output B
148   std::string out_file_b = webrtc::test::OutputPath() + "outB.pcm";
149   printf("Output file at side B: %s\n", out_file_b.c_str());
150   printf("Sampling frequency (in Hz) of the above file: %u\n", frequencyHz);
151   _outFileB.Open(out_file_b, frequencyHz, "wb");
152   ref_file_name = webrtc::test::OutputPath() + "ref_outB.pcm";
153   _outFileRefB.Open(ref_file_name, frequencyHz, "wb");
154
155   //--- Set A-to-B channel
156   _channel_A2B = new Channel;
157   _acmA->RegisterTransportCallback(_channel_A2B);
158   _channel_A2B->RegisterReceiverACM(_acmB.get());
159   //--- Do the same for the reference
160   _channelRef_A2B = new Channel;
161   _acmRefA->RegisterTransportCallback(_channelRef_A2B);
162   _channelRef_A2B->RegisterReceiverACM(_acmRefB.get());
163
164   //--- Set B-to-A channel
165   _channel_B2A = new Channel;
166   _acmB->RegisterTransportCallback(_channel_B2A);
167   _channel_B2A->RegisterReceiverACM(_acmA.get());
168   //--- Do the same for reference
169   _channelRef_B2A = new Channel;
170   _acmRefB->RegisterTransportCallback(_channelRef_B2A);
171   _channelRef_B2A->RegisterReceiverACM(_acmRefA.get());
172
173   // The clicks will be more obvious when we
174   // are in FAX mode.
175   EXPECT_EQ(_acmB->SetPlayoutMode(fax), 0);
176   EXPECT_EQ(_acmRefB->SetPlayoutMode(fax), 0);
177 }
178
179 void TwoWayCommunication::SetUpAutotest() {
180   CodecInst codecInst_A;
181   CodecInst codecInst_B;
182   CodecInst dummyCodec;
183
184   EXPECT_EQ(0, _acmA->Codec("ISAC", &codecInst_A, 16000, 1));
185   EXPECT_EQ(0, _acmB->Codec("L16", &codecInst_B, 8000, 1));
186   EXPECT_EQ(0, _acmA->Codec(6, &dummyCodec));
187
188   //--- Set A codecs
189   EXPECT_EQ(0, _acmA->RegisterSendCodec(codecInst_A));
190   EXPECT_EQ(0, _acmA->RegisterReceiveCodec(codecInst_B));
191 #ifdef WEBRTC_DTMF_DETECTION
192   _dtmfDetectorA = new(DTMFDetector);
193   EXPECT_EQ(0, _acmA->RegisterIncomingMessagesCallback(_dtmfDetectorA, ACMUSA));
194 #endif
195
196   //--- Set ref-A codecs
197   EXPECT_GT(_acmRefA->RegisterSendCodec(codecInst_A), -1);
198   EXPECT_GT(_acmRefA->RegisterReceiveCodec(codecInst_B), -1);
199
200   //--- Set B codecs
201   EXPECT_GT(_acmB->RegisterSendCodec(codecInst_B), -1);
202   EXPECT_GT(_acmB->RegisterReceiveCodec(codecInst_A), -1);
203 #ifdef WEBRTC_DTMF_DETECTION
204   _dtmfDetectorB = new(DTMFDetector);
205   EXPECT_EQ(0, _acmB->RegisterIncomingMessagesCallback(_dtmfDetectorB, ACMUSA));
206 #endif
207
208   //--- Set ref-B codecs
209   EXPECT_EQ(0, _acmRefB->RegisterSendCodec(codecInst_B));
210   EXPECT_EQ(0, _acmRefB->RegisterReceiveCodec(codecInst_A));
211
212   uint16_t frequencyHz;
213
214   //--- Input A and B
215   std::string in_file_name = webrtc::test::ResourcePath(
216       "audio_coding/testfile32kHz", "pcm");
217   frequencyHz = 16000;
218   _inFileA.Open(in_file_name, frequencyHz, "rb");
219   _inFileB.Open(in_file_name, frequencyHz, "rb");
220
221   //--- Output A
222   std::string output_file_a = webrtc::test::OutputPath() + "outAutotestA.pcm";
223   frequencyHz = 16000;
224   _outFileA.Open(output_file_a, frequencyHz, "wb");
225   std::string output_ref_file_a = webrtc::test::OutputPath()
226       + "ref_outAutotestA.pcm";
227   _outFileRefA.Open(output_ref_file_a, frequencyHz, "wb");
228
229   //--- Output B
230   std::string output_file_b = webrtc::test::OutputPath() + "outAutotestB.pcm";
231   frequencyHz = 16000;
232   _outFileB.Open(output_file_b, frequencyHz, "wb");
233   std::string output_ref_file_b = webrtc::test::OutputPath()
234       + "ref_outAutotestB.pcm";
235   _outFileRefB.Open(output_ref_file_b, frequencyHz, "wb");
236
237   //--- Set A-to-B channel
238   _channel_A2B = new Channel;
239   _acmA->RegisterTransportCallback(_channel_A2B);
240   _channel_A2B->RegisterReceiverACM(_acmB.get());
241   //--- Do the same for the reference
242   _channelRef_A2B = new Channel;
243   _acmRefA->RegisterTransportCallback(_channelRef_A2B);
244   _channelRef_A2B->RegisterReceiverACM(_acmRefB.get());
245
246   //--- Set B-to-A channel
247   _channel_B2A = new Channel;
248   _acmB->RegisterTransportCallback(_channel_B2A);
249   _channel_B2A->RegisterReceiverACM(_acmA.get());
250   //--- Do the same for reference
251   _channelRef_B2A = new Channel;
252   _acmRefB->RegisterTransportCallback(_channelRef_B2A);
253   _channelRef_B2A->RegisterReceiverACM(_acmRefA.get());
254
255   // The clicks will be more obvious when we
256   // are in FAX mode.
257   EXPECT_EQ(0, _acmB->SetPlayoutMode(fax));
258   EXPECT_EQ(0, _acmRefB->SetPlayoutMode(fax));
259 }
260
261 void TwoWayCommunication::Perform() {
262   if (_testMode == 0) {
263     SetUpAutotest();
264   } else {
265     SetUp();
266   }
267   unsigned int msecPassed = 0;
268   unsigned int secPassed = 0;
269
270   int32_t outFreqHzA = _outFileA.SamplingFrequency();
271   int32_t outFreqHzB = _outFileB.SamplingFrequency();
272
273   AudioFrame audioFrame;
274
275   CodecInst codecInst_B;
276   CodecInst dummy;
277
278   EXPECT_EQ(0, _acmB->SendCodec(&codecInst_B));
279
280   // In the following loop we tests that the code can handle misuse of the APIs.
281   // In the middle of a session with data flowing between two sides, called A
282   // and B, APIs will be called, like ResetEncoder(), and the code should
283   // continue to run, and be able to recover.
284   bool expect_error_add = false;
285   bool expect_error_process = false;
286   while (!_inFileA.EndOfFile() && !_inFileB.EndOfFile()) {
287     msecPassed += 10;
288     EXPECT_GT(_inFileA.Read10MsData(audioFrame), 0);
289     EXPECT_EQ(0, _acmA->Add10MsData(audioFrame));
290     EXPECT_EQ(0, _acmRefA->Add10MsData(audioFrame));
291
292     EXPECT_GT(_inFileB.Read10MsData(audioFrame), 0);
293
294     // Expect call to pass except for the time when no send codec is registered.
295     if (!expect_error_add) {
296       EXPECT_EQ(0, _acmB->Add10MsData(audioFrame));
297     } else {
298       EXPECT_EQ(-1, _acmB->Add10MsData(audioFrame));
299     }
300     // Expect to pass except for the time when there either is no send codec
301     // registered, or no receive codec.
302     if (!expect_error_process) {
303       EXPECT_GT(_acmB->Process(), -1);
304     } else {
305       EXPECT_EQ(_acmB->Process(), -1);
306     }
307     EXPECT_EQ(0, _acmRefB->Add10MsData(audioFrame));
308     EXPECT_GT(_acmA->Process(), -1);
309     EXPECT_GT(_acmRefA->Process(), -1);
310     EXPECT_GT(_acmRefB->Process(), -1);
311     EXPECT_EQ(0, _acmA->PlayoutData10Ms(outFreqHzA, &audioFrame));
312     _outFileA.Write10MsData(audioFrame);
313     EXPECT_EQ(0, _acmRefA->PlayoutData10Ms(outFreqHzA, &audioFrame));
314     _outFileRefA.Write10MsData(audioFrame);
315     EXPECT_EQ(0, _acmB->PlayoutData10Ms(outFreqHzB, &audioFrame));
316     _outFileB.Write10MsData(audioFrame);
317     EXPECT_EQ(0, _acmRefB->PlayoutData10Ms(outFreqHzB, &audioFrame));
318     _outFileRefB.Write10MsData(audioFrame);
319
320     // Update time counters each time a second of data has passed.
321     if (msecPassed >= 1000) {
322       msecPassed = 0;
323       secPassed++;
324     }
325     // Call RestEncoder for ACM on side A, and InitializeSender for ACM on
326     // side B.
327     if (((secPassed % 5) == 4) && (msecPassed == 0)) {
328       EXPECT_EQ(0, _acmA->ResetEncoder());
329       EXPECT_EQ(0, _acmB->InitializeSender());
330       expect_error_add = true;
331       expect_error_process = true;
332     }
333     // Re-register send codec on side B.
334     if (((secPassed % 5) == 4) && (msecPassed >= 990)) {
335       EXPECT_EQ(0, _acmB->RegisterSendCodec(codecInst_B));
336       EXPECT_EQ(0, _acmB->SendCodec(&dummy));
337       expect_error_add = false;
338       expect_error_process = false;
339     }
340     // Reset decoder on side B, and initialize receiver on side A.
341     if (((secPassed % 7) == 6) && (msecPassed == 0)) {
342       EXPECT_EQ(0, _acmB->ResetDecoder());
343       EXPECT_EQ(0, _acmA->InitializeReceiver());
344     }
345     // Re-register codec on side A.
346     if (((secPassed % 7) == 6) && (msecPassed >= 990)) {
347       EXPECT_EQ(0, _acmA->RegisterReceiveCodec(codecInst_B));
348     }
349   }
350 }
351
352 }  // namespace webrtc