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/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 353