blob: ac3f7091b3c04e446c96a10f92acc761b464bc8e [file] [log] [blame]
peahd0263542017-01-03 12:20:341/*
2 * Copyright (c) 2016 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_processing/aec3/echo_canceller3.h"
12
13#include <deque>
14#include <memory>
15#include <sstream>
16#include <string>
17#include <utility>
18#include <vector>
19
peah522d71b2017-02-23 13:16:2620#include "webrtc/modules/audio_processing/aec3/aec3_common.h"
peahd0263542017-01-03 12:20:3421#include "webrtc/modules/audio_processing/aec3/block_processor.h"
22#include "webrtc/modules/audio_processing/aec3/frame_blocker.h"
23#include "webrtc/modules/audio_processing/aec3/mock/mock_block_processor.h"
24#include "webrtc/modules/audio_processing/audio_buffer.h"
25#include "webrtc/test/gmock.h"
26#include "webrtc/test/gtest.h"
27
28namespace webrtc {
29namespace {
30
31using testing::StrictMock;
32using testing::_;
33
34// Populates the frame with linearly increasing sample values for each band,
35// with a band-specific offset, in order to allow simple bitexactness
36// verification for each band.
37void PopulateInputFrame(size_t frame_length,
38 size_t num_bands,
39 size_t frame_index,
40 float* const* frame,
41 int offset) {
42 for (size_t k = 0; k < num_bands; ++k) {
43 for (size_t i = 0; i < frame_length; ++i) {
44 float value = static_cast<int>(frame_index * frame_length + i) + offset;
45 frame[k][i] = (value > 0 ? 5000 * k + value : 0);
46 }
47 }
48}
49
peah522d71b2017-02-23 13:16:2650// Populates the frame with linearly increasing sample values.
51void PopulateInputFrame(size_t frame_length,
52 size_t frame_index,
53 float* frame,
54 int offset) {
55 for (size_t i = 0; i < frame_length; ++i) {
56 float value = static_cast<int>(frame_index * frame_length + i) + offset;
57 frame[i] = std::max(value, 0.f);
58 }
59}
60
peahd0263542017-01-03 12:20:3461// Verifies the that samples in the output frame are identical to the samples
62// that were produced for the input frame, with an offset in order to compensate
63// for buffering delays.
64bool VerifyOutputFrameBitexactness(size_t frame_length,
65 size_t num_bands,
66 size_t frame_index,
67 const float* const* frame,
68 int offset) {
69 float reference_frame_data[kMaxNumBands][2 * kSubFrameLength];
70 float* reference_frame[kMaxNumBands];
71 for (size_t k = 0; k < num_bands; ++k) {
72 reference_frame[k] = &reference_frame_data[k][0];
73 }
74
75 PopulateInputFrame(frame_length, num_bands, frame_index, reference_frame,
76 offset);
77 for (size_t k = 0; k < num_bands; ++k) {
78 for (size_t i = 0; i < frame_length; ++i) {
79 if (reference_frame[k][i] != frame[k][i]) {
80 return false;
81 }
82 }
83 }
84
85 return true;
86}
87
88// Class for testing that the capture data is properly received by the block
89// processor and that the processor data is properly passed to the
90// EchoCanceller3 output.
91class CaptureTransportVerificationProcessor : public BlockProcessor {
92 public:
93 explicit CaptureTransportVerificationProcessor(size_t num_bands) {}
94 ~CaptureTransportVerificationProcessor() override = default;
95
peah69221db2017-01-27 11:28:1996 void ProcessCapture(bool level_change,
peahd0263542017-01-03 12:20:3497 bool saturated_microphone_signal,
98 std::vector<std::vector<float>>* capture_block) override {
99 }
100
peahcf02cf12017-04-05 21:18:07101 void BufferRender(const std::vector<std::vector<float>>& block) override {}
peahd0263542017-01-03 12:20:34102
peah69221db2017-01-27 11:28:19103 void UpdateEchoLeakageStatus(bool leakage_detected) override {}
peahd0263542017-01-03 12:20:34104
105 private:
106 RTC_DISALLOW_IMPLICIT_CONSTRUCTORS(CaptureTransportVerificationProcessor);
107};
108
109// Class for testing that the render data is properly received by the block
110// processor.
111class RenderTransportVerificationProcessor : public BlockProcessor {
112 public:
113 explicit RenderTransportVerificationProcessor(size_t num_bands) {}
114 ~RenderTransportVerificationProcessor() override = default;
115
peah69221db2017-01-27 11:28:19116 void ProcessCapture(bool level_change,
peahd0263542017-01-03 12:20:34117 bool saturated_microphone_signal,
118 std::vector<std::vector<float>>* capture_block) override {
119 std::vector<std::vector<float>> render_block =
120 received_render_blocks_.front();
121 received_render_blocks_.pop_front();
122 capture_block->swap(render_block);
123 }
124
peahcf02cf12017-04-05 21:18:07125 void BufferRender(const std::vector<std::vector<float>>& block) override {
126 received_render_blocks_.push_back(block);
peahd0263542017-01-03 12:20:34127 }
128
peah69221db2017-01-27 11:28:19129 void UpdateEchoLeakageStatus(bool leakage_detected) override {}
peahd0263542017-01-03 12:20:34130
131 private:
132 std::deque<std::vector<std::vector<float>>> received_render_blocks_;
133 RTC_DISALLOW_IMPLICIT_CONSTRUCTORS(RenderTransportVerificationProcessor);
134};
135
136class EchoCanceller3Tester {
137 public:
138 explicit EchoCanceller3Tester(int sample_rate_hz)
139 : sample_rate_hz_(sample_rate_hz),
140 num_bands_(NumBandsForRate(sample_rate_hz_)),
141 frame_length_(sample_rate_hz_ == 8000 ? 80 : 160),
142 fullband_frame_length_(rtc::CheckedDivExact(sample_rate_hz_, 100)),
143 capture_buffer_(fullband_frame_length_,
144 1,
145 fullband_frame_length_,
146 1,
147 fullband_frame_length_),
148 render_buffer_(fullband_frame_length_,
149 1,
150 fullband_frame_length_,
151 1,
152 fullband_frame_length_) {}
153
154 // Verifies that the capture data is properly received by the block processor
155 // and that the processor data is properly passed to the EchoCanceller3
156 // output.
157 void RunCaptureTransportVerificationTest() {
158 EchoCanceller3 aec3(
159 sample_rate_hz_, false,
160 std::unique_ptr<BlockProcessor>(
161 new CaptureTransportVerificationProcessor(num_bands_)));
162
163 for (size_t frame_index = 0; frame_index < kNumFramesToProcess;
164 ++frame_index) {
165 aec3.AnalyzeCapture(&capture_buffer_);
166 OptionalBandSplit();
167 PopulateInputFrame(frame_length_, num_bands_, frame_index,
168 &capture_buffer_.split_bands_f(0)[0], 0);
peah522d71b2017-02-23 13:16:26169 PopulateInputFrame(frame_length_, frame_index,
170 &render_buffer_.channels_f()[0][0], 0);
peahd0263542017-01-03 12:20:34171
peahcf02cf12017-04-05 21:18:07172 aec3.AnalyzeRender(&render_buffer_);
peahd0263542017-01-03 12:20:34173 aec3.ProcessCapture(&capture_buffer_, false);
174 EXPECT_TRUE(VerifyOutputFrameBitexactness(
175 frame_length_, num_bands_, frame_index,
176 &capture_buffer_.split_bands_f(0)[0], -64));
177 }
178 }
179
180 // Test method for testing that the render data is properly received by the
181 // block processor.
182 void RunRenderTransportVerificationTest() {
183 EchoCanceller3 aec3(
184 sample_rate_hz_, false,
185 std::unique_ptr<BlockProcessor>(
186 new RenderTransportVerificationProcessor(num_bands_)));
187
188 for (size_t frame_index = 0; frame_index < kNumFramesToProcess;
189 ++frame_index) {
190 aec3.AnalyzeCapture(&capture_buffer_);
191 OptionalBandSplit();
192 PopulateInputFrame(frame_length_, num_bands_, frame_index,
193 &capture_buffer_.split_bands_f(0)[0], 100);
peahcf02cf12017-04-05 21:18:07194 PopulateInputFrame(frame_length_, num_bands_, frame_index,
195 &render_buffer_.split_bands_f(0)[0], 0);
peahd0263542017-01-03 12:20:34196
peahcf02cf12017-04-05 21:18:07197 aec3.AnalyzeRender(&render_buffer_);
peahd0263542017-01-03 12:20:34198 aec3.ProcessCapture(&capture_buffer_, false);
199 EXPECT_TRUE(VerifyOutputFrameBitexactness(
peahcf02cf12017-04-05 21:18:07200 frame_length_, num_bands_, frame_index,
201 &capture_buffer_.split_bands_f(0)[0], -64));
peahd0263542017-01-03 12:20:34202 }
203 }
204
205 // Verifies that information about echo path changes are properly propagated
206 // to the block processor.
207 // The cases tested are:
208 // -That no set echo path change flags are received when there is no echo path
209 // change.
210 // -That set echo path change flags are received and continues to be received
211 // as long as echo path changes are flagged.
212 // -That set echo path change flags are no longer received when echo path
213 // change events stop being flagged.
214 enum class EchoPathChangeTestVariant { kNone, kOneSticky, kOneNonSticky };
215
216 void RunEchoPathChangeVerificationTest(
217 EchoPathChangeTestVariant echo_path_change_test_variant) {
218 const size_t num_full_blocks_per_frame =
219 rtc::CheckedDivExact(LowestBandRate(sample_rate_hz_), 100) / kBlockSize;
220 const size_t expected_num_block_to_process =
221 (kNumFramesToProcess *
222 rtc::CheckedDivExact(LowestBandRate(sample_rate_hz_), 100)) /
223 kBlockSize;
224 std::unique_ptr<testing::StrictMock<webrtc::test::MockBlockProcessor>>
225 block_processor_mock(
226 new StrictMock<webrtc::test::MockBlockProcessor>());
227 EXPECT_CALL(*block_processor_mock, BufferRender(_))
peahcf02cf12017-04-05 21:18:07228 .Times(expected_num_block_to_process);
peah69221db2017-01-27 11:28:19229 EXPECT_CALL(*block_processor_mock, UpdateEchoLeakageStatus(_)).Times(0);
peahd0263542017-01-03 12:20:34230
231 switch (echo_path_change_test_variant) {
232 case EchoPathChangeTestVariant::kNone:
233 EXPECT_CALL(*block_processor_mock, ProcessCapture(false, _, _))
234 .Times(expected_num_block_to_process);
235 break;
236 case EchoPathChangeTestVariant::kOneSticky:
237 EXPECT_CALL(*block_processor_mock, ProcessCapture(true, _, _))
238 .Times(expected_num_block_to_process);
239 break;
240 case EchoPathChangeTestVariant::kOneNonSticky:
241 EXPECT_CALL(*block_processor_mock, ProcessCapture(true, _, _))
242 .Times(num_full_blocks_per_frame);
243 EXPECT_CALL(*block_processor_mock, ProcessCapture(false, _, _))
244 .Times(expected_num_block_to_process - num_full_blocks_per_frame);
245 break;
246 }
247
248 EchoCanceller3 aec3(sample_rate_hz_, false,
249 std::move(block_processor_mock));
250
251 for (size_t frame_index = 0; frame_index < kNumFramesToProcess;
252 ++frame_index) {
253 bool echo_path_change = false;
254 switch (echo_path_change_test_variant) {
255 case EchoPathChangeTestVariant::kNone:
256 break;
257 case EchoPathChangeTestVariant::kOneSticky:
258 echo_path_change = true;
259 break;
260 case EchoPathChangeTestVariant::kOneNonSticky:
261 if (frame_index == 0) {
262 echo_path_change = true;
263 }
264 break;
265 }
266
267 aec3.AnalyzeCapture(&capture_buffer_);
268 OptionalBandSplit();
269
270 PopulateInputFrame(frame_length_, num_bands_, frame_index,
271 &capture_buffer_.split_bands_f(0)[0], 0);
peah522d71b2017-02-23 13:16:26272 PopulateInputFrame(frame_length_, frame_index,
273 &render_buffer_.channels_f()[0][0], 0);
peahd0263542017-01-03 12:20:34274
peahcf02cf12017-04-05 21:18:07275 aec3.AnalyzeRender(&render_buffer_);
peahd0263542017-01-03 12:20:34276 aec3.ProcessCapture(&capture_buffer_, echo_path_change);
277 }
278 }
279
280 // Test for verifying that echo leakage information is being properly passed
281 // to the processor.
282 // The cases tested are:
283 // -That no method calls are received when they should not.
284 // -That false values are received each time they are flagged.
285 // -That true values are received each time they are flagged.
286 // -That a false value is received when flagged after a true value has been
287 // flagged.
288 enum class EchoLeakageTestVariant {
289 kNone,
290 kFalseSticky,
291 kTrueSticky,
292 kTrueNonSticky
293 };
294
295 void RunEchoLeakageVerificationTest(
296 EchoLeakageTestVariant leakage_report_variant) {
297 const size_t expected_num_block_to_process =
298 (kNumFramesToProcess *
299 rtc::CheckedDivExact(LowestBandRate(sample_rate_hz_), 100)) /
300 kBlockSize;
301 std::unique_ptr<testing::StrictMock<webrtc::test::MockBlockProcessor>>
302 block_processor_mock(
303 new StrictMock<webrtc::test::MockBlockProcessor>());
304 EXPECT_CALL(*block_processor_mock, BufferRender(_))
peahcf02cf12017-04-05 21:18:07305 .Times(expected_num_block_to_process);
peahd0263542017-01-03 12:20:34306 EXPECT_CALL(*block_processor_mock, ProcessCapture(_, _, _))
307 .Times(expected_num_block_to_process);
308
309 switch (leakage_report_variant) {
310 case EchoLeakageTestVariant::kNone:
peah69221db2017-01-27 11:28:19311 EXPECT_CALL(*block_processor_mock, UpdateEchoLeakageStatus(_)).Times(0);
peahd0263542017-01-03 12:20:34312 break;
313 case EchoLeakageTestVariant::kFalseSticky:
peah69221db2017-01-27 11:28:19314 EXPECT_CALL(*block_processor_mock, UpdateEchoLeakageStatus(false))
315 .Times(1);
peahd0263542017-01-03 12:20:34316 break;
317 case EchoLeakageTestVariant::kTrueSticky:
peah69221db2017-01-27 11:28:19318 EXPECT_CALL(*block_processor_mock, UpdateEchoLeakageStatus(true))
319 .Times(1);
peahd0263542017-01-03 12:20:34320 break;
321 case EchoLeakageTestVariant::kTrueNonSticky: {
322 testing::InSequence s;
peah69221db2017-01-27 11:28:19323 EXPECT_CALL(*block_processor_mock, UpdateEchoLeakageStatus(true))
324 .Times(1);
325 EXPECT_CALL(*block_processor_mock, UpdateEchoLeakageStatus(false))
peahd0263542017-01-03 12:20:34326 .Times(kNumFramesToProcess - 1);
327 } break;
328 }
329
330 EchoCanceller3 aec3(sample_rate_hz_, false,
331 std::move(block_processor_mock));
332
333 for (size_t frame_index = 0; frame_index < kNumFramesToProcess;
334 ++frame_index) {
335 switch (leakage_report_variant) {
336 case EchoLeakageTestVariant::kNone:
337 break;
338 case EchoLeakageTestVariant::kFalseSticky:
339 if (frame_index == 0) {
peah69221db2017-01-27 11:28:19340 aec3.UpdateEchoLeakageStatus(false);
peahd0263542017-01-03 12:20:34341 }
342 break;
343 case EchoLeakageTestVariant::kTrueSticky:
344 if (frame_index == 0) {
peah69221db2017-01-27 11:28:19345 aec3.UpdateEchoLeakageStatus(true);
peahd0263542017-01-03 12:20:34346 }
347 break;
348 case EchoLeakageTestVariant::kTrueNonSticky:
349 if (frame_index == 0) {
peah69221db2017-01-27 11:28:19350 aec3.UpdateEchoLeakageStatus(true);
peahd0263542017-01-03 12:20:34351 } else {
peah69221db2017-01-27 11:28:19352 aec3.UpdateEchoLeakageStatus(false);
peahd0263542017-01-03 12:20:34353 }
354 break;
355 }
356
357 aec3.AnalyzeCapture(&capture_buffer_);
358 OptionalBandSplit();
359
360 PopulateInputFrame(frame_length_, num_bands_, frame_index,
361 &capture_buffer_.split_bands_f(0)[0], 0);
peah522d71b2017-02-23 13:16:26362 PopulateInputFrame(frame_length_, frame_index,
363 &render_buffer_.channels_f()[0][0], 0);
peahd0263542017-01-03 12:20:34364
peahcf02cf12017-04-05 21:18:07365 aec3.AnalyzeRender(&render_buffer_);
peahd0263542017-01-03 12:20:34366 aec3.ProcessCapture(&capture_buffer_, false);
367 }
368 }
369
370 // This verifies that saturation information is properly passed to the
371 // BlockProcessor.
372 // The cases tested are:
373 // -That no saturation event is passed to the processor if there is no
374 // saturation.
375 // -That one frame with one negative saturated sample value is reported to be
376 // saturated and that following non-saturated frames are properly reported as
377 // not being saturated.
378 // -That one frame with one positive saturated sample value is reported to be
379 // saturated and that following non-saturated frames are properly reported as
380 // not being saturated.
381 enum class SaturationTestVariant { kNone, kOneNegative, kOnePositive };
382
383 void RunCaptureSaturationVerificationTest(
384 SaturationTestVariant saturation_variant) {
385 const size_t num_full_blocks_per_frame =
386 rtc::CheckedDivExact(LowestBandRate(sample_rate_hz_), 100) / kBlockSize;
387 const size_t expected_num_block_to_process =
388 (kNumFramesToProcess *
389 rtc::CheckedDivExact(LowestBandRate(sample_rate_hz_), 100)) /
390 kBlockSize;
391 std::unique_ptr<testing::StrictMock<webrtc::test::MockBlockProcessor>>
392 block_processor_mock(
393 new StrictMock<webrtc::test::MockBlockProcessor>());
394 EXPECT_CALL(*block_processor_mock, BufferRender(_))
peahcf02cf12017-04-05 21:18:07395 .Times(expected_num_block_to_process);
peah69221db2017-01-27 11:28:19396 EXPECT_CALL(*block_processor_mock, UpdateEchoLeakageStatus(_)).Times(0);
peahd0263542017-01-03 12:20:34397
398 switch (saturation_variant) {
399 case SaturationTestVariant::kNone:
400 EXPECT_CALL(*block_processor_mock, ProcessCapture(_, false, _))
401 .Times(expected_num_block_to_process);
402 break;
403 case SaturationTestVariant::kOneNegative: {
404 testing::InSequence s;
405 EXPECT_CALL(*block_processor_mock, ProcessCapture(_, true, _))
406 .Times(num_full_blocks_per_frame);
407 EXPECT_CALL(*block_processor_mock, ProcessCapture(_, false, _))
408 .Times(expected_num_block_to_process - num_full_blocks_per_frame);
409 } break;
410 case SaturationTestVariant::kOnePositive: {
411 testing::InSequence s;
412 EXPECT_CALL(*block_processor_mock, ProcessCapture(_, true, _))
413 .Times(num_full_blocks_per_frame);
414 EXPECT_CALL(*block_processor_mock, ProcessCapture(_, false, _))
415 .Times(expected_num_block_to_process - num_full_blocks_per_frame);
416 } break;
417 }
418
419 EchoCanceller3 aec3(sample_rate_hz_, false,
420 std::move(block_processor_mock));
peahd0263542017-01-03 12:20:34421 for (size_t frame_index = 0; frame_index < kNumFramesToProcess;
422 ++frame_index) {
423 for (int k = 0; k < fullband_frame_length_; ++k) {
424 capture_buffer_.channels_f()[0][k] = 0.f;
425 }
426 switch (saturation_variant) {
427 case SaturationTestVariant::kNone:
428 break;
429 case SaturationTestVariant::kOneNegative:
430 if (frame_index == 0) {
431 capture_buffer_.channels_f()[0][10] = -32768.f;
432 }
433 break;
434 case SaturationTestVariant::kOnePositive:
435 if (frame_index == 0) {
436 capture_buffer_.channels_f()[0][10] = 32767.f;
437 }
438 break;
439 }
440
441 aec3.AnalyzeCapture(&capture_buffer_);
442 OptionalBandSplit();
443
444 PopulateInputFrame(frame_length_, num_bands_, frame_index,
445 &capture_buffer_.split_bands_f(0)[0], 0);
peahcf02cf12017-04-05 21:18:07446 PopulateInputFrame(frame_length_, num_bands_, frame_index,
447 &render_buffer_.split_bands_f(0)[0], 0);
peahd0263542017-01-03 12:20:34448
peahcf02cf12017-04-05 21:18:07449 aec3.AnalyzeRender(&render_buffer_);
peahd0263542017-01-03 12:20:34450 aec3.ProcessCapture(&capture_buffer_, false);
451 }
452 }
453
454 // This test verifies that the swapqueue is able to handle jitter in the
455 // capture and render API calls.
456 void RunRenderSwapQueueVerificationTest() {
457 EchoCanceller3 aec3(
458 sample_rate_hz_, false,
459 std::unique_ptr<BlockProcessor>(
460 new RenderTransportVerificationProcessor(num_bands_)));
461
peahcf02cf12017-04-05 21:18:07462 for (size_t frame_index = 0; frame_index < kRenderTransferQueueSize;
peahd0263542017-01-03 12:20:34463 ++frame_index) {
464 if (sample_rate_hz_ > 16000) {
465 render_buffer_.SplitIntoFrequencyBands();
466 }
peahcf02cf12017-04-05 21:18:07467 PopulateInputFrame(frame_length_, num_bands_, frame_index,
468 &render_buffer_.split_bands_f(0)[0], 0);
peahd0263542017-01-03 12:20:34469
peahcf02cf12017-04-05 21:18:07470 if (sample_rate_hz_ > 16000) {
471 render_buffer_.SplitIntoFrequencyBands();
472 }
473
474 aec3.AnalyzeRender(&render_buffer_);
peahd0263542017-01-03 12:20:34475 }
476
peahcf02cf12017-04-05 21:18:07477 for (size_t frame_index = 0; frame_index < kRenderTransferQueueSize;
peahd0263542017-01-03 12:20:34478 ++frame_index) {
479 aec3.AnalyzeCapture(&capture_buffer_);
480 if (sample_rate_hz_ > 16000) {
481 capture_buffer_.SplitIntoFrequencyBands();
482 }
483
484 PopulateInputFrame(frame_length_, num_bands_, frame_index,
485 &capture_buffer_.split_bands_f(0)[0], 0);
486
487 aec3.ProcessCapture(&capture_buffer_, false);
488 EXPECT_TRUE(VerifyOutputFrameBitexactness(
peahcf02cf12017-04-05 21:18:07489 frame_length_, num_bands_, frame_index,
490 &capture_buffer_.split_bands_f(0)[0], -64));
peahd0263542017-01-03 12:20:34491 }
492 }
493
494 // This test verifies that a buffer overrun in the render swapqueue is
495 // properly reported.
496 void RunRenderPipelineSwapQueueOverrunReturnValueTest() {
peah697a5902017-06-30 14:06:10497 EchoCanceller3 aec3(AudioProcessing::Config::EchoCanceller3(),
498 sample_rate_hz_, false);
peahd0263542017-01-03 12:20:34499
peahcf02cf12017-04-05 21:18:07500 constexpr size_t kRenderTransferQueueSize = 30;
peahd0263542017-01-03 12:20:34501 for (size_t k = 0; k < 2; ++k) {
peahcf02cf12017-04-05 21:18:07502 for (size_t frame_index = 0; frame_index < kRenderTransferQueueSize;
peahd0263542017-01-03 12:20:34503 ++frame_index) {
504 if (sample_rate_hz_ > 16000) {
505 render_buffer_.SplitIntoFrequencyBands();
506 }
peah522d71b2017-02-23 13:16:26507 PopulateInputFrame(frame_length_, frame_index,
508 &render_buffer_.channels_f()[0][0], 0);
peahd0263542017-01-03 12:20:34509
510 if (k == 0) {
peahcf02cf12017-04-05 21:18:07511 aec3.AnalyzeRender(&render_buffer_);
peahd0263542017-01-03 12:20:34512 } else {
peahcf02cf12017-04-05 21:18:07513 aec3.AnalyzeRender(&render_buffer_);
peahd0263542017-01-03 12:20:34514 }
515 }
516 }
517 }
518
519#if RTC_DCHECK_IS_ON && GTEST_HAS_DEATH_TEST && !defined(WEBRTC_ANDROID)
520 // Verifies the that the check for the number of bands in the AnalyzeRender
521 // input is correct by adjusting the sample rates of EchoCanceller3 and the
522 // input AudioBuffer to have a different number of bands.
523 void RunAnalyzeRenderNumBandsCheckVerification() {
524 // Set aec3_sample_rate_hz to be different from sample_rate_hz_ in such a
525 // way that the number of bands for the rates are different.
526 const int aec3_sample_rate_hz = sample_rate_hz_ == 48000 ? 32000 : 48000;
peah697a5902017-06-30 14:06:10527 EchoCanceller3 aec3(AudioProcessing::Config::EchoCanceller3(),
528 aec3_sample_rate_hz, false);
peah522d71b2017-02-23 13:16:26529 PopulateInputFrame(frame_length_, 0, &render_buffer_.channels_f()[0][0], 0);
peahd0263542017-01-03 12:20:34530
531 EXPECT_DEATH(aec3.AnalyzeRender(&render_buffer_), "");
532 }
533
534 // Verifies the that the check for the number of bands in the ProcessCapture
535 // input is correct by adjusting the sample rates of EchoCanceller3 and the
536 // input AudioBuffer to have a different number of bands.
537 void RunProcessCaptureNumBandsCheckVerification() {
538 // Set aec3_sample_rate_hz to be different from sample_rate_hz_ in such a
539 // way that the number of bands for the rates are different.
540 const int aec3_sample_rate_hz = sample_rate_hz_ == 48000 ? 32000 : 48000;
peah697a5902017-06-30 14:06:10541 EchoCanceller3 aec3(AudioProcessing::Config::EchoCanceller3(),
542 aec3_sample_rate_hz, false);
peahd0263542017-01-03 12:20:34543 PopulateInputFrame(frame_length_, num_bands_, 0,
544 &capture_buffer_.split_bands_f(0)[0], 100);
545 EXPECT_DEATH(aec3.ProcessCapture(&capture_buffer_, false), "");
546 }
547
548 // Verifies the that the check for the frame length in the AnalyzeRender input
549 // is correct by adjusting the sample rates of EchoCanceller3 and the input
550 // AudioBuffer to have a different frame lengths.
551 void RunAnalyzeRenderFrameLengthCheckVerification() {
552 // Set aec3_sample_rate_hz to be different from sample_rate_hz_ in such a
553 // way that the band frame lengths are different.
554 const int aec3_sample_rate_hz = sample_rate_hz_ == 8000 ? 16000 : 8000;
peah697a5902017-06-30 14:06:10555 EchoCanceller3 aec3(AudioProcessing::Config::EchoCanceller3(),
556 aec3_sample_rate_hz, false);
peahd0263542017-01-03 12:20:34557
558 OptionalBandSplit();
peah522d71b2017-02-23 13:16:26559 PopulateInputFrame(frame_length_, 0, &render_buffer_.channels_f()[0][0], 0);
peahd0263542017-01-03 12:20:34560
561 EXPECT_DEATH(aec3.AnalyzeRender(&render_buffer_), "");
562 }
563
564 // Verifies the that the check for the frame length in the AnalyzeRender input
565 // is correct by adjusting the sample rates of EchoCanceller3 and the input
566 // AudioBuffer to have a different frame lengths.
567 void RunProcessCaptureFrameLengthCheckVerification() {
568 // Set aec3_sample_rate_hz to be different from sample_rate_hz_ in such a
569 // way that the band frame lengths are different.
570 const int aec3_sample_rate_hz = sample_rate_hz_ == 8000 ? 16000 : 8000;
peah697a5902017-06-30 14:06:10571 EchoCanceller3 aec3(AudioProcessing::Config::EchoCanceller3(),
572 aec3_sample_rate_hz, false);
peahd0263542017-01-03 12:20:34573
574 OptionalBandSplit();
575 PopulateInputFrame(frame_length_, num_bands_, 0,
576 &capture_buffer_.split_bands_f(0)[0], 100);
577
578 EXPECT_DEATH(aec3.ProcessCapture(&capture_buffer_, false), "");
579 }
580
581#endif
582
583 private:
584 void OptionalBandSplit() {
585 if (sample_rate_hz_ > 16000) {
586 capture_buffer_.SplitIntoFrequencyBands();
587 render_buffer_.SplitIntoFrequencyBands();
588 }
589 }
590
591 static constexpr size_t kNumFramesToProcess = 20;
592 const int sample_rate_hz_;
593 const size_t num_bands_;
594 const size_t frame_length_;
595 const int fullband_frame_length_;
596 AudioBuffer capture_buffer_;
597 AudioBuffer render_buffer_;
598
599 RTC_DISALLOW_IMPLICIT_CONSTRUCTORS(EchoCanceller3Tester);
600};
601
602std::string ProduceDebugText(int sample_rate_hz) {
603 std::ostringstream ss;
604 ss << "Sample rate: " << sample_rate_hz;
605 return ss.str();
606}
607
608std::string ProduceDebugText(int sample_rate_hz, int variant) {
609 std::ostringstream ss;
610 ss << "Sample rate: " << sample_rate_hz << ", variant: " << variant;
611 return ss.str();
612}
613
614} // namespace
615
616TEST(EchoCanceller3Buffering, CaptureBitexactness) {
617 for (auto rate : {8000, 16000, 32000, 48000}) {
618 SCOPED_TRACE(ProduceDebugText(rate));
619 EchoCanceller3Tester(rate).RunCaptureTransportVerificationTest();
620 }
621}
622
623TEST(EchoCanceller3Buffering, RenderBitexactness) {
624 for (auto rate : {8000, 16000, 32000, 48000}) {
625 SCOPED_TRACE(ProduceDebugText(rate));
626 EchoCanceller3Tester(rate).RunRenderTransportVerificationTest();
627 }
628}
629
630TEST(EchoCanceller3Buffering, RenderSwapQueue) {
peahcf02cf12017-04-05 21:18:07631 for (auto rate : {8000, 16000}) {
peahd0263542017-01-03 12:20:34632 SCOPED_TRACE(ProduceDebugText(rate));
633 EchoCanceller3Tester(rate).RunRenderSwapQueueVerificationTest();
634 }
635}
636
637TEST(EchoCanceller3Buffering, RenderSwapQueueOverrunReturnValue) {
638 for (auto rate : {8000, 16000, 32000, 48000}) {
639 SCOPED_TRACE(ProduceDebugText(rate));
640 EchoCanceller3Tester(rate)
641 .RunRenderPipelineSwapQueueOverrunReturnValueTest();
642 }
643}
644
645TEST(EchoCanceller3Messaging, CaptureSaturation) {
646 auto variants = {EchoCanceller3Tester::SaturationTestVariant::kNone,
647 EchoCanceller3Tester::SaturationTestVariant::kOneNegative,
648 EchoCanceller3Tester::SaturationTestVariant::kOnePositive};
649 for (auto rate : {8000, 16000, 32000, 48000}) {
650 for (auto variant : variants) {
651 SCOPED_TRACE(ProduceDebugText(rate, static_cast<int>(variant)));
652 EchoCanceller3Tester(rate).RunCaptureSaturationVerificationTest(variant);
653 }
654 }
655}
656
657TEST(EchoCanceller3Messaging, EchoPathChange) {
658 auto variants = {
659 EchoCanceller3Tester::EchoPathChangeTestVariant::kNone,
660 EchoCanceller3Tester::EchoPathChangeTestVariant::kOneSticky,
661 EchoCanceller3Tester::EchoPathChangeTestVariant::kOneNonSticky};
662 for (auto rate : {8000, 16000, 32000, 48000}) {
663 for (auto variant : variants) {
664 SCOPED_TRACE(ProduceDebugText(rate, static_cast<int>(variant)));
665 EchoCanceller3Tester(rate).RunEchoPathChangeVerificationTest(variant);
666 }
667 }
668}
669
670TEST(EchoCanceller3Messaging, EchoLeakage) {
671 auto variants = {
672 EchoCanceller3Tester::EchoLeakageTestVariant::kNone,
673 EchoCanceller3Tester::EchoLeakageTestVariant::kFalseSticky,
674 EchoCanceller3Tester::EchoLeakageTestVariant::kTrueSticky,
675 EchoCanceller3Tester::EchoLeakageTestVariant::kTrueNonSticky};
676 for (auto rate : {8000, 16000, 32000, 48000}) {
677 for (auto variant : variants) {
678 SCOPED_TRACE(ProduceDebugText(rate, static_cast<int>(variant)));
679 EchoCanceller3Tester(rate).RunEchoLeakageVerificationTest(variant);
680 }
681 }
682}
683
peah697a5902017-06-30 14:06:10684TEST(EchoCanceller3, ConfigValidation) {
685 AudioProcessing::Config::EchoCanceller3 config;
peah697a5902017-06-30 14:06:10686 EXPECT_TRUE(EchoCanceller3::Validate(config));
peah697a5902017-06-30 14:06:10687}
688
peahd0263542017-01-03 12:20:34689#if RTC_DCHECK_IS_ON && GTEST_HAS_DEATH_TEST && !defined(WEBRTC_ANDROID)
peahd0263542017-01-03 12:20:34690
691TEST(EchoCanceller3InputCheck, WrongCaptureNumBandsCheckVerification) {
692 for (auto rate : {8000, 16000, 32000, 48000}) {
693 SCOPED_TRACE(ProduceDebugText(rate));
694 EchoCanceller3Tester(rate).RunProcessCaptureNumBandsCheckVerification();
695 }
696}
697
peah522d71b2017-02-23 13:16:26698// TODO(peah): Re-enable the test once the issue with memory leaks during DEATH
699// tests on test bots has been fixed.
700TEST(EchoCanceller3InputCheck,
701 DISABLED_WrongRenderFrameLengthCheckVerification) {
peahd0263542017-01-03 12:20:34702 for (auto rate : {8000, 16000}) {
703 SCOPED_TRACE(ProduceDebugText(rate));
704 EchoCanceller3Tester(rate).RunAnalyzeRenderFrameLengthCheckVerification();
705 }
706}
707
708TEST(EchoCanceller3InputCheck, WrongCaptureFrameLengthCheckVerification) {
709 for (auto rate : {8000, 16000}) {
710 SCOPED_TRACE(ProduceDebugText(rate));
711 EchoCanceller3Tester(rate).RunProcessCaptureFrameLengthCheckVerification();
712 }
713}
714
715// Verifiers that the verification for null input to the render analysis api
716// call works.
717TEST(EchoCanceller3InputCheck, NullRenderAnalysisParameter) {
peah697a5902017-06-30 14:06:10718 EXPECT_DEATH(
719 EchoCanceller3(AudioProcessing::Config::EchoCanceller3(), 8000, false)
720 .AnalyzeRender(nullptr),
721 "");
peahd0263542017-01-03 12:20:34722}
723
724// Verifiers that the verification for null input to the capture analysis api
725// call works.
726TEST(EchoCanceller3InputCheck, NullCaptureAnalysisParameter) {
peah697a5902017-06-30 14:06:10727 EXPECT_DEATH(
728 EchoCanceller3(AudioProcessing::Config::EchoCanceller3(), 8000, false)
729 .AnalyzeCapture(nullptr),
730 "");
peahd0263542017-01-03 12:20:34731}
732
733// Verifiers that the verification for null input to the capture processing api
734// call works.
735TEST(EchoCanceller3InputCheck, NullCaptureProcessingParameter) {
peah697a5902017-06-30 14:06:10736 EXPECT_DEATH(
737 EchoCanceller3(AudioProcessing::Config::EchoCanceller3(), 8000, false)
738 .ProcessCapture(nullptr, false),
739 "");
peahd0263542017-01-03 12:20:34740}
741
peah21920892017-02-08 13:08:56742// Verifies the check for correct sample rate.
peahcf02cf12017-04-05 21:18:07743// TODO(peah): Re-enable the test once the issue with memory leaks during DEATH
744// tests on test bots has been fixed.
745TEST(EchoCanceller3InputCheck, DISABLED_WrongSampleRate) {
peah21920892017-02-08 13:08:56746 ApmDataDumper data_dumper(0);
peah697a5902017-06-30 14:06:10747 EXPECT_DEATH(
748 EchoCanceller3(AudioProcessing::Config::EchoCanceller3(), 8001, false),
749 "");
peah21920892017-02-08 13:08:56750}
751
peahd0263542017-01-03 12:20:34752#endif
753
754} // namespace webrtc