OLD | NEW |
(Empty) | |
| 1 /* |
| 2 * Copyright (c) 2017 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/aec_state.h" |
| 12 |
| 13 #include "webrtc/test/gtest.h" |
| 14 |
| 15 namespace webrtc { |
| 16 |
| 17 // Verify the general functionality of AecState |
| 18 TEST(AecState, NormalUsage) { |
| 19 ApmDataDumper data_dumper(42); |
| 20 AecState state; |
| 21 FftBuffer X_buffer(30, std::vector<size_t>(1, 30)); |
| 22 std::array<float, kFftLengthBy2Plus1> E2_main; |
| 23 std::array<float, kFftLengthBy2Plus1> E2_shadow; |
| 24 std::array<float, kFftLengthBy2Plus1> Y2; |
| 25 std::array<float, kBlockSize> x; |
| 26 DelayHandler delay_handler; |
| 27 EchoPathVariability echo_path_variability(false, false); |
| 28 x.fill(0.f); |
| 29 |
| 30 std::vector<std::array<float, kFftLengthBy2Plus1>> |
| 31 converged_filter_frequency_response(10); |
| 32 for (auto& v : converged_filter_frequency_response) { |
| 33 v.fill(0.01f); |
| 34 } |
| 35 std::vector<std::array<float, kFftLengthBy2Plus1>> |
| 36 diverged_filter_frequency_response = converged_filter_frequency_response; |
| 37 converged_filter_frequency_response[2].fill(100.f); |
| 38 |
| 39 // Verify that model based aec feasibility and linear AEC usability are false |
| 40 // when the filter is diverged and there is no external delay reported. |
| 41 delay_handler.UpdateDelays(diverged_filter_frequency_response, |
| 42 rtc::Optional<size_t>()); |
| 43 state.Update(X_buffer, E2_main, E2_shadow, Y2, x, delay_handler, |
| 44 echo_path_variability, false); |
| 45 EXPECT_FALSE(state.ModelBasedAecFeasible()); |
| 46 EXPECT_FALSE(state.UsableLinearEstimate()); |
| 47 |
| 48 // Verify that model based aec feasibility is true and that linear AEC |
| 49 // usability is false when the filter is diverged and there is an external |
| 50 // delay reported. |
| 51 delay_handler.UpdateDelays(diverged_filter_frequency_response, |
| 52 rtc::Optional<size_t>(2)); |
| 53 state.Update(X_buffer, E2_main, E2_shadow, Y2, x, delay_handler, |
| 54 echo_path_variability, false); |
| 55 EXPECT_FALSE(state.ModelBasedAecFeasible()); |
| 56 for (int k = 0; k < 50; ++k) { |
| 57 state.Update(X_buffer, E2_main, E2_shadow, Y2, x, delay_handler, |
| 58 echo_path_variability, false); |
| 59 } |
| 60 EXPECT_TRUE(state.ModelBasedAecFeasible()); |
| 61 EXPECT_FALSE(state.UsableLinearEstimate()); |
| 62 |
| 63 // Verify that linear AEC usability is true when the filter is converged |
| 64 delay_handler.UpdateDelays(converged_filter_frequency_response, |
| 65 rtc::Optional<size_t>(2)); |
| 66 for (int k = 0; k < 50; ++k) { |
| 67 state.Update(X_buffer, E2_main, E2_shadow, Y2, x, delay_handler, |
| 68 echo_path_variability, false); |
| 69 } |
| 70 EXPECT_TRUE(state.UsableLinearEstimate()); |
| 71 |
| 72 // Verify that linear AEC usability becomes false after an echo path change is |
| 73 // reported |
| 74 echo_path_variability = EchoPathVariability(true, false); |
| 75 state.Update(X_buffer, E2_main, E2_shadow, Y2, x, delay_handler, |
| 76 echo_path_variability, false); |
| 77 EXPECT_FALSE(state.UsableLinearEstimate()); |
| 78 |
| 79 // Verify that the active render detection works as intended. |
| 80 x.fill(101.f); |
| 81 state.Update(X_buffer, E2_main, E2_shadow, Y2, x, delay_handler, |
| 82 echo_path_variability, false); |
| 83 EXPECT_TRUE(state.ActiveRender()); |
| 84 |
| 85 x.fill(0.f); |
| 86 for (int k = 0; k < 200; ++k) { |
| 87 state.Update(X_buffer, E2_main, E2_shadow, Y2, x, delay_handler, |
| 88 echo_path_variability, false); |
| 89 } |
| 90 EXPECT_FALSE(state.ActiveRender()); |
| 91 |
| 92 x.fill(101.f); |
| 93 state.Update(X_buffer, E2_main, E2_shadow, Y2, x, delay_handler, |
| 94 echo_path_variability, false); |
| 95 EXPECT_TRUE(state.ActiveRender()); |
| 96 |
| 97 // Verify that echo leakage is properly reported. |
| 98 state.Update(X_buffer, E2_main, E2_shadow, Y2, x, delay_handler, |
| 99 echo_path_variability, false); |
| 100 EXPECT_FALSE(state.EchoLeakageDetected()); |
| 101 |
| 102 state.Update(X_buffer, E2_main, E2_shadow, Y2, x, delay_handler, |
| 103 echo_path_variability, true); |
| 104 EXPECT_TRUE(state.EchoLeakageDetected()); |
| 105 |
| 106 // Verify that the bands containing reliable filter estimates are properly |
| 107 // reported. |
| 108 echo_path_variability = EchoPathVariability(false, false); |
| 109 delay_handler.UpdateDelays(converged_filter_frequency_response, |
| 110 rtc::Optional<size_t>(2)); |
| 111 for (int k = 0; k < 200; ++k) { |
| 112 state.Update(X_buffer, E2_main, E2_shadow, Y2, x, delay_handler, |
| 113 echo_path_variability, false); |
| 114 } |
| 115 |
| 116 FftData X; |
| 117 X.re.fill(10000.f); |
| 118 X.im.fill(0.f); |
| 119 for (size_t k = 0; k < X_buffer.Buffer().size(); ++k) { |
| 120 X_buffer.Insert(X); |
| 121 } |
| 122 |
| 123 Y2.fill(10.f * 1000.f * 1000.f); |
| 124 E2_main.fill(100.f * Y2[0]); |
| 125 E2_shadow.fill(100.f * Y2[0]); |
| 126 state.Update(X_buffer, E2_main, E2_shadow, Y2, x, delay_handler, |
| 127 echo_path_variability, false); |
| 128 |
| 129 E2_main.fill(0.1f * Y2[0]); |
| 130 E2_shadow.fill(E2_main[0]); |
| 131 for (size_t k = 0; k < Y2.size(); k += 2) { |
| 132 E2_main[k] = Y2[k]; |
| 133 E2_shadow[k] = Y2[k]; |
| 134 } |
| 135 state.Update(X_buffer, E2_main, E2_shadow, Y2, x, delay_handler, |
| 136 echo_path_variability, false); |
| 137 |
| 138 const std::array<bool, kFftLengthBy2Plus1>& reliable_bands = |
| 139 state.BandsWithReliableFilter(); |
| 140 |
| 141 EXPECT_EQ(reliable_bands[0], reliable_bands[1]); |
| 142 for (size_t k = 1; k < kFftLengthBy2 - 5; ++k) { |
| 143 EXPECT_EQ(2 * E2_main[k] < Y2[k], reliable_bands[k]); |
| 144 } |
| 145 for (size_t k = kFftLengthBy2 - 5; k < reliable_bands.size(); ++k) { |
| 146 EXPECT_EQ(reliable_bands[kFftLengthBy2 - 6], reliable_bands[k]); |
| 147 } |
| 148 |
| 149 // Verify that the ERL is properly estimated |
| 150 Y2.fill(10.f * X.re[0] * X.re[0]); |
| 151 for (size_t k = 0; k < 100000; ++k) { |
| 152 state.Update(X_buffer, E2_main, E2_shadow, Y2, x, delay_handler, |
| 153 echo_path_variability, false); |
| 154 } |
| 155 |
| 156 ASSERT_TRUE(state.UsableLinearEstimate()); |
| 157 const std::array<float, kFftLengthBy2Plus1>& erl = state.Erl(); |
| 158 std::for_each(erl.begin(), erl.end(), |
| 159 [](float a) { EXPECT_NEAR(10.f, a, 0.1); }); |
| 160 |
| 161 // Verify that the ERLE is properly estimated |
| 162 E2_main.fill(1.f * X.re[0] * X.re[0]); |
| 163 Y2.fill(10.f * E2_main[0]); |
| 164 for (size_t k = 0; k < 10000; ++k) { |
| 165 state.Update(X_buffer, E2_main, E2_shadow, Y2, x, delay_handler, |
| 166 echo_path_variability, false); |
| 167 } |
| 168 ASSERT_TRUE(state.UsableLinearEstimate()); |
| 169 std::for_each(state.Erle().begin(), state.Erle().end(), |
| 170 [](float a) { EXPECT_NEAR(8.f, a, 0.1); }); |
| 171 |
| 172 E2_main.fill(1.f * X.re[0] * X.re[0]); |
| 173 Y2.fill(5.f * E2_main[0]); |
| 174 for (size_t k = 0; k < 10000; ++k) { |
| 175 state.Update(X_buffer, E2_main, E2_shadow, Y2, x, delay_handler, |
| 176 echo_path_variability, false); |
| 177 } |
| 178 ASSERT_TRUE(state.UsableLinearEstimate()); |
| 179 std::for_each(state.Erle().begin(), state.Erle().end(), |
| 180 [](float a) { EXPECT_NEAR(5.f, a, 0.1); }); |
| 181 } |
| 182 |
| 183 } // namespace webrtc |
OLD | NEW |