forked from OSchip/llvm-project
[sanitizer] when dumping coverage bitset, dump seperate file for every module, instead of dumping a single combined bitset
llvm-svn: 231319
This commit is contained in:
parent
e9c019a7a6
commit
07aee9c2c6
|
@ -450,6 +450,7 @@ class InternalMmapVectorNoCtor {
|
||||||
}
|
}
|
||||||
|
|
||||||
void clear() { size_ = 0; }
|
void clear() { size_ = 0; }
|
||||||
|
bool empty() const { return size() == 0; }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void Resize(uptr new_capacity) {
|
void Resize(uptr new_capacity) {
|
||||||
|
|
|
@ -77,12 +77,15 @@ class CoverageData {
|
||||||
uptr cache_size);
|
uptr cache_size);
|
||||||
void DumpCallerCalleePairs();
|
void DumpCallerCalleePairs();
|
||||||
void DumpTrace();
|
void DumpTrace();
|
||||||
|
void DumpAsBitSet();
|
||||||
|
|
||||||
ALWAYS_INLINE
|
ALWAYS_INLINE
|
||||||
void TraceBasicBlock(s32 *id);
|
void TraceBasicBlock(s32 *id);
|
||||||
|
|
||||||
void InitializeGuardArray(s32 *guards);
|
void InitializeGuardArray(s32 *guards);
|
||||||
void InitializeGuards(s32 *guards, uptr n, const char *module_name);
|
void InitializeGuards(s32 *guards, uptr n, const char *module_name,
|
||||||
|
uptr caller_pc);
|
||||||
|
void UpdateModuleNameVec(uptr caller_pc, uptr range_beg, uptr range_end);
|
||||||
void InitializeCounters(u8 *counters, uptr n);
|
void InitializeCounters(u8 *counters, uptr n);
|
||||||
void ReinitializeGuards();
|
void ReinitializeGuards();
|
||||||
uptr GetNumberOf8bitCounters();
|
uptr GetNumberOf8bitCounters();
|
||||||
|
@ -113,8 +116,14 @@ class CoverageData {
|
||||||
// Vector of coverage guard arrays, protected by mu.
|
// Vector of coverage guard arrays, protected by mu.
|
||||||
InternalMmapVectorNoCtor<s32*> guard_array_vec;
|
InternalMmapVectorNoCtor<s32*> guard_array_vec;
|
||||||
|
|
||||||
// Vector of module (compilation unit) names.
|
struct NamedPcRange {
|
||||||
InternalMmapVectorNoCtor<const char*> comp_unit_name_vec;
|
const char *name;
|
||||||
|
uptr beg, end; // elements [beg,end) in pc_array.
|
||||||
|
};
|
||||||
|
|
||||||
|
// Vector of module and compilation unit pc ranges.
|
||||||
|
InternalMmapVectorNoCtor<NamedPcRange> comp_unit_name_vec;
|
||||||
|
InternalMmapVectorNoCtor<NamedPcRange> module_name_vec;
|
||||||
|
|
||||||
struct CounterAndSize {
|
struct CounterAndSize {
|
||||||
u8 *counters;
|
u8 *counters;
|
||||||
|
@ -311,16 +320,33 @@ void CoverageData::InitializeCounters(u8 *counters, uptr n) {
|
||||||
num_8bit_counters += n;
|
num_8bit_counters += n;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void CoverageData::UpdateModuleNameVec(uptr caller_pc, uptr range_beg,
|
||||||
|
uptr range_end) {
|
||||||
|
auto sym = Symbolizer::GetOrInit();
|
||||||
|
if (!sym)
|
||||||
|
return;
|
||||||
|
const char *module_name = sym->GetModuleNameForPc(caller_pc);
|
||||||
|
if (!module_name) return;
|
||||||
|
if (module_name_vec.empty() || module_name_vec.back().name != module_name)
|
||||||
|
module_name_vec.push_back({module_name, range_beg, range_end});
|
||||||
|
else
|
||||||
|
module_name_vec.back().end = range_end;
|
||||||
|
}
|
||||||
|
|
||||||
void CoverageData::InitializeGuards(s32 *guards, uptr n,
|
void CoverageData::InitializeGuards(s32 *guards, uptr n,
|
||||||
const char *module_name) {
|
const char *comp_unit_name,
|
||||||
|
uptr caller_pc) {
|
||||||
// The array 'guards' has n+1 elements, we use the element zero
|
// The array 'guards' has n+1 elements, we use the element zero
|
||||||
// to store 'n'.
|
// to store 'n'.
|
||||||
CHECK_LT(n, 1 << 30);
|
CHECK_LT(n, 1 << 30);
|
||||||
guards[0] = static_cast<s32>(n);
|
guards[0] = static_cast<s32>(n);
|
||||||
InitializeGuardArray(guards);
|
InitializeGuardArray(guards);
|
||||||
SpinMutexLock l(&mu);
|
SpinMutexLock l(&mu);
|
||||||
comp_unit_name_vec.push_back(module_name);
|
uptr range_end = atomic_load(&pc_array_index, memory_order_relaxed);
|
||||||
|
uptr range_beg = range_end - n;
|
||||||
|
comp_unit_name_vec.push_back({comp_unit_name, range_beg, range_end});
|
||||||
guard_array_vec.push_back(guards);
|
guard_array_vec.push_back(guards);
|
||||||
|
UpdateModuleNameVec(caller_pc, range_beg, range_end);
|
||||||
}
|
}
|
||||||
|
|
||||||
// If guard is negative, atomically set it to -guard and store the PC in
|
// If guard is negative, atomically set it to -guard and store the PC in
|
||||||
|
@ -539,7 +565,7 @@ void CoverageData::DumpTrace() {
|
||||||
if (fd < 0) return;
|
if (fd < 0) return;
|
||||||
out.clear();
|
out.clear();
|
||||||
for (uptr i = 0; i < comp_unit_name_vec.size(); i++)
|
for (uptr i = 0; i < comp_unit_name_vec.size(); i++)
|
||||||
out.append("%s\n", comp_unit_name_vec[i]);
|
out.append("%s\n", comp_unit_name_vec[i].name);
|
||||||
internal_write(fd, out.data(), out.length());
|
internal_write(fd, out.data(), out.length());
|
||||||
internal_close(fd);
|
internal_close(fd);
|
||||||
|
|
||||||
|
@ -614,24 +640,31 @@ void CoverageData::TraceBasicBlock(s32 *id) {
|
||||||
tr_event_pointer++;
|
tr_event_pointer++;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void CovDumpAsBitSet() {
|
void CoverageData::DumpAsBitSet() {
|
||||||
if (!common_flags()->coverage_bitset) return;
|
if (!common_flags()->coverage_bitset) return;
|
||||||
if (!coverage_data.size()) return;
|
if (!size()) return;
|
||||||
int fd = CovOpenFile(/* packed */false, "combined", "bitset-sancov");
|
InternalScopedBuffer<char> out(size());
|
||||||
if (fd < 0) return;
|
for (uptr m = 0; m < module_name_vec.size(); m++) {
|
||||||
uptr n = coverage_data.size();
|
uptr n_set_bits = 0;
|
||||||
uptr n_set_bits = 0;
|
auto r = module_name_vec[m];
|
||||||
InternalScopedBuffer<char> out(n);
|
CHECK(r.name);
|
||||||
for (uptr i = 0; i < n; i++) {
|
CHECK_LE(r.beg, r.end);
|
||||||
uptr pc = coverage_data.data()[i];
|
CHECK_LE(r.end, size());
|
||||||
out[i] = pc ? '1' : '0';
|
for (uptr i = r.beg; i < r.end; i++) {
|
||||||
if (pc)
|
uptr pc = data()[i];
|
||||||
n_set_bits++;
|
out[i] = pc ? '1' : '0';
|
||||||
|
if (pc)
|
||||||
|
n_set_bits++;
|
||||||
|
}
|
||||||
|
const char *base_name = StripModuleName(r.name);
|
||||||
|
int fd = CovOpenFile(/* packed */ false, base_name, "bitset-sancov");
|
||||||
|
if (fd < 0) return;
|
||||||
|
internal_write(fd, out.data() + r.beg, r.end - r.beg);
|
||||||
|
internal_close(fd);
|
||||||
|
VReport(1,
|
||||||
|
" CovDump: bitset of %zd bits written for '%s', %zd bits are set\n",
|
||||||
|
r.end - r.beg, base_name, n_set_bits);
|
||||||
}
|
}
|
||||||
internal_write(fd, out.data(), n);
|
|
||||||
internal_close(fd);
|
|
||||||
VReport(1, " CovDump: bitset of %zd bits written, %zd bits are set\n", n,
|
|
||||||
n_set_bits);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Dump the coverage on disk.
|
// Dump the coverage on disk.
|
||||||
|
@ -640,7 +673,7 @@ static void CovDump() {
|
||||||
#if !SANITIZER_WINDOWS
|
#if !SANITIZER_WINDOWS
|
||||||
if (atomic_fetch_add(&dump_once_guard, 1, memory_order_relaxed))
|
if (atomic_fetch_add(&dump_once_guard, 1, memory_order_relaxed))
|
||||||
return;
|
return;
|
||||||
CovDumpAsBitSet();
|
coverage_data.DumpAsBitSet();
|
||||||
coverage_data.DumpTrace();
|
coverage_data.DumpTrace();
|
||||||
if (!common_flags()->coverage_pcs) return;
|
if (!common_flags()->coverage_pcs) return;
|
||||||
uptr size = coverage_data.size();
|
uptr size = coverage_data.size();
|
||||||
|
@ -770,8 +803,8 @@ SANITIZER_INTERFACE_ATTRIBUTE void __sanitizer_cov_init() {
|
||||||
SANITIZER_INTERFACE_ATTRIBUTE void __sanitizer_cov_dump() { CovDump(); }
|
SANITIZER_INTERFACE_ATTRIBUTE void __sanitizer_cov_dump() { CovDump(); }
|
||||||
SANITIZER_INTERFACE_ATTRIBUTE void
|
SANITIZER_INTERFACE_ATTRIBUTE void
|
||||||
__sanitizer_cov_module_init(s32 *guards, uptr npcs, u8 *counters,
|
__sanitizer_cov_module_init(s32 *guards, uptr npcs, u8 *counters,
|
||||||
const char *module_name) {
|
const char *comp_unit_name) {
|
||||||
coverage_data.InitializeGuards(guards, npcs, module_name);
|
coverage_data.InitializeGuards(guards, npcs, comp_unit_name, GET_CALLER_PC());
|
||||||
coverage_data.InitializeCounters(counters, npcs);
|
coverage_data.InitializeCounters(counters, npcs);
|
||||||
if (!common_flags()->coverage_direct) return;
|
if (!common_flags()->coverage_direct) return;
|
||||||
if (SANITIZER_ANDROID && coverage_enabled) {
|
if (SANITIZER_ANDROID && coverage_enabled) {
|
||||||
|
|
|
@ -90,6 +90,13 @@ class Symbolizer {
|
||||||
uptr *module_address) {
|
uptr *module_address) {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
const char *GetModuleNameForPc(uptr pc) {
|
||||||
|
const char *module_name = 0;
|
||||||
|
uptr unused;
|
||||||
|
if (GetModuleNameAndOffsetForPC(pc, &module_name, &unused))
|
||||||
|
return module_name;
|
||||||
|
return nullptr;
|
||||||
|
}
|
||||||
virtual bool CanReturnFileLineInfo() {
|
virtual bool CanReturnFileLineInfo() {
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
|
@ -19,11 +19,11 @@ int main(int argc, char **argv) {
|
||||||
sink = 0;
|
sink = 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
// CHECK1: CovDump: bitset of 1 bits written, 1 bits are set
|
// CHECK1: CovDump: bitset of 1 bits written for '{{.*}}', 1 bits are set
|
||||||
// CHECK1: 1 PCs written
|
// CHECK1: 1 PCs written
|
||||||
// CHECK2: CovDump: bitset of 3 bits written, 2 bits are set
|
// CHECK2: CovDump: bitset of 3 bits written for '{{.*}}', 2 bits are set
|
||||||
// CHECK2: 2 PCs written
|
// CHECK2: 2 PCs written
|
||||||
// CHECK3: CovDump: bitset of 4 bits written, 3 bits are set
|
// CHECK3: CovDump: bitset of 4 bits written for '{{.*}}', 3 bits are set
|
||||||
// CHECK3: 3 PCs written
|
// CHECK3: 3 PCs written
|
||||||
// CHECK3_NOBITSET-NOT: bitset of
|
// CHECK3_NOBITSET-NOT: bitset of
|
||||||
// CHECK3_NOPCS-NOT: PCs written
|
// CHECK3_NOPCS-NOT: PCs written
|
||||||
|
|
Loading…
Reference in New Issue