From a4c8514258b6fb4b03ddcdbb675768e80086d314 Mon Sep 17 00:00:00 2001 From: Sam Zackrisson Date: Wed, 10 Oct 2018 10:44:43 +0200 Subject: [PATCH] Add JSON parsing and corresponding ToString to EchoCanceller3Config MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit Bug: webrtc:9535 Change-Id: I51eaaac4009a30536444292a32938b21e69386bf Reviewed-on: https://webrtc-review.googlesource.com/c/102980 Commit-Queue: Sam Zackrisson Reviewed-by: Alex Loiko Reviewed-by: Per Ã…hgren Cr-Commit-Position: refs/heads/master@{#25083} --- api/audio/BUILD.gn | 18 + api/audio/echo_canceller3_config.cc | 214 ++++++ api/audio/echo_canceller3_config.h | 4 + api/audio/echo_canceller3_config_json.cc | 579 ++++++++++++++++ api/audio/echo_canceller3_config_json.h | 32 + api/audio/test/BUILD.gn | 4 + .../echo_canceller3_config_json_unittest.cc | 41 ++ .../test/echo_canceller3_config_unittest.cc | 47 ++ modules/audio_processing/BUILD.gn | 1 + .../test/audio_processing_simulator.cc | 616 +----------------- 10 files changed, 956 insertions(+), 600 deletions(-) create mode 100644 api/audio/echo_canceller3_config_json.cc create mode 100644 api/audio/echo_canceller3_config_json.h create mode 100644 api/audio/test/echo_canceller3_config_json_unittest.cc create mode 100644 api/audio/test/echo_canceller3_config_unittest.cc diff --git a/api/audio/BUILD.gn b/api/audio/BUILD.gn index 286a5a6f1c..43dc420aef 100644 --- a/api/audio/BUILD.gn +++ b/api/audio/BUILD.gn @@ -39,6 +39,24 @@ rtc_source_set("aec3_config") { "echo_canceller3_config.cc", "echo_canceller3_config.h", ] + deps = [ + "../../rtc_base:rtc_base_approved", + "../../rtc_base:safe_minmax", + ] +} + +rtc_source_set("aec3_config_json") { + visibility = [ "*" ] + sources = [ + "echo_canceller3_config_json.cc", + "echo_canceller3_config_json.h", + ] + deps = [ + ":aec3_config", + "../../rtc_base:rtc_base_approved", + "../../rtc_base:rtc_json", + "//third_party/abseil-cpp/absl/strings:strings", + ] } rtc_source_set("aec3_factory") { diff --git a/api/audio/echo_canceller3_config.cc b/api/audio/echo_canceller3_config.cc index 3b03d13c1c..036ef9c5b2 100644 --- a/api/audio/echo_canceller3_config.cc +++ b/api/audio/echo_canceller3_config.cc @@ -9,7 +9,34 @@ */ #include "api/audio/echo_canceller3_config.h" +#include + +#include "rtc_base/logging.h" +#include "rtc_base/numerics/safe_minmax.h" + namespace webrtc { +namespace { +bool Limit(float* value, float min, float max) { + float clamped = rtc::SafeClamp(*value, min, max); + bool res = *value == clamped; + *value = clamped; + return res; +} + +bool Limit(size_t* value, size_t min, size_t max) { + size_t clamped = rtc::SafeClamp(*value, min, max); + bool res = *value == clamped; + *value = clamped; + return res; +} + +bool Limit(int* value, int min, int max) { + int clamped = rtc::SafeClamp(*value, min, max); + bool res = *value == clamped; + *value = clamped; + return res; +} +} // namespace EchoCanceller3Config::EchoCanceller3Config() = default; EchoCanceller3Config::EchoCanceller3Config(const EchoCanceller3Config& e) = @@ -51,4 +78,191 @@ EchoCanceller3Config::Suppressor::Tuning::Tuning(MaskingThresholds mask_lf, EchoCanceller3Config::Suppressor::Tuning::Tuning( const EchoCanceller3Config::Suppressor::Tuning& e) = default; +bool EchoCanceller3Config::Validate(EchoCanceller3Config* config) { + RTC_DCHECK(config); + EchoCanceller3Config* c = config; + bool res = true; + + if (c->delay.down_sampling_factor != 4 && + c->delay.down_sampling_factor != 8) { + c->delay.down_sampling_factor = 4; + res = false; + } + + if (c->delay.num_filters == 0) { + c->delay.num_filters = 1; + res = false; + } + if (c->delay.api_call_jitter_blocks == 0) { + c->delay.api_call_jitter_blocks = 1; + res = false; + } + + if (c->delay.api_call_jitter_blocks == 0) { + c->delay.api_call_jitter_blocks = 1; + res = false; + } + if (c->delay.delay_headroom_blocks <= 1 && + c->delay.hysteresis_limit_1_blocks == 1) { + c->delay.hysteresis_limit_1_blocks = 0; + res = false; + } + res = res && Limit(&c->delay.delay_estimate_smoothing, 0.f, 1.f); + res = res && Limit(&c->delay.delay_candidate_detection_threshold, 0.f, 1.f); + res = res && Limit(&c->delay.delay_selection_thresholds.initial, 1, 250); + res = res && Limit(&c->delay.delay_selection_thresholds.converged, 1, 250); + + res = res && Limit(&c->filter.main.length_blocks, 1, 50); + res = res && Limit(&c->filter.main.leakage_converged, 0.f, 1000.f); + res = res && Limit(&c->filter.main.leakage_diverged, 0.f, 1000.f); + res = res && Limit(&c->filter.main.error_floor, 0.f, 1000.f); + res = res && Limit(&c->filter.main.noise_gate, 0.f, 100000000.f); + + res = res && Limit(&c->filter.main_initial.length_blocks, 1, 50); + res = res && Limit(&c->filter.main_initial.leakage_converged, 0.f, 1000.f); + res = res && Limit(&c->filter.main_initial.leakage_diverged, 0.f, 1000.f); + res = res && Limit(&c->filter.main_initial.error_floor, 0.f, 1000.f); + res = res && Limit(&c->filter.main_initial.noise_gate, 0.f, 100000000.f); + + if (c->filter.main.length_blocks < c->filter.main_initial.length_blocks) { + c->filter.main_initial.length_blocks = c->filter.main.length_blocks; + res = false; + } + + res = res && Limit(&c->filter.shadow.length_blocks, 1, 50); + res = res && Limit(&c->filter.shadow.rate, 0.f, 1.f); + res = res && Limit(&c->filter.shadow.noise_gate, 0.f, 100000000.f); + + res = res && Limit(&c->filter.shadow_initial.length_blocks, 1, 50); + res = res && Limit(&c->filter.shadow_initial.rate, 0.f, 1.f); + res = res && Limit(&c->filter.shadow_initial.noise_gate, 0.f, 100000000.f); + + if (c->filter.shadow.length_blocks < c->filter.shadow_initial.length_blocks) { + c->filter.shadow_initial.length_blocks = c->filter.shadow.length_blocks; + res = false; + } + + res = res && Limit(&c->filter.config_change_duration_blocks, 0, 100000); + res = res && Limit(&c->filter.initial_state_seconds, 0.f, 100.f); + + res = res && Limit(&c->erle.min, 1.f, 100000.f); + res = res && Limit(&c->erle.max_l, 1.f, 100000.f); + res = res && Limit(&c->erle.max_h, 1.f, 100000.f); + if (c->erle.min > c->erle.max_l || c->erle.min > c->erle.max_h) { + c->erle.min = std::min(c->erle.max_l, c->erle.max_h); + res = false; + } + + res = res && Limit(&c->ep_strength.lf, 0.f, 1000000.f); + res = res && Limit(&c->ep_strength.mf, 0.f, 1000000.f); + res = res && Limit(&c->ep_strength.hf, 0.f, 1000000.f); + res = res && Limit(&c->ep_strength.default_len, 0.f, 1.f); + + res = res && Limit(&c->gain_mask.m0, 0.f, 1.f); + res = res && Limit(&c->gain_mask.m1, 0.f, 1.f); + res = res && Limit(&c->gain_mask.m2, 0.f, 1.f); + res = res && Limit(&c->gain_mask.m3, 0.f, 1.f); + res = res && Limit(&c->gain_mask.m5, 0.f, 1.f); + res = res && Limit(&c->gain_mask.m6, 0.f, 1.f); + res = res && Limit(&c->gain_mask.m7, 0.f, 1.f); + res = res && Limit(&c->gain_mask.m8, 0.f, 1.f); + res = res && Limit(&c->gain_mask.m9, 0.f, 1.f); + + res = res && Limit(&c->gain_mask.gain_curve_offset, 0.f, 1000.f); + res = res && Limit(&c->gain_mask.gain_curve_slope, 0.f, 1000.f); + res = res && Limit(&c->gain_mask.temporal_masking_lf, 0.f, 1000.f); + res = res && Limit(&c->gain_mask.temporal_masking_hf, 0.f, 1000.f); + res = res && Limit(&c->gain_mask.temporal_masking_lf_bands, 0, 65); + + res = res && + Limit(&c->echo_audibility.low_render_limit, 0.f, 32768.f * 32768.f); + res = res && + Limit(&c->echo_audibility.normal_render_limit, 0.f, 32768.f * 32768.f); + res = res && Limit(&c->echo_audibility.floor_power, 0.f, 32768.f * 32768.f); + res = res && Limit(&c->echo_audibility.audibility_threshold_lf, 0.f, + 32768.f * 32768.f); + res = res && Limit(&c->echo_audibility.audibility_threshold_mf, 0.f, + 32768.f * 32768.f); + res = res && Limit(&c->echo_audibility.audibility_threshold_hf, 0.f, + 32768.f * 32768.f); + + res = res && + Limit(&c->render_levels.active_render_limit, 0.f, 32768.f * 32768.f); + res = res && Limit(&c->render_levels.poor_excitation_render_limit, 0.f, + 32768.f * 32768.f); + res = res && Limit(&c->render_levels.poor_excitation_render_limit_ds8, 0.f, + 32768.f * 32768.f); + + res = + res && Limit(&c->echo_removal_control.gain_rampup.initial_gain, 0.f, 1.f); + res = res && Limit(&c->echo_removal_control.gain_rampup.first_non_zero_gain, + 0.f, 1.f); + res = res && Limit(&c->echo_removal_control.gain_rampup.non_zero_gain_blocks, + 0, 100000); + res = res && + Limit(&c->echo_removal_control.gain_rampup.full_gain_blocks, 0, 100000); + + res = res && Limit(&c->echo_model.noise_floor_hold, 0, 1000); + res = res && Limit(&c->echo_model.min_noise_floor_power, 0, 2000000.f); + res = res && Limit(&c->echo_model.stationary_gate_slope, 0, 1000000.f); + res = res && Limit(&c->echo_model.noise_gate_power, 0, 1000000.f); + res = res && Limit(&c->echo_model.noise_gate_slope, 0, 1000000.f); + res = res && Limit(&c->echo_model.render_pre_window_size, 0, 100); + res = res && Limit(&c->echo_model.render_post_window_size, 0, 100); + res = res && Limit(&c->echo_model.render_pre_window_size_init, 0, 100); + res = res && Limit(&c->echo_model.render_post_window_size_init, 0, 100); + res = res && Limit(&c->echo_model.nonlinear_hold, 0, 100); + res = res && Limit(&c->echo_model.nonlinear_release, 0, 1.f); + + res = res && + Limit(&c->suppressor.normal_tuning.mask_lf.enr_transparent, 0.f, 100.f); + res = res && + Limit(&c->suppressor.normal_tuning.mask_lf.enr_suppress, 0.f, 100.f); + res = res && + Limit(&c->suppressor.normal_tuning.mask_lf.emr_transparent, 0.f, 100.f); + res = res && + Limit(&c->suppressor.normal_tuning.mask_hf.enr_transparent, 0.f, 100.f); + res = res && + Limit(&c->suppressor.normal_tuning.mask_hf.enr_suppress, 0.f, 100.f); + res = res && + Limit(&c->suppressor.normal_tuning.mask_hf.emr_transparent, 0.f, 100.f); + res = res && Limit(&c->suppressor.normal_tuning.max_inc_factor, 0.f, 100.f); + res = + res && Limit(&c->suppressor.normal_tuning.max_dec_factor_lf, 0.f, 100.f); + + res = res && Limit(&c->suppressor.nearend_tuning.mask_lf.enr_transparent, 0.f, + 100.f); + res = res && + Limit(&c->suppressor.nearend_tuning.mask_lf.enr_suppress, 0.f, 100.f); + res = res && Limit(&c->suppressor.nearend_tuning.mask_lf.emr_transparent, 0.f, + 100.f); + res = res && Limit(&c->suppressor.nearend_tuning.mask_hf.enr_transparent, 0.f, + 100.f); + res = res && + Limit(&c->suppressor.nearend_tuning.mask_hf.enr_suppress, 0.f, 100.f); + res = res && Limit(&c->suppressor.nearend_tuning.mask_hf.emr_transparent, 0.f, + 100.f); + res = res && Limit(&c->suppressor.nearend_tuning.max_inc_factor, 0.f, 100.f); + res = + res && Limit(&c->suppressor.nearend_tuning.max_dec_factor_lf, 0.f, 100.f); + + res = res && Limit(&c->suppressor.dominant_nearend_detection.enr_threshold, + 0.f, 1000000.f); + res = res && Limit(&c->suppressor.dominant_nearend_detection.snr_threshold, + 0.f, 1000000.f); + res = res && Limit(&c->suppressor.dominant_nearend_detection.hold_duration, 0, + 10000); + res = + res && Limit(&c->suppressor.dominant_nearend_detection.trigger_threshold, + 0, 10000); + + res = res && Limit(&c->suppressor.high_bands_suppression.enr_threshold, 0.f, + 1000000.f); + res = res && Limit(&c->suppressor.high_bands_suppression.max_gain_during_echo, + 0.f, 1.f); + + res = res && Limit(&c->suppressor.floor_first_increase, 0.f, 1000000.f); + + return res; +} } // namespace webrtc diff --git a/api/audio/echo_canceller3_config.h b/api/audio/echo_canceller3_config.h index 113a06ec8e..64e3383946 100644 --- a/api/audio/echo_canceller3_config.h +++ b/api/audio/echo_canceller3_config.h @@ -17,6 +17,10 @@ namespace webrtc { // Configuration struct for EchoCanceller3 struct EchoCanceller3Config { + // Checks and updates the parameters in a config to lie within reasonable + // ranges. Returns true if and only of the config did not need to be changed. + static bool Validate(EchoCanceller3Config* config); + EchoCanceller3Config(); EchoCanceller3Config(const EchoCanceller3Config& e); struct Delay { diff --git a/api/audio/echo_canceller3_config_json.cc b/api/audio/echo_canceller3_config_json.cc new file mode 100644 index 0000000000..6599ed6c99 --- /dev/null +++ b/api/audio/echo_canceller3_config_json.cc @@ -0,0 +1,579 @@ +/* + * Copyright (c) 2018 The WebRTC project authors. All Rights Reserved. + * + * Use of this source code is governed by a BSD-style license + * that can be found in the LICENSE file in the root of the source + * tree. An additional intellectual property rights grant can be found + * in the file PATENTS. All contributing project authors may + * be found in the AUTHORS file in the root of the source tree. + */ +#include "api/audio/echo_canceller3_config_json.h" + +#include +#include + +#include "rtc_base/logging.h" +#include "rtc_base/strings/json.h" +#include "rtc_base/strings/string_builder.h" + +namespace webrtc { +namespace { +void ReadParam(const Json::Value& root, std::string param_name, bool* param) { + RTC_DCHECK(param); + bool v; + if (rtc::GetBoolFromJsonObject(root, param_name, &v)) { + *param = v; + } +} + +void ReadParam(const Json::Value& root, std::string param_name, size_t* param) { + RTC_DCHECK(param); + int v; + if (rtc::GetIntFromJsonObject(root, param_name, &v)) { + *param = v; + } +} + +void ReadParam(const Json::Value& root, std::string param_name, int* param) { + RTC_DCHECK(param); + int v; + if (rtc::GetIntFromJsonObject(root, param_name, &v)) { + *param = v; + } +} + +void ReadParam(const Json::Value& root, std::string param_name, float* param) { + RTC_DCHECK(param); + double v; + if (rtc::GetDoubleFromJsonObject(root, param_name, &v)) { + *param = static_cast(v); + } +} + +void ReadParam(const Json::Value& root, + std::string param_name, + EchoCanceller3Config::Filter::MainConfiguration* param) { + RTC_CHECK(param); + Json::Value json_array; + if (rtc::GetValueFromJsonObject(root, param_name, &json_array)) { + std::vector v; + rtc::JsonArrayToDoubleVector(json_array, &v); + if (v.size() != 6) { + RTC_LOG(LS_ERROR) << "Incorrect array size for " << param_name; + RTC_CHECK(false); + } + param->length_blocks = static_cast(v[0]); + param->leakage_converged = static_cast(v[1]); + param->leakage_diverged = static_cast(v[2]); + param->error_floor = static_cast(v[3]); + param->error_ceil = static_cast(v[4]); + param->noise_gate = static_cast(v[5]); + } +} + +void ReadParam(const Json::Value& root, + std::string param_name, + EchoCanceller3Config::Filter::ShadowConfiguration* param) { + RTC_CHECK(param); + Json::Value json_array; + if (rtc::GetValueFromJsonObject(root, param_name, &json_array)) { + std::vector v; + rtc::JsonArrayToDoubleVector(json_array, &v); + if (v.size() != 3) { + RTC_LOG(LS_ERROR) << "Incorrect array size for " << param_name; + RTC_CHECK(false); + } + param->length_blocks = static_cast(v[0]); + param->rate = static_cast(v[1]); + param->noise_gate = static_cast(v[2]); + } +} + +void ReadParam(const Json::Value& root, + std::string param_name, + EchoCanceller3Config::Suppressor::MaskingThresholds* param) { + RTC_CHECK(param); + Json::Value json_array; + if (rtc::GetValueFromJsonObject(root, param_name, &json_array)) { + std::vector v; + rtc::JsonArrayToDoubleVector(json_array, &v); + if (v.size() != 3) { + RTC_LOG(LS_ERROR) << "Incorrect array size for " << param_name; + RTC_CHECK(false); + } + param->enr_transparent = static_cast(v[0]); + param->enr_suppress = static_cast(v[1]); + param->emr_transparent = static_cast(v[2]); + } +} +} // namespace + +EchoCanceller3Config Aec3ConfigFromJsonString(absl::string_view json_string) { + EchoCanceller3Config cfg; + + Json::Value root; + bool success = Json::Reader().parse(std::string(json_string), root); + if (!success) { + RTC_LOG(LS_ERROR) << "Incorrect JSON format: " << json_string; + return EchoCanceller3Config(); + } + + Json::Value aec3_root; + success = rtc::GetValueFromJsonObject(root, "aec3", &aec3_root); + if (!success) { + RTC_LOG(LS_ERROR) << "Missing AEC3 config field: " << json_string; + return EchoCanceller3Config(); + } + + Json::Value section; + if (rtc::GetValueFromJsonObject(aec3_root, "delay", §ion)) { + ReadParam(section, "default_delay", &cfg.delay.default_delay); + ReadParam(section, "down_sampling_factor", &cfg.delay.down_sampling_factor); + ReadParam(section, "num_filters", &cfg.delay.num_filters); + ReadParam(section, "api_call_jitter_blocks", + &cfg.delay.api_call_jitter_blocks); + ReadParam(section, "min_echo_path_delay_blocks", + &cfg.delay.min_echo_path_delay_blocks); + ReadParam(section, "delay_headroom_blocks", + &cfg.delay.delay_headroom_blocks); + ReadParam(section, "hysteresis_limit_1_blocks", + &cfg.delay.hysteresis_limit_1_blocks); + ReadParam(section, "hysteresis_limit_2_blocks", + &cfg.delay.hysteresis_limit_2_blocks); + ReadParam(section, "skew_hysteresis_blocks", + &cfg.delay.skew_hysteresis_blocks); + ReadParam(section, "fixed_capture_delay_samples", + &cfg.delay.fixed_capture_delay_samples); + ReadParam(section, "delay_estimate_smoothing", + &cfg.delay.delay_estimate_smoothing); + ReadParam(section, "delay_candidate_detection_threshold", + &cfg.delay.delay_candidate_detection_threshold); + + Json::Value subsection; + if (rtc::GetValueFromJsonObject(section, "delay_selection_thresholds", + &subsection)) { + ReadParam(subsection, "initial", + &cfg.delay.delay_selection_thresholds.initial); + ReadParam(subsection, "converged", + &cfg.delay.delay_selection_thresholds.converged); + } + } + + if (rtc::GetValueFromJsonObject(aec3_root, "filter", §ion)) { + ReadParam(section, "main", &cfg.filter.main); + ReadParam(section, "shadow", &cfg.filter.shadow); + ReadParam(section, "main_initial", &cfg.filter.main_initial); + ReadParam(section, "shadow_initial", &cfg.filter.shadow_initial); + ReadParam(section, "config_change_duration_blocks", + &cfg.filter.config_change_duration_blocks); + ReadParam(section, "initial_state_seconds", + &cfg.filter.initial_state_seconds); + ReadParam(section, "conservative_initial_phase", + &cfg.filter.conservative_initial_phase); + ReadParam(section, "enable_shadow_filter_output_usage", + &cfg.filter.enable_shadow_filter_output_usage); + } + + if (rtc::GetValueFromJsonObject(aec3_root, "erle", §ion)) { + ReadParam(section, "min", &cfg.erle.min); + ReadParam(section, "max_l", &cfg.erle.max_l); + ReadParam(section, "max_h", &cfg.erle.max_h); + ReadParam(section, "onset_detection", &cfg.erle.onset_detection); + } + + if (rtc::GetValueFromJsonObject(aec3_root, "ep_strength", §ion)) { + ReadParam(section, "lf", &cfg.ep_strength.lf); + ReadParam(section, "mf", &cfg.ep_strength.mf); + ReadParam(section, "hf", &cfg.ep_strength.hf); + ReadParam(section, "default_len", &cfg.ep_strength.default_len); + ReadParam(section, "reverb_based_on_render", + &cfg.ep_strength.reverb_based_on_render); + ReadParam(section, "echo_can_saturate", &cfg.ep_strength.echo_can_saturate); + ReadParam(section, "bounded_erl", &cfg.ep_strength.bounded_erl); + } + + if (rtc::GetValueFromJsonObject(aec3_root, "gain_mask", §ion)) { + ReadParam(section, "m1", &cfg.gain_mask.m1); + ReadParam(section, "m2", &cfg.gain_mask.m2); + ReadParam(section, "m3", &cfg.gain_mask.m3); + ReadParam(section, "m5", &cfg.gain_mask.m5); + ReadParam(section, "m6", &cfg.gain_mask.m6); + ReadParam(section, "m7", &cfg.gain_mask.m7); + ReadParam(section, "m8", &cfg.gain_mask.m8); + ReadParam(section, "m9", &cfg.gain_mask.m9); + + ReadParam(section, "gain_curve_offset", &cfg.gain_mask.gain_curve_offset); + ReadParam(section, "gain_curve_slope", &cfg.gain_mask.gain_curve_slope); + ReadParam(section, "temporal_masking_lf", + &cfg.gain_mask.temporal_masking_lf); + ReadParam(section, "temporal_masking_hf", + &cfg.gain_mask.temporal_masking_hf); + ReadParam(section, "temporal_masking_lf_bands", + &cfg.gain_mask.temporal_masking_lf_bands); + } + + if (rtc::GetValueFromJsonObject(aec3_root, "echo_audibility", §ion)) { + ReadParam(section, "low_render_limit", + &cfg.echo_audibility.low_render_limit); + ReadParam(section, "normal_render_limit", + &cfg.echo_audibility.normal_render_limit); + + ReadParam(section, "floor_power", &cfg.echo_audibility.floor_power); + ReadParam(section, "audibility_threshold_lf", + &cfg.echo_audibility.audibility_threshold_lf); + ReadParam(section, "audibility_threshold_mf", + &cfg.echo_audibility.audibility_threshold_mf); + ReadParam(section, "audibility_threshold_hf", + &cfg.echo_audibility.audibility_threshold_hf); + ReadParam(section, "use_stationary_properties", + &cfg.echo_audibility.use_stationary_properties); + ReadParam(section, "use_stationary_properties_at_init", + &cfg.echo_audibility.use_stationarity_properties_at_init); + } + + if (rtc::GetValueFromJsonObject(aec3_root, "echo_removal_control", + §ion)) { + Json::Value subsection; + if (rtc::GetValueFromJsonObject(section, "gain_rampup", &subsection)) { + ReadParam(subsection, "initial_gain", + &cfg.echo_removal_control.gain_rampup.initial_gain); + ReadParam(subsection, "first_non_zero_gain", + &cfg.echo_removal_control.gain_rampup.first_non_zero_gain); + ReadParam(subsection, "non_zero_gain_blocks", + &cfg.echo_removal_control.gain_rampup.non_zero_gain_blocks); + ReadParam(subsection, "full_gain_blocks", + &cfg.echo_removal_control.gain_rampup.full_gain_blocks); + } + ReadParam(section, "has_clock_drift", + &cfg.echo_removal_control.has_clock_drift); + ReadParam(section, "linear_and_stable_echo_path", + &cfg.echo_removal_control.linear_and_stable_echo_path); + } + + if (rtc::GetValueFromJsonObject(aec3_root, "echo_model", §ion)) { + Json::Value subsection; + ReadParam(section, "noise_floor_hold", &cfg.echo_model.noise_floor_hold); + ReadParam(section, "min_noise_floor_power", + &cfg.echo_model.min_noise_floor_power); + ReadParam(section, "stationary_gate_slope", + &cfg.echo_model.stationary_gate_slope); + ReadParam(section, "noise_gate_power", &cfg.echo_model.noise_gate_power); + ReadParam(section, "noise_gate_slope", &cfg.echo_model.noise_gate_slope); + ReadParam(section, "render_pre_window_size", + &cfg.echo_model.render_pre_window_size); + ReadParam(section, "render_post_window_size", + &cfg.echo_model.render_post_window_size); + ReadParam(section, "render_pre_window_size_init", + &cfg.echo_model.render_pre_window_size_init); + ReadParam(section, "render_post_window_size_init", + &cfg.echo_model.render_post_window_size_init); + ReadParam(section, "nonlinear_hold", &cfg.echo_model.nonlinear_hold); + ReadParam(section, "nonlinear_release", &cfg.echo_model.nonlinear_release); + } + + Json::Value subsection; + if (rtc::GetValueFromJsonObject(aec3_root, "suppressor", §ion)) { + ReadParam(section, "nearend_average_blocks", + &cfg.suppressor.nearend_average_blocks); + + if (rtc::GetValueFromJsonObject(section, "normal_tuning", &subsection)) { + ReadParam(subsection, "mask_lf", &cfg.suppressor.normal_tuning.mask_lf); + ReadParam(subsection, "mask_hf", &cfg.suppressor.normal_tuning.mask_hf); + ReadParam(subsection, "max_inc_factor", + &cfg.suppressor.normal_tuning.max_inc_factor); + ReadParam(subsection, "max_dec_factor_lf", + &cfg.suppressor.normal_tuning.max_dec_factor_lf); + } + + if (rtc::GetValueFromJsonObject(section, "nearend_tuning", &subsection)) { + ReadParam(subsection, "mask_lf", &cfg.suppressor.nearend_tuning.mask_lf); + ReadParam(subsection, "mask_hf", &cfg.suppressor.nearend_tuning.mask_hf); + ReadParam(subsection, "max_inc_factor", + &cfg.suppressor.nearend_tuning.max_inc_factor); + ReadParam(subsection, "max_dec_factor_lf", + &cfg.suppressor.nearend_tuning.max_dec_factor_lf); + } + + if (rtc::GetValueFromJsonObject(section, "dominant_nearend_detection", + &subsection)) { + ReadParam(subsection, "enr_threshold", + &cfg.suppressor.dominant_nearend_detection.enr_threshold); + ReadParam(subsection, "snr_threshold", + &cfg.suppressor.dominant_nearend_detection.snr_threshold); + ReadParam(subsection, "hold_duration", + &cfg.suppressor.dominant_nearend_detection.hold_duration); + ReadParam(subsection, "trigger_threshold", + &cfg.suppressor.dominant_nearend_detection.trigger_threshold); + } + + if (rtc::GetValueFromJsonObject(section, "high_bands_suppression", + &subsection)) { + ReadParam(subsection, "enr_threshold", + &cfg.suppressor.high_bands_suppression.enr_threshold); + ReadParam(subsection, "max_gain_during_echo", + &cfg.suppressor.high_bands_suppression.max_gain_during_echo); + } + + ReadParam(section, "floor_first_increase", + &cfg.suppressor.floor_first_increase); + ReadParam(section, "enforce_transparent", + &cfg.suppressor.enforce_transparent); + ReadParam(section, "enforce_empty_higher_bands", + &cfg.suppressor.enforce_empty_higher_bands); + } + return cfg; +} + +std::string Aec3ConfigToJsonString(const EchoCanceller3Config& config) { + rtc::StringBuilder ost; + ost << "{"; + ost << "\"aec3\": {"; + ost << "\"delay\": {"; + ost << "\"default_delay\": " << config.delay.default_delay << ","; + ost << "\"down_sampling_factor\": " << config.delay.down_sampling_factor + << ","; + ost << "\"num_filters\": " << config.delay.num_filters << ","; + ost << "\"api_call_jitter_blocks\": " << config.delay.api_call_jitter_blocks + << ","; + ost << "\"min_echo_path_delay_blocks\": " + << config.delay.min_echo_path_delay_blocks << ","; + ost << "\"delay_headroom_blocks\": " << config.delay.delay_headroom_blocks + << ","; + ost << "\"hysteresis_limit_1_blocks\": " + << config.delay.hysteresis_limit_1_blocks << ","; + ost << "\"hysteresis_limit_2_blocks\": " + << config.delay.hysteresis_limit_2_blocks << ","; + ost << "\"skew_hysteresis_blocks\": " << config.delay.skew_hysteresis_blocks + << ","; + ost << "\"fixed_capture_delay_samples\": " + << config.delay.fixed_capture_delay_samples << ","; + ost << "\"delay_estimate_smoothing\": " + << config.delay.delay_estimate_smoothing << ","; + ost << "\"delay_candidate_detection_threshold\": " + << config.delay.delay_candidate_detection_threshold << ","; + + ost << "\"delay_selection_thresholds\": {"; + ost << "\"initial\": " << config.delay.delay_selection_thresholds.initial + << ","; + ost << "\"converged\": " << config.delay.delay_selection_thresholds.converged; + ost << "}"; + + ost << "},"; + + ost << "\"filter\": {"; + ost << "\"main\": ["; + ost << config.filter.main.length_blocks << ","; + ost << config.filter.main.leakage_converged << ","; + ost << config.filter.main.leakage_diverged << ","; + ost << config.filter.main.error_floor << ","; + ost << config.filter.main.error_ceil << ","; + ost << config.filter.main.noise_gate; + ost << "],"; + + ost << "\"shadow\": ["; + ost << config.filter.shadow.length_blocks << ","; + ost << config.filter.shadow.rate << ","; + ost << config.filter.shadow.noise_gate; + ost << "],"; + + ost << "\"main_initial\": ["; + ost << config.filter.main_initial.length_blocks << ","; + ost << config.filter.main_initial.leakage_converged << ","; + ost << config.filter.main_initial.leakage_diverged << ","; + ost << config.filter.main_initial.error_floor << ","; + ost << config.filter.main_initial.error_ceil << ","; + ost << config.filter.main_initial.noise_gate; + ost << "],"; + + ost << "\"shadow_initial\": ["; + ost << config.filter.shadow_initial.length_blocks << ","; + ost << config.filter.shadow_initial.rate << ","; + ost << config.filter.shadow_initial.noise_gate; + ost << "],"; + + ost << "\"config_change_duration_blocks\": " + << config.filter.config_change_duration_blocks << ","; + ost << "\"initial_state_seconds\": " << config.filter.initial_state_seconds + << ","; + ost << "\"conservative_initial_phase\": " + << (config.filter.conservative_initial_phase ? "true" : "false") << ","; + ost << "\"enable_shadow_filter_output_usage\": " + << (config.filter.enable_shadow_filter_output_usage ? "true" : "false"); + + ost << "},"; + + ost << "\"erle\": {"; + ost << "\"min\": " << config.erle.min << ","; + ost << "\"max_l\": " << config.erle.max_l << ","; + ost << "\"max_h\": " << config.erle.max_h << ","; + ost << "\"onset_detection\": " + << (config.erle.onset_detection ? "true" : "false"); + ost << "},"; + + ost << "\"ep_strength\": {"; + ost << "\"lf\": " << config.ep_strength.lf << ","; + ost << "\"mf\": " << config.ep_strength.mf << ","; + ost << "\"hf\": " << config.ep_strength.hf << ","; + ost << "\"default_len\": " << config.ep_strength.default_len << ","; + ost << "\"reverb_based_on_render\": " + << (config.ep_strength.reverb_based_on_render ? "true" : "false") << ","; + ost << "\"echo_can_saturate\": " + << (config.ep_strength.echo_can_saturate ? "true" : "false") << ","; + ost << "\"bounded_erl\": " + << (config.ep_strength.bounded_erl ? "true" : "false"); + + ost << "},"; + + ost << "\"gain_mask\": {"; + ost << "\"m0\": " << config.gain_mask.m0 << ","; + ost << "\"m1\": " << config.gain_mask.m1 << ","; + ost << "\"m2\": " << config.gain_mask.m2 << ","; + ost << "\"m3\": " << config.gain_mask.m3 << ","; + ost << "\"m5\": " << config.gain_mask.m5 << ","; + ost << "\"m6\": " << config.gain_mask.m6 << ","; + ost << "\"m7\": " << config.gain_mask.m7 << ","; + ost << "\"m8\": " << config.gain_mask.m8 << ","; + ost << "\"m9\": " << config.gain_mask.m9 << ","; + ost << "\"gain_curve_offset\": " << config.gain_mask.gain_curve_offset << ","; + ost << "\"gain_curve_slope\": " << config.gain_mask.gain_curve_slope << ","; + ost << "\"temporal_masking_lf\": " << config.gain_mask.temporal_masking_lf + << ","; + ost << "\"temporal_masking_hf\": " << config.gain_mask.temporal_masking_hf + << ","; + ost << "\"temporal_masking_lf_bands\": " + << config.gain_mask.temporal_masking_lf_bands; + ost << "},"; + + ost << "\"echo_audibility\": {"; + ost << "\"low_render_limit\": " << config.echo_audibility.low_render_limit + << ","; + ost << "\"normal_render_limit\": " + << config.echo_audibility.normal_render_limit << ","; + ost << "\"floor_power\": " << config.echo_audibility.floor_power << ","; + ost << "\"audibility_threshold_lf\": " + << config.echo_audibility.audibility_threshold_lf << ","; + ost << "\"audibility_threshold_mf\": " + << config.echo_audibility.audibility_threshold_mf << ","; + ost << "\"audibility_threshold_hf\": " + << config.echo_audibility.audibility_threshold_hf << ","; + ost << "\"use_stationary_properties\": " + << (config.echo_audibility.use_stationary_properties ? "true" : "false") + << ","; + ost << "\"use_stationarity_properties_at_init\": " + << (config.echo_audibility.use_stationarity_properties_at_init ? "true" + : "false"); + ost << "},"; + + ost << "\"render_levels\": {"; + ost << "\"active_render_limit\": " << config.render_levels.active_render_limit + << ","; + ost << "\"poor_excitation_render_limit\": " + << config.render_levels.poor_excitation_render_limit << ","; + ost << "\"poor_excitation_render_limit_ds8\": " + << config.render_levels.poor_excitation_render_limit_ds8; + ost << "},"; + + ost << "\"echo_removal_control\": {"; + ost << "\"gain_rampup\": {"; + ost << "\"initial_gain\": " + << config.echo_removal_control.gain_rampup.initial_gain << ","; + ost << "\"first_non_zero_gain\": " + << config.echo_removal_control.gain_rampup.first_non_zero_gain << ","; + ost << "\"non_zero_gain_blocks\": " + << config.echo_removal_control.gain_rampup.non_zero_gain_blocks << ","; + ost << "\"full_gain_blocks\": " + << config.echo_removal_control.gain_rampup.full_gain_blocks; + ost << "},"; + ost << "\"has_clock_drift\": " + << (config.echo_removal_control.has_clock_drift ? "true" : "false") + << ","; + ost << "\"linear_and_stable_echo_path\": " + << (config.echo_removal_control.linear_and_stable_echo_path ? "true" + : "false"); + + ost << "},"; + + ost << "\"echo_model\": {"; + ost << "\"noise_floor_hold\": " << config.echo_model.noise_floor_hold << ","; + ost << "\"min_noise_floor_power\": " + << config.echo_model.min_noise_floor_power << ","; + ost << "\"stationary_gate_slope\": " + << config.echo_model.stationary_gate_slope << ","; + ost << "\"noise_gate_power\": " << config.echo_model.noise_gate_power << ","; + ost << "\"noise_gate_slope\": " << config.echo_model.noise_gate_slope << ","; + ost << "\"render_pre_window_size\": " + << config.echo_model.render_pre_window_size << ","; + ost << "\"render_post_window_size\": " + << config.echo_model.render_post_window_size << ","; + ost << "\"render_pre_window_size_init\": " + << config.echo_model.render_pre_window_size_init << ","; + ost << "\"render_post_window_size_init\": " + << config.echo_model.render_post_window_size_init << ","; + ost << "\"nonlinear_hold\": " << config.echo_model.nonlinear_hold << ","; + ost << "\"nonlinear_release\": " << config.echo_model.nonlinear_release; + ost << "},"; + + ost << "\"suppressor\": {"; + ost << "\"nearend_average_blocks\": " + << config.suppressor.nearend_average_blocks << ","; + ost << "\"normal_tuning\": {"; + ost << "\"mask_lf\": ["; + ost << config.suppressor.normal_tuning.mask_lf.enr_transparent << ","; + ost << config.suppressor.normal_tuning.mask_lf.enr_suppress << ","; + ost << config.suppressor.normal_tuning.mask_lf.emr_transparent; + ost << "],"; + ost << "\"mask_hf\": ["; + ost << config.suppressor.normal_tuning.mask_hf.enr_transparent << ","; + ost << config.suppressor.normal_tuning.mask_hf.enr_suppress << ","; + ost << config.suppressor.normal_tuning.mask_hf.emr_transparent; + ost << "],"; + ost << "\"max_inc_factor\": " + << config.suppressor.normal_tuning.max_inc_factor << ","; + ost << "\"max_dec_factor_lf\": " + << config.suppressor.normal_tuning.max_dec_factor_lf; + ost << "},"; + ost << "\"nearend_tuning\": {"; + ost << "\"mask_lf\": ["; + ost << config.suppressor.nearend_tuning.mask_lf.enr_transparent << ","; + ost << config.suppressor.nearend_tuning.mask_lf.enr_suppress << ","; + ost << config.suppressor.nearend_tuning.mask_lf.emr_transparent; + ost << "],"; + ost << "\"mask_hf\": ["; + ost << config.suppressor.nearend_tuning.mask_hf.enr_transparent << ","; + ost << config.suppressor.nearend_tuning.mask_hf.enr_suppress << ","; + ost << config.suppressor.nearend_tuning.mask_hf.emr_transparent; + ost << "],"; + ost << "\"max_inc_factor\": " + << config.suppressor.nearend_tuning.max_inc_factor << ","; + ost << "\"max_dec_factor_lf\": " + << config.suppressor.nearend_tuning.max_dec_factor_lf; + ost << "},"; + ost << "\"dominant_nearend_detection\": {"; + ost << "\"enr_threshold\": " + << config.suppressor.dominant_nearend_detection.enr_threshold << ","; + ost << "\"snr_threshold\": " + << config.suppressor.dominant_nearend_detection.snr_threshold << ","; + ost << "\"hold_duration\": " + << config.suppressor.dominant_nearend_detection.hold_duration << ","; + ost << "\"trigger_threshold\": " + << config.suppressor.dominant_nearend_detection.trigger_threshold; + ost << "},"; + ost << "\"high_bands_suppression\": {"; + ost << "\"enr_threshold\": " + << config.suppressor.high_bands_suppression.enr_threshold << ","; + ost << "\"max_gain_during_echo\": " + << config.suppressor.high_bands_suppression.max_gain_during_echo; + ost << "},"; + ost << "\"floor_first_increase\": " << config.suppressor.floor_first_increase + << ","; + ost << "\"enforce_transparent\": " + << (config.suppressor.enforce_transparent ? "true" : "false") << ","; + ost << "\"enforce_empty_higher_bands\": " + << (config.suppressor.enforce_empty_higher_bands ? "true" : "false"); + ost << "}"; + ost << "}"; + ost << "}"; + + return ost.Release(); +} +} // namespace webrtc diff --git a/api/audio/echo_canceller3_config_json.h b/api/audio/echo_canceller3_config_json.h new file mode 100644 index 0000000000..b315bf07f5 --- /dev/null +++ b/api/audio/echo_canceller3_config_json.h @@ -0,0 +1,32 @@ +/* + * Copyright (c) 2018 The WebRTC project authors. All Rights Reserved. + * + * Use of this source code is governed by a BSD-style license + * that can be found in the LICENSE file in the root of the source + * tree. An additional intellectual property rights grant can be found + * in the file PATENTS. All contributing project authors may + * be found in the AUTHORS file in the root of the source tree. + */ + +#ifndef API_AUDIO_ECHO_CANCELLER3_CONFIG_JSON_H_ +#define API_AUDIO_ECHO_CANCELLER3_CONFIG_JSON_H_ + +#include + +#include "absl/strings/string_view.h" +#include "api/audio/echo_canceller3_config.h" + +namespace webrtc { +// Parses a JSON-encoded string into an Aec3 config. Fields corresponds to +// substruct names, with the addition that there must be a top-level node +// "aec3". Returns default config values for anything that cannot be parsed from +// the string. +EchoCanceller3Config Aec3ConfigFromJsonString(absl::string_view json_string); + +// Encodes an Aec3 config in JSON format. Fields corresponds to substruct names, +// with the addition that the top-level node is named "aec3". +std::string Aec3ConfigToJsonString(const EchoCanceller3Config& config); + +} // namespace webrtc + +#endif // API_AUDIO_ECHO_CANCELLER3_CONFIG_JSON_H_ diff --git a/api/audio/test/BUILD.gn b/api/audio/test/BUILD.gn index 9a797788f6..4e04a8a64a 100644 --- a/api/audio/test/BUILD.gn +++ b/api/audio/test/BUILD.gn @@ -17,8 +17,12 @@ if (rtc_include_tests) { testonly = true sources = [ "audio_frame_unittest.cc", + "echo_canceller3_config_json_unittest.cc", + "echo_canceller3_config_unittest.cc", ] deps = [ + "..:aec3_config", + "..:aec3_config_json", "..:audio_frame_api", "../../../rtc_base:rtc_base_approved", "../../../test:test_support", diff --git a/api/audio/test/echo_canceller3_config_json_unittest.cc b/api/audio/test/echo_canceller3_config_json_unittest.cc new file mode 100644 index 0000000000..7c3a445c60 --- /dev/null +++ b/api/audio/test/echo_canceller3_config_json_unittest.cc @@ -0,0 +1,41 @@ +/* + * Copyright 2018 The WebRTC Project Authors. All rights reserved. + * + * Use of this source code is governed by a BSD-style license + * that can be found in the LICENSE file in the root of the source + * tree. An additional intellectual property rights grant can be found + * in the file PATENTS. All contributing project authors may + * be found in the AUTHORS file in the root of the source tree. + */ + +#include "api/audio/echo_canceller3_config_json.h" +#include "api/audio/echo_canceller3_config.h" +#include "test/gtest.h" + +namespace webrtc { + +TEST(EchoCanceller3JsonHelpers, ToStringAndParseJson) { + EchoCanceller3Config cfg; + cfg.delay.down_sampling_factor = 1u; + cfg.filter.shadow_initial.length_blocks = 7u; + cfg.suppressor.normal_tuning.mask_hf.enr_suppress = .5f; + std::string json_string = Aec3ConfigToJsonString(cfg); + EchoCanceller3Config cfg_transformed = Aec3ConfigFromJsonString(json_string); + + // Expect unchanged values to remain default. + EXPECT_EQ(cfg.filter.main.error_floor, + cfg_transformed.filter.main.error_floor); + EXPECT_EQ(cfg.ep_strength.default_len, + cfg_transformed.ep_strength.default_len); + EXPECT_EQ(cfg.suppressor.normal_tuning.mask_lf.enr_suppress, + cfg_transformed.suppressor.normal_tuning.mask_lf.enr_suppress); + + // Expect changed values to carry through the transformation. + EXPECT_EQ(cfg.delay.down_sampling_factor, + cfg_transformed.delay.down_sampling_factor); + EXPECT_EQ(cfg.filter.shadow_initial.length_blocks, + cfg_transformed.filter.shadow_initial.length_blocks); + EXPECT_EQ(cfg.suppressor.normal_tuning.mask_hf.enr_suppress, + cfg_transformed.suppressor.normal_tuning.mask_hf.enr_suppress); +} +} // namespace webrtc diff --git a/api/audio/test/echo_canceller3_config_unittest.cc b/api/audio/test/echo_canceller3_config_unittest.cc new file mode 100644 index 0000000000..ffe72ac958 --- /dev/null +++ b/api/audio/test/echo_canceller3_config_unittest.cc @@ -0,0 +1,47 @@ +/* + * Copyright 2018 The WebRTC Project Authors. All rights reserved. + * + * Use of this source code is governed by a BSD-style license + * that can be found in the LICENSE file in the root of the source + * tree. An additional intellectual property rights grant can be found + * in the file PATENTS. All contributing project authors may + * be found in the AUTHORS file in the root of the source tree. + */ + +#include + +#include "api/audio/echo_canceller3_config.h" +#include "api/audio/echo_canceller3_config_json.h" +#include "test/gtest.h" + +namespace webrtc { + +TEST(EchoCanceller3Config, ValidConfigIsNotModified) { + EchoCanceller3Config config; + EXPECT_TRUE(EchoCanceller3Config::Validate(&config)); + EchoCanceller3Config default_config; + EXPECT_EQ(Aec3ConfigToJsonString(config), + Aec3ConfigToJsonString(default_config)); +} + +TEST(EchoCanceller3Config, InvalidConfigIsCorrected) { + // Change a parameter and validate. + EchoCanceller3Config config; + config.echo_model.min_noise_floor_power = -1600000.f; + EXPECT_FALSE(EchoCanceller3Config::Validate(&config)); + EXPECT_GE(config.echo_model.min_noise_floor_power, 0.f); + // Verify remaining parameters are unchanged. + EchoCanceller3Config default_config; + config.echo_model.min_noise_floor_power = + default_config.echo_model.min_noise_floor_power; + EXPECT_EQ(Aec3ConfigToJsonString(config), + Aec3ConfigToJsonString(default_config)); +} + +TEST(EchoCanceller3Config, ValidatedConfigsAreValid) { + EchoCanceller3Config config; + config.delay.down_sampling_factor = 983; + EXPECT_FALSE(EchoCanceller3Config::Validate(&config)); + EXPECT_TRUE(EchoCanceller3Config::Validate(&config)); +} +} // namespace webrtc diff --git a/modules/audio_processing/BUILD.gn b/modules/audio_processing/BUILD.gn index 61b4d495ed..a51b6a08e7 100644 --- a/modules/audio_processing/BUILD.gn +++ b/modules/audio_processing/BUILD.gn @@ -530,6 +530,7 @@ if (rtc_include_tests) { ":audioproc_protobuf_utils", ":audioproc_test_utils", ":runtime_settings_protobuf_utils", + "../../api/audio:aec3_config_json", "../../api/audio:aec3_factory", "../../common_audio:common_audio", "../../rtc_base:checks", diff --git a/modules/audio_processing/test/audio_processing_simulator.cc b/modules/audio_processing/test/audio_processing_simulator.cc index 0e06a9adee..cd682d0ae1 100644 --- a/modules/audio_processing/test/audio_processing_simulator.cc +++ b/modules/audio_processing/test/audio_processing_simulator.cc @@ -18,6 +18,7 @@ #include #include "absl/memory/memory.h" +#include "api/audio/echo_canceller3_config_json.h" #include "api/audio/echo_canceller3_factory.h" #include "common_audio/include/audio_util.h" #include "modules/audio_processing/aec_dump/aec_dump_factory.h" @@ -34,606 +35,21 @@ namespace webrtc { namespace test { namespace { - -// Prints out the currently used AEC3 parameter values in JSON format. -void PrintAec3ParameterValues(const EchoCanceller3Config& cfg) { - std::cout << "{"; - std::cout << "\"delay\": {"; - std::cout << "\"default_delay\": " << cfg.delay.default_delay << ","; - std::cout << "\"down_sampling_factor\": " << cfg.delay.down_sampling_factor - << ","; - std::cout << "\"num_filters\": " << cfg.delay.num_filters << ","; - std::cout << "\"api_call_jitter_blocks\": " - << cfg.delay.api_call_jitter_blocks << ","; - std::cout << "\"min_echo_path_delay_blocks\": " - << cfg.delay.min_echo_path_delay_blocks << ","; - std::cout << "\"delay_headroom_blocks\": " << cfg.delay.delay_headroom_blocks - << ","; - std::cout << "\"hysteresis_limit_1_blocks\": " - << cfg.delay.hysteresis_limit_1_blocks << ","; - std::cout << "\"hysteresis_limit_2_blocks\": " - << cfg.delay.hysteresis_limit_2_blocks << ","; - std::cout << "\"skew_hysteresis_blocks\": " - << cfg.delay.skew_hysteresis_blocks << ","; - std::cout << "\"fixed_capture_delay_samples\": " - << cfg.delay.fixed_capture_delay_samples << ","; - std::cout << "\"delay_estimate_smoothing\": " - << cfg.delay.delay_estimate_smoothing << ","; - std::cout << "\"delay_candidate_detection_threshold\": " - << cfg.delay.delay_candidate_detection_threshold << ","; - - std::cout << "\"delay_selection_thresholds\": {"; - std::cout << "\"initial\": " << cfg.delay.delay_selection_thresholds.initial - << ","; - std::cout << "\"converged\": " - << cfg.delay.delay_selection_thresholds.converged; - std::cout << "}"; - - std::cout << "},"; - - std::cout << "\"filter\": {"; - std::cout << "\"main\": ["; - std::cout << cfg.filter.main.length_blocks << ","; - std::cout << cfg.filter.main.leakage_converged << ","; - std::cout << cfg.filter.main.leakage_diverged << ","; - std::cout << cfg.filter.main.error_floor << ","; - std::cout << cfg.filter.main.error_ceil << ","; - std::cout << cfg.filter.main.noise_gate; - std::cout << "],"; - - std::cout << "\"shadow\": ["; - std::cout << cfg.filter.shadow.length_blocks << ","; - std::cout << cfg.filter.shadow.rate << ","; - std::cout << cfg.filter.shadow.noise_gate; - std::cout << "],"; - - std::cout << "\"main_initial\": ["; - std::cout << cfg.filter.main_initial.length_blocks << ","; - std::cout << cfg.filter.main_initial.leakage_converged << ","; - std::cout << cfg.filter.main_initial.leakage_diverged << ","; - std::cout << cfg.filter.main_initial.error_floor << ","; - std::cout << cfg.filter.main_initial.error_ceil << ","; - std::cout << cfg.filter.main_initial.noise_gate; - std::cout << "],"; - - std::cout << "\"shadow_initial\": ["; - std::cout << cfg.filter.shadow_initial.length_blocks << ","; - std::cout << cfg.filter.shadow_initial.rate << ","; - std::cout << cfg.filter.shadow_initial.noise_gate; - std::cout << "],"; - - std::cout << "\"config_change_duration_blocks\": " - << cfg.filter.config_change_duration_blocks << ","; - std::cout << "\"initial_state_seconds\": " << cfg.filter.initial_state_seconds - << ","; - std::cout << "\"conservative_initial_phase\": " - << (cfg.filter.conservative_initial_phase ? "true" : "false") - << ","; - std::cout << "\"enable_shadow_filter_output_usage\": " - << (cfg.filter.enable_shadow_filter_output_usage ? "true" - : "false"); - - std::cout << "},"; - - std::cout << "\"erle\": {"; - std::cout << "\"min\": " << cfg.erle.min << ","; - std::cout << "\"max_l\": " << cfg.erle.max_l << ","; - std::cout << "\"max_h\": " << cfg.erle.max_h << ","; - std::cout << "\"onset_detection\": " - << (cfg.erle.onset_detection ? "true" : "false"); - std::cout << "},"; - - std::cout << "\"ep_strength\": {"; - std::cout << "\"lf\": " << cfg.ep_strength.lf << ","; - std::cout << "\"mf\": " << cfg.ep_strength.mf << ","; - std::cout << "\"hf\": " << cfg.ep_strength.hf << ","; - std::cout << "\"default_len\": " << cfg.ep_strength.default_len << ","; - std::cout << "\"reverb_based_on_render\": " - << (cfg.ep_strength.reverb_based_on_render ? "true" : "false") - << ","; - std::cout << "\"echo_can_saturate\": " - << (cfg.ep_strength.echo_can_saturate ? "true" : "false") << ","; - std::cout << "\"bounded_erl\": " - << (cfg.ep_strength.bounded_erl ? "true" : "false"); - - std::cout << "},"; - - std::cout << "\"gain_mask\": {"; - std::cout << "\"m0\": " << cfg.gain_mask.m0 << ","; - std::cout << "\"m1\": " << cfg.gain_mask.m1 << ","; - std::cout << "\"m2\": " << cfg.gain_mask.m2 << ","; - std::cout << "\"m3\": " << cfg.gain_mask.m3 << ","; - std::cout << "\"m5\": " << cfg.gain_mask.m5 << ","; - std::cout << "\"m6\": " << cfg.gain_mask.m6 << ","; - std::cout << "\"m7\": " << cfg.gain_mask.m7 << ","; - std::cout << "\"m8\": " << cfg.gain_mask.m8 << ","; - std::cout << "\"m9\": " << cfg.gain_mask.m9 << ","; - std::cout << "\"gain_curve_offset\": " << cfg.gain_mask.gain_curve_offset - << ","; - std::cout << "\"gain_curve_slope\": " << cfg.gain_mask.gain_curve_slope - << ","; - std::cout << "\"temporal_masking_lf\": " << cfg.gain_mask.temporal_masking_lf - << ","; - std::cout << "\"temporal_masking_hf\": " << cfg.gain_mask.temporal_masking_hf - << ","; - std::cout << "\"temporal_masking_lf_bands\": " - << cfg.gain_mask.temporal_masking_lf_bands; - std::cout << "},"; - - std::cout << "\"echo_audibility\": {"; - std::cout << "\"low_render_limit\": " << cfg.echo_audibility.low_render_limit - << ","; - std::cout << "\"normal_render_limit\": " - << cfg.echo_audibility.normal_render_limit << ","; - std::cout << "\"floor_power\": " << cfg.echo_audibility.floor_power << ","; - std::cout << "\"audibility_threshold_lf\": " - << cfg.echo_audibility.audibility_threshold_lf << ","; - std::cout << "\"audibility_threshold_mf\": " - << cfg.echo_audibility.audibility_threshold_mf << ","; - std::cout << "\"audibility_threshold_hf\": " - << cfg.echo_audibility.audibility_threshold_hf << ","; - std::cout << "\"use_stationary_properties\": " - << (cfg.echo_audibility.use_stationary_properties ? "true" - : "false") - << ","; - std::cout << "\"use_stationarity_properties_at_init\": " - << (cfg.echo_audibility.use_stationarity_properties_at_init - ? "true" - : "false"); - std::cout << "},"; - - std::cout << "\"render_levels\": {"; - std::cout << "\"active_render_limit\": " - << cfg.render_levels.active_render_limit << ","; - std::cout << "\"poor_excitation_render_limit\": " - << cfg.render_levels.poor_excitation_render_limit << ","; - std::cout << "\"poor_excitation_render_limit_ds8\": " - << cfg.render_levels.poor_excitation_render_limit_ds8; - std::cout << "},"; - - std::cout << "\"echo_removal_control\": {"; - std::cout << "\"gain_rampup\": {"; - std::cout << "\"initial_gain\": " - << cfg.echo_removal_control.gain_rampup.initial_gain << ","; - std::cout << "\"first_non_zero_gain\": " - << cfg.echo_removal_control.gain_rampup.first_non_zero_gain << ","; - std::cout << "\"non_zero_gain_blocks\": " - << cfg.echo_removal_control.gain_rampup.non_zero_gain_blocks << ","; - std::cout << "\"full_gain_blocks\": " - << cfg.echo_removal_control.gain_rampup.full_gain_blocks; - std::cout << "},"; - std::cout << "\"has_clock_drift\": " - << (cfg.echo_removal_control.has_clock_drift ? "true" : "false") - << ","; - std::cout << "\"linear_and_stable_echo_path\": " - << (cfg.echo_removal_control.linear_and_stable_echo_path ? "true" - : "false"); - - std::cout << "},"; - - std::cout << "\"echo_model\": {"; - std::cout << "\"noise_floor_hold\": " << cfg.echo_model.noise_floor_hold - << ","; - std::cout << "\"min_noise_floor_power\": " - << cfg.echo_model.min_noise_floor_power << ","; - std::cout << "\"stationary_gate_slope\": " - << cfg.echo_model.stationary_gate_slope << ","; - std::cout << "\"noise_gate_power\": " << cfg.echo_model.noise_gate_power - << ","; - std::cout << "\"noise_gate_slope\": " << cfg.echo_model.noise_gate_slope - << ","; - std::cout << "\"render_pre_window_size\": " - << cfg.echo_model.render_pre_window_size << ","; - std::cout << "\"render_post_window_size\": " - << cfg.echo_model.render_post_window_size << ","; - std::cout << "\"render_pre_window_size_init\": " - << cfg.echo_model.render_pre_window_size_init << ","; - std::cout << "\"render_post_window_size_init\": " - << cfg.echo_model.render_post_window_size_init << ","; - std::cout << "\"nonlinear_hold\": " << cfg.echo_model.nonlinear_hold << ","; - std::cout << "\"nonlinear_release\": " << cfg.echo_model.nonlinear_release; - std::cout << "},"; - - std::cout << "\"suppressor\": {"; - std::cout << "\"nearend_average_blocks\": " - << cfg.suppressor.nearend_average_blocks << ","; - std::cout << "\"normal_tuning\": {"; - std::cout << "\"mask_lf\": ["; - std::cout << cfg.suppressor.normal_tuning.mask_lf.enr_transparent << ","; - std::cout << cfg.suppressor.normal_tuning.mask_lf.enr_suppress << ","; - std::cout << cfg.suppressor.normal_tuning.mask_lf.emr_transparent; - std::cout << "],"; - std::cout << "\"mask_hf\": ["; - std::cout << cfg.suppressor.normal_tuning.mask_hf.enr_transparent << ","; - std::cout << cfg.suppressor.normal_tuning.mask_hf.enr_suppress << ","; - std::cout << cfg.suppressor.normal_tuning.mask_hf.emr_transparent; - std::cout << "],"; - std::cout << "\"max_inc_factor\": " - << cfg.suppressor.normal_tuning.max_inc_factor << ","; - std::cout << "\"max_dec_factor_lf\": " - << cfg.suppressor.normal_tuning.max_dec_factor_lf; - std::cout << "},"; - std::cout << "\"nearend_tuning\": {"; - std::cout << "\"mask_lf\": ["; - std::cout << cfg.suppressor.nearend_tuning.mask_lf.enr_transparent << ","; - std::cout << cfg.suppressor.nearend_tuning.mask_lf.enr_suppress << ","; - std::cout << cfg.suppressor.nearend_tuning.mask_lf.emr_transparent; - std::cout << "],"; - std::cout << "\"mask_hf\": ["; - std::cout << cfg.suppressor.nearend_tuning.mask_hf.enr_transparent << ","; - std::cout << cfg.suppressor.nearend_tuning.mask_hf.enr_suppress << ","; - std::cout << cfg.suppressor.nearend_tuning.mask_hf.emr_transparent; - std::cout << "],"; - std::cout << "\"max_inc_factor\": " - << cfg.suppressor.nearend_tuning.max_inc_factor << ","; - std::cout << "\"max_dec_factor_lf\": " - << cfg.suppressor.nearend_tuning.max_dec_factor_lf; - std::cout << "},"; - std::cout << "\"dominant_nearend_detection\": {"; - std::cout << "\"enr_threshold\": " - << cfg.suppressor.dominant_nearend_detection.enr_threshold << ","; - std::cout << "\"snr_threshold\": " - << cfg.suppressor.dominant_nearend_detection.snr_threshold << ","; - std::cout << "\"hold_duration\": " - << cfg.suppressor.dominant_nearend_detection.hold_duration << ","; - std::cout << "\"trigger_threshold\": " - << cfg.suppressor.dominant_nearend_detection.trigger_threshold; - std::cout << "},"; - std::cout << "\"high_bands_suppression\": {"; - std::cout << "\"enr_threshold\": " - << cfg.suppressor.high_bands_suppression.enr_threshold << ","; - std::cout << "\"max_gain_during_echo\": " - << cfg.suppressor.high_bands_suppression.max_gain_during_echo; - std::cout << "},"; - std::cout << "\"floor_first_increase\": " - << cfg.suppressor.floor_first_increase << ","; - std::cout << "\"enforce_transparent\": " - << (cfg.suppressor.enforce_transparent ? "true" : "false") << ","; - std::cout << "\"enforce_empty_higher_bands\": " - << (cfg.suppressor.enforce_empty_higher_bands ? "true" : "false"); - std::cout << "}"; - std::cout << "}"; - std::cout << std::endl; +// Helper for reading JSON from a file and parsing it to an AEC3 configuration. +EchoCanceller3Config ReadAec3ConfigFromJsonFile(const std::string& filename) { + std::string json_string; + std::string s; + std::ifstream f(filename.c_str()); + if (f.fail()) { + std::cout << "Failed to open the file " << filename << std::endl; + RTC_CHECK(false); + } + while (std::getline(f, s)) { + json_string += s; + } + return Aec3ConfigFromJsonString(json_string); } -// Class for parsing the AEC3 parameters from a JSON file and producing a config -// struct. -class Aec3ParametersParser { - public: - static EchoCanceller3Config Parse(const std::string& filename) { - return Aec3ParametersParser().ParseFile(filename); - } - - private: - Aec3ParametersParser() = default; - - void ReadParam(const Json::Value& root, - std::string param_name, - bool* param) const { - RTC_CHECK(param); - bool v; - if (rtc::GetBoolFromJsonObject(root, param_name, &v)) { - *param = v; - } - } - - void ReadParam(const Json::Value& root, - std::string param_name, - size_t* param) const { - RTC_CHECK(param); - int v; - if (rtc::GetIntFromJsonObject(root, param_name, &v)) { - *param = v; - } - } - - void ReadParam(const Json::Value& root, - std::string param_name, - int* param) const { - RTC_CHECK(param); - int v; - if (rtc::GetIntFromJsonObject(root, param_name, &v)) { - *param = v; - } - } - - void ReadParam(const Json::Value& root, - std::string param_name, - float* param) const { - RTC_CHECK(param); - double v; - if (rtc::GetDoubleFromJsonObject(root, param_name, &v)) { - *param = static_cast(v); - } - } - - void ReadParam(const Json::Value& root, - std::string param_name, - EchoCanceller3Config::Filter::MainConfiguration* param) const { - RTC_CHECK(param); - Json::Value json_array; - if (rtc::GetValueFromJsonObject(root, param_name, &json_array)) { - std::vector v; - rtc::JsonArrayToDoubleVector(json_array, &v); - if (v.size() != 6) { - std::cout << "Incorrect array size for " << param_name << std::endl; - RTC_CHECK(false); - } - param->length_blocks = static_cast(v[0]); - param->leakage_converged = static_cast(v[1]); - param->leakage_diverged = static_cast(v[2]); - param->error_floor = static_cast(v[3]); - param->error_ceil = static_cast(v[4]); - param->noise_gate = static_cast(v[5]); - } - } - - void ReadParam( - const Json::Value& root, - std::string param_name, - EchoCanceller3Config::Filter::ShadowConfiguration* param) const { - RTC_CHECK(param); - Json::Value json_array; - if (rtc::GetValueFromJsonObject(root, param_name, &json_array)) { - std::vector v; - rtc::JsonArrayToDoubleVector(json_array, &v); - if (v.size() != 3) { - std::cout << "Incorrect array size for " << param_name << std::endl; - RTC_CHECK(false); - } - param->length_blocks = static_cast(v[0]); - param->rate = static_cast(v[1]); - param->noise_gate = static_cast(v[2]); - } - } - - void ReadParam( - const Json::Value& root, - std::string param_name, - EchoCanceller3Config::Suppressor::MaskingThresholds* param) const { - RTC_CHECK(param); - Json::Value json_array; - if (rtc::GetValueFromJsonObject(root, param_name, &json_array)) { - std::vector v; - rtc::JsonArrayToDoubleVector(json_array, &v); - if (v.size() != 3) { - std::cout << "Incorrect array size for " << param_name << std::endl; - RTC_CHECK(false); - } - param->enr_transparent = static_cast(v[0]); - param->enr_suppress = static_cast(v[1]); - param->emr_transparent = static_cast(v[2]); - } - } - - EchoCanceller3Config ParseFile(const std::string& filename) const { - EchoCanceller3Config cfg; - Json::Value root; - std::string s; - std::string json_string; - std::ifstream f(filename.c_str()); - - if (f.fail()) { - std::cout << "Failed to open the file " << filename << std::endl; - RTC_CHECK(false); - } - - while (std::getline(f, s)) { - json_string += s; - } - bool success = Json::Reader().parse(json_string, root); - if (!success) { - std::cout << "Incorrect JSON format:" << std::endl; - std::cout << json_string << std::endl; - RTC_CHECK(false); - } - - Json::Value section; - if (rtc::GetValueFromJsonObject(root, "delay", §ion)) { - ReadParam(section, "default_delay", &cfg.delay.default_delay); - ReadParam(section, "down_sampling_factor", - &cfg.delay.down_sampling_factor); - ReadParam(section, "num_filters", &cfg.delay.num_filters); - ReadParam(section, "api_call_jitter_blocks", - &cfg.delay.api_call_jitter_blocks); - ReadParam(section, "min_echo_path_delay_blocks", - &cfg.delay.min_echo_path_delay_blocks); - ReadParam(section, "delay_headroom_blocks", - &cfg.delay.delay_headroom_blocks); - ReadParam(section, "hysteresis_limit_1_blocks", - &cfg.delay.hysteresis_limit_1_blocks); - ReadParam(section, "hysteresis_limit_2_blocks", - &cfg.delay.hysteresis_limit_2_blocks); - ReadParam(section, "skew_hysteresis_blocks", - &cfg.delay.skew_hysteresis_blocks); - ReadParam(section, "fixed_capture_delay_samples", - &cfg.delay.fixed_capture_delay_samples); - ReadParam(section, "delay_estimate_smoothing", - &cfg.delay.delay_estimate_smoothing); - ReadParam(section, "delay_candidate_detection_threshold", - &cfg.delay.delay_candidate_detection_threshold); - - Json::Value subsection; - if (rtc::GetValueFromJsonObject(section, "delay_selection_thresholds", - &subsection)) { - ReadParam(subsection, "initial", - &cfg.delay.delay_selection_thresholds.initial); - ReadParam(subsection, "converged", - &cfg.delay.delay_selection_thresholds.converged); - } - } - - if (rtc::GetValueFromJsonObject(root, "filter", §ion)) { - ReadParam(section, "main", &cfg.filter.main); - ReadParam(section, "shadow", &cfg.filter.shadow); - ReadParam(section, "main_initial", &cfg.filter.main_initial); - ReadParam(section, "shadow_initial", &cfg.filter.shadow_initial); - ReadParam(section, "config_change_duration_blocks", - &cfg.filter.config_change_duration_blocks); - ReadParam(section, "initial_state_seconds", - &cfg.filter.initial_state_seconds); - ReadParam(section, "conservative_initial_phase", - &cfg.filter.conservative_initial_phase); - ReadParam(section, "enable_shadow_filter_output_usage", - &cfg.filter.enable_shadow_filter_output_usage); - } - - if (rtc::GetValueFromJsonObject(root, "erle", §ion)) { - ReadParam(section, "min", &cfg.erle.min); - ReadParam(section, "max_l", &cfg.erle.max_l); - ReadParam(section, "max_h", &cfg.erle.max_h); - ReadParam(section, "onset_detection", &cfg.erle.onset_detection); - } - - if (rtc::GetValueFromJsonObject(root, "ep_strength", §ion)) { - ReadParam(section, "lf", &cfg.ep_strength.lf); - ReadParam(section, "mf", &cfg.ep_strength.mf); - ReadParam(section, "hf", &cfg.ep_strength.hf); - ReadParam(section, "default_len", &cfg.ep_strength.default_len); - ReadParam(section, "reverb_based_on_render", - &cfg.ep_strength.reverb_based_on_render); - ReadParam(section, "echo_can_saturate", - &cfg.ep_strength.echo_can_saturate); - ReadParam(section, "bounded_erl", &cfg.ep_strength.bounded_erl); - } - - if (rtc::GetValueFromJsonObject(root, "gain_mask", §ion)) { - ReadParam(section, "m1", &cfg.gain_mask.m1); - ReadParam(section, "m2", &cfg.gain_mask.m2); - ReadParam(section, "m3", &cfg.gain_mask.m3); - ReadParam(section, "m5", &cfg.gain_mask.m5); - ReadParam(section, "m6", &cfg.gain_mask.m6); - ReadParam(section, "m7", &cfg.gain_mask.m7); - ReadParam(section, "m8", &cfg.gain_mask.m8); - ReadParam(section, "m9", &cfg.gain_mask.m9); - - ReadParam(section, "gain_curve_offset", &cfg.gain_mask.gain_curve_offset); - ReadParam(section, "gain_curve_slope", &cfg.gain_mask.gain_curve_slope); - ReadParam(section, "temporal_masking_lf", - &cfg.gain_mask.temporal_masking_lf); - ReadParam(section, "temporal_masking_hf", - &cfg.gain_mask.temporal_masking_hf); - ReadParam(section, "temporal_masking_lf_bands", - &cfg.gain_mask.temporal_masking_lf_bands); - } - - if (rtc::GetValueFromJsonObject(root, "echo_audibility", §ion)) { - ReadParam(section, "low_render_limit", - &cfg.echo_audibility.low_render_limit); - ReadParam(section, "normal_render_limit", - &cfg.echo_audibility.normal_render_limit); - - ReadParam(section, "floor_power", &cfg.echo_audibility.floor_power); - ReadParam(section, "audibility_threshold_lf", - &cfg.echo_audibility.audibility_threshold_lf); - ReadParam(section, "audibility_threshold_mf", - &cfg.echo_audibility.audibility_threshold_mf); - ReadParam(section, "audibility_threshold_hf", - &cfg.echo_audibility.audibility_threshold_hf); - ReadParam(section, "use_stationary_properties", - &cfg.echo_audibility.use_stationary_properties); - ReadParam(section, "use_stationary_properties_at_init", - &cfg.echo_audibility.use_stationarity_properties_at_init); - } - - if (rtc::GetValueFromJsonObject(root, "echo_removal_control", §ion)) { - Json::Value subsection; - if (rtc::GetValueFromJsonObject(section, "gain_rampup", &subsection)) { - ReadParam(subsection, "initial_gain", - &cfg.echo_removal_control.gain_rampup.initial_gain); - ReadParam(subsection, "first_non_zero_gain", - &cfg.echo_removal_control.gain_rampup.first_non_zero_gain); - ReadParam(subsection, "non_zero_gain_blocks", - &cfg.echo_removal_control.gain_rampup.non_zero_gain_blocks); - ReadParam(subsection, "full_gain_blocks", - &cfg.echo_removal_control.gain_rampup.full_gain_blocks); - } - ReadParam(section, "has_clock_drift", - &cfg.echo_removal_control.has_clock_drift); - ReadParam(section, "linear_and_stable_echo_path", - &cfg.echo_removal_control.linear_and_stable_echo_path); - } - - if (rtc::GetValueFromJsonObject(root, "echo_model", §ion)) { - Json::Value subsection; - ReadParam(section, "noise_floor_hold", &cfg.echo_model.noise_floor_hold); - ReadParam(section, "min_noise_floor_power", - &cfg.echo_model.min_noise_floor_power); - ReadParam(section, "stationary_gate_slope", - &cfg.echo_model.stationary_gate_slope); - ReadParam(section, "noise_gate_power", &cfg.echo_model.noise_gate_power); - ReadParam(section, "noise_gate_slope", &cfg.echo_model.noise_gate_slope); - ReadParam(section, "render_pre_window_size", - &cfg.echo_model.render_pre_window_size); - ReadParam(section, "render_post_window_size", - &cfg.echo_model.render_post_window_size); - ReadParam(section, "render_pre_window_size_init", - &cfg.echo_model.render_pre_window_size_init); - ReadParam(section, "render_post_window_size_init", - &cfg.echo_model.render_post_window_size_init); - ReadParam(section, "nonlinear_hold", &cfg.echo_model.nonlinear_hold); - ReadParam(section, "nonlinear_release", - &cfg.echo_model.nonlinear_release); - } - - Json::Value subsection; - if (rtc::GetValueFromJsonObject(root, "suppressor", §ion)) { - ReadParam(section, "nearend_average_blocks", - &cfg.suppressor.nearend_average_blocks); - - if (rtc::GetValueFromJsonObject(section, "normal_tuning", &subsection)) { - ReadParam(subsection, "mask_lf", &cfg.suppressor.normal_tuning.mask_lf); - ReadParam(subsection, "mask_hf", &cfg.suppressor.normal_tuning.mask_hf); - ReadParam(subsection, "max_inc_factor", - &cfg.suppressor.normal_tuning.max_inc_factor); - ReadParam(subsection, "max_dec_factor_lf", - &cfg.suppressor.normal_tuning.max_dec_factor_lf); - } - - if (rtc::GetValueFromJsonObject(section, "nearend_tuning", &subsection)) { - ReadParam(subsection, "mask_lf", - &cfg.suppressor.nearend_tuning.mask_lf); - ReadParam(subsection, "mask_hf", - &cfg.suppressor.nearend_tuning.mask_hf); - ReadParam(subsection, "max_inc_factor", - &cfg.suppressor.nearend_tuning.max_inc_factor); - ReadParam(subsection, "max_dec_factor_lf", - &cfg.suppressor.nearend_tuning.max_dec_factor_lf); - } - - if (rtc::GetValueFromJsonObject(section, "dominant_nearend_detection", - &subsection)) { - ReadParam(subsection, "enr_threshold", - &cfg.suppressor.dominant_nearend_detection.enr_threshold); - ReadParam(subsection, "snr_threshold", - &cfg.suppressor.dominant_nearend_detection.snr_threshold); - ReadParam(subsection, "hold_duration", - &cfg.suppressor.dominant_nearend_detection.hold_duration); - ReadParam(subsection, "trigger_threshold", - &cfg.suppressor.dominant_nearend_detection.trigger_threshold); - } - - if (rtc::GetValueFromJsonObject(section, "high_bands_suppression", - &subsection)) { - ReadParam(subsection, "enr_threshold", - &cfg.suppressor.high_bands_suppression.enr_threshold); - ReadParam(subsection, "max_gain_during_echo", - &cfg.suppressor.high_bands_suppression.max_gain_during_echo); - } - - ReadParam(section, "floor_first_increase", - &cfg.suppressor.floor_first_increase); - ReadParam(section, "enforce_transparent", - &cfg.suppressor.enforce_transparent); - ReadParam(section, "enforce_empty_higher_bands", - &cfg.suppressor.enforce_empty_higher_bands); - } - - std::cout << std::endl; - return cfg; - } -}; - void CopyFromAudioFrame(const AudioFrame& src, ChannelBuffer* dest) { RTC_CHECK_EQ(src.num_channels_, dest->num_channels()); RTC_CHECK_EQ(src.samples_per_channel_, dest->num_frames()); @@ -947,7 +363,7 @@ void AudioProcessingSimulator::CreateAudioProcessor() { if (settings_.use_verbose_logging) { std::cout << "Reading AEC3 Parameters from JSON input." << std::endl; } - cfg = Aec3ParametersParser::Parse(*settings_.aec3_settings_filename); + cfg = ReadAec3ConfigFromJsonFile(*settings_.aec3_settings_filename); } echo_control_factory.reset(new EchoCanceller3Factory(cfg)); @@ -955,7 +371,7 @@ void AudioProcessingSimulator::CreateAudioProcessor() { if (!settings_.use_quiet_output) { std::cout << "AEC3 settings:" << std::endl; } - PrintAec3ParameterValues(cfg); + std::cout << Aec3ConfigToJsonString(cfg) << std::endl; } }