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 "webrtc/modules/audio_coding/test/TestVADDTX.h" 12 13 #include <string> 14 15 #include "webrtc/engine_configurations.h" 16 #include "webrtc/modules/audio_coding/test/PCMFile.h" 17 #include "webrtc/modules/audio_coding/test/utility.h" 18 #include "webrtc/test/testsupport/fileutils.h" 19 20 namespace webrtc { 21 22 #ifdef WEBRTC_CODEC_ISAC 23 const CodecInst kIsacWb = {103, "ISAC", 16000, 480, 1, 32000}; 24 const CodecInst kIsacSwb = {104, "ISAC", 32000, 960, 1, 56000}; 25 #endif 26 27 #ifdef WEBRTC_CODEC_ILBC 28 const CodecInst kIlbc = {102, "ILBC", 8000, 240, 1, 13300}; 29 #endif 30 31 #ifdef WEBRTC_CODEC_OPUS 32 const CodecInst kOpus = {120, "opus", 48000, 960, 1, 64000}; 33 const CodecInst kOpusStereo = {120, "opus", 48000, 960, 2, 64000}; 34 #endif 35 36 ActivityMonitor::ActivityMonitor() { 37 ResetStatistics(); 38 } 39 40 int32_t ActivityMonitor::InFrameType(FrameType frame_type) { 41 counter_[frame_type]++; 42 return 0; 43 } 44 45 void ActivityMonitor::PrintStatistics() { 46 printf("\n"); 47 printf("kEmptyFrame %u\n", counter_[kEmptyFrame]); 48 printf("kAudioFrameSpeech %u\n", counter_[kAudioFrameSpeech]); 49 printf("kAudioFrameCN %u\n", counter_[kAudioFrameCN]); 50 printf("kVideoFrameKey %u\n", counter_[kVideoFrameKey]); 51 printf("kVideoFrameDelta %u\n", counter_[kVideoFrameDelta]); 52 printf("\n\n"); 53 } 54 55 void ActivityMonitor::ResetStatistics() { 56 memset(counter_, 0, sizeof(counter_)); 57 } 58 59 void ActivityMonitor::GetStatistics(uint32_t* counter) { 60 memcpy(counter, counter_, sizeof(counter_)); 61 } 62 63 TestVadDtx::TestVadDtx() 64 : acm_send_(AudioCodingModule::Create(0)), 65 acm_receive_(AudioCodingModule::Create(1)), 66 channel_(new Channel), 67 monitor_(new ActivityMonitor) { 68 EXPECT_EQ(0, acm_send_->RegisterTransportCallback(channel_.get())); 69 channel_->RegisterReceiverACM(acm_receive_.get()); 70 EXPECT_EQ(0, acm_send_->RegisterVADCallback(monitor_.get())); 71 } 72 73 void TestVadDtx::RegisterCodec(CodecInst codec_param) { 74 // Set the codec for sending and receiving. 75 EXPECT_EQ(0, acm_send_->RegisterSendCodec(codec_param)); 76 EXPECT_EQ(0, acm_receive_->RegisterReceiveCodec(codec_param)); 77 channel_->SetIsStereo(codec_param.channels > 1); 78 } 79 80 // Encoding a file and see if the numbers that various packets occur follow 81 // the expectation. 82 void TestVadDtx::Run(std::string in_filename, int frequency, int channels, 83 std::string out_filename, bool append, 84 const int* expects) { 85 monitor_->ResetStatistics(); 86 87 PCMFile in_file; 88 in_file.Open(in_filename, frequency, "rb"); 89 in_file.ReadStereo(channels > 1); 90 // Set test length to 1000 ms (100 blocks of 10 ms each). 91 in_file.SetNum10MsBlocksToRead(100); 92 // Fast-forward both files 500 ms (50 blocks). The first second of the file is 93 // silence, but we want to keep half of that to test silence periods. 94 in_file.FastForward(50); 95 96 PCMFile out_file; 97 if (append) { 98 out_file.Open(out_filename, kOutputFreqHz, "ab"); 99 } else { 100 out_file.Open(out_filename, kOutputFreqHz, "wb"); 101 } 102 103 uint16_t frame_size_samples = in_file.PayloadLength10Ms(); 104 uint32_t time_stamp = 0x12345678; 105 AudioFrame audio_frame; 106 while (!in_file.EndOfFile()) { 107 in_file.Read10MsData(audio_frame); 108 audio_frame.timestamp_ = time_stamp; 109 time_stamp += frame_size_samples; 110 EXPECT_GE(acm_send_->Add10MsData(audio_frame), 0); 111 acm_receive_->PlayoutData10Ms(kOutputFreqHz, &audio_frame); 112 out_file.Write10MsData(audio_frame); 113 } 114 115 in_file.Close(); 116 out_file.Close(); 117 118 #ifdef PRINT_STAT 119 monitor_->PrintStatistics(); 120 #endif 121 122 uint32_t stats[5]; 123 monitor_->GetStatistics(stats); 124 monitor_->ResetStatistics(); 125 126 for (const auto& st : stats) { 127 int i = &st - stats; // Calculate the current position in stats. 128 switch (expects[i]) { 129 case 0: { 130 EXPECT_EQ(0u, st) << "stats[" << i << "] error."; 131 break; 132 } 133 case 1: { 134 EXPECT_GT(st, 0u) << "stats[" << i << "] error."; 135 break; 136 } 137 } 138 } 139 } 140 141 // Following is the implementation of TestWebRtcVadDtx. 142 TestWebRtcVadDtx::TestWebRtcVadDtx() 143 : vad_enabled_(false), 144 dtx_enabled_(false), 145 output_file_num_(0) { 146 } 147 148 void TestWebRtcVadDtx::Perform() { 149 // Go through various test cases. 150 #ifdef WEBRTC_CODEC_ISAC 151 // Register iSAC WB as send codec 152 RegisterCodec(kIsacWb); 153 RunTestCases(); 154 155 // Register iSAC SWB as send codec 156 RegisterCodec(kIsacSwb); 157 RunTestCases(); 158 #endif 159 160 #ifdef WEBRTC_CODEC_ILBC 161 // Register iLBC as send codec 162 RegisterCodec(kIlbc); 163 RunTestCases(); 164 #endif 165 166 #ifdef WEBRTC_CODEC_OPUS 167 // Register Opus as send codec 168 RegisterCodec(kOpus); 169 RunTestCases(); 170 #endif 171 } 172 173 // Test various configurations on VAD/DTX. 174 void TestWebRtcVadDtx::RunTestCases() { 175 // #1 DTX = OFF, VAD = OFF, VADNormal 176 SetVAD(false, false, VADNormal); 177 Test(true); 178 179 // #2 DTX = ON, VAD = ON, VADAggr 180 SetVAD(true, true, VADAggr); 181 Test(false); 182 183 // #3 DTX = ON, VAD = ON, VADLowBitrate 184 SetVAD(true, true, VADLowBitrate); 185 Test(false); 186 187 // #4 DTX = ON, VAD = ON, VADVeryAggr 188 SetVAD(true, true, VADVeryAggr); 189 Test(false); 190 191 // #5 DTX = ON, VAD = ON, VADNormal 192 SetVAD(true, true, VADNormal); 193 Test(false); 194 } 195 196 // Set the expectation and run the test. 197 void TestWebRtcVadDtx::Test(bool new_outfile) { 198 int expects[] = {-1, 1, dtx_enabled_, 0, 0}; 199 if (new_outfile) { 200 output_file_num_++; 201 } 202 std::stringstream out_filename; 203 out_filename << webrtc::test::OutputPath() 204 << "testWebRtcVadDtx_outFile_" 205 << output_file_num_ 206 << ".pcm"; 207 Run(webrtc::test::ResourcePath("audio_coding/testfile32kHz", "pcm"), 208 32000, 1, out_filename.str(), !new_outfile, expects); 209 } 210 211 void TestWebRtcVadDtx::SetVAD(bool enable_dtx, bool enable_vad, 212 ACMVADMode vad_mode) { 213 ACMVADMode mode; 214 EXPECT_EQ(0, acm_send_->SetVAD(enable_dtx, enable_vad, vad_mode)); 215 EXPECT_EQ(0, acm_send_->VAD(&dtx_enabled_, &vad_enabled_, &mode)); 216 217 auto codec_param = acm_send_->SendCodec(); 218 ASSERT_TRUE(codec_param); 219 if (STR_CASE_CMP(codec_param->plname, "opus") == 0) { 220 // If send codec is Opus, WebRTC VAD/DTX cannot be used. 221 enable_dtx = enable_vad = false; 222 } 223 224 EXPECT_EQ(dtx_enabled_ , enable_dtx); // DTX should be set as expected. 225 226 if (dtx_enabled_) { 227 EXPECT_TRUE(vad_enabled_); // WebRTC DTX cannot run without WebRTC VAD. 228 } else { 229 // Using no DTX should not affect setting of VAD. 230 EXPECT_EQ(enable_vad, vad_enabled_); 231 } 232 } 233 234 // Following is the implementation of TestOpusDtx. 235 void TestOpusDtx::Perform() { 236 #ifdef WEBRTC_CODEC_ISAC 237 // If we set other codec than Opus, DTX cannot be switched on. 238 RegisterCodec(kIsacWb); 239 EXPECT_EQ(-1, acm_send_->EnableOpusDtx()); 240 EXPECT_EQ(0, acm_send_->DisableOpusDtx()); 241 #endif 242 243 #ifdef WEBRTC_CODEC_OPUS 244 int expects[] = {0, 1, 0, 0, 0}; 245 246 // Register Opus as send codec 247 std::string out_filename = webrtc::test::OutputPath() + 248 "testOpusDtx_outFile_mono.pcm"; 249 RegisterCodec(kOpus); 250 EXPECT_EQ(0, acm_send_->DisableOpusDtx()); 251 252 Run(webrtc::test::ResourcePath("audio_coding/testfile32kHz", "pcm"), 253 32000, 1, out_filename, false, expects); 254 255 EXPECT_EQ(0, acm_send_->EnableOpusDtx()); 256 expects[kEmptyFrame] = 1; 257 Run(webrtc::test::ResourcePath("audio_coding/testfile32kHz", "pcm"), 258 32000, 1, out_filename, true, expects); 259 260 // Register stereo Opus as send codec 261 out_filename = webrtc::test::OutputPath() + "testOpusDtx_outFile_stereo.pcm"; 262 RegisterCodec(kOpusStereo); 263 EXPECT_EQ(0, acm_send_->DisableOpusDtx()); 264 expects[kEmptyFrame] = 0; 265 Run(webrtc::test::ResourcePath("audio_coding/teststereo32kHz", "pcm"), 266 32000, 2, out_filename, false, expects); 267 268 EXPECT_EQ(0, acm_send_->EnableOpusDtx()); 269 270 expects[kEmptyFrame] = 1; 271 Run(webrtc::test::ResourcePath("audio_coding/teststereo32kHz", "pcm"), 272 32000, 2, out_filename, true, expects); 273 #endif 274 } 275 276 } // namespace webrtc 277