xref: /aosp_15_r20/external/webrtc/api/numerics/samples_stats_counter.cc (revision d9f758449e529ab9291ac668be2861e7a55c2422)
1*d9f75844SAndroid Build Coastguard Worker /*
2*d9f75844SAndroid Build Coastguard Worker  *  Copyright (c) 2018 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 "api/numerics/samples_stats_counter.h"
12*d9f75844SAndroid Build Coastguard Worker 
13*d9f75844SAndroid Build Coastguard Worker #include <algorithm>
14*d9f75844SAndroid Build Coastguard Worker #include <cmath>
15*d9f75844SAndroid Build Coastguard Worker 
16*d9f75844SAndroid Build Coastguard Worker #include "absl/algorithm/container.h"
17*d9f75844SAndroid Build Coastguard Worker #include "rtc_base/time_utils.h"
18*d9f75844SAndroid Build Coastguard Worker 
19*d9f75844SAndroid Build Coastguard Worker namespace webrtc {
20*d9f75844SAndroid Build Coastguard Worker 
21*d9f75844SAndroid Build Coastguard Worker SamplesStatsCounter::SamplesStatsCounter() = default;
SamplesStatsCounter(size_t expected_samples_count)22*d9f75844SAndroid Build Coastguard Worker SamplesStatsCounter::SamplesStatsCounter(size_t expected_samples_count) {
23*d9f75844SAndroid Build Coastguard Worker   samples_.reserve(expected_samples_count);
24*d9f75844SAndroid Build Coastguard Worker }
25*d9f75844SAndroid Build Coastguard Worker 
26*d9f75844SAndroid Build Coastguard Worker SamplesStatsCounter::~SamplesStatsCounter() = default;
27*d9f75844SAndroid Build Coastguard Worker SamplesStatsCounter::SamplesStatsCounter(const SamplesStatsCounter&) = default;
28*d9f75844SAndroid Build Coastguard Worker SamplesStatsCounter& SamplesStatsCounter::operator=(
29*d9f75844SAndroid Build Coastguard Worker     const SamplesStatsCounter&) = default;
30*d9f75844SAndroid Build Coastguard Worker SamplesStatsCounter::SamplesStatsCounter(SamplesStatsCounter&&) = default;
31*d9f75844SAndroid Build Coastguard Worker SamplesStatsCounter& SamplesStatsCounter::operator=(SamplesStatsCounter&&) =
32*d9f75844SAndroid Build Coastguard Worker     default;
33*d9f75844SAndroid Build Coastguard Worker 
AddSample(double value)34*d9f75844SAndroid Build Coastguard Worker void SamplesStatsCounter::AddSample(double value) {
35*d9f75844SAndroid Build Coastguard Worker   AddSample(StatsSample{value, Timestamp::Micros(rtc::TimeMicros())});
36*d9f75844SAndroid Build Coastguard Worker }
37*d9f75844SAndroid Build Coastguard Worker 
AddSample(StatsSample sample)38*d9f75844SAndroid Build Coastguard Worker void SamplesStatsCounter::AddSample(StatsSample sample) {
39*d9f75844SAndroid Build Coastguard Worker   stats_.AddSample(sample.value);
40*d9f75844SAndroid Build Coastguard Worker   samples_.push_back(sample);
41*d9f75844SAndroid Build Coastguard Worker   sorted_ = false;
42*d9f75844SAndroid Build Coastguard Worker }
43*d9f75844SAndroid Build Coastguard Worker 
AddSamples(const SamplesStatsCounter & other)44*d9f75844SAndroid Build Coastguard Worker void SamplesStatsCounter::AddSamples(const SamplesStatsCounter& other) {
45*d9f75844SAndroid Build Coastguard Worker   stats_.MergeStatistics(other.stats_);
46*d9f75844SAndroid Build Coastguard Worker   samples_.insert(samples_.end(), other.samples_.begin(), other.samples_.end());
47*d9f75844SAndroid Build Coastguard Worker   sorted_ = false;
48*d9f75844SAndroid Build Coastguard Worker }
49*d9f75844SAndroid Build Coastguard Worker 
GetPercentile(double percentile)50*d9f75844SAndroid Build Coastguard Worker double SamplesStatsCounter::GetPercentile(double percentile) {
51*d9f75844SAndroid Build Coastguard Worker   RTC_DCHECK(!IsEmpty());
52*d9f75844SAndroid Build Coastguard Worker   RTC_CHECK_GE(percentile, 0);
53*d9f75844SAndroid Build Coastguard Worker   RTC_CHECK_LE(percentile, 1);
54*d9f75844SAndroid Build Coastguard Worker   if (!sorted_) {
55*d9f75844SAndroid Build Coastguard Worker     absl::c_sort(samples_, [](const StatsSample& a, const StatsSample& b) {
56*d9f75844SAndroid Build Coastguard Worker       return a.value < b.value;
57*d9f75844SAndroid Build Coastguard Worker     });
58*d9f75844SAndroid Build Coastguard Worker     sorted_ = true;
59*d9f75844SAndroid Build Coastguard Worker   }
60*d9f75844SAndroid Build Coastguard Worker   const double raw_rank = percentile * (samples_.size() - 1);
61*d9f75844SAndroid Build Coastguard Worker   double int_part;
62*d9f75844SAndroid Build Coastguard Worker   double fract_part = std::modf(raw_rank, &int_part);
63*d9f75844SAndroid Build Coastguard Worker   size_t rank = static_cast<size_t>(int_part);
64*d9f75844SAndroid Build Coastguard Worker   if (fract_part >= 1.0) {
65*d9f75844SAndroid Build Coastguard Worker     // It can happen due to floating point calculation error.
66*d9f75844SAndroid Build Coastguard Worker     rank++;
67*d9f75844SAndroid Build Coastguard Worker     fract_part -= 1.0;
68*d9f75844SAndroid Build Coastguard Worker   }
69*d9f75844SAndroid Build Coastguard Worker 
70*d9f75844SAndroid Build Coastguard Worker   RTC_DCHECK_GE(rank, 0);
71*d9f75844SAndroid Build Coastguard Worker   RTC_DCHECK_LT(rank, samples_.size());
72*d9f75844SAndroid Build Coastguard Worker   RTC_DCHECK_GE(fract_part, 0);
73*d9f75844SAndroid Build Coastguard Worker   RTC_DCHECK_LT(fract_part, 1);
74*d9f75844SAndroid Build Coastguard Worker   RTC_DCHECK(rank + fract_part == raw_rank);
75*d9f75844SAndroid Build Coastguard Worker 
76*d9f75844SAndroid Build Coastguard Worker   const double low = samples_[rank].value;
77*d9f75844SAndroid Build Coastguard Worker   const double high = samples_[std::min(rank + 1, samples_.size() - 1)].value;
78*d9f75844SAndroid Build Coastguard Worker   return low + fract_part * (high - low);
79*d9f75844SAndroid Build Coastguard Worker }
80*d9f75844SAndroid Build Coastguard Worker 
operator *(const SamplesStatsCounter & counter,double value)81*d9f75844SAndroid Build Coastguard Worker SamplesStatsCounter operator*(const SamplesStatsCounter& counter,
82*d9f75844SAndroid Build Coastguard Worker                               double value) {
83*d9f75844SAndroid Build Coastguard Worker   SamplesStatsCounter out;
84*d9f75844SAndroid Build Coastguard Worker   for (const auto& sample : counter.GetTimedSamples()) {
85*d9f75844SAndroid Build Coastguard Worker     out.AddSample(
86*d9f75844SAndroid Build Coastguard Worker         SamplesStatsCounter::StatsSample{sample.value * value, sample.time});
87*d9f75844SAndroid Build Coastguard Worker   }
88*d9f75844SAndroid Build Coastguard Worker   return out;
89*d9f75844SAndroid Build Coastguard Worker }
90*d9f75844SAndroid Build Coastguard Worker 
operator /(const SamplesStatsCounter & counter,double value)91*d9f75844SAndroid Build Coastguard Worker SamplesStatsCounter operator/(const SamplesStatsCounter& counter,
92*d9f75844SAndroid Build Coastguard Worker                               double value) {
93*d9f75844SAndroid Build Coastguard Worker   SamplesStatsCounter out;
94*d9f75844SAndroid Build Coastguard Worker   for (const auto& sample : counter.GetTimedSamples()) {
95*d9f75844SAndroid Build Coastguard Worker     out.AddSample(
96*d9f75844SAndroid Build Coastguard Worker         SamplesStatsCounter::StatsSample{sample.value / value, sample.time});
97*d9f75844SAndroid Build Coastguard Worker   }
98*d9f75844SAndroid Build Coastguard Worker   return out;
99*d9f75844SAndroid Build Coastguard Worker }
100*d9f75844SAndroid Build Coastguard Worker 
101*d9f75844SAndroid Build Coastguard Worker }  // namespace webrtc
102