105 lines
2.8 KiB
C++
105 lines
2.8 KiB
C++
/*
|
|
* Copyright (c) 2016 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 "video/quality_threshold.h"
|
|
|
|
#include "rtc_base/checks.h"
|
|
|
|
namespace webrtc {
|
|
|
|
QualityThreshold::QualityThreshold(int low_threshold,
|
|
int high_threshold,
|
|
float fraction,
|
|
int max_measurements)
|
|
: buffer_(new int[max_measurements]),
|
|
max_measurements_(max_measurements),
|
|
fraction_(fraction),
|
|
low_threshold_(low_threshold),
|
|
high_threshold_(high_threshold),
|
|
until_full_(max_measurements),
|
|
next_index_(0),
|
|
sum_(0),
|
|
count_low_(0),
|
|
count_high_(0),
|
|
num_high_states_(0),
|
|
num_certain_states_(0) {
|
|
RTC_CHECK_GT(fraction, 0.5f);
|
|
RTC_CHECK_GT(max_measurements, 1);
|
|
RTC_CHECK_LT(low_threshold, high_threshold);
|
|
}
|
|
|
|
QualityThreshold::~QualityThreshold() = default;
|
|
|
|
void QualityThreshold::AddMeasurement(int measurement) {
|
|
int prev_val = until_full_ > 0 ? 0 : buffer_[next_index_];
|
|
buffer_[next_index_] = measurement;
|
|
next_index_ = (next_index_ + 1) % max_measurements_;
|
|
|
|
sum_ += measurement - prev_val;
|
|
|
|
if (until_full_ == 0) {
|
|
if (prev_val <= low_threshold_) {
|
|
--count_low_;
|
|
} else if (prev_val >= high_threshold_) {
|
|
--count_high_;
|
|
}
|
|
}
|
|
|
|
if (measurement <= low_threshold_) {
|
|
++count_low_;
|
|
} else if (measurement >= high_threshold_) {
|
|
++count_high_;
|
|
}
|
|
|
|
float sufficient_majority = fraction_ * max_measurements_;
|
|
if (count_high_ >= sufficient_majority) {
|
|
is_high_ = true;
|
|
} else if (count_low_ >= sufficient_majority) {
|
|
is_high_ = false;
|
|
}
|
|
|
|
if (until_full_ > 0)
|
|
--until_full_;
|
|
|
|
if (is_high_) {
|
|
if (*is_high_)
|
|
++num_high_states_;
|
|
++num_certain_states_;
|
|
}
|
|
}
|
|
|
|
absl::optional<bool> QualityThreshold::IsHigh() const {
|
|
return is_high_;
|
|
}
|
|
|
|
absl::optional<double> QualityThreshold::CalculateVariance() const {
|
|
if (until_full_ > 0) {
|
|
return absl::nullopt;
|
|
}
|
|
|
|
double variance = 0;
|
|
double mean = static_cast<double>(sum_) / max_measurements_;
|
|
for (int i = 0; i < max_measurements_; ++i) {
|
|
variance += (buffer_[i] - mean) * (buffer_[i] - mean);
|
|
}
|
|
return variance / (max_measurements_ - 1);
|
|
}
|
|
|
|
absl::optional<double> QualityThreshold::FractionHigh(
|
|
int min_required_samples) const {
|
|
RTC_DCHECK_GT(min_required_samples, 0);
|
|
if (num_certain_states_ < min_required_samples)
|
|
return absl::nullopt;
|
|
|
|
return static_cast<double>(num_high_states_) / num_certain_states_;
|
|
}
|
|
|
|
} // namespace webrtc
|