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
|
|
|
// 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;
|
|
|
|
}
|
|
|
|
|
2014-12-03 03:48:40 +08:00
|
|
|
SymbolizedStack *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-12-03 03:48:40 +08:00
|
|
|
SymbolizedStack *frame = SymbolizedStack::New(addr);
|
|
|
|
if (__tsan_symbolize_external(addr, func_buf, sizeof(func_buf), file_buf,
|
|
|
|
sizeof(file_buf), &line, &col)) {
|
|
|
|
frame->info.function = internal_strdup(func_buf);
|
|
|
|
frame->info.file = internal_strdup(file_buf);
|
|
|
|
frame->info.line = line;
|
|
|
|
frame->info.column = col;
|
|
|
|
}
|
|
|
|
return frame;
|
2013-09-22 08:14:57 +08:00
|
|
|
}
|
2014-12-03 03:48:40 +08:00
|
|
|
return Symbolizer::GetOrInit()->SymbolizePC(addr);
|
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
|