2012-07-05 15:18:29 +08:00
|
|
|
//===-- tsan_symbolize.cc -------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// 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 ThreadSanitizer (TSan), a race detector.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "tsan_symbolize.h"
|
|
|
|
|
|
|
|
#include "sanitizer_common/sanitizer_common.h"
|
|
|
|
#include "sanitizer_common/sanitizer_placement_new.h"
|
|
|
|
#include "sanitizer_common/sanitizer_symbolizer.h"
|
|
|
|
#include "tsan_flags.h"
|
|
|
|
#include "tsan_report.h"
|
2013-01-29 21:03:07 +08:00
|
|
|
#include "tsan_rtl.h"
|
2012-07-05 15:18:29 +08:00
|
|
|
|
|
|
|
namespace __tsan {
|
|
|
|
|
2013-11-01 05:44:07 +08:00
|
|
|
void EnterSymbolizer() {
|
|
|
|
ThreadState *thr = cur_thread();
|
|
|
|
CHECK(!thr->in_symbolizer);
|
|
|
|
thr->in_symbolizer = true;
|
2014-04-11 23:36:54 +08:00
|
|
|
thr->ignore_interceptors++;
|
2013-11-01 05:44:07 +08:00
|
|
|
}
|
2013-01-29 21:03:07 +08:00
|
|
|
|
2013-11-01 05:44:07 +08:00
|
|
|
void ExitSymbolizer() {
|
|
|
|
ThreadState *thr = cur_thread();
|
|
|
|
CHECK(thr->in_symbolizer);
|
|
|
|
thr->in_symbolizer = false;
|
2014-04-11 23:36:54 +08:00
|
|
|
thr->ignore_interceptors--;
|
2013-11-01 05:44:07 +08:00
|
|
|
}
|
2013-01-29 21:03:07 +08:00
|
|
|
|
2013-09-22 08:14:57 +08:00
|
|
|
// Denotes fake PC values that come from JIT/JAVA/etc.
|
|
|
|
// For such PC values __tsan_symbolize_external() will be called.
|
|
|
|
const uptr kExternalPCBit = 1ULL << 60;
|
|
|
|
|
|
|
|
// May be overriden by JIT/JAVA/etc,
|
|
|
|
// whatever produces PCs marked with kExternalPCBit.
|
|
|
|
extern "C" bool __tsan_symbolize_external(uptr pc,
|
|
|
|
char *func_buf, uptr func_siz,
|
|
|
|
char *file_buf, uptr file_siz,
|
|
|
|
int *line, int *col)
|
|
|
|
SANITIZER_WEAK_ATTRIBUTE;
|
|
|
|
|
|
|
|
bool __tsan_symbolize_external(uptr pc,
|
|
|
|
char *func_buf, uptr func_siz,
|
|
|
|
char *file_buf, uptr file_siz,
|
|
|
|
int *line, int *col) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2012-07-05 15:18:29 +08:00
|
|
|
ReportStack *SymbolizeCode(uptr addr) {
|
2013-09-22 08:14:57 +08:00
|
|
|
// Check if PC comes from non-native land.
|
|
|
|
if (addr & kExternalPCBit) {
|
|
|
|
// Declare static to not consume too much stack space.
|
|
|
|
// We symbolize reports in a single thread, so this is fine.
|
|
|
|
static char func_buf[1024];
|
|
|
|
static char file_buf[1024];
|
|
|
|
int line, col;
|
2014-11-05 04:52:37 +08:00
|
|
|
ReportStack *ent = ReportStack::New(addr);
|
2013-09-22 08:14:57 +08:00
|
|
|
if (!__tsan_symbolize_external(addr, func_buf, sizeof(func_buf),
|
|
|
|
file_buf, sizeof(file_buf), &line, &col))
|
2014-11-05 04:52:37 +08:00
|
|
|
return ent;
|
|
|
|
ent->info.function = internal_strdup(func_buf);
|
|
|
|
ent->info.file = internal_strdup(file_buf);
|
|
|
|
ent->info.line = line;
|
|
|
|
ent->info.column = col;
|
2013-09-22 08:14:57 +08:00
|
|
|
return ent;
|
|
|
|
}
|
2013-01-29 17:35:14 +08:00
|
|
|
static const uptr kMaxAddrFrames = 16;
|
|
|
|
InternalScopedBuffer<AddressInfo> addr_frames(kMaxAddrFrames);
|
|
|
|
for (uptr i = 0; i < kMaxAddrFrames; i++)
|
|
|
|
new(&addr_frames[i]) AddressInfo();
|
2014-09-11 06:45:09 +08:00
|
|
|
uptr addr_frames_num = Symbolizer::GetOrInit()->SymbolizePC(
|
2013-10-26 07:03:29 +08:00
|
|
|
addr, addr_frames.data(), kMaxAddrFrames);
|
2013-01-29 17:35:14 +08:00
|
|
|
if (addr_frames_num == 0)
|
2014-11-05 04:52:37 +08:00
|
|
|
return ReportStack::New(addr);
|
2013-01-29 17:35:14 +08:00
|
|
|
ReportStack *top = 0;
|
|
|
|
ReportStack *bottom = 0;
|
|
|
|
for (uptr i = 0; i < addr_frames_num; i++) {
|
2014-11-05 04:52:37 +08:00
|
|
|
ReportStack *cur_entry = ReportStack::New(addr);
|
|
|
|
cur_entry->info = addr_frames[i];
|
2013-01-29 17:35:14 +08:00
|
|
|
if (i == 0)
|
|
|
|
top = cur_entry;
|
|
|
|
else
|
|
|
|
bottom->next = cur_entry;
|
|
|
|
bottom = cur_entry;
|
2012-07-05 15:18:29 +08:00
|
|
|
}
|
2013-01-29 17:35:14 +08:00
|
|
|
return top;
|
2012-07-05 15:18:29 +08:00
|
|
|
}
|
|
|
|
|
2013-01-11 15:23:51 +08:00
|
|
|
ReportLocation *SymbolizeData(uptr addr) {
|
|
|
|
DataInfo info;
|
2014-09-11 06:45:09 +08:00
|
|
|
if (!Symbolizer::GetOrInit()->SymbolizeData(addr, &info))
|
2013-01-11 15:23:51 +08:00
|
|
|
return 0;
|
2014-11-05 06:07:57 +08:00
|
|
|
ReportLocation *ent = ReportLocation::New(ReportLocationGlobal);
|
|
|
|
ent->global = info;
|
2013-01-11 15:23:51 +08:00
|
|
|
return ent;
|
2012-07-05 15:18:29 +08:00
|
|
|
}
|
|
|
|
|
2013-03-19 18:24:01 +08:00
|
|
|
void SymbolizeFlush() {
|
2014-09-11 06:45:09 +08:00
|
|
|
Symbolizer::GetOrInit()->Flush();
|
2013-03-19 18:24:01 +08:00
|
|
|
}
|
|
|
|
|
2012-07-05 15:18:29 +08:00
|
|
|
} // namespace __tsan
|