• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
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 "testing/gtest/include/gtest/gtest.h"
22 #include "webrtc/engine_configurations.h"
23 #include "webrtc/common_types.h"
24 #include "webrtc/modules/audio_coding/test/PCMFile.h"
25 #include "webrtc/modules/audio_coding/test/utility.h"
26 #include "webrtc/system_wrappers/include/trace.h"
27 #include "webrtc/test/testsupport/fileutils.h"
28 
29 namespace webrtc {
30 
31 #define MAX_FILE_NAME_LENGTH_BYTE 500
32 
TwoWayCommunication(int testMode)33 TwoWayCommunication::TwoWayCommunication(int testMode)
34     : _acmA(AudioCodingModule::Create(1)),
35       _acmRefA(AudioCodingModule::Create(3)),
36       _testMode(testMode) {
37   AudioCodingModule::Config config;
38   // The clicks will be more obvious in FAX mode. TODO(henrik.lundin) Really?
39   config.neteq_config.playout_mode = kPlayoutFax;
40   config.id = 2;
41   _acmB.reset(AudioCodingModule::Create(config));
42   config.id = 4;
43   _acmRefB.reset(AudioCodingModule::Create(config));
44 }
45 
~TwoWayCommunication()46 TwoWayCommunication::~TwoWayCommunication() {
47   delete _channel_A2B;
48   delete _channel_B2A;
49   delete _channelRef_A2B;
50   delete _channelRef_B2A;
51 #ifdef WEBRTC_DTMF_DETECTION
52   if (_dtmfDetectorA != NULL) {
53     delete _dtmfDetectorA;
54   }
55   if (_dtmfDetectorB != NULL) {
56     delete _dtmfDetectorB;
57   }
58 #endif
59   _inFileA.Close();
60   _inFileB.Close();
61   _outFileA.Close();
62   _outFileB.Close();
63   _outFileRefA.Close();
64   _outFileRefB.Close();
65 }
66 
ChooseCodec(uint8_t * codecID_A,uint8_t * codecID_B)67 void TwoWayCommunication::ChooseCodec(uint8_t* codecID_A,
68                                       uint8_t* codecID_B) {
69   rtc::scoped_ptr<AudioCodingModule> tmpACM(AudioCodingModule::Create(0));
70   uint8_t noCodec = tmpACM->NumberOfCodecs();
71   CodecInst codecInst;
72   printf("List of Supported Codecs\n");
73   printf("========================\n");
74   for (uint8_t codecCntr = 0; codecCntr < noCodec; codecCntr++) {
75     EXPECT_EQ(tmpACM->Codec(codecCntr, &codecInst), 0);
76     printf("%d- %s\n", codecCntr, codecInst.plname);
77   }
78   printf("\nChoose a send codec for side A [0]: ");
79   char myStr[15] = "";
80   EXPECT_TRUE(fgets(myStr, 10, stdin) != NULL);
81   *codecID_A = (uint8_t) atoi(myStr);
82 
83   printf("\nChoose a send codec for side B [0]: ");
84   EXPECT_TRUE(fgets(myStr, 10, stdin) != NULL);
85   *codecID_B = (uint8_t) atoi(myStr);
86 
87   printf("\n");
88 }
89 
SetUp()90 void TwoWayCommunication::SetUp() {
91   uint8_t codecID_A;
92   uint8_t codecID_B;
93 
94   ChooseCodec(&codecID_A, &codecID_B);
95   CodecInst codecInst_A;
96   CodecInst codecInst_B;
97   CodecInst dummyCodec;
98   EXPECT_EQ(0, _acmA->Codec(codecID_A, &codecInst_A));
99   EXPECT_EQ(0, _acmB->Codec(codecID_B, &codecInst_B));
100   EXPECT_EQ(0, _acmA->Codec(6, &dummyCodec));
101 
102   //--- Set A codecs
103   EXPECT_EQ(0, _acmA->RegisterSendCodec(codecInst_A));
104   EXPECT_EQ(0, _acmA->RegisterReceiveCodec(codecInst_B));
105   //--- Set ref-A codecs
106   EXPECT_EQ(0, _acmRefA->RegisterSendCodec(codecInst_A));
107   EXPECT_EQ(0, _acmRefA->RegisterReceiveCodec(codecInst_B));
108 
109   //--- Set B codecs
110   EXPECT_EQ(0, _acmB->RegisterSendCodec(codecInst_B));
111   EXPECT_EQ(0, _acmB->RegisterReceiveCodec(codecInst_A));
112 
113   //--- Set ref-B codecs
114   EXPECT_EQ(0, _acmRefB->RegisterSendCodec(codecInst_B));
115   EXPECT_EQ(0, _acmRefB->RegisterReceiveCodec(codecInst_A));
116 
117   uint16_t frequencyHz;
118 
119   //--- Input A
120   std::string in_file_name = webrtc::test::ResourcePath(
121       "audio_coding/testfile32kHz", "pcm");
122   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
136   in_file_name = webrtc::test::ResourcePath("audio_coding/testfile32kHz",
137                                             "pcm");
138   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);
154   _channel_A2B->RegisterReceiverACM(_acmB.get());
155   //--- Do the same for the reference
156   _channelRef_A2B = new Channel;
157   _acmRefA->RegisterTransportCallback(_channelRef_A2B);
158   _channelRef_A2B->RegisterReceiverACM(_acmRefB.get());
159 
160   //--- Set B-to-A channel
161   _channel_B2A = new Channel;
162   _acmB->RegisterTransportCallback(_channel_B2A);
163   _channel_B2A->RegisterReceiverACM(_acmA.get());
164   //--- Do the same for reference
165   _channelRef_B2A = new Channel;
166   _acmRefB->RegisterTransportCallback(_channelRef_B2A);
167   _channelRef_B2A->RegisterReceiverACM(_acmRefA.get());
168 }
169 
SetUpAutotest()170 void TwoWayCommunication::SetUpAutotest() {
171   CodecInst codecInst_A;
172   CodecInst codecInst_B;
173   CodecInst dummyCodec;
174 
175   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));
178 
179   //--- Set A codecs
180   EXPECT_EQ(0, _acmA->RegisterSendCodec(codecInst_A));
181   EXPECT_EQ(0, _acmA->RegisterReceiveCodec(codecInst_B));
182 
183   //--- Set ref-A codecs
184   EXPECT_GT(_acmRefA->RegisterSendCodec(codecInst_A), -1);
185   EXPECT_GT(_acmRefA->RegisterReceiveCodec(codecInst_B), -1);
186 
187   //--- Set B codecs
188   EXPECT_GT(_acmB->RegisterSendCodec(codecInst_B), -1);
189   EXPECT_GT(_acmB->RegisterReceiveCodec(codecInst_A), -1);
190 
191   //--- Set ref-B codecs
192   EXPECT_EQ(0, _acmRefB->RegisterSendCodec(codecInst_B));
193   EXPECT_EQ(0, _acmRefB->RegisterReceiveCodec(codecInst_A));
194 
195   uint16_t frequencyHz;
196 
197   //--- Input A and B
198   std::string in_file_name = webrtc::test::ResourcePath(
199       "audio_coding/testfile32kHz", "pcm");
200   frequencyHz = 16000;
201   _inFileA.Open(in_file_name, frequencyHz, "rb");
202   _inFileB.Open(in_file_name, frequencyHz, "rb");
203 
204   //--- Output A
205   std::string output_file_a = webrtc::test::OutputPath() + "outAutotestA.pcm";
206   frequencyHz = 16000;
207   _outFileA.Open(output_file_a, frequencyHz, "wb");
208   std::string output_ref_file_a = webrtc::test::OutputPath()
209       + "ref_outAutotestA.pcm";
210   _outFileRefA.Open(output_ref_file_a, frequencyHz, "wb");
211 
212   //--- Output B
213   std::string output_file_b = webrtc::test::OutputPath() + "outAutotestB.pcm";
214   frequencyHz = 16000;
215   _outFileB.Open(output_file_b, frequencyHz, "wb");
216   std::string output_ref_file_b = webrtc::test::OutputPath()
217       + "ref_outAutotestB.pcm";
218   _outFileRefB.Open(output_ref_file_b, frequencyHz, "wb");
219 
220   //--- Set A-to-B channel
221   _channel_A2B = new Channel;
222   _acmA->RegisterTransportCallback(_channel_A2B);
223   _channel_A2B->RegisterReceiverACM(_acmB.get());
224   //--- Do the same for the reference
225   _channelRef_A2B = new Channel;
226   _acmRefA->RegisterTransportCallback(_channelRef_A2B);
227   _channelRef_A2B->RegisterReceiverACM(_acmRefB.get());
228 
229   //--- Set B-to-A channel
230   _channel_B2A = new Channel;
231   _acmB->RegisterTransportCallback(_channel_B2A);
232   _channel_B2A->RegisterReceiverACM(_acmA.get());
233   //--- Do the same for reference
234   _channelRef_B2A = new Channel;
235   _acmRefB->RegisterTransportCallback(_channelRef_B2A);
236   _channelRef_B2A->RegisterReceiverACM(_acmRefA.get());
237 }
238 
Perform()239 void TwoWayCommunication::Perform() {
240   if (_testMode == 0) {
241     SetUpAutotest();
242   } else {
243     SetUp();
244   }
245   unsigned int msecPassed = 0;
246   unsigned int secPassed = 0;
247 
248   int32_t outFreqHzA = _outFileA.SamplingFrequency();
249   int32_t outFreqHzB = _outFileB.SamplingFrequency();
250 
251   AudioFrame audioFrame;
252 
253   auto codecInst_B = _acmB->SendCodec();
254   ASSERT_TRUE(codecInst_B);
255 
256   // In the following loop we tests that the code can handle misuse of the APIs.
257   // In the middle of a session with data flowing between two sides, called A
258   // and B, APIs will be called, and the code should continue to run, and be
259   // able to recover.
260   while (!_inFileA.EndOfFile() && !_inFileB.EndOfFile()) {
261     msecPassed += 10;
262     EXPECT_GT(_inFileA.Read10MsData(audioFrame), 0);
263     EXPECT_GE(_acmA->Add10MsData(audioFrame), 0);
264     EXPECT_GE(_acmRefA->Add10MsData(audioFrame), 0);
265 
266     EXPECT_GT(_inFileB.Read10MsData(audioFrame), 0);
267 
268     EXPECT_GE(_acmB->Add10MsData(audioFrame), 0);
269     EXPECT_GE(_acmRefB->Add10MsData(audioFrame), 0);
270     EXPECT_EQ(0, _acmA->PlayoutData10Ms(outFreqHzA, &audioFrame));
271     _outFileA.Write10MsData(audioFrame);
272     EXPECT_EQ(0, _acmRefA->PlayoutData10Ms(outFreqHzA, &audioFrame));
273     _outFileRefA.Write10MsData(audioFrame);
274     EXPECT_EQ(0, _acmB->PlayoutData10Ms(outFreqHzB, &audioFrame));
275     _outFileB.Write10MsData(audioFrame);
276     EXPECT_EQ(0, _acmRefB->PlayoutData10Ms(outFreqHzB, &audioFrame));
277     _outFileRefB.Write10MsData(audioFrame);
278 
279     // Update time counters each time a second of data has passed.
280     if (msecPassed >= 1000) {
281       msecPassed = 0;
282       secPassed++;
283     }
284     // Re-register send codec on side B.
285     if (((secPassed % 5) == 4) && (msecPassed >= 990)) {
286       EXPECT_EQ(0, _acmB->RegisterSendCodec(*codecInst_B));
287       EXPECT_TRUE(_acmB->SendCodec());
288     }
289     // Initialize receiver on side A.
290     if (((secPassed % 7) == 6) && (msecPassed == 0))
291       EXPECT_EQ(0, _acmA->InitializeReceiver());
292     // Re-register codec on side A.
293     if (((secPassed % 7) == 6) && (msecPassed >= 990)) {
294       EXPECT_EQ(0, _acmA->RegisterReceiveCodec(*codecInst_B));
295     }
296   }
297 }
298 
299 }  // namespace webrtc
300