blob: 87cd61c8e739f01075edb17565b2114529bc2657 [file] [log] [blame]
andrew@webrtc.orgb015cbe2012-10-22 18:19:231/*
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
kjellander0c1546c2015-11-26 12:44:5411#include "webrtc/modules/audio_coding/test/EncodeDecodeTest.h"
andrew@webrtc.orgb015cbe2012-10-22 18:19:2312
kwiberg4df87572016-02-15 04:40:5713#include <memory>
minyue@webrtc.org91c0a252014-05-23 15:16:5114#include <sstream>
andrew@webrtc.orgb015cbe2012-10-22 18:19:2315#include <stdio.h>
16#include <stdlib.h>
andrew@webrtc.orgb015cbe2012-10-22 18:19:2317
tina.legrand@webrtc.orgcf9ab122013-03-15 13:29:1718#include "webrtc/common_types.h"
kwiberg77c17ed2016-10-24 20:47:0919#include "webrtc/modules/audio_coding/codecs/audio_format_conversion.h"
kwiberg36a24792016-10-01 05:29:4320#include "webrtc/modules/audio_coding/include/audio_coding_module.h"
kjellander0c1546c2015-11-26 12:44:5421#include "webrtc/modules/audio_coding/test/utility.h"
Henrik Kjellander78f65d02015-10-28 17:17:4022#include "webrtc/system_wrappers/include/trace.h"
kwiberg36a24792016-10-01 05:29:4323#include "webrtc/test/gtest.h"
tina.legrand@webrtc.orgcf9ab122013-03-15 13:29:1724#include "webrtc/test/testsupport/fileutils.h"
andrew@webrtc.orgb015cbe2012-10-22 18:19:2325
26namespace webrtc {
27
tina.legrand@webrtc.orgb31332e2013-05-03 07:34:1228TestPacketization::TestPacketization(RTPStream *rtpStream, uint16_t frequency)
andrew@webrtc.orgb015cbe2012-10-22 18:19:2329 : _rtpStream(rtpStream),
30 _frequency(frequency),
31 _seqNo(0) {
32}
33
tina.legrand@webrtc.orgb31332e2013-05-03 07:34:1234TestPacketization::~TestPacketization() {
35}
andrew@webrtc.orgb015cbe2012-10-22 18:19:2336
pbos@webrtc.orgfbda0fc2013-04-09 00:28:0637int32_t TestPacketization::SendData(
tina.legrand@webrtc.orgb31332e2013-05-03 07:34:1238 const FrameType /* frameType */, const uint8_t payloadType,
39 const uint32_t timeStamp, const uint8_t* payloadData,
pkasting@chromium.org0ab923a2014-11-20 22:28:1440 const size_t payloadSize,
andrew@webrtc.orgb015cbe2012-10-22 18:19:2341 const RTPFragmentationHeader* /* fragmentation */) {
42 _rtpStream->Write(payloadType, timeStamp, _seqNo++, payloadData, payloadSize,
43 _frequency);
44 return 1;
45}
46
47Sender::Sender()
48 : _acm(NULL),
49 _pcmFile(),
50 _audioFrame(),
51 _packetization(NULL) {
52}
53
minyue@webrtc.org91c0a252014-05-23 15:16:5154void Sender::Setup(AudioCodingModule *acm, RTPStream *rtpStream,
Peter Kasting80590d92016-01-13 00:26:3555 std::string in_file_name, int sample_rate, size_t channels) {
andrew@webrtc.orgb015cbe2012-10-22 18:19:2356 struct CodecInst sendCodec;
57 int noOfCodecs = acm->NumberOfCodecs();
58 int codecNo;
59
60 // Open input file
minyue@webrtc.org91c0a252014-05-23 15:16:5161 const std::string file_name = webrtc::test::ResourcePath(in_file_name, "pcm");
62 _pcmFile.Open(file_name, sample_rate, "rb");
63 if (channels == 2) {
64 _pcmFile.ReadStereo(true);
65 }
Henrik Lundin9f2a01c2015-12-10 15:24:3966 // Set test length to 500 ms (50 blocks of 10 ms each).
67 _pcmFile.SetNum10MsBlocksToRead(50);
68 // Fast-forward 1 second (100 blocks) since the file starts with silence.
69 _pcmFile.FastForward(100);
andrew@webrtc.orgb015cbe2012-10-22 18:19:2370
71 // Set the codec for the current test.
72 if ((testMode == 0) || (testMode == 1)) {
73 // Set the codec id.
74 codecNo = codeId;
75 } else {
76 // Choose codec on command line.
77 printf("List of supported codec.\n");
78 for (int n = 0; n < noOfCodecs; n++) {
tina.legrand@webrtc.org584b6882013-08-27 07:33:5179 EXPECT_EQ(0, acm->Codec(n, &sendCodec));
andrew@webrtc.orgb015cbe2012-10-22 18:19:2380 printf("%d %s\n", n, sendCodec.plname);
81 }
82 printf("Choose your codec:");
83 ASSERT_GT(scanf("%d", &codecNo), 0);
84 }
85
tina.legrand@webrtc.org584b6882013-08-27 07:33:5186 EXPECT_EQ(0, acm->Codec(codecNo, &sendCodec));
minyue@webrtc.org91c0a252014-05-23 15:16:5187
88 sendCodec.channels = channels;
tina.legrand@webrtc.org584b6882013-08-27 07:33:5189
90 EXPECT_EQ(0, acm->RegisterSendCodec(sendCodec));
andrew@webrtc.orgb015cbe2012-10-22 18:19:2391 _packetization = new TestPacketization(rtpStream, sendCodec.plfreq);
tina.legrand@webrtc.org584b6882013-08-27 07:33:5192 EXPECT_EQ(0, acm->RegisterTransportCallback(_packetization));
andrew@webrtc.orgb015cbe2012-10-22 18:19:2393
94 _acm = acm;
95}
96
97void Sender::Teardown() {
98 _pcmFile.Close();
99 delete _packetization;
100}
101
102bool Sender::Add10MsData() {
103 if (!_pcmFile.EndOfFile()) {
tina.legrand@webrtc.org584b6882013-08-27 07:33:51104 EXPECT_GT(_pcmFile.Read10MsData(_audioFrame), 0);
pbos@webrtc.orgfbda0fc2013-04-09 00:28:06105 int32_t ok = _acm->Add10MsData(_audioFrame);
henrik.lundin@webrtc.org57277692015-03-02 12:29:30106 EXPECT_GE(ok, 0);
107 return ok >= 0 ? true : false;
andrew@webrtc.orgb015cbe2012-10-22 18:19:23108 }
109 return false;
110}
111
andrew@webrtc.orgb015cbe2012-10-22 18:19:23112void Sender::Run() {
113 while (true) {
114 if (!Add10MsData()) {
115 break;
116 }
andrew@webrtc.orgb015cbe2012-10-22 18:19:23117 }
118}
119
120Receiver::Receiver()
121 : _playoutLengthSmpls(WEBRTC_10MS_PCM_AUDIO),
122 _payloadSizeBytes(MAX_INCOMING_PAYLOAD) {
123}
124
minyue@webrtc.org91c0a252014-05-23 15:16:51125void Receiver::Setup(AudioCodingModule *acm, RTPStream *rtpStream,
Peter Kasting80590d92016-01-13 00:26:35126 std::string out_file_name, size_t channels) {
henrike@webrtc.org22c283b2014-08-11 21:06:30127 struct CodecInst recvCodec = CodecInst();
andrew@webrtc.orgb015cbe2012-10-22 18:19:23128 int noOfCodecs;
tina.legrand@webrtc.org584b6882013-08-27 07:33:51129 EXPECT_EQ(0, acm->InitializeReceiver());
andrew@webrtc.orgb015cbe2012-10-22 18:19:23130
131 noOfCodecs = acm->NumberOfCodecs();
132 for (int i = 0; i < noOfCodecs; i++) {
minyue@webrtc.org91c0a252014-05-23 15:16:51133 EXPECT_EQ(0, acm->Codec(i, &recvCodec));
134 if (recvCodec.channels == channels)
kwiberg77c17ed2016-10-24 20:47:09135 EXPECT_EQ(true, acm->RegisterReceiveCodec(recvCodec.pltype,
136 CodecInstToSdp(recvCodec)));
minyue@webrtc.org91c0a252014-05-23 15:16:51137 // Forces mono/stereo for Opus.
138 if (!strcmp(recvCodec.plname, "opus")) {
139 recvCodec.channels = channels;
kwiberg77c17ed2016-10-24 20:47:09140 EXPECT_EQ(true, acm->RegisterReceiveCodec(recvCodec.pltype,
141 CodecInstToSdp(recvCodec)));
minyue@webrtc.org91c0a252014-05-23 15:16:51142 }
andrew@webrtc.orgb015cbe2012-10-22 18:19:23143 }
144
145 int playSampFreq;
146 std::string file_name;
147 std::stringstream file_stream;
minyue@webrtc.org91c0a252014-05-23 15:16:51148 file_stream << webrtc::test::OutputPath() << out_file_name
tina.legrand@webrtc.orgb31332e2013-05-03 07:34:12149 << static_cast<int>(codeId) << ".pcm";
andrew@webrtc.orgb015cbe2012-10-22 18:19:23150 file_name = file_stream.str();
151 _rtpStream = rtpStream;
152
153 if (testMode == 1) {
tina.legrand@webrtc.orgb31332e2013-05-03 07:34:12154 playSampFreq = recvCodec.plfreq;
andrew@webrtc.orgb015cbe2012-10-22 18:19:23155 _pcmFile.Open(file_name, recvCodec.plfreq, "wb+");
156 } else if (testMode == 0) {
tina.legrand@webrtc.orgb31332e2013-05-03 07:34:12157 playSampFreq = 32000;
andrew@webrtc.orgb015cbe2012-10-22 18:19:23158 _pcmFile.Open(file_name, 32000, "wb+");
159 } else {
160 printf("\nValid output frequencies:\n");
161 printf("8000\n16000\n32000\n-1,");
162 printf("which means output frequency equal to received signal frequency");
163 printf("\n\nChoose output sampling frequency: ");
164 ASSERT_GT(scanf("%d", &playSampFreq), 0);
minyue@webrtc.org91c0a252014-05-23 15:16:51165 file_name = webrtc::test::OutputPath() + out_file_name + ".pcm";
andrew@webrtc.orgb015cbe2012-10-22 18:19:23166 _pcmFile.Open(file_name, playSampFreq, "wb+");
167 }
168
169 _realPayloadSizeBytes = 0;
pbos@webrtc.orgfbda0fc2013-04-09 00:28:06170 _playoutBuffer = new int16_t[WEBRTC_10MS_PCM_AUDIO];
andrew@webrtc.orgb015cbe2012-10-22 18:19:23171 _frequency = playSampFreq;
172 _acm = acm;
173 _firstTime = true;
174}
175
176void Receiver::Teardown() {
tina.legrand@webrtc.orgb31332e2013-05-03 07:34:12177 delete[] _playoutBuffer;
andrew@webrtc.orgb015cbe2012-10-22 18:19:23178 _pcmFile.Close();
tina.legrand@webrtc.org584b6882013-08-27 07:33:51179 if (testMode > 1) {
andrew@webrtc.orgb015cbe2012-10-22 18:19:23180 Trace::ReturnTrace();
tina.legrand@webrtc.org584b6882013-08-27 07:33:51181 }
andrew@webrtc.orgb015cbe2012-10-22 18:19:23182}
183
184bool Receiver::IncomingPacket() {
185 if (!_rtpStream->EndOfFile()) {
186 if (_firstTime) {
187 _firstTime = false;
188 _realPayloadSizeBytes = _rtpStream->Read(&_rtpInfo, _incomingPayload,
189 _payloadSizeBytes, &_nextTime);
190 if (_realPayloadSizeBytes == 0) {
191 if (_rtpStream->EndOfFile()) {
192 _firstTime = true;
193 return true;
194 } else {
andrew@webrtc.orgb015cbe2012-10-22 18:19:23195 return false;
196 }
197 }
tina.legrand@webrtc.orgb31332e2013-05-03 07:34:12198 }
andrew@webrtc.orgb015cbe2012-10-22 18:19:23199
tina.legrand@webrtc.org584b6882013-08-27 07:33:51200 EXPECT_EQ(0, _acm->IncomingPacket(_incomingPayload, _realPayloadSizeBytes,
201 _rtpInfo));
tina.legrand@webrtc.orgb31332e2013-05-03 07:34:12202 _realPayloadSizeBytes = _rtpStream->Read(&_rtpInfo, _incomingPayload,
203 _payloadSizeBytes, &_nextTime);
andrew@webrtc.orgb015cbe2012-10-22 18:19:23204 if (_realPayloadSizeBytes == 0 && _rtpStream->EndOfFile()) {
205 _firstTime = true;
206 }
207 }
208 return true;
209}
210
211bool Receiver::PlayoutData() {
212 AudioFrame audioFrame;
henrik.lundinb8896d22016-05-17 19:21:55213 bool muted;
214 int32_t ok = _acm->PlayoutData10Ms(_frequency, &audioFrame, &muted);
215 if (muted) {
216 ADD_FAILURE();
217 return false;
218 }
tina.legrand@webrtc.org584b6882013-08-27 07:33:51219 EXPECT_EQ(0, ok);
220 if (ok < 0){
221 return false;
andrew@webrtc.orgb015cbe2012-10-22 18:19:23222 }
223 if (_playoutLengthSmpls == 0) {
224 return false;
225 }
minyue@webrtc.org91c0a252014-05-23 15:16:51226 _pcmFile.Write10MsData(audioFrame.data_,
227 audioFrame.samples_per_channel_ * audioFrame.num_channels_);
andrew@webrtc.orgb015cbe2012-10-22 18:19:23228 return true;
229}
230
231void Receiver::Run() {
pbos@webrtc.orgfbda0fc2013-04-09 00:28:06232 uint8_t counter500Ms = 50;
233 uint32_t clock = 0;
andrew@webrtc.orgb015cbe2012-10-22 18:19:23234
235 while (counter500Ms > 0) {
236 if (clock == 0 || clock >= _nextTime) {
tina.legrand@webrtc.org584b6882013-08-27 07:33:51237 EXPECT_TRUE(IncomingPacket());
andrew@webrtc.orgb015cbe2012-10-22 18:19:23238 if (clock == 0) {
239 clock = _nextTime;
240 }
241 }
242 if ((clock % 10) == 0) {
243 if (!PlayoutData()) {
244 clock++;
245 continue;
246 }
247 }
248 if (_rtpStream->EndOfFile()) {
249 counter500Ms--;
250 }
251 clock++;
252 }
253}
254
henrik.lundin@webrtc.orgc68f88e2014-04-17 08:29:10255EncodeDecodeTest::EncodeDecodeTest() {
andrew@webrtc.orgb015cbe2012-10-22 18:19:23256 _testMode = 2;
257 Trace::CreateTrace();
tina.legrand@webrtc.orgb31332e2013-05-03 07:34:12258 Trace::SetTraceFile(
259 (webrtc::test::OutputPath() + "acm_encdec_trace.txt").c_str());
andrew@webrtc.orgb015cbe2012-10-22 18:19:23260}
261
henrik.lundin@webrtc.orgc68f88e2014-04-17 08:29:10262EncodeDecodeTest::EncodeDecodeTest(int testMode) {
andrew@webrtc.orgb015cbe2012-10-22 18:19:23263 //testMode == 0 for autotest
264 //testMode == 1 for testing all codecs/parameters
265 //testMode > 1 for specific user-input test (as it was used before)
tina.legrand@webrtc.orgb31332e2013-05-03 07:34:12266 _testMode = testMode;
267 if (_testMode != 0) {
268 Trace::CreateTrace();
269 Trace::SetTraceFile(
270 (webrtc::test::OutputPath() + "acm_encdec_trace.txt").c_str());
271 }
andrew@webrtc.orgb015cbe2012-10-22 18:19:23272}
273
274void EncodeDecodeTest::Perform() {
andrew@webrtc.orgb015cbe2012-10-22 18:19:23275 int numCodecs = 1;
tina.legrand@webrtc.orgb31332e2013-05-03 07:34:12276 int codePars[3]; // Frequency, packet size, rate.
277 int numPars[52]; // Number of codec parameters sets (freq, pacsize, rate)
278 // to test, for a given codec.
andrew@webrtc.orgb015cbe2012-10-22 18:19:23279
280 codePars[0] = 0;
281 codePars[1] = 0;
282 codePars[2] = 0;
283
kwiberg4df87572016-02-15 04:40:57284 std::unique_ptr<AudioCodingModule> acm(AudioCodingModule::Create(0));
andrew@webrtc.orgb015cbe2012-10-22 18:19:23285 struct CodecInst sendCodecTmp;
286 numCodecs = acm->NumberOfCodecs();
287
andrew@webrtc.orgb015cbe2012-10-22 18:19:23288 if (_testMode != 2) {
289 for (int n = 0; n < numCodecs; n++) {
tina.legrand@webrtc.org584b6882013-08-27 07:33:51290 EXPECT_EQ(0, acm->Codec(n, &sendCodecTmp));
andrew@webrtc.orgb015cbe2012-10-22 18:19:23291 if (STR_CASE_CMP(sendCodecTmp.plname, "telephone-event") == 0) {
292 numPars[n] = 0;
293 } else if (STR_CASE_CMP(sendCodecTmp.plname, "cn") == 0) {
294 numPars[n] = 0;
295 } else if (STR_CASE_CMP(sendCodecTmp.plname, "red") == 0) {
296 numPars[n] = 0;
297 } else if (sendCodecTmp.channels == 2) {
298 numPars[n] = 0;
299 } else {
300 numPars[n] = 1;
andrew@webrtc.orgb015cbe2012-10-22 18:19:23301 }
302 }
303 } else {
304 numCodecs = 1;
305 numPars[0] = 1;
306 }
307
308 _receiver.testMode = _testMode;
309
310 // Loop over all mono codecs:
311 for (int codeId = 0; codeId < numCodecs; codeId++) {
312 // Only encode using real mono encoders, not telephone-event and cng.
313 for (int loopPars = 1; loopPars <= numPars[codeId]; loopPars++) {
tina.legrand@webrtc.org584b6882013-08-27 07:33:51314 // Encode all data to file.
pbos@webrtc.orgf719f202014-10-01 10:05:40315 std::string fileName = EncodeToFile(1, codeId, codePars, _testMode);
andrew@webrtc.orgb015cbe2012-10-22 18:19:23316
317 RTPFile rtpFile;
andrew@webrtc.orgb015cbe2012-10-22 18:19:23318 rtpFile.Open(fileName.c_str(), "rb");
319
320 _receiver.codeId = codeId;
321
322 rtpFile.ReadHeader();
minyue@webrtc.org91c0a252014-05-23 15:16:51323 _receiver.Setup(acm.get(), &rtpFile, "encodeDecode_out", 1);
andrew@webrtc.orgb015cbe2012-10-22 18:19:23324 _receiver.Run();
325 _receiver.Teardown();
326 rtpFile.Close();
andrew@webrtc.orgb015cbe2012-10-22 18:19:23327 }
328 }
tina.legrand@webrtc.org584b6882013-08-27 07:33:51329
tina.legrand@webrtc.org584b6882013-08-27 07:33:51330 // End tracing.
331 if (_testMode == 1) {
andrew@webrtc.orgb015cbe2012-10-22 18:19:23332 Trace::ReturnTrace();
tina.legrand@webrtc.org584b6882013-08-27 07:33:51333 }
andrew@webrtc.orgb015cbe2012-10-22 18:19:23334}
335
pbos@webrtc.orgf719f202014-10-01 10:05:40336std::string EncodeDecodeTest::EncodeToFile(int fileType,
337 int codeId,
338 int* codePars,
339 int testMode) {
kwiberg4df87572016-02-15 04:40:57340 std::unique_ptr<AudioCodingModule> acm(AudioCodingModule::Create(1));
andrew@webrtc.orgb015cbe2012-10-22 18:19:23341 RTPFile rtpFile;
pbos@webrtc.orgf719f202014-10-01 10:05:40342 std::string fileName = webrtc::test::TempFilename(webrtc::test::OutputPath(),
343 "encode_decode_rtp");
andrew@webrtc.orgb015cbe2012-10-22 18:19:23344 rtpFile.Open(fileName.c_str(), "wb+");
345 rtpFile.WriteHeader();
346
tina.legrand@webrtc.org584b6882013-08-27 07:33:51347 // Store for auto_test and logging.
andrew@webrtc.orgb015cbe2012-10-22 18:19:23348 _sender.testMode = testMode;
349 _sender.codeId = codeId;
350
minyue@webrtc.org91c0a252014-05-23 15:16:51351 _sender.Setup(acm.get(), &rtpFile, "audio_coding/testfile32kHz", 32000, 1);
kwiberg85b36b92015-11-03 19:20:50352 if (acm->SendCodec()) {
andrew@webrtc.orgb015cbe2012-10-22 18:19:23353 _sender.Run();
354 }
355 _sender.Teardown();
356 rtpFile.Close();
pbos@webrtc.orgf719f202014-10-01 10:05:40357
358 return fileName;
andrew@webrtc.orgb015cbe2012-10-22 18:19:23359}
360
tina.legrand@webrtc.orgb31332e2013-05-03 07:34:12361} // namespace webrtc