xref: /aosp_15_r20/external/webrtc/video/quality_threshold.cc (revision d9f758449e529ab9291ac668be2861e7a55c2422)
1*d9f75844SAndroid Build Coastguard Worker /*
2*d9f75844SAndroid Build Coastguard Worker  *  Copyright (c) 2016 The WebRTC project authors. All Rights Reserved.
3*d9f75844SAndroid Build Coastguard Worker  *
4*d9f75844SAndroid Build Coastguard Worker  *  Use of this source code is governed by a BSD-style license
5*d9f75844SAndroid Build Coastguard Worker  *  that can be found in the LICENSE file in the root of the source
6*d9f75844SAndroid Build Coastguard Worker  *  tree. An additional intellectual property rights grant can be found
7*d9f75844SAndroid Build Coastguard Worker  *  in the file PATENTS.  All contributing project authors may
8*d9f75844SAndroid Build Coastguard Worker  *  be found in the AUTHORS file in the root of the source tree.
9*d9f75844SAndroid Build Coastguard Worker  */
10*d9f75844SAndroid Build Coastguard Worker 
11*d9f75844SAndroid Build Coastguard Worker #include "video/quality_threshold.h"
12*d9f75844SAndroid Build Coastguard Worker 
13*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/checks.h"
14*d9f75844SAndroid Build Coastguard Worker 
15*d9f75844SAndroid Build Coastguard Worker namespace webrtc {
16*d9f75844SAndroid Build Coastguard Worker 
QualityThreshold(int low_threshold,int high_threshold,float fraction,int max_measurements)17*d9f75844SAndroid Build Coastguard Worker QualityThreshold::QualityThreshold(int low_threshold,
18*d9f75844SAndroid Build Coastguard Worker                                    int high_threshold,
19*d9f75844SAndroid Build Coastguard Worker                                    float fraction,
20*d9f75844SAndroid Build Coastguard Worker                                    int max_measurements)
21*d9f75844SAndroid Build Coastguard Worker     : buffer_(new int[max_measurements]),
22*d9f75844SAndroid Build Coastguard Worker       max_measurements_(max_measurements),
23*d9f75844SAndroid Build Coastguard Worker       fraction_(fraction),
24*d9f75844SAndroid Build Coastguard Worker       low_threshold_(low_threshold),
25*d9f75844SAndroid Build Coastguard Worker       high_threshold_(high_threshold),
26*d9f75844SAndroid Build Coastguard Worker       until_full_(max_measurements),
27*d9f75844SAndroid Build Coastguard Worker       next_index_(0),
28*d9f75844SAndroid Build Coastguard Worker       sum_(0),
29*d9f75844SAndroid Build Coastguard Worker       count_low_(0),
30*d9f75844SAndroid Build Coastguard Worker       count_high_(0),
31*d9f75844SAndroid Build Coastguard Worker       num_high_states_(0),
32*d9f75844SAndroid Build Coastguard Worker       num_certain_states_(0) {
33*d9f75844SAndroid Build Coastguard Worker   RTC_CHECK_GT(fraction, 0.5f);
34*d9f75844SAndroid Build Coastguard Worker   RTC_CHECK_GT(max_measurements, 1);
35*d9f75844SAndroid Build Coastguard Worker   RTC_CHECK_LT(low_threshold, high_threshold);
36*d9f75844SAndroid Build Coastguard Worker }
37*d9f75844SAndroid Build Coastguard Worker 
38*d9f75844SAndroid Build Coastguard Worker QualityThreshold::~QualityThreshold() = default;
39*d9f75844SAndroid Build Coastguard Worker 
AddMeasurement(int measurement)40*d9f75844SAndroid Build Coastguard Worker void QualityThreshold::AddMeasurement(int measurement) {
41*d9f75844SAndroid Build Coastguard Worker   int prev_val = until_full_ > 0 ? 0 : buffer_[next_index_];
42*d9f75844SAndroid Build Coastguard Worker   buffer_[next_index_] = measurement;
43*d9f75844SAndroid Build Coastguard Worker   next_index_ = (next_index_ + 1) % max_measurements_;
44*d9f75844SAndroid Build Coastguard Worker 
45*d9f75844SAndroid Build Coastguard Worker   sum_ += measurement - prev_val;
46*d9f75844SAndroid Build Coastguard Worker 
47*d9f75844SAndroid Build Coastguard Worker   if (until_full_ == 0) {
48*d9f75844SAndroid Build Coastguard Worker     if (prev_val <= low_threshold_) {
49*d9f75844SAndroid Build Coastguard Worker       --count_low_;
50*d9f75844SAndroid Build Coastguard Worker     } else if (prev_val >= high_threshold_) {
51*d9f75844SAndroid Build Coastguard Worker       --count_high_;
52*d9f75844SAndroid Build Coastguard Worker     }
53*d9f75844SAndroid Build Coastguard Worker   }
54*d9f75844SAndroid Build Coastguard Worker 
55*d9f75844SAndroid Build Coastguard Worker   if (measurement <= low_threshold_) {
56*d9f75844SAndroid Build Coastguard Worker     ++count_low_;
57*d9f75844SAndroid Build Coastguard Worker   } else if (measurement >= high_threshold_) {
58*d9f75844SAndroid Build Coastguard Worker     ++count_high_;
59*d9f75844SAndroid Build Coastguard Worker   }
60*d9f75844SAndroid Build Coastguard Worker 
61*d9f75844SAndroid Build Coastguard Worker   float sufficient_majority = fraction_ * max_measurements_;
62*d9f75844SAndroid Build Coastguard Worker   if (count_high_ >= sufficient_majority) {
63*d9f75844SAndroid Build Coastguard Worker     is_high_ = true;
64*d9f75844SAndroid Build Coastguard Worker   } else if (count_low_ >= sufficient_majority) {
65*d9f75844SAndroid Build Coastguard Worker     is_high_ = false;
66*d9f75844SAndroid Build Coastguard Worker   }
67*d9f75844SAndroid Build Coastguard Worker 
68*d9f75844SAndroid Build Coastguard Worker   if (until_full_ > 0)
69*d9f75844SAndroid Build Coastguard Worker     --until_full_;
70*d9f75844SAndroid Build Coastguard Worker 
71*d9f75844SAndroid Build Coastguard Worker   if (is_high_) {
72*d9f75844SAndroid Build Coastguard Worker     if (*is_high_)
73*d9f75844SAndroid Build Coastguard Worker       ++num_high_states_;
74*d9f75844SAndroid Build Coastguard Worker     ++num_certain_states_;
75*d9f75844SAndroid Build Coastguard Worker   }
76*d9f75844SAndroid Build Coastguard Worker }
77*d9f75844SAndroid Build Coastguard Worker 
IsHigh() const78*d9f75844SAndroid Build Coastguard Worker absl::optional<bool> QualityThreshold::IsHigh() const {
79*d9f75844SAndroid Build Coastguard Worker   return is_high_;
80*d9f75844SAndroid Build Coastguard Worker }
81*d9f75844SAndroid Build Coastguard Worker 
CalculateVariance() const82*d9f75844SAndroid Build Coastguard Worker absl::optional<double> QualityThreshold::CalculateVariance() const {
83*d9f75844SAndroid Build Coastguard Worker   if (until_full_ > 0) {
84*d9f75844SAndroid Build Coastguard Worker     return absl::nullopt;
85*d9f75844SAndroid Build Coastguard Worker   }
86*d9f75844SAndroid Build Coastguard Worker 
87*d9f75844SAndroid Build Coastguard Worker   double variance = 0;
88*d9f75844SAndroid Build Coastguard Worker   double mean = static_cast<double>(sum_) / max_measurements_;
89*d9f75844SAndroid Build Coastguard Worker   for (int i = 0; i < max_measurements_; ++i) {
90*d9f75844SAndroid Build Coastguard Worker     variance += (buffer_[i] - mean) * (buffer_[i] - mean);
91*d9f75844SAndroid Build Coastguard Worker   }
92*d9f75844SAndroid Build Coastguard Worker   return variance / (max_measurements_ - 1);
93*d9f75844SAndroid Build Coastguard Worker }
94*d9f75844SAndroid Build Coastguard Worker 
FractionHigh(int min_required_samples) const95*d9f75844SAndroid Build Coastguard Worker absl::optional<double> QualityThreshold::FractionHigh(
96*d9f75844SAndroid Build Coastguard Worker     int min_required_samples) const {
97*d9f75844SAndroid Build Coastguard Worker   RTC_DCHECK_GT(min_required_samples, 0);
98*d9f75844SAndroid Build Coastguard Worker   if (num_certain_states_ < min_required_samples)
99*d9f75844SAndroid Build Coastguard Worker     return absl::nullopt;
100*d9f75844SAndroid Build Coastguard Worker 
101*d9f75844SAndroid Build Coastguard Worker   return static_cast<double>(num_high_states_) / num_certain_states_;
102*d9f75844SAndroid Build Coastguard Worker }
103*d9f75844SAndroid Build Coastguard Worker 
104*d9f75844SAndroid Build Coastguard Worker }  // namespace webrtc
105