blob: 3ca7fd217d57592802f6f6ec38a940cf9676f152 [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
kjellander@webrtc.org3c0aae12014-09-04 09:55:40 +000023#include "testing/gtest/include/gtest/gtest.h"
turaj@webrtc.org6ea3d1c2013-10-02 21:44:33 +000024#include "webrtc/engine_configurations.h"
turaj@webrtc.org6ea3d1c2013-10-02 21:44:33 +000025#include "webrtc/common_types.h"
kjellander3e6db232015-11-26 04:44:54 -080026#include "webrtc/modules/audio_coding/test/PCMFile.h"
27#include "webrtc/modules/audio_coding/test/utility.h"
Henrik Kjellander98f53512015-10-28 18:17:40 +010028#include "webrtc/system_wrappers/include/trace.h"
pbos@webrtc.org2ab209e2013-08-09 08:49:48 +000029#include "webrtc/test/testsupport/fileutils.h"
niklase@google.com470e71d2011-07-07 08:21:25 +000030
tina.legrand@webrtc.org554ae1a2011-12-16 10:09:04 +000031namespace webrtc {
niklase@google.com470e71d2011-07-07 08:21:25 +000032
33#define MAX_FILE_NAME_LENGTH_BYTE 500
34
henrik.lundin@webrtc.orgadaf8092014-04-17 08:29:10 +000035TwoWayCommunication::TwoWayCommunication(int testMode)
36 : _acmA(AudioCodingModule::Create(1)),
henrik.lundin@webrtc.orgadaf8092014-04-17 08:29:10 +000037 _acmRefA(AudioCodingModule::Create(3)),
henrik.lundin1bd0e032015-09-28 06:12:17 -070038 _testMode(testMode) {
39 AudioCodingModule::Config config;
40 // The clicks will be more obvious in FAX mode. TODO(henrik.lundin) Really?
41 config.neteq_config.playout_mode = kPlayoutFax;
42 config.id = 2;
43 _acmB.reset(AudioCodingModule::Create(config));
44 config.id = 4;
45 _acmRefB.reset(AudioCodingModule::Create(config));
46}
niklase@google.com470e71d2011-07-07 08:21:25 +000047
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000048TwoWayCommunication::~TwoWayCommunication() {
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000049 delete _channel_A2B;
50 delete _channel_B2A;
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000051 delete _channelRef_A2B;
52 delete _channelRef_B2A;
niklase@google.com470e71d2011-07-07 08:21:25 +000053#ifdef WEBRTC_DTMF_DETECTION
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +000054 if (_dtmfDetectorA != NULL) {
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000055 delete _dtmfDetectorA;
56 }
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +000057 if (_dtmfDetectorB != NULL) {
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000058 delete _dtmfDetectorB;
59 }
niklase@google.com470e71d2011-07-07 08:21:25 +000060#endif
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000061 _inFileA.Close();
62 _inFileB.Close();
63 _outFileA.Close();
64 _outFileB.Close();
65 _outFileRefA.Close();
66 _outFileRefB.Close();
niklase@google.com470e71d2011-07-07 08:21:25 +000067}
68
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +000069void TwoWayCommunication::ChooseCodec(uint8_t* codecID_A,
70 uint8_t* codecID_B) {
kwiberg37478382016-02-14 20:40:57 -080071 std::unique_ptr<AudioCodingModule> tmpACM(AudioCodingModule::Create(0));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000072 uint8_t noCodec = tmpACM->NumberOfCodecs();
73 CodecInst codecInst;
74 printf("List of Supported Codecs\n");
75 printf("========================\n");
76 for (uint8_t codecCntr = 0; codecCntr < noCodec; codecCntr++) {
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +000077 EXPECT_EQ(tmpACM->Codec(codecCntr, &codecInst), 0);
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000078 printf("%d- %s\n", codecCntr, codecInst.plname);
79 }
80 printf("\nChoose a send codec for side A [0]: ");
81 char myStr[15] = "";
82 EXPECT_TRUE(fgets(myStr, 10, stdin) != NULL);
83 *codecID_A = (uint8_t) atoi(myStr);
niklase@google.com470e71d2011-07-07 08:21:25 +000084
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000085 printf("\nChoose a send codec for side B [0]: ");
86 EXPECT_TRUE(fgets(myStr, 10, stdin) != NULL);
87 *codecID_B = (uint8_t) atoi(myStr);
niklase@google.com470e71d2011-07-07 08:21:25 +000088
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000089 printf("\n");
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000090}
niklase@google.com470e71d2011-07-07 08:21:25 +000091
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +000092void TwoWayCommunication::SetUp() {
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000093 uint8_t codecID_A;
94 uint8_t codecID_B;
95
96 ChooseCodec(&codecID_A, &codecID_B);
97 CodecInst codecInst_A;
98 CodecInst codecInst_B;
99 CodecInst dummyCodec;
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000100 EXPECT_EQ(0, _acmA->Codec(codecID_A, &codecInst_A));
101 EXPECT_EQ(0, _acmB->Codec(codecID_B, &codecInst_B));
102 EXPECT_EQ(0, _acmA->Codec(6, &dummyCodec));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000103
104 //--- Set A codecs
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000105 EXPECT_EQ(0, _acmA->RegisterSendCodec(codecInst_A));
106 EXPECT_EQ(0, _acmA->RegisterReceiveCodec(codecInst_B));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000107 //--- Set ref-A codecs
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000108 EXPECT_EQ(0, _acmRefA->RegisterSendCodec(codecInst_A));
109 EXPECT_EQ(0, _acmRefA->RegisterReceiveCodec(codecInst_B));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000110
111 //--- Set B codecs
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000112 EXPECT_EQ(0, _acmB->RegisterSendCodec(codecInst_B));
113 EXPECT_EQ(0, _acmB->RegisterReceiveCodec(codecInst_A));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000114
115 //--- Set ref-B codecs
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000116 EXPECT_EQ(0, _acmRefB->RegisterSendCodec(codecInst_B));
117 EXPECT_EQ(0, _acmRefB->RegisterReceiveCodec(codecInst_A));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000118
119 uint16_t frequencyHz;
120
121 //--- Input A
122 std::string in_file_name = webrtc::test::ResourcePath(
123 "audio_coding/testfile32kHz", "pcm");
124 frequencyHz = 32000;
125 printf("Enter input file at side A [%s]: ", in_file_name.c_str());
126 PCMFile::ChooseFile(&in_file_name, 499, &frequencyHz);
127 _inFileA.Open(in_file_name, frequencyHz, "rb");
128
129 //--- Output A
130 std::string out_file_a = webrtc::test::OutputPath() + "outA.pcm";
131 printf("Output file at side A: %s\n", out_file_a.c_str());
132 printf("Sampling frequency (in Hz) of the above file: %u\n", frequencyHz);
133 _outFileA.Open(out_file_a, frequencyHz, "wb");
134 std::string ref_file_name = webrtc::test::OutputPath() + "ref_outA.pcm";
135 _outFileRefA.Open(ref_file_name, frequencyHz, "wb");
136
137 //--- Input B
138 in_file_name = webrtc::test::ResourcePath("audio_coding/testfile32kHz",
139 "pcm");
140 frequencyHz = 32000;
141 printf("\n\nEnter input file at side B [%s]: ", in_file_name.c_str());
142 PCMFile::ChooseFile(&in_file_name, 499, &frequencyHz);
143 _inFileB.Open(in_file_name, frequencyHz, "rb");
144
145 //--- Output B
146 std::string out_file_b = webrtc::test::OutputPath() + "outB.pcm";
147 printf("Output file at side B: %s\n", out_file_b.c_str());
148 printf("Sampling frequency (in Hz) of the above file: %u\n", frequencyHz);
149 _outFileB.Open(out_file_b, frequencyHz, "wb");
150 ref_file_name = webrtc::test::OutputPath() + "ref_outB.pcm";
151 _outFileRefB.Open(ref_file_name, frequencyHz, "wb");
152
153 //--- Set A-to-B channel
154 _channel_A2B = new Channel;
155 _acmA->RegisterTransportCallback(_channel_A2B);
andrew@webrtc.org89df0922013-09-12 01:27:43 +0000156 _channel_A2B->RegisterReceiverACM(_acmB.get());
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000157 //--- Do the same for the reference
158 _channelRef_A2B = new Channel;
159 _acmRefA->RegisterTransportCallback(_channelRef_A2B);
andrew@webrtc.org89df0922013-09-12 01:27:43 +0000160 _channelRef_A2B->RegisterReceiverACM(_acmRefB.get());
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000161
162 //--- Set B-to-A channel
163 _channel_B2A = new Channel;
164 _acmB->RegisterTransportCallback(_channel_B2A);
andrew@webrtc.org89df0922013-09-12 01:27:43 +0000165 _channel_B2A->RegisterReceiverACM(_acmA.get());
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000166 //--- Do the same for reference
167 _channelRef_B2A = new Channel;
168 _acmRefB->RegisterTransportCallback(_channelRef_B2A);
andrew@webrtc.org89df0922013-09-12 01:27:43 +0000169 _channelRef_B2A->RegisterReceiverACM(_acmRefA.get());
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000170}
171
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000172void TwoWayCommunication::SetUpAutotest() {
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000173 CodecInst codecInst_A;
174 CodecInst codecInst_B;
175 CodecInst dummyCodec;
176
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000177 EXPECT_EQ(0, _acmA->Codec("ISAC", &codecInst_A, 16000, 1));
178 EXPECT_EQ(0, _acmB->Codec("L16", &codecInst_B, 8000, 1));
179 EXPECT_EQ(0, _acmA->Codec(6, &dummyCodec));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000180
181 //--- Set A codecs
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000182 EXPECT_EQ(0, _acmA->RegisterSendCodec(codecInst_A));
183 EXPECT_EQ(0, _acmA->RegisterReceiveCodec(codecInst_B));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000184
185 //--- Set ref-A codecs
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000186 EXPECT_GT(_acmRefA->RegisterSendCodec(codecInst_A), -1);
187 EXPECT_GT(_acmRefA->RegisterReceiveCodec(codecInst_B), -1);
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);
191 EXPECT_GT(_acmB->RegisterReceiveCodec(codecInst_A), -1);
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000192
193 //--- Set ref-B codecs
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000194 EXPECT_EQ(0, _acmRefB->RegisterSendCodec(codecInst_B));
195 EXPECT_EQ(0, _acmRefB->RegisterReceiveCodec(codecInst_A));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000196
197 uint16_t frequencyHz;
198
199 //--- Input A and B
200 std::string in_file_name = webrtc::test::ResourcePath(
201 "audio_coding/testfile32kHz", "pcm");
202 frequencyHz = 16000;
203 _inFileA.Open(in_file_name, frequencyHz, "rb");
204 _inFileB.Open(in_file_name, frequencyHz, "rb");
205
206 //--- Output A
207 std::string output_file_a = webrtc::test::OutputPath() + "outAutotestA.pcm";
208 frequencyHz = 16000;
209 _outFileA.Open(output_file_a, frequencyHz, "wb");
210 std::string output_ref_file_a = webrtc::test::OutputPath()
211 + "ref_outAutotestA.pcm";
212 _outFileRefA.Open(output_ref_file_a, frequencyHz, "wb");
213
214 //--- Output B
215 std::string output_file_b = webrtc::test::OutputPath() + "outAutotestB.pcm";
216 frequencyHz = 16000;
217 _outFileB.Open(output_file_b, frequencyHz, "wb");
218 std::string output_ref_file_b = webrtc::test::OutputPath()
219 + "ref_outAutotestB.pcm";
220 _outFileRefB.Open(output_ref_file_b, frequencyHz, "wb");
221
222 //--- Set A-to-B channel
223 _channel_A2B = new Channel;
224 _acmA->RegisterTransportCallback(_channel_A2B);
andrew@webrtc.org89df0922013-09-12 01:27:43 +0000225 _channel_A2B->RegisterReceiverACM(_acmB.get());
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000226 //--- Do the same for the reference
227 _channelRef_A2B = new Channel;
228 _acmRefA->RegisterTransportCallback(_channelRef_A2B);
andrew@webrtc.org89df0922013-09-12 01:27:43 +0000229 _channelRef_A2B->RegisterReceiverACM(_acmRefB.get());
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000230
231 //--- Set B-to-A channel
232 _channel_B2A = new Channel;
233 _acmB->RegisterTransportCallback(_channel_B2A);
andrew@webrtc.org89df0922013-09-12 01:27:43 +0000234 _channel_B2A->RegisterReceiverACM(_acmA.get());
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000235 //--- Do the same for reference
236 _channelRef_B2A = new Channel;
237 _acmRefB->RegisterTransportCallback(_channelRef_B2A);
andrew@webrtc.org89df0922013-09-12 01:27:43 +0000238 _channelRef_B2A->RegisterReceiverACM(_acmRefA.get());
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000239}
240
241void TwoWayCommunication::Perform() {
242 if (_testMode == 0) {
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000243 SetUpAutotest();
244 } else {
245 SetUp();
246 }
247 unsigned int msecPassed = 0;
248 unsigned int secPassed = 0;
249
250 int32_t outFreqHzA = _outFileA.SamplingFrequency();
251 int32_t outFreqHzB = _outFileB.SamplingFrequency();
252
253 AudioFrame audioFrame;
254
kwiberg1fd4a4a2015-11-03 11:20:50 -0800255 auto codecInst_B = _acmB->SendCodec();
256 ASSERT_TRUE(codecInst_B);
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000257
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000258 // In the following loop we tests that the code can handle misuse of the APIs.
259 // In the middle of a session with data flowing between two sides, called A
Karl Wibergdd00f112015-08-25 09:37:04 +0200260 // and B, APIs will be called, and the code should continue to run, and be
261 // able to recover.
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000262 while (!_inFileA.EndOfFile() && !_inFileB.EndOfFile()) {
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000263 msecPassed += 10;
264 EXPECT_GT(_inFileA.Read10MsData(audioFrame), 0);
henrik.lundin@webrtc.orgf56c1622015-03-02 12:29:30 +0000265 EXPECT_GE(_acmA->Add10MsData(audioFrame), 0);
266 EXPECT_GE(_acmRefA->Add10MsData(audioFrame), 0);
niklase@google.com470e71d2011-07-07 08:21:25 +0000267
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000268 EXPECT_GT(_inFileB.Read10MsData(audioFrame), 0);
niklase@google.com470e71d2011-07-07 08:21:25 +0000269
Henrik Lundin45c64492015-03-30 19:00:44 +0200270 EXPECT_GE(_acmB->Add10MsData(audioFrame), 0);
henrik.lundin@webrtc.orgf56c1622015-03-02 12:29:30 +0000271 EXPECT_GE(_acmRefB->Add10MsData(audioFrame), 0);
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000272 EXPECT_EQ(0, _acmA->PlayoutData10Ms(outFreqHzA, &audioFrame));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000273 _outFileA.Write10MsData(audioFrame);
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000274 EXPECT_EQ(0, _acmRefA->PlayoutData10Ms(outFreqHzA, &audioFrame));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000275 _outFileRefA.Write10MsData(audioFrame);
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000276 EXPECT_EQ(0, _acmB->PlayoutData10Ms(outFreqHzB, &audioFrame));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000277 _outFileB.Write10MsData(audioFrame);
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000278 EXPECT_EQ(0, _acmRefB->PlayoutData10Ms(outFreqHzB, &audioFrame));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000279 _outFileRefB.Write10MsData(audioFrame);
niklase@google.com470e71d2011-07-07 08:21:25 +0000280
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000281 // Update time counters each time a second of data has passed.
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000282 if (msecPassed >= 1000) {
283 msecPassed = 0;
284 secPassed++;
285 }
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000286 // Re-register send codec on side B.
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000287 if (((secPassed % 5) == 4) && (msecPassed >= 990)) {
kwiberg1fd4a4a2015-11-03 11:20:50 -0800288 EXPECT_EQ(0, _acmB->RegisterSendCodec(*codecInst_B));
289 EXPECT_TRUE(_acmB->SendCodec());
niklase@google.com470e71d2011-07-07 08:21:25 +0000290 }
Karl Wibergdd00f112015-08-25 09:37:04 +0200291 // Initialize receiver on side A.
292 if (((secPassed % 7) == 6) && (msecPassed == 0))
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000293 EXPECT_EQ(0, _acmA->InitializeReceiver());
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000294 // Re-register codec on side A.
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000295 if (((secPassed % 7) == 6) && (msecPassed >= 990)) {
kwiberg1fd4a4a2015-11-03 11:20:50 -0800296 EXPECT_EQ(0, _acmA->RegisterReceiveCodec(*codecInst_B));
niklase@google.com470e71d2011-07-07 08:21:25 +0000297 }
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000298 }
niklase@google.com470e71d2011-07-07 08:21:25 +0000299}
300
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000301} // namespace webrtc