Per Åhgren | b6b00dc | 2018-02-20 22:18:27 +0100 | [diff] [blame] | 1 | /* |
| 2 | * Copyright (c) 2018 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 | #include "api/audio/echo_canceller3_config.h" |
| 11 | |
Sam Zackrisson | a4c8514 | 2018-10-10 10:44:43 +0200 | [diff] [blame] | 12 | #include <algorithm> |
Sam Zackrisson | 877dc89 | 2018-10-23 14:17:38 +0200 | [diff] [blame] | 13 | #include <cmath> |
Sam Zackrisson | a4c8514 | 2018-10-10 10:44:43 +0200 | [diff] [blame] | 14 | |
Yves Gerey | 988cc08 | 2018-10-23 12:03:01 +0200 | [diff] [blame] | 15 | #include "rtc_base/checks.h" |
Sam Zackrisson | a4c8514 | 2018-10-10 10:44:43 +0200 | [diff] [blame] | 16 | #include "rtc_base/numerics/safe_minmax.h" |
| 17 | |
Per Åhgren | b6b00dc | 2018-02-20 22:18:27 +0100 | [diff] [blame] | 18 | namespace webrtc { |
Sam Zackrisson | a4c8514 | 2018-10-10 10:44:43 +0200 | [diff] [blame] | 19 | namespace { |
| 20 | bool Limit(float* value, float min, float max) { |
| 21 | float clamped = rtc::SafeClamp(*value, min, max); |
Sam Zackrisson | 877dc89 | 2018-10-23 14:17:38 +0200 | [diff] [blame] | 22 | clamped = std::isfinite(clamped) ? clamped : min; |
Sam Zackrisson | a4c8514 | 2018-10-10 10:44:43 +0200 | [diff] [blame] | 23 | bool res = *value == clamped; |
| 24 | *value = clamped; |
| 25 | return res; |
| 26 | } |
| 27 | |
| 28 | bool Limit(size_t* value, size_t min, size_t max) { |
| 29 | size_t clamped = rtc::SafeClamp(*value, min, max); |
| 30 | bool res = *value == clamped; |
| 31 | *value = clamped; |
| 32 | return res; |
| 33 | } |
| 34 | |
| 35 | bool Limit(int* value, int min, int max) { |
| 36 | int clamped = rtc::SafeClamp(*value, min, max); |
| 37 | bool res = *value == clamped; |
| 38 | *value = clamped; |
| 39 | return res; |
| 40 | } |
| 41 | } // namespace |
Per Åhgren | b6b00dc | 2018-02-20 22:18:27 +0100 | [diff] [blame] | 42 | |
| 43 | EchoCanceller3Config::EchoCanceller3Config() = default; |
Per Åhgren | 251c735 | 2018-03-28 16:31:57 +0200 | [diff] [blame] | 44 | EchoCanceller3Config::EchoCanceller3Config(const EchoCanceller3Config& e) = |
| 45 | default; |
Per Åhgren | 398689f | 2018-08-23 11:38:27 +0200 | [diff] [blame] | 46 | EchoCanceller3Config::Delay::Delay() = default; |
| 47 | EchoCanceller3Config::Delay::Delay(const EchoCanceller3Config::Delay& e) = |
| 48 | default; |
| 49 | |
Jesús de Vicente Peña | dd09287 | 2018-05-25 16:55:11 +0200 | [diff] [blame] | 50 | EchoCanceller3Config::EchoModel::EchoModel() = default; |
| 51 | EchoCanceller3Config::EchoModel::EchoModel( |
| 52 | const EchoCanceller3Config::EchoModel& e) = default; |
| 53 | |
Per Åhgren | 524e878 | 2018-08-24 22:48:49 +0200 | [diff] [blame] | 54 | EchoCanceller3Config::Suppressor::Suppressor() = default; |
| 55 | EchoCanceller3Config::Suppressor::Suppressor( |
| 56 | const EchoCanceller3Config::Suppressor& e) = default; |
| 57 | |
| 58 | EchoCanceller3Config::Suppressor::MaskingThresholds::MaskingThresholds( |
| 59 | float enr_transparent, |
| 60 | float enr_suppress, |
| 61 | float emr_transparent) |
| 62 | : enr_transparent(enr_transparent), |
| 63 | enr_suppress(enr_suppress), |
| 64 | emr_transparent(emr_transparent) {} |
| 65 | EchoCanceller3Config::Suppressor::Suppressor::MaskingThresholds:: |
| 66 | MaskingThresholds( |
| 67 | const EchoCanceller3Config::Suppressor::MaskingThresholds& e) = default; |
| 68 | |
| 69 | EchoCanceller3Config::Suppressor::Tuning::Tuning(MaskingThresholds mask_lf, |
| 70 | MaskingThresholds mask_hf, |
| 71 | float max_inc_factor, |
| 72 | float max_dec_factor_lf) |
| 73 | : mask_lf(mask_lf), |
| 74 | mask_hf(mask_hf), |
| 75 | max_inc_factor(max_inc_factor), |
| 76 | max_dec_factor_lf(max_dec_factor_lf) {} |
| 77 | EchoCanceller3Config::Suppressor::Tuning::Tuning( |
| 78 | const EchoCanceller3Config::Suppressor::Tuning& e) = default; |
| 79 | |
Sam Zackrisson | a4c8514 | 2018-10-10 10:44:43 +0200 | [diff] [blame] | 80 | bool EchoCanceller3Config::Validate(EchoCanceller3Config* config) { |
| 81 | RTC_DCHECK(config); |
| 82 | EchoCanceller3Config* c = config; |
| 83 | bool res = true; |
| 84 | |
| 85 | if (c->delay.down_sampling_factor != 4 && |
| 86 | c->delay.down_sampling_factor != 8) { |
| 87 | c->delay.down_sampling_factor = 4; |
| 88 | res = false; |
| 89 | } |
Gustaf Ullberg | 9249fbf | 2019-03-14 11:24:54 +0100 | [diff] [blame] | 90 | |
Sam Zackrisson | 877dc89 | 2018-10-23 14:17:38 +0200 | [diff] [blame] | 91 | res = res & Limit(&c->delay.default_delay, 0, 5000); |
| 92 | res = res & Limit(&c->delay.num_filters, 0, 5000); |
Gustaf Ullberg | 9249fbf | 2019-03-14 11:24:54 +0100 | [diff] [blame] | 93 | res = res & Limit(&c->delay.delay_headroom_samples, 0, 5000); |
| 94 | res = res & Limit(&c->delay.hysteresis_limit_blocks, 0, 5000); |
Sam Zackrisson | 877dc89 | 2018-10-23 14:17:38 +0200 | [diff] [blame] | 95 | res = res & Limit(&c->delay.fixed_capture_delay_samples, 0, 5000); |
| 96 | res = res & Limit(&c->delay.delay_estimate_smoothing, 0.f, 1.f); |
| 97 | res = res & Limit(&c->delay.delay_candidate_detection_threshold, 0.f, 1.f); |
| 98 | res = res & Limit(&c->delay.delay_selection_thresholds.initial, 1, 250); |
| 99 | res = res & Limit(&c->delay.delay_selection_thresholds.converged, 1, 250); |
Sam Zackrisson | a4c8514 | 2018-10-10 10:44:43 +0200 | [diff] [blame] | 100 | |
Sam Zackrisson | 877dc89 | 2018-10-23 14:17:38 +0200 | [diff] [blame] | 101 | res = res & Limit(&c->filter.main.length_blocks, 1, 50); |
| 102 | res = res & Limit(&c->filter.main.leakage_converged, 0.f, 1000.f); |
| 103 | res = res & Limit(&c->filter.main.leakage_diverged, 0.f, 1000.f); |
| 104 | res = res & Limit(&c->filter.main.error_floor, 0.f, 1000.f); |
| 105 | res = res & Limit(&c->filter.main.error_ceil, 0.f, 100000000.f); |
| 106 | res = res & Limit(&c->filter.main.noise_gate, 0.f, 100000000.f); |
Sam Zackrisson | a4c8514 | 2018-10-10 10:44:43 +0200 | [diff] [blame] | 107 | |
Sam Zackrisson | 877dc89 | 2018-10-23 14:17:38 +0200 | [diff] [blame] | 108 | res = res & Limit(&c->filter.main_initial.length_blocks, 1, 50); |
| 109 | res = res & Limit(&c->filter.main_initial.leakage_converged, 0.f, 1000.f); |
| 110 | res = res & Limit(&c->filter.main_initial.leakage_diverged, 0.f, 1000.f); |
| 111 | res = res & Limit(&c->filter.main_initial.error_floor, 0.f, 1000.f); |
| 112 | res = res & Limit(&c->filter.main_initial.error_ceil, 0.f, 100000000.f); |
| 113 | res = res & Limit(&c->filter.main_initial.noise_gate, 0.f, 100000000.f); |
Sam Zackrisson | a4c8514 | 2018-10-10 10:44:43 +0200 | [diff] [blame] | 114 | |
| 115 | if (c->filter.main.length_blocks < c->filter.main_initial.length_blocks) { |
| 116 | c->filter.main_initial.length_blocks = c->filter.main.length_blocks; |
| 117 | res = false; |
| 118 | } |
| 119 | |
Sam Zackrisson | 877dc89 | 2018-10-23 14:17:38 +0200 | [diff] [blame] | 120 | res = res & Limit(&c->filter.shadow.length_blocks, 1, 50); |
| 121 | res = res & Limit(&c->filter.shadow.rate, 0.f, 1.f); |
| 122 | res = res & Limit(&c->filter.shadow.noise_gate, 0.f, 100000000.f); |
Sam Zackrisson | a4c8514 | 2018-10-10 10:44:43 +0200 | [diff] [blame] | 123 | |
Sam Zackrisson | 877dc89 | 2018-10-23 14:17:38 +0200 | [diff] [blame] | 124 | res = res & Limit(&c->filter.shadow_initial.length_blocks, 1, 50); |
| 125 | res = res & Limit(&c->filter.shadow_initial.rate, 0.f, 1.f); |
| 126 | res = res & Limit(&c->filter.shadow_initial.noise_gate, 0.f, 100000000.f); |
Sam Zackrisson | a4c8514 | 2018-10-10 10:44:43 +0200 | [diff] [blame] | 127 | |
| 128 | if (c->filter.shadow.length_blocks < c->filter.shadow_initial.length_blocks) { |
| 129 | c->filter.shadow_initial.length_blocks = c->filter.shadow.length_blocks; |
| 130 | res = false; |
| 131 | } |
| 132 | |
Sam Zackrisson | 877dc89 | 2018-10-23 14:17:38 +0200 | [diff] [blame] | 133 | res = res & Limit(&c->filter.config_change_duration_blocks, 0, 100000); |
| 134 | res = res & Limit(&c->filter.initial_state_seconds, 0.f, 100.f); |
Sam Zackrisson | a4c8514 | 2018-10-10 10:44:43 +0200 | [diff] [blame] | 135 | |
Sam Zackrisson | 877dc89 | 2018-10-23 14:17:38 +0200 | [diff] [blame] | 136 | res = res & Limit(&c->erle.min, 1.f, 100000.f); |
| 137 | res = res & Limit(&c->erle.max_l, 1.f, 100000.f); |
| 138 | res = res & Limit(&c->erle.max_h, 1.f, 100000.f); |
Sam Zackrisson | a4c8514 | 2018-10-10 10:44:43 +0200 | [diff] [blame] | 139 | if (c->erle.min > c->erle.max_l || c->erle.min > c->erle.max_h) { |
| 140 | c->erle.min = std::min(c->erle.max_l, c->erle.max_h); |
| 141 | res = false; |
| 142 | } |
Jesús de Vicente Peña | 44974e1 | 2018-11-20 12:54:23 +0100 | [diff] [blame] | 143 | res = res & Limit(&c->erle.num_sections, 1, c->filter.main.length_blocks); |
Sam Zackrisson | a4c8514 | 2018-10-10 10:44:43 +0200 | [diff] [blame] | 144 | |
Per Åhgren | e8efbbd | 2019-03-14 11:29:39 +0100 | [diff] [blame] | 145 | res = res & Limit(&c->ep_strength.default_gain, 0.f, 1000000.f); |
Jesús de Vicente Peña | 44974e1 | 2018-11-20 12:54:23 +0100 | [diff] [blame] | 146 | res = res & Limit(&c->ep_strength.default_len, -1.f, 1.f); |
Sam Zackrisson | 8ee06a7 | 2018-10-23 12:32:42 +0200 | [diff] [blame] | 147 | |
| 148 | res = |
Sam Zackrisson | 877dc89 | 2018-10-23 14:17:38 +0200 | [diff] [blame] | 149 | res & Limit(&c->echo_audibility.low_render_limit, 0.f, 32768.f * 32768.f); |
| 150 | res = res & |
| 151 | Limit(&c->echo_audibility.normal_render_limit, 0.f, 32768.f * 32768.f); |
| 152 | res = res & Limit(&c->echo_audibility.floor_power, 0.f, 32768.f * 32768.f); |
| 153 | res = res & Limit(&c->echo_audibility.audibility_threshold_lf, 0.f, |
| 154 | 32768.f * 32768.f); |
| 155 | res = res & Limit(&c->echo_audibility.audibility_threshold_mf, 0.f, |
| 156 | 32768.f * 32768.f); |
| 157 | res = res & Limit(&c->echo_audibility.audibility_threshold_hf, 0.f, |
| 158 | 32768.f * 32768.f); |
| 159 | |
| 160 | res = res & |
| 161 | Limit(&c->render_levels.active_render_limit, 0.f, 32768.f * 32768.f); |
| 162 | res = res & Limit(&c->render_levels.poor_excitation_render_limit, 0.f, |
| 163 | 32768.f * 32768.f); |
| 164 | res = res & Limit(&c->render_levels.poor_excitation_render_limit_ds8, 0.f, |
| 165 | 32768.f * 32768.f); |
| 166 | |
Sam Zackrisson | 877dc89 | 2018-10-23 14:17:38 +0200 | [diff] [blame] | 167 | res = res & Limit(&c->echo_model.noise_floor_hold, 0, 1000); |
| 168 | res = res & Limit(&c->echo_model.min_noise_floor_power, 0, 2000000.f); |
| 169 | res = res & Limit(&c->echo_model.stationary_gate_slope, 0, 1000000.f); |
| 170 | res = res & Limit(&c->echo_model.noise_gate_power, 0, 1000000.f); |
| 171 | res = res & Limit(&c->echo_model.noise_gate_slope, 0, 1000000.f); |
| 172 | res = res & Limit(&c->echo_model.render_pre_window_size, 0, 100); |
| 173 | res = res & Limit(&c->echo_model.render_post_window_size, 0, 100); |
Sam Zackrisson | a4c8514 | 2018-10-10 10:44:43 +0200 | [diff] [blame] | 174 | |
Sam Zackrisson | 877dc89 | 2018-10-23 14:17:38 +0200 | [diff] [blame] | 175 | res = res & Limit(&c->suppressor.nearend_average_blocks, 1, 5000); |
| 176 | |
| 177 | res = res & |
Sam Zackrisson | a4c8514 | 2018-10-10 10:44:43 +0200 | [diff] [blame] | 178 | Limit(&c->suppressor.normal_tuning.mask_lf.enr_transparent, 0.f, 100.f); |
Sam Zackrisson | 877dc89 | 2018-10-23 14:17:38 +0200 | [diff] [blame] | 179 | res = res & |
Sam Zackrisson | a4c8514 | 2018-10-10 10:44:43 +0200 | [diff] [blame] | 180 | Limit(&c->suppressor.normal_tuning.mask_lf.enr_suppress, 0.f, 100.f); |
Sam Zackrisson | 877dc89 | 2018-10-23 14:17:38 +0200 | [diff] [blame] | 181 | res = res & |
Sam Zackrisson | a4c8514 | 2018-10-10 10:44:43 +0200 | [diff] [blame] | 182 | Limit(&c->suppressor.normal_tuning.mask_lf.emr_transparent, 0.f, 100.f); |
Sam Zackrisson | 877dc89 | 2018-10-23 14:17:38 +0200 | [diff] [blame] | 183 | res = res & |
Sam Zackrisson | a4c8514 | 2018-10-10 10:44:43 +0200 | [diff] [blame] | 184 | Limit(&c->suppressor.normal_tuning.mask_hf.enr_transparent, 0.f, 100.f); |
Sam Zackrisson | 877dc89 | 2018-10-23 14:17:38 +0200 | [diff] [blame] | 185 | res = res & |
Sam Zackrisson | a4c8514 | 2018-10-10 10:44:43 +0200 | [diff] [blame] | 186 | Limit(&c->suppressor.normal_tuning.mask_hf.enr_suppress, 0.f, 100.f); |
Sam Zackrisson | 877dc89 | 2018-10-23 14:17:38 +0200 | [diff] [blame] | 187 | res = res & |
Sam Zackrisson | a4c8514 | 2018-10-10 10:44:43 +0200 | [diff] [blame] | 188 | Limit(&c->suppressor.normal_tuning.mask_hf.emr_transparent, 0.f, 100.f); |
Sam Zackrisson | 877dc89 | 2018-10-23 14:17:38 +0200 | [diff] [blame] | 189 | res = res & Limit(&c->suppressor.normal_tuning.max_inc_factor, 0.f, 100.f); |
| 190 | res = res & Limit(&c->suppressor.normal_tuning.max_dec_factor_lf, 0.f, 100.f); |
Sam Zackrisson | a4c8514 | 2018-10-10 10:44:43 +0200 | [diff] [blame] | 191 | |
Sam Zackrisson | 877dc89 | 2018-10-23 14:17:38 +0200 | [diff] [blame] | 192 | res = res & Limit(&c->suppressor.nearend_tuning.mask_lf.enr_transparent, 0.f, |
| 193 | 100.f); |
| 194 | res = res & |
Sam Zackrisson | 848273a | 2018-10-23 12:13:42 +0000 | [diff] [blame] | 195 | Limit(&c->suppressor.nearend_tuning.mask_lf.enr_suppress, 0.f, 100.f); |
Sam Zackrisson | 877dc89 | 2018-10-23 14:17:38 +0200 | [diff] [blame] | 196 | res = res & Limit(&c->suppressor.nearend_tuning.mask_lf.emr_transparent, 0.f, |
| 197 | 100.f); |
| 198 | res = res & Limit(&c->suppressor.nearend_tuning.mask_hf.enr_transparent, 0.f, |
| 199 | 100.f); |
| 200 | res = res & |
Sam Zackrisson | 848273a | 2018-10-23 12:13:42 +0000 | [diff] [blame] | 201 | Limit(&c->suppressor.nearend_tuning.mask_hf.enr_suppress, 0.f, 100.f); |
Sam Zackrisson | 877dc89 | 2018-10-23 14:17:38 +0200 | [diff] [blame] | 202 | res = res & Limit(&c->suppressor.nearend_tuning.mask_hf.emr_transparent, 0.f, |
| 203 | 100.f); |
| 204 | res = res & Limit(&c->suppressor.nearend_tuning.max_inc_factor, 0.f, 100.f); |
Sam Zackrisson | 848273a | 2018-10-23 12:13:42 +0000 | [diff] [blame] | 205 | res = |
Sam Zackrisson | 877dc89 | 2018-10-23 14:17:38 +0200 | [diff] [blame] | 206 | res & Limit(&c->suppressor.nearend_tuning.max_dec_factor_lf, 0.f, 100.f); |
Sam Zackrisson | a4c8514 | 2018-10-10 10:44:43 +0200 | [diff] [blame] | 207 | |
Sam Zackrisson | 877dc89 | 2018-10-23 14:17:38 +0200 | [diff] [blame] | 208 | res = res & Limit(&c->suppressor.dominant_nearend_detection.enr_threshold, |
| 209 | 0.f, 1000000.f); |
| 210 | res = res & Limit(&c->suppressor.dominant_nearend_detection.snr_threshold, |
| 211 | 0.f, 1000000.f); |
| 212 | res = res & Limit(&c->suppressor.dominant_nearend_detection.hold_duration, 0, |
| 213 | 10000); |
| 214 | res = res & Limit(&c->suppressor.dominant_nearend_detection.trigger_threshold, |
| 215 | 0, 10000); |
Sam Zackrisson | a4c8514 | 2018-10-10 10:44:43 +0200 | [diff] [blame] | 216 | |
Sam Zackrisson | 877dc89 | 2018-10-23 14:17:38 +0200 | [diff] [blame] | 217 | res = res & Limit(&c->suppressor.high_bands_suppression.enr_threshold, 0.f, |
| 218 | 1000000.f); |
| 219 | res = res & Limit(&c->suppressor.high_bands_suppression.max_gain_during_echo, |
| 220 | 0.f, 1.f); |
Sam Zackrisson | 848273a | 2018-10-23 12:13:42 +0000 | [diff] [blame] | 221 | |
Sam Zackrisson | 877dc89 | 2018-10-23 14:17:38 +0200 | [diff] [blame] | 222 | res = res & Limit(&c->suppressor.floor_first_increase, 0.f, 1000000.f); |
Sam Zackrisson | a4c8514 | 2018-10-10 10:44:43 +0200 | [diff] [blame] | 223 | |
| 224 | return res; |
| 225 | } |
Per Åhgren | b6b00dc | 2018-02-20 22:18:27 +0100 | [diff] [blame] | 226 | } // namespace webrtc |