From c85c9cc40d05c89b3fa6de997590ab488c13cb2e Mon Sep 17 00:00:00 2001 From: "Matthias P. Braendli" Date: Wed, 7 Mar 2018 15:28:21 +0100 Subject: Introduce slow peak statistic, averaging over 5 minutes --- doc/show_dabmux_stats.py | 10 ++-- doc/stats_dabmux_munin.py | 25 ++++++++-- src/ManagementServer.cpp | 113 +++++++++++++++++++++++++++++----------------- src/ManagementServer.h | 5 +- 4 files changed, 104 insertions(+), 49 deletions(-) diff --git a/doc/show_dabmux_stats.py b/doc/show_dabmux_stats.py index 1216593..d226208 100755 --- a/doc/show_dabmux_stats.py +++ b/doc/show_dabmux_stats.py @@ -46,13 +46,15 @@ if len(sys.argv) == 1: data = sock.recv() values = json.loads(data)['values'] - tmpl = "{ident:20}{maxfill:>8}{minfill:>8}{under:>8}{over:>8}{peakleft:>8}{peakright:>8}{state:>16}" + tmpl = "{ident:20}{maxfill:>8}{minfill:>8}{under:>8}{over:>8}{audioleft:>8}{audioright:>8}{peakleft:>8}{peakright:>8}{state:>16}" print(tmpl.format( ident="id", maxfill="max", minfill="min", under="under", over="over", + audioleft="audio L", + audioright="audio R", peakleft="peak L", peakright="peak R", state="state")) @@ -69,8 +71,10 @@ if len(sys.argv) == 1: minfill=v['min_fill'], under=v['num_underruns'], over=v['num_overruns'], - peakleft=v['peak_left'], - peakright=v['peak_right'], + audioleft=v['peak_left'], + audioright=v['peak_right'], + peakleft=v['peak_left_slow'], + peakright=v['peak_right_slow'], state=v['state'])) diff --git a/doc/stats_dabmux_munin.py b/doc/stats_dabmux_munin.py index 7a134ef..6a52011 100755 --- a/doc/stats_dabmux_munin.py +++ b/doc/stats_dabmux_munin.py @@ -69,24 +69,36 @@ overruns.type COUNTER multigraph audio_levels_{ident} graph_title Contribution {ident} audio level (peak) -graph_order left right +graph_order left left_slow right right_slow graph_args --base 1000 graph_vlabel peak audio level during last ${{graph_period}} graph_category encoders -graph_info This graph shows the audio level of both channels of the {ident} ZMQ input +graph_info This graph shows the audio level and peak of both channels of the {ident} ZMQ input left.info Left channel audio level -left.label Left channel audio level +left.label Left level left.min -90 left.max 0 left.warning -40:0 left.critical -80:0 +left_slow.info Left channel audio peak over last 5 minutes +left_slow.label Left peak +left_slow.min -90 +left_slow.max 0 +left_slow.warning -40:0 +left_slow.critical -80:0 right.info Right channel audio level -right.label Right channel audio level +right.label Right level right.min -90 right.max 0 right.warning -40:0 right.critical -80:0 +right_slow.info Right channel audio peak over last 5 minutes +right_slow.label Right peak +right_slow.min -90 +right_slow.max 0 +right_slow.warning -40:0 +right_slow.critical -80:0 multigraph state_{ident} graph_title State of contribution {ident} @@ -235,6 +247,11 @@ if len(sys.argv) == 1: munin_values += "left.value {}\n".format(v['peak_left']) munin_values += "right.value {}\n".format(v['peak_right']) + if 'peak_left_slow' in v: + # If ODR-DabMux is v2.0.0-3 or older, it doesn't export the slow peaks + munin_values += "left_slow.value {}\n".format(v['peak_left_slow']) + munin_values += "right_slow.value {}\n".format(v['peak_right_slow']) + if 'state' in v: # If ODR-DabMux is v1.3.1-3 or older, it doesn't export state re_state = re.compile(r"\w+ \((\d+)\)") diff --git a/src/ManagementServer.cpp b/src/ManagementServer.cpp index 092b257..16a21b1 100644 --- a/src/ManagementServer.cpp +++ b/src/ManagementServer.cpp @@ -107,10 +107,13 @@ static constexpr auto BUFFER_STATS_KEEP_DURATION = std::chrono::seconds(30); /* Audio level information changes faster than buffer levels, so it makes sense - * to poll much faster. If we keep too much data, we will hide the interesting - * short-time fluctuations. */ + * to poll much faster. If we take the peak over too much data, we will hide + * the interesting short-time fluctuations. At the same time, we want to have a + * statistic that also catches the rare peaks, for slow pollers. */ static constexpr auto -PEAK_STATS_KEEP_DURATION = std::chrono::milliseconds(500); +PEAK_STATS_SHORT_WINDOW = std::chrono::milliseconds(500); +static constexpr auto +PEAK_STATS_KEEP_DURATION = std::chrono::minutes(5); ManagementServer& get_mgmt_server() { @@ -383,7 +386,7 @@ void InputStat::notifyBuffer(long bufsize) { unique_lock lock(m_mutex); - m_buffer_fill_stats.push_back(bufsize); + m_buffer_fill_stats.push_front(bufsize); using namespace std::chrono; const auto time_now = steady_clock::now(); @@ -392,7 +395,7 @@ void InputStat::notifyBuffer(long bufsize) auto total_length = insertion_interval * m_buffer_fill_stats.size(); if (total_length > BUFFER_STATS_KEEP_DURATION) { - m_buffer_fill_stats.pop_front(); + m_buffer_fill_stats.pop_back(); } } m_time_last_buffer_notify = time_now; @@ -402,49 +405,56 @@ void InputStat::notifyPeakLevels(int peak_left, int peak_right) { unique_lock lock(m_mutex); - m_peaks_left.push_back(peak_left); - m_peaks_right.push_back(peak_right); + m_peaks_left.push_front(peak_left); + m_peaks_right.push_front(peak_right); using namespace std::chrono; const auto time_now = steady_clock::now(); - if (m_peaks_left.size() > 1) { - auto insertion_interval = time_now - m_time_last_peak_notify; - auto peaks_total_length = insertion_interval * m_peaks_left.size(); + + if (m_peaks_left.size() < 2) { + m_time_last_peak_notify = time_now; + } + else { + const auto insertion_interval = time_now - m_time_last_peak_notify; + const auto peaks_total_length = insertion_interval * m_peaks_left.size(); if (peaks_total_length > PEAK_STATS_KEEP_DURATION) { - m_peaks_left.pop_front(); - m_peaks_right.pop_front(); + m_peaks_left.pop_back(); + m_peaks_right.pop_back(); } - } - m_time_last_peak_notify = time_now; - if (m_peaks_left.empty() or m_peaks_right.empty()) { - throw std::logic_error("Peak statistics empty!"); - } + m_time_last_peak_notify = time_now; - const auto max_left = *max_element(m_peaks_left.begin(), m_peaks_left.end()); - const auto max_right = *max_element(m_peaks_right.begin(), m_peaks_right.end()); + // Calculate the peak over the short window + m_short_window_length = PEAK_STATS_SHORT_WINDOW / insertion_interval; + const size_t short_window = std::max( + m_peaks_left.size(), m_short_window_length); + const auto max_left = *max_element(m_peaks_left.begin(), + m_peaks_left.begin() + short_window); + const auto max_right = *max_element(m_peaks_right.begin(), + m_peaks_right.begin() + short_window); - // State + // State - // using the lower of the two channels allows us to detect if only one - // channel is silent. - const int lower_peak = max_left < max_right ? max_left : max_right; + // using the lower of the two channels allows us to detect if only one + // channel is silent. + const int lower_peak = max_left < max_right ? max_left : max_right; - const int16_t int16_max = std::numeric_limits::max(); - int peak_dB = lower_peak ? - round(20*log10((double)lower_peak / int16_max)) : - -90; + const int16_t int16_max = std::numeric_limits::max(); + int peak_dB = lower_peak ? + round(20*log10((double)lower_peak / int16_max)) : + -90; - if (peak_dB < INPUT_AUDIO_LEVEL_THRESHOLD) { - if (m_silence_counter < INPUT_AUDIO_LEVEL_COUNT_SATURATION) { - m_silence_counter++; + if (peak_dB < INPUT_AUDIO_LEVEL_THRESHOLD) { + if (m_silence_counter < INPUT_AUDIO_LEVEL_COUNT_SATURATION) { + m_silence_counter++; + } } - } - else { - if (m_silence_counter > 0) { - m_silence_counter--; + else { + if (m_silence_counter > 0) { + m_silence_counter--; + } } } } @@ -462,7 +472,7 @@ void InputStat::notifyUnderrun(void) m_glitch_counter++; } else { - // As we don't receive level notifications anymore, clear the + // As we don't receive level notifications anymore, clear the // audio level information m_peaks_left.clear(); m_peaks_right.clear(); @@ -491,17 +501,27 @@ std::string InputStat::encodeValuesJSON() unique_lock lock(m_mutex); + int peak_left_short = 0; + int peak_right_short = 0; int peak_left = 0; int peak_right = 0; if (not m_peaks_left.empty() and not m_peaks_right.empty()) { peak_left = *max_element(m_peaks_left.begin(), m_peaks_left.end()); peak_right = *max_element(m_peaks_right.begin(), m_peaks_right.end()); - } - /* convert to dB */ - int dB_l = peak_left ? round(20*log10((double)peak_left / int16_max)) : -90; - int dB_r = peak_right ? round(20*log10((double)peak_right / int16_max)) : -90; + if (m_peaks_left.size() >= m_short_window_length and + m_peaks_right.size() >= m_short_window_length) { + peak_left_short = *max_element(m_peaks_left.begin(), + m_peaks_left.begin() + m_short_window_length); + peak_right_short = *max_element(m_peaks_right.begin(), + m_peaks_right.begin() + m_short_window_length); + } + else { + peak_left_short = peak_left; + peak_right_short = peak_right; + } + } long min_fill_buffer = MIN_FILL_BUFFER_UNDEF; long max_fill_buffer = 0; @@ -512,12 +532,23 @@ std::string InputStat::encodeValuesJSON() max_fill_buffer = *buffer_min_max_fill.second; } + /* convert to dB */ + auto to_dB = [](int p) { + int dB = -90; + if (p) { + dB = round(20*log10((double)p / int16_max)); + } + return dB; + }; + ss << "{ \"inputstat\" : {" "\"min_fill\": " << min_fill_buffer << ", " "\"max_fill\": " << max_fill_buffer << ", " - "\"peak_left\": " << dB_l << ", " - "\"peak_right\": " << dB_r << ", " + "\"peak_left\": " << to_dB(peak_left_short) << ", " + "\"peak_right\": " << to_dB(peak_right_short) << ", " + "\"peak_left_slow\": " << to_dB(peak_left) << ", " + "\"peak_right_slow\": " << to_dB(peak_right) << ", " "\"num_underruns\": " << m_num_underruns << ", " "\"num_overruns\": " << m_num_overruns << ", "; diff --git a/src/ManagementServer.h b/src/ManagementServer.h index 274dece..17b8bda 100644 --- a/src/ManagementServer.h +++ b/src/ManagementServer.h @@ -117,11 +117,14 @@ class InputStat uint32_t m_num_overruns; // Peak audio levels (linear 16-bit PCM) for the two channels. - // Keep a FIFO of values from the last few seconds + // Keep a FIFO of values from the last minutes, apply + // a short window to also see short-term fluctuations. std::deque m_peaks_left; std::deque m_peaks_right; std::chrono::time_point m_time_last_peak_notify; + size_t m_short_window_length = 0; + /************* STATE ***************/ /* Variables used for determining the input state */ int m_glitch_counter; // saturating counter -- cgit v1.2.3