blob: 93ba4ddde70a0e85ae02e8773360de590fe8ed7c [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"
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
Karl Wiberg3a6b6bd2018-09-26 10:38:45 +020035TwoWayCommunication::TwoWayCommunication()
Karl Wiberg5817d3d2018-04-06 10:06:42 +020036 : _acmA(AudioCodingModule::Create(
37 AudioCodingModule::Config(CreateBuiltinAudioDecoderFactory()))),
38 _acmRefA(AudioCodingModule::Create(
Karl Wiberg3a6b6bd2018-09-26 10:38:45 +020039 AudioCodingModule::Config(CreateBuiltinAudioDecoderFactory()))) {
henrik.lundin1bd0e032015-09-28 06:12:17 -070040 AudioCodingModule::Config config;
Henrik Lundin7687ad52018-07-02 10:14:46 +020041 // The clicks will be more obvious if time-stretching is not allowed.
42 // TODO(henrik.lundin) Really?
43 config.neteq_config.for_test_no_time_stretching = true;
ossue3525782016-05-25 07:37:43 -070044 config.decoder_factory = CreateBuiltinAudioDecoderFactory();
henrik.lundin1bd0e032015-09-28 06:12:17 -070045 _acmB.reset(AudioCodingModule::Create(config));
henrik.lundin1bd0e032015-09-28 06:12:17 -070046 _acmRefB.reset(AudioCodingModule::Create(config));
47}
niklase@google.com470e71d2011-07-07 08:21:25 +000048
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000049TwoWayCommunication::~TwoWayCommunication() {
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000050 delete _channel_A2B;
51 delete _channel_B2A;
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000052 delete _channelRef_A2B;
53 delete _channelRef_B2A;
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000054 _inFileA.Close();
55 _inFileB.Close();
56 _outFileA.Close();
57 _outFileB.Close();
58 _outFileRefA.Close();
59 _outFileRefB.Close();
niklase@google.com470e71d2011-07-07 08:21:25 +000060}
61
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +000062void TwoWayCommunication::SetUpAutotest() {
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000063 CodecInst codecInst_A;
64 CodecInst codecInst_B;
65 CodecInst dummyCodec;
66
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +000067 EXPECT_EQ(0, _acmA->Codec("ISAC", &codecInst_A, 16000, 1));
68 EXPECT_EQ(0, _acmB->Codec("L16", &codecInst_B, 8000, 1));
69 EXPECT_EQ(0, _acmA->Codec(6, &dummyCodec));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000070
71 //--- Set A codecs
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +000072 EXPECT_EQ(0, _acmA->RegisterSendCodec(codecInst_A));
kwibergda2bf4e2016-10-24 13:47:09 -070073 EXPECT_EQ(true, _acmA->RegisterReceiveCodec(codecInst_B.pltype,
74 CodecInstToSdp(codecInst_B)));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000075
76 //--- Set ref-A codecs
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +000077 EXPECT_GT(_acmRefA->RegisterSendCodec(codecInst_A), -1);
kwibergda2bf4e2016-10-24 13:47:09 -070078 EXPECT_EQ(true, _acmRefA->RegisterReceiveCodec(codecInst_B.pltype,
79 CodecInstToSdp(codecInst_B)));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000080
81 //--- Set B codecs
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +000082 EXPECT_GT(_acmB->RegisterSendCodec(codecInst_B), -1);
kwibergda2bf4e2016-10-24 13:47:09 -070083 EXPECT_EQ(true, _acmB->RegisterReceiveCodec(codecInst_A.pltype,
84 CodecInstToSdp(codecInst_A)));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000085
86 //--- Set ref-B codecs
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +000087 EXPECT_EQ(0, _acmRefB->RegisterSendCodec(codecInst_B));
kwibergda2bf4e2016-10-24 13:47:09 -070088 EXPECT_EQ(true, _acmRefB->RegisterReceiveCodec(codecInst_A.pltype,
89 CodecInstToSdp(codecInst_A)));
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000090
91 uint16_t frequencyHz;
92
93 //--- Input A and B
Yves Gerey665174f2018-06-19 15:03:05 +020094 std::string in_file_name =
95 webrtc::test::ResourcePath("audio_coding/testfile32kHz", "pcm");
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +000096 frequencyHz = 16000;
97 _inFileA.Open(in_file_name, frequencyHz, "rb");
98 _inFileB.Open(in_file_name, frequencyHz, "rb");
99
100 //--- Output A
101 std::string output_file_a = webrtc::test::OutputPath() + "outAutotestA.pcm";
102 frequencyHz = 16000;
103 _outFileA.Open(output_file_a, frequencyHz, "wb");
Yves Gerey665174f2018-06-19 15:03:05 +0200104 std::string output_ref_file_a =
105 webrtc::test::OutputPath() + "ref_outAutotestA.pcm";
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000106 _outFileRefA.Open(output_ref_file_a, frequencyHz, "wb");
107
108 //--- Output B
109 std::string output_file_b = webrtc::test::OutputPath() + "outAutotestB.pcm";
110 frequencyHz = 16000;
111 _outFileB.Open(output_file_b, frequencyHz, "wb");
Yves Gerey665174f2018-06-19 15:03:05 +0200112 std::string output_ref_file_b =
113 webrtc::test::OutputPath() + "ref_outAutotestB.pcm";
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000114 _outFileRefB.Open(output_ref_file_b, frequencyHz, "wb");
115
116 //--- Set A-to-B channel
117 _channel_A2B = new Channel;
118 _acmA->RegisterTransportCallback(_channel_A2B);
andrew@webrtc.org89df0922013-09-12 01:27:43 +0000119 _channel_A2B->RegisterReceiverACM(_acmB.get());
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000120 //--- Do the same for the reference
121 _channelRef_A2B = new Channel;
122 _acmRefA->RegisterTransportCallback(_channelRef_A2B);
andrew@webrtc.org89df0922013-09-12 01:27:43 +0000123 _channelRef_A2B->RegisterReceiverACM(_acmRefB.get());
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000124
125 //--- Set B-to-A channel
126 _channel_B2A = new Channel;
127 _acmB->RegisterTransportCallback(_channel_B2A);
andrew@webrtc.org89df0922013-09-12 01:27:43 +0000128 _channel_B2A->RegisterReceiverACM(_acmA.get());
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000129 //--- Do the same for reference
130 _channelRef_B2A = new Channel;
131 _acmRefB->RegisterTransportCallback(_channelRef_B2A);
andrew@webrtc.org89df0922013-09-12 01:27:43 +0000132 _channelRef_B2A->RegisterReceiverACM(_acmRefA.get());
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000133}
134
135void TwoWayCommunication::Perform() {
Karl Wiberg3a6b6bd2018-09-26 10:38:45 +0200136 SetUpAutotest();
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000137 unsigned int msecPassed = 0;
138 unsigned int secPassed = 0;
139
140 int32_t outFreqHzA = _outFileA.SamplingFrequency();
141 int32_t outFreqHzB = _outFileB.SamplingFrequency();
142
143 AudioFrame audioFrame;
144
kwiberg1fd4a4a2015-11-03 11:20:50 -0800145 auto codecInst_B = _acmB->SendCodec();
146 ASSERT_TRUE(codecInst_B);
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000147
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000148 // In the following loop we tests that the code can handle misuse of the APIs.
149 // In the middle of a session with data flowing between two sides, called A
Karl Wibergdd00f112015-08-25 09:37:04 +0200150 // and B, APIs will be called, and the code should continue to run, and be
151 // able to recover.
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000152 while (!_inFileA.EndOfFile() && !_inFileB.EndOfFile()) {
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000153 msecPassed += 10;
154 EXPECT_GT(_inFileA.Read10MsData(audioFrame), 0);
henrik.lundin@webrtc.orgf56c1622015-03-02 12:29:30 +0000155 EXPECT_GE(_acmA->Add10MsData(audioFrame), 0);
156 EXPECT_GE(_acmRefA->Add10MsData(audioFrame), 0);
niklase@google.com470e71d2011-07-07 08:21:25 +0000157
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000158 EXPECT_GT(_inFileB.Read10MsData(audioFrame), 0);
niklase@google.com470e71d2011-07-07 08:21:25 +0000159
Henrik Lundin45c64492015-03-30 19:00:44 +0200160 EXPECT_GE(_acmB->Add10MsData(audioFrame), 0);
henrik.lundin@webrtc.orgf56c1622015-03-02 12:29:30 +0000161 EXPECT_GE(_acmRefB->Add10MsData(audioFrame), 0);
henrik.lundind4ccb002016-05-17 12:21:55 -0700162 bool muted;
163 EXPECT_EQ(0, _acmA->PlayoutData10Ms(outFreqHzA, &audioFrame, &muted));
164 ASSERT_FALSE(muted);
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000165 _outFileA.Write10MsData(audioFrame);
henrik.lundind4ccb002016-05-17 12:21:55 -0700166 EXPECT_EQ(0, _acmRefA->PlayoutData10Ms(outFreqHzA, &audioFrame, &muted));
167 ASSERT_FALSE(muted);
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000168 _outFileRefA.Write10MsData(audioFrame);
henrik.lundind4ccb002016-05-17 12:21:55 -0700169 EXPECT_EQ(0, _acmB->PlayoutData10Ms(outFreqHzB, &audioFrame, &muted));
170 ASSERT_FALSE(muted);
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000171 _outFileB.Write10MsData(audioFrame);
henrik.lundind4ccb002016-05-17 12:21:55 -0700172 EXPECT_EQ(0, _acmRefB->PlayoutData10Ms(outFreqHzB, &audioFrame, &muted));
173 ASSERT_FALSE(muted);
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000174 _outFileRefB.Write10MsData(audioFrame);
niklase@google.com470e71d2011-07-07 08:21:25 +0000175
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000176 // Update time counters each time a second of data has passed.
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000177 if (msecPassed >= 1000) {
178 msecPassed = 0;
179 secPassed++;
180 }
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000181 // Re-register send codec on side B.
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000182 if (((secPassed % 5) == 4) && (msecPassed >= 990)) {
kwiberg1fd4a4a2015-11-03 11:20:50 -0800183 EXPECT_EQ(0, _acmB->RegisterSendCodec(*codecInst_B));
184 EXPECT_TRUE(_acmB->SendCodec());
niklase@google.com470e71d2011-07-07 08:21:25 +0000185 }
Karl Wibergdd00f112015-08-25 09:37:04 +0200186 // Initialize receiver on side A.
187 if (((secPassed % 7) == 6) && (msecPassed == 0))
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000188 EXPECT_EQ(0, _acmA->InitializeReceiver());
tina.legrand@webrtc.orgee92b662013-08-27 07:33:51 +0000189 // Re-register codec on side A.
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000190 if (((secPassed % 7) == 6) && (msecPassed >= 990)) {
kwibergda2bf4e2016-10-24 13:47:09 -0700191 EXPECT_EQ(true, _acmA->RegisterReceiveCodec(
192 codecInst_B->pltype, CodecInstToSdp(*codecInst_B)));
niklase@google.com470e71d2011-07-07 08:21:25 +0000193 }
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000194 }
niklase@google.com470e71d2011-07-07 08:21:25 +0000195}
196
tina.legrand@webrtc.orgd5726a12013-05-03 07:34:12 +0000197} // namespace webrtc