/*
* Copyright 2018 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
#include "histogram.h"
#include <sstream>
#include <cmath>
#define LOG_TAG "TuningFork"
#include "Log.h"
#include "clearcutserializer.h"
namespace tuningfork {
Histogram::Histogram(float start_ms, float end_ms, int num_buckets_between)
: start_ms_(start_ms), end_ms_(end_ms),
bucket_dt_ms_((end_ms_ - start_ms_) / num_buckets_between),
num_buckets_(num_buckets_between + 2),
buckets_(num_buckets_), auto_range_(start_ms_ == 0 && end_ms_ == 0), count_(0) {
std::fill(buckets_.begin(), buckets_.end(), 0);
if (auto_range_)
samples_.reserve(num_buckets_);
else if (bucket_dt_ms_ <= 0)
ALOGE("Histogram end needs to be larger than histogram begin");
}
Histogram::Histogram(const Settings::Histogram &hs)
: Histogram(hs.bucket_min, hs.bucket_max, hs.n_buckets) {
}
void Histogram::Add(Sample dt_ms) {
if (auto_range_) {
samples_.push_back(dt_ms);
if (samples_.size() == samples_.capacity()) {
CalcBucketsFromSamples();
}
} else {
int i = (dt_ms - start_ms_) / bucket_dt_ms_;
if (i < 0)
buckets_[0]++;
else if (i + 1 >= num_buckets_)
buckets_[num_buckets_ - 1]++;
else
buckets_[i + 1]++;
++count_;
}
}
void Histogram::CalcBucketsFromSamples() {
Sample min_dt = std::numeric_limits<Sample>::max();
Sample max_dt = std::numeric_limits<Sample>::min();
Sample sum = 0;
Sample sum2 = 0;
for (Sample d: samples_) {
if (d < min_dt) min_dt = d;
if (d > max_dt) max_dt = d;
sum += d;
sum2 += d * d;
}
size_t n = samples_.size();
Sample mean = sum / n;
Sample var = sum2 / n - mean * mean;
if (var < 0) var = 0; // Can be negative due to rounding errors
Sample stddev = sqrt(var);
start_ms_ = std::max(mean - kAutoSizeNumStdDev * stddev, 0.0);
end_ms_ = mean + kAutoSizeNumStdDev * stddev;
bucket_dt_ms_ = (end_ms_ - start_ms_) / (num_buckets_ - 2);
if (bucket_dt_ms_ < kAutoSizeMinBucketSizeMs) {
bucket_dt_ms_ = kAutoSizeMinBucketSizeMs;
Sample w = bucket_dt_ms_ * (num_buckets_ - 2);
start_ms_ = mean - w / 2;
end_ms_ = mean + w / 2;
}
auto_range_ = false;
for (Sample d: samples_) {
Add(d);
}
}
std::string Histogram::ToJSON() const {
std::stringstream str;
str.precision(2);
str << std::fixed;
if (auto_range_) {
str << "{\"pmax\":[],\"cnts\":[]}";
} else {
str << "{\"pmax\":[";
Sample x = start_ms_;
for (int i = 0; i < num_buckets_ - 1; ++i) {
str << x << ",";
x += bucket_dt_ms_;
}
str << "99999],\"cnts\":[";
for (int i = 0; i < num_buckets_ - 1; ++i) {
str << buckets_[i] << ",";
}
if (num_buckets_ > 0)
str << buckets_.back();
str << "]}";
}
return str.str();
}
void Histogram::Clear(bool autorange) {
std::fill(buckets_.begin(), buckets_.end(), 0);
samples_.clear();
auto_range_ = autorange;
count_ = 0;
}
} // namespace tuningfork