forked from OSchip/llvm-project
209 lines
7.1 KiB
C++
209 lines
7.1 KiB
C++
//===-- cache_frag.cpp ----------------------------------------------------===//
|
|
//
|
|
// The LLVM Compiler Infrastructure
|
|
//
|
|
// This file is distributed under the University of Illinois Open Source
|
|
// License. See LICENSE.TXT for details.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
//
|
|
// This file is a part of EfficiencySanitizer, a family of performance tuners.
|
|
//
|
|
// This file contains cache fragmentation-specific code.
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "esan.h"
|
|
#include "esan_flags.h"
|
|
#include "sanitizer_common/sanitizer_addrhashmap.h"
|
|
#include "sanitizer_common/sanitizer_common.h"
|
|
#include "sanitizer_common/sanitizer_placement_new.h"
|
|
#include <string.h>
|
|
|
|
namespace __esan {
|
|
|
|
//===-- Struct field access counter runtime -------------------------------===//
|
|
|
|
// This should be kept consistent with LLVM's EfficiencySanitizer StructInfo.
|
|
struct StructInfo {
|
|
const char *StructName;
|
|
u32 Size;
|
|
u32 NumFields;
|
|
u32 *FieldOffset; // auxiliary struct field info.
|
|
u32 *FieldSize; // auxiliary struct field info.
|
|
const char **FieldTypeName; // auxiliary struct field info.
|
|
u64 *FieldCounters;
|
|
u64 *ArrayCounter;
|
|
bool hasAuxFieldInfo() { return FieldOffset != nullptr; }
|
|
};
|
|
|
|
// This should be kept consistent with LLVM's EfficiencySanitizer CacheFragInfo.
|
|
// The tool-specific information per compilation unit (module).
|
|
struct CacheFragInfo {
|
|
const char *UnitName;
|
|
u32 NumStructs;
|
|
StructInfo *Structs;
|
|
};
|
|
|
|
struct StructCounter {
|
|
StructInfo *Struct;
|
|
u64 Count; // The total access count of the struct.
|
|
u64 Ratio; // Difference ratio for the struct layout access.
|
|
};
|
|
|
|
// We use StructHashMap to keep track of an unique copy of StructCounter.
|
|
typedef AddrHashMap<StructCounter, 31051> StructHashMap;
|
|
struct Context {
|
|
StructHashMap StructMap;
|
|
u32 NumStructs;
|
|
u64 TotalCount; // The total access count of all structs.
|
|
};
|
|
static Context *Ctx;
|
|
|
|
static void reportStructSummary() {
|
|
// FIXME: provide a better struct field access summary report.
|
|
Report("%s: total struct field access count = %llu\n", SanitizerToolName,
|
|
Ctx->TotalCount);
|
|
}
|
|
|
|
// FIXME: we are still exploring proper ways to evaluate the difference between
|
|
// struct field counts. Currently, we use a simple formula to calculate the
|
|
// difference ratio: V1/V2.
|
|
static inline u64 computeDifferenceRatio(u64 Val1, u64 Val2) {
|
|
if (Val2 > Val1) {
|
|
Swap(Val1, Val2);
|
|
}
|
|
if (Val2 == 0)
|
|
Val2 = 1;
|
|
return (Val1 / Val2);
|
|
}
|
|
|
|
static void reportStructCounter(StructHashMap::Handle &Handle) {
|
|
const u32 TypePrintLimit = 512;
|
|
const char *type, *start, *end;
|
|
StructInfo *Struct = Handle->Struct;
|
|
// Union field address calculation is done via bitcast instead of GEP,
|
|
// so the count for union is always 0.
|
|
// We skip the union report to avoid confusion.
|
|
if (strncmp(Struct->StructName, "union.", 6) == 0)
|
|
return;
|
|
// Remove the '.' after class/struct during print.
|
|
if (strncmp(Struct->StructName, "class.", 6) == 0) {
|
|
type = "class";
|
|
start = &Struct->StructName[6];
|
|
} else {
|
|
type = "struct";
|
|
start = &Struct->StructName[7];
|
|
}
|
|
// Remove the suffixes with '#' during print.
|
|
end = strchr(start, '#');
|
|
CHECK(end != nullptr);
|
|
Report(" %s %.*s\n", type, end - start, start);
|
|
Report(" size = %u, count = %llu, ratio = %llu, array access = %llu\n",
|
|
Struct->Size, Handle->Count, Handle->Ratio, *Struct->ArrayCounter);
|
|
if (Struct->hasAuxFieldInfo()) {
|
|
for (u32 i = 0; i < Struct->NumFields; ++i) {
|
|
Report(" #%2u: offset = %u,\t size = %u,"
|
|
"\t count = %llu,\t type = %.*s\n",
|
|
i, Struct->FieldOffset[i], Struct->FieldSize[i],
|
|
Struct->FieldCounters[i], TypePrintLimit, Struct->FieldTypeName[i]);
|
|
}
|
|
} else {
|
|
for (u32 i = 0; i < Struct->NumFields; ++i) {
|
|
Report(" #%2u: count = %llu\n", i, Struct->FieldCounters[i]);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void computeStructRatio(StructHashMap::Handle &Handle) {
|
|
Handle->Ratio = 0;
|
|
Handle->Count = Handle->Struct->FieldCounters[0];
|
|
for (u32 i = 1; i < Handle->Struct->NumFields; ++i) {
|
|
Handle->Count += Handle->Struct->FieldCounters[i];
|
|
Handle->Ratio += computeDifferenceRatio(
|
|
Handle->Struct->FieldCounters[i - 1], Handle->Struct->FieldCounters[i]);
|
|
}
|
|
Ctx->TotalCount += Handle->Count;
|
|
if (Handle->Ratio >= (u64)getFlags()->report_threshold ||
|
|
(Verbosity() >= 1 && Handle->Count > 0))
|
|
reportStructCounter(Handle);
|
|
}
|
|
|
|
static void registerStructInfo(CacheFragInfo *CacheFrag) {
|
|
for (u32 i = 0; i < CacheFrag->NumStructs; ++i) {
|
|
StructInfo *Struct = &CacheFrag->Structs[i];
|
|
StructHashMap::Handle H(&Ctx->StructMap, (uptr)Struct->FieldCounters);
|
|
if (H.created()) {
|
|
VPrintf(2, " Register %s: %u fields\n", Struct->StructName,
|
|
Struct->NumFields);
|
|
H->Struct = Struct;
|
|
++Ctx->NumStructs;
|
|
} else {
|
|
VPrintf(2, " Duplicated %s: %u fields\n", Struct->StructName,
|
|
Struct->NumFields);
|
|
}
|
|
}
|
|
}
|
|
|
|
static void unregisterStructInfo(CacheFragInfo *CacheFrag) {
|
|
// FIXME: if the library is unloaded before finalizeCacheFrag, we should
|
|
// collect the result for later report.
|
|
for (u32 i = 0; i < CacheFrag->NumStructs; ++i) {
|
|
StructInfo *Struct = &CacheFrag->Structs[i];
|
|
StructHashMap::Handle H(&Ctx->StructMap, (uptr)Struct->FieldCounters, true);
|
|
if (H.exists()) {
|
|
VPrintf(2, " Unregister %s: %u fields\n", Struct->StructName,
|
|
Struct->NumFields);
|
|
// FIXME: we should move this call to finalizeCacheFrag once we can
|
|
// iterate over the hash map there.
|
|
computeStructRatio(H);
|
|
--Ctx->NumStructs;
|
|
} else {
|
|
VPrintf(2, " Duplicated %s: %u fields\n", Struct->StructName,
|
|
Struct->NumFields);
|
|
}
|
|
}
|
|
static bool Reported = false;
|
|
if (Ctx->NumStructs == 0 && !Reported) {
|
|
Reported = true;
|
|
reportStructSummary();
|
|
}
|
|
}
|
|
|
|
//===-- Init/exit functions -----------------------------------------------===//
|
|
|
|
void processCacheFragCompilationUnitInit(void *Ptr) {
|
|
CacheFragInfo *CacheFrag = (CacheFragInfo *)Ptr;
|
|
VPrintf(2, "in esan::%s: %s with %u class(es)/struct(s)\n", __FUNCTION__,
|
|
CacheFrag->UnitName, CacheFrag->NumStructs);
|
|
registerStructInfo(CacheFrag);
|
|
}
|
|
|
|
void processCacheFragCompilationUnitExit(void *Ptr) {
|
|
CacheFragInfo *CacheFrag = (CacheFragInfo *)Ptr;
|
|
VPrintf(2, "in esan::%s: %s with %u class(es)/struct(s)\n", __FUNCTION__,
|
|
CacheFrag->UnitName, CacheFrag->NumStructs);
|
|
unregisterStructInfo(CacheFrag);
|
|
}
|
|
|
|
void initializeCacheFrag() {
|
|
VPrintf(2, "in esan::%s\n", __FUNCTION__);
|
|
// We use placement new to initialize Ctx before C++ static initializaion.
|
|
// We make CtxMem 8-byte aligned for atomic operations in AddrHashMap.
|
|
static u64 CtxMem[sizeof(Context) / sizeof(u64) + 1];
|
|
Ctx = new (CtxMem) Context();
|
|
Ctx->NumStructs = 0;
|
|
}
|
|
|
|
int finalizeCacheFrag() {
|
|
VPrintf(2, "in esan::%s\n", __FUNCTION__);
|
|
return 0;
|
|
}
|
|
|
|
void reportCacheFrag() {
|
|
VPrintf(2, "in esan::%s\n", __FUNCTION__);
|
|
// FIXME: Not yet implemented. We need to iterate over all of the
|
|
// compilation unit data.
|
|
}
|
|
|
|
} // namespace __esan
|