sapling/eden/fs/store/StatsFetchContext.cpp
Chad Austin 4e1f60fc24 log checkout type and fetch counts to scuba
Summary: In addition to duration and success, log object fetch counts and checkout type to Scuba.

Reviewed By: fanzeyi

Differential Revision: D19334276

fbshipit-source-id: dabf52427f2ebda2b58df93194df39d52f4fcb4f
2020-02-05 16:05:30 -08:00

90 lines
3.0 KiB
C++

/*
* Copyright (c) Facebook, Inc. and its affiliates.
*
* This software may be used and distributed according to the terms of the
* GNU General Public License version 2.
*/
#include "eden/fs/store/StatsFetchContext.h"
#include <folly/logging/xlog.h>
namespace facebook {
namespace eden {
StatsFetchContext::StatsFetchContext(const StatsFetchContext& other) {
for (size_t y = 0; y < ObjectFetchContext::kObjectTypeEnumMax; ++y) {
for (size_t x = 0; x < ObjectFetchContext::kOriginEnumMax; ++x) {
// This could almost certainly use a more relaxed memory ordering.
counts_[y][x] = other.counts_[y][x].load();
}
}
}
void StatsFetchContext::didFetch(ObjectType type, const Hash&, Origin origin) {
XCHECK(type < ObjectFetchContext::kObjectTypeEnumMax)
<< "type is out of range: " << type;
XCHECK(origin < ObjectFetchContext::kOriginEnumMax)
<< "origin is out of range: " << type;
counts_[type][origin].fetch_add(1, std::memory_order_acq_rel);
}
uint64_t StatsFetchContext::countFetchesOfType(ObjectType type) const {
XCHECK(type < ObjectFetchContext::kObjectTypeEnumMax)
<< "type is out of range: " << type;
uint64_t result = 0;
for (unsigned origin = 0; origin < ObjectFetchContext::kOriginEnumMax;
++origin) {
result += counts_[type][origin].load(std::memory_order_acquire);
}
return result;
}
void StatsFetchContext::merge(const StatsFetchContext& other) {
for (unsigned type = 0; type < ObjectFetchContext::kObjectTypeEnumMax;
++type) {
for (unsigned origin = 0; origin < ObjectFetchContext::kOriginEnumMax;
++origin) {
counts_[type][origin] += other.counts_[type][origin];
}
}
}
uint64_t StatsFetchContext::countFetchesOfTypeAndOrigin(
ObjectType type,
Origin origin) const {
XCHECK(type < ObjectFetchContext::kObjectTypeEnumMax)
<< "type is out of range: " << type;
XCHECK(origin < ObjectFetchContext::kOriginEnumMax)
<< "origin is out of range: " << type;
return counts_[type][origin].load(std::memory_order_acquire);
}
FetchStatistics StatsFetchContext::computeStatistics() const {
auto computePercent = [&](uint64_t n, uint64_t d) -> unsigned short {
DCHECK_LE(n, d) << n << " > " << d;
if (d == 0) {
return 0;
}
return (1000 * n / d + 5) / 10;
};
auto computeAccessStats = [&](ObjectFetchContext::ObjectType type) {
uint64_t fromMemory = counts_[type][ObjectFetchContext::FromMemoryCache];
uint64_t fromDisk = counts_[type][ObjectFetchContext::FromDiskCache];
uint64_t fromBackingStore =
counts_[type][ObjectFetchContext::FromBackingStore];
uint64_t total = fromMemory + fromDisk + fromBackingStore;
return FetchStatistics::Access{
total, fromBackingStore, computePercent(fromMemory + fromDisk, total)};
};
auto result = FetchStatistics{};
result.tree = computeAccessStats(ObjectFetchContext::Tree);
result.blob = computeAccessStats(ObjectFetchContext::Blob);
result.metadata = computeAccessStats(ObjectFetchContext::BlobMetadata);
return result;
}
} // namespace eden
} // namespace facebook