#include "net/disk_cache/blockfile/stats.h"
#include <algorithm>
#include <array>
#include <bit>
#include <cstdint>
#include "base/check.h"
#include "base/containers/span.h"
#include "base/format_macros.h"
#include "base/strings/string_util.h"
#include "base/strings/stringprintf.h"
namespace {
const int32_t kDiskSignature = 0xF01427E0;
struct OnDiskStats {
int32_t signature;
uint32_t size;
int data_sizes[disk_cache::Stats::kDataSizesLength];
int64_t counters[disk_cache::Stats::MAX_COUNTER];
};
static_assert(sizeof(OnDiskStats) < 512, "needs more than 2 blocks");
constexpr auto kCounterNames = std::to_array<const char*>({
"Open miss", "Open hit", "Create miss",
"Create hit", "Resurrect hit", "Create error",
"Trim entry", "Doom entry", "Doom cache",
"Invalid entry", "Open entries", "Max entries",
"Timer", "Read data", "Write data",
"Open rankings", "Get rankings", "Fatal error",
"Last report", "Last report timer", "Doom recent entries",
"unused",
});
static_assert(std::size(kCounterNames) == disk_cache::Stats::MAX_COUNTER,
"update the names");
}
namespace disk_cache {
bool VerifyStats(OnDiskStats* stats) {
if (stats->signature != kDiskSignature)
return false;
auto stats_bytes = base::byte_span_from_ref(*stats);
if (stats->size > sizeof(*stats)) {
std::ranges::fill(stats_bytes, 0);
stats->signature = kDiskSignature;
} else if (stats->size < sizeof(*stats)) {
std::ranges::fill(stats_bytes.subspan(stats->size), 0);
stats->size = sizeof(*stats);
}
return true;
}
Stats::Stats() = default;
Stats::~Stats() = default;
bool Stats::Init(base::span<uint8_t> data, Addr address) {
OnDiskStats local_stats;
OnDiskStats* stats = &local_stats;
auto local_stats_bytes = base::byte_span_from_ref(local_stats);
if (data.empty()) {
std::ranges::fill(local_stats_bytes, 0);
local_stats.signature = kDiskSignature;
local_stats.size = sizeof(local_stats);
} else if (data.size() >= sizeof(*stats)) {
stats = reinterpret_cast<OnDiskStats*>(data.data());
if (!VerifyStats(stats)) {
std::ranges::fill(local_stats_bytes, 0);
if (base::byte_span_from_ref(*stats).first(sizeof(local_stats)) ==
local_stats_bytes) {
local_stats.signature = kDiskSignature;
local_stats.size = sizeof(local_stats);
stats = &local_stats;
} else {
return false;
}
}
} else {
return false;
}
storage_addr_ = address;
base::as_writable_byte_span(data_sizes_)
.copy_from_nonoverlapping(base::as_byte_span(stats->data_sizes));
base::as_writable_byte_span(counters_).copy_from_nonoverlapping(
base::as_byte_span(stats->counters));
SetCounter(UNUSED, 0);
return true;
}
void Stats::InitSizeHistogram() {
static bool first_time = true;
if (!first_time)
return;
first_time = false;
for (int& data_size : data_sizes_) {
if (data_size < 0)
data_size = 0;
}
}
int Stats::StorageSize() {
static_assert(sizeof(OnDiskStats) <= 256 * 2, "use more blocks");
return 256 * 2;
}
void Stats::ModifyStorageStats(int32_t old_size, int32_t new_size) {
int new_index = GetStatsBucket(new_size);
int old_index = GetStatsBucket(old_size);
if (new_size)
data_sizes_[new_index]++;
if (old_size)
data_sizes_[old_index]--;
}
void Stats::OnEvent(Counters an_event) {
DCHECK(an_event >= MIN_COUNTER && an_event < MAX_COUNTER);
counters_[an_event]++;
}
void Stats::SetCounter(Counters counter, int64_t value) {
DCHECK(counter >= MIN_COUNTER && counter < MAX_COUNTER);
counters_[counter] = value;
}
int64_t Stats::GetCounter(Counters counter) const {
DCHECK(counter >= MIN_COUNTER && counter < MAX_COUNTER);
return counters_[counter];
}
void Stats::GetItems(StatsItems* items) {
std::pair<std::string, std::string> item;
for (int i = 0; i < kDataSizesLength; i++) {
item.first = base::StringPrintf("Size%02d", i);
item.second = base::StringPrintf("0x%08x", data_sizes_[i]);
items->push_back(item);
}
for (int i = MIN_COUNTER; i < MAX_COUNTER; i++) {
item.first = kCounterNames[i];
item.second = base::StringPrintf("0x%" PRIx64, counters_[i]);
items->push_back(item);
}
}
void Stats::ResetRatios() {
SetCounter(OPEN_HIT, 0);
SetCounter(OPEN_MISS, 0);
SetCounter(RESURRECT_HIT, 0);
SetCounter(CREATE_HIT, 0);
}
int Stats::GetLargeEntriesSize() {
int total = 0;
for (int bucket = 20; bucket < kDataSizesLength; bucket++)
total += data_sizes_[bucket] * GetBucketRange(bucket);
return total;
}
int Stats::SerializeStats(base::span<uint8_t> data, Addr* address) {
if (data.size() < sizeof(OnDiskStats)) {
return 0;
}
OnDiskStats* stats = reinterpret_cast<OnDiskStats*>(data.data());
stats->signature = kDiskSignature;
stats->size = sizeof(*stats);
base::as_writable_byte_span(stats->data_sizes)
.copy_from_nonoverlapping(base::as_byte_span(data_sizes_));
base::as_writable_byte_span(stats->counters)
.copy_from_nonoverlapping(base::as_byte_span(counters_));
*address = storage_addr_;
return sizeof(*stats);
}
int Stats::GetBucketRange(size_t i) const {
CHECK_LE(i, static_cast<size_t>(kDataSizesLength));
if (i < 2)
return static_cast<int>(1024 * i);
if (i < 12)
return static_cast<int>(2048 * (i - 1));
if (i < 17)
return static_cast<int>(4096 * (i - 11)) + 20 * 1024;
int n = 64 * 1024;
i -= 17;
n <<= i;
return n;
}
int Stats::GetStatsBucket(int32_t size) {
if (size < 1024)
return 0;
if (size < 20 * 1024)
return size / 2048 + 1;
if (size < 40 * 1024)
return (size - 20 * 1024) / 4096 + 11;
int result = std::bit_width<uint32_t>(size);
static_assert(kDataSizesLength > 16, "update the scale");
if (result >= kDataSizesLength)
result = kDataSizesLength - 1;
return result;
}
int Stats::GetRatio(Counters hit, Counters miss) const {
int64_t ratio = GetCounter(hit) * 100;
if (!ratio)
return 0;
ratio /= (GetCounter(hit) + GetCounter(miss));
return static_cast<int>(ratio);
}
}