blob: ac516663b0d7f32277ff6944792dc7e007dc29ab [file] [log] [blame]
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +00001/*
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
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020011#include "modules/audio_coding/test/TestRedFec.h"
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +000012
Fredrik Solenberg657b2962018-12-05 10:30:25 +010013#include <utility>
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +000014
Fredrik Solenberg657b2962018-12-05 10:30:25 +010015#include "absl/strings/match.h"
Karl Wibergc2c4d042018-10-04 12:38:03 +020016#include "api/audio_codecs/L16/audio_decoder_L16.h"
17#include "api/audio_codecs/L16/audio_encoder_L16.h"
18#include "api/audio_codecs/audio_decoder_factory_template.h"
19#include "api/audio_codecs/audio_encoder_factory_template.h"
20#include "api/audio_codecs/g711/audio_decoder_g711.h"
21#include "api/audio_codecs/g711/audio_encoder_g711.h"
22#include "api/audio_codecs/g722/audio_decoder_g722.h"
23#include "api/audio_codecs/g722/audio_encoder_g722.h"
24#include "api/audio_codecs/isac/audio_decoder_isac_float.h"
25#include "api/audio_codecs/isac/audio_encoder_isac_float.h"
26#include "api/audio_codecs/opus/audio_decoder_opus.h"
27#include "api/audio_codecs/opus/audio_encoder_opus.h"
Karl Wibergc2c4d042018-10-04 12:38:03 +020028#include "modules/audio_coding/codecs/cng/audio_encoder_cng.h"
29#include "modules/audio_coding/codecs/red/audio_encoder_copy_red.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020030#include "modules/audio_coding/include/audio_coding_module_typedefs.h"
Jonas Olsson366a50c2018-09-06 13:41:30 +020031#include "rtc_base/strings/string_builder.h"
Fredrik Solenberg657b2962018-12-05 10:30:25 +010032#include "test/gtest.h"
Mirko Bonadei92ea95e2017-09-15 06:47:31 +020033#include "test/testsupport/fileutils.h"
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +000034
35namespace webrtc {
36
37TestRedFec::TestRedFec()
Karl Wibergc2c4d042018-10-04 12:38:03 +020038 : encoder_factory_(CreateAudioEncoderFactory<AudioEncoderG711,
39 AudioEncoderG722,
40 AudioEncoderIsacFloat,
41 AudioEncoderL16,
42 AudioEncoderOpus>()),
43 decoder_factory_(CreateAudioDecoderFactory<AudioDecoderG711,
44 AudioDecoderG722,
45 AudioDecoderIsacFloat,
46 AudioDecoderL16,
47 AudioDecoderOpus>()),
48 _acmA(AudioCodingModule::Create(
49 AudioCodingModule::Config(decoder_factory_))),
Karl Wiberg5817d3d2018-04-06 10:06:42 +020050 _acmB(AudioCodingModule::Create(
Karl Wibergc2c4d042018-10-04 12:38:03 +020051 AudioCodingModule::Config(decoder_factory_))),
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +000052 _channelA2B(NULL),
Karl Wiberg5817d3d2018-04-06 10:06:42 +020053 _testCntr(0) {}
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +000054
55TestRedFec::~TestRedFec() {
56 if (_channelA2B != NULL) {
57 delete _channelA2B;
58 _channelA2B = NULL;
59 }
60}
61
62void TestRedFec::Perform() {
63 const std::string file_name = webrtc::test::ResourcePath(
64 "audio_coding/testfile32kHz", "pcm");
65 _inFileA.Open(file_name, 32000, "rb");
66
67 ASSERT_EQ(0, _acmA->InitializeReceiver());
68 ASSERT_EQ(0, _acmB->InitializeReceiver());
69
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +000070 // Create and connect the channel
71 _channelA2B = new Channel;
72 _acmA->RegisterTransportCallback(_channelA2B);
73 _channelA2B->RegisterReceiverACM(_acmB.get());
74
Karl Wibergc2c4d042018-10-04 12:38:03 +020075 RegisterSendCodec(_acmA, {"L16", 8000, 1}, Vad::kVadAggressive, true);
minyue@webrtc.org41d2bef2015-03-23 12:57:45 +000076
77 OpenOutFile(_testCntr);
78 Run();
79 _outFileB.Close();
80
Karl Wibergc2c4d042018-10-04 12:38:03 +020081 // Switch to another 8 kHz codec; RED should remain switched on.
82 RegisterSendCodec(_acmA, {"PCMU", 8000, 1}, Vad::kVadAggressive, true);
minyue@webrtc.org41d2bef2015-03-23 12:57:45 +000083 OpenOutFile(_testCntr);
84 Run();
85 _outFileB.Close();
86
Karl Wibergc2c4d042018-10-04 12:38:03 +020087 // Switch to a 16 kHz codec; RED should be switched off.
88 RegisterSendCodec(_acmA, {"G722", 8000, 1}, Vad::kVadAggressive, false);
minyue@webrtc.org41d2bef2015-03-23 12:57:45 +000089
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +000090 OpenOutFile(_testCntr);
Karl Wibergc2c4d042018-10-04 12:38:03 +020091 RegisterSendCodec(_acmA, {"G722", 8000, 1}, Vad::kVadAggressive, false);
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +000092 Run();
Karl Wibergc2c4d042018-10-04 12:38:03 +020093 RegisterSendCodec(_acmA, {"G722", 8000, 1}, Vad::kVadAggressive, false);
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +000094 Run();
95 _outFileB.Close();
96
Karl Wibergc2c4d042018-10-04 12:38:03 +020097 RegisterSendCodec(_acmA, {"ISAC", 16000, 1}, Vad::kVadVeryAggressive, false);
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +000098 OpenOutFile(_testCntr);
99 Run();
100 _outFileB.Close();
101
Karl Wibergc2c4d042018-10-04 12:38:03 +0200102 // Switch to a 32 kHz codec; RED should be switched off.
103 RegisterSendCodec(_acmA, {"ISAC", 32000, 1}, Vad::kVadVeryAggressive, false);
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000104 OpenOutFile(_testCntr);
105 Run();
106 _outFileB.Close();
107
Karl Wibergc2c4d042018-10-04 12:38:03 +0200108 RegisterSendCodec(_acmA, {"ISAC", 32000, 1}, absl::nullopt, false);
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000109
110 _channelA2B->SetFECTestWithPacketLoss(true);
minyue@webrtc.org41d2bef2015-03-23 12:57:45 +0000111 // Following tests are under packet losses.
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000112
Karl Wibergc2c4d042018-10-04 12:38:03 +0200113 // Switch to a 16 kHz codec; RED should be switched off.
114 RegisterSendCodec(_acmA, {"G722", 8000, 1}, Vad::kVadAggressive, false);
115
116 OpenOutFile(_testCntr);
117 Run();
118 _outFileB.Close();
minyue@webrtc.org41d2bef2015-03-23 12:57:45 +0000119
minyue@webrtc.org41d2bef2015-03-23 12:57:45 +0000120 // Switch to a 16 kHz codec, RED should have been switched off.
Karl Wibergc2c4d042018-10-04 12:38:03 +0200121 RegisterSendCodec(_acmA, {"ISAC", 16000, 1}, Vad::kVadVeryAggressive, false);
minyue@webrtc.org41d2bef2015-03-23 12:57:45 +0000122
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000123 OpenOutFile(_testCntr);
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000124 Run();
125 _outFileB.Close();
126
minyue@webrtc.org41d2bef2015-03-23 12:57:45 +0000127 // Switch to a 32 kHz codec, RED should have been switched off.
Karl Wibergc2c4d042018-10-04 12:38:03 +0200128 RegisterSendCodec(_acmA, {"ISAC", 32000, 1}, Vad::kVadVeryAggressive, false);
minyue@webrtc.org41d2bef2015-03-23 12:57:45 +0000129
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000130 OpenOutFile(_testCntr);
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000131 Run();
132 _outFileB.Close();
133
Karl Wibergc2c4d042018-10-04 12:38:03 +0200134 RegisterSendCodec(_acmA, {"ISAC", 32000, 1}, absl::nullopt, false);
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000135
Karl Wibergc2c4d042018-10-04 12:38:03 +0200136 RegisterSendCodec(_acmA, {"opus", 48000, 2}, absl::nullopt, false);
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000137
138 // _channelA2B imposes 25% packet loss rate.
139 EXPECT_EQ(0, _acmA->SetPacketLossRate(25));
140
Karl Wibergc2c4d042018-10-04 12:38:03 +0200141 _acmA->ModifyEncoder([&](std::unique_ptr<AudioEncoder>* enc) {
142 EXPECT_EQ(true, (*enc)->SetFec(true));
143 });
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000144
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000145 OpenOutFile(_testCntr);
146 Run();
147
minyue@webrtc.org41d2bef2015-03-23 12:57:45 +0000148 // Switch to L16 with RED.
Karl Wibergc2c4d042018-10-04 12:38:03 +0200149 RegisterSendCodec(_acmA, {"L16", 8000, 1}, absl::nullopt, true);
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000150 Run();
151
152 // Switch to Opus again.
Karl Wibergc2c4d042018-10-04 12:38:03 +0200153 RegisterSendCodec(_acmA, {"opus", 48000, 2}, absl::nullopt, false);
154 _acmA->ModifyEncoder([&](std::unique_ptr<AudioEncoder>* enc) {
155 EXPECT_EQ(true, (*enc)->SetFec(false));
156 });
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000157 Run();
158
Karl Wibergc2c4d042018-10-04 12:38:03 +0200159 _acmA->ModifyEncoder([&](std::unique_ptr<AudioEncoder>* enc) {
160 EXPECT_EQ(true, (*enc)->SetFec(true));
161 });
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000162 _outFileB.Close();
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000163}
164
Karl Wibergc2c4d042018-10-04 12:38:03 +0200165void TestRedFec::RegisterSendCodec(
166 const std::unique_ptr<AudioCodingModule>& acm,
167 const SdpAudioFormat& codec_format,
168 absl::optional<Vad::Aggressiveness> vad_mode,
169 bool use_red) {
170 constexpr int payload_type = 17, cn_payload_type = 27, red_payload_type = 37;
171 const auto& other_acm = &acm == &_acmA ? _acmB : _acmA;
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000172
Karl Wibergc2c4d042018-10-04 12:38:03 +0200173 auto encoder = encoder_factory_->MakeAudioEncoder(payload_type, codec_format,
174 absl::nullopt);
175 EXPECT_NE(encoder, nullptr);
Fredrik Solenberg657b2962018-12-05 10:30:25 +0100176 std::map<int, SdpAudioFormat> receive_codecs = {{payload_type, codec_format}};
Niels Möller2edab4c2018-10-22 09:48:08 +0200177 if (!absl::EqualsIgnoreCase(codec_format.name, "opus")) {
Karl Wibergc2c4d042018-10-04 12:38:03 +0200178 if (vad_mode.has_value()) {
Karl Wiberg23659362018-11-01 11:13:44 +0100179 AudioEncoderCngConfig config;
Karl Wibergc2c4d042018-10-04 12:38:03 +0200180 config.speech_encoder = std::move(encoder);
181 config.num_channels = 1;
182 config.payload_type = cn_payload_type;
183 config.vad_mode = vad_mode.value();
Karl Wiberg23659362018-11-01 11:13:44 +0100184 encoder = CreateComfortNoiseEncoder(std::move(config));
Fredrik Solenberg657b2962018-12-05 10:30:25 +0100185 receive_codecs.emplace(
186 std::make_pair(cn_payload_type,
187 SdpAudioFormat("CN", codec_format.clockrate_hz, 1)));
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000188 }
Karl Wibergc2c4d042018-10-04 12:38:03 +0200189 if (use_red) {
190 AudioEncoderCopyRed::Config config;
191 config.payload_type = red_payload_type;
192 config.speech_encoder = std::move(encoder);
193 encoder = absl::make_unique<AudioEncoderCopyRed>(std::move(config));
Fredrik Solenberg657b2962018-12-05 10:30:25 +0100194 receive_codecs.emplace(
195 std::make_pair(red_payload_type,
196 SdpAudioFormat("red", codec_format.clockrate_hz, 1)));
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000197 }
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000198 }
Karl Wibergc2c4d042018-10-04 12:38:03 +0200199 acm->SetEncoder(std::move(encoder));
Fredrik Solenberg657b2962018-12-05 10:30:25 +0100200 other_acm->SetReceiveCodecs(receive_codecs);
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000201}
202
203void TestRedFec::Run() {
204 AudioFrame audioFrame;
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000205 int32_t outFreqHzB = _outFileB.SamplingFrequency();
Henrik Lundin4d682082015-12-10 16:24:39 +0100206 // Set test length to 500 ms (50 blocks of 10 ms each).
207 _inFileA.SetNum10MsBlocksToRead(50);
208 // Fast-forward 1 second (100 blocks) since the file starts with silence.
209 _inFileA.FastForward(100);
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000210
211 while (!_inFileA.EndOfFile()) {
212 EXPECT_GT(_inFileA.Read10MsData(audioFrame), 0);
henrik.lundin@webrtc.orgf56c1622015-03-02 12:29:30 +0000213 EXPECT_GE(_acmA->Add10MsData(audioFrame), 0);
henrik.lundind4ccb002016-05-17 12:21:55 -0700214 bool muted;
215 EXPECT_EQ(0, _acmB->PlayoutData10Ms(outFreqHzB, &audioFrame, &muted));
216 ASSERT_FALSE(muted);
yujo36b1a5f2017-06-12 12:45:32 -0700217 _outFileB.Write10MsData(audioFrame.data(), audioFrame.samples_per_channel_);
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000218 }
219 _inFileA.Rewind();
220}
221
222void TestRedFec::OpenOutFile(int16_t test_number) {
223 std::string file_name;
Jonas Olsson366a50c2018-09-06 13:41:30 +0200224 rtc::StringBuilder file_stream;
minyue@webrtc.orgaa5ea1c2014-05-23 15:16:51 +0000225 file_stream << webrtc::test::OutputPath();
226 file_stream << "TestRedFec_outFile_";
227 file_stream << test_number << ".pcm";
228 file_name = file_stream.str();
229 _outFileB.Open(file_name, 16000, "wb");
230}
231
232} // namespace webrtc