forked from OSchip/llvm-project
[UBSan] Call UBSan initialization as early as possible.
Specifically, use .preinit_array initialization on Linux and dynamic global initializer on another platforms. Historically UBSan didn't have any initialization code and its runtime was stateless. This is no longer the case - UBSan relies on some non-trivial functionality from sanitizer_common (e.g. online symbolization) and is now configurable by runtime flags. Additionally, we've dropped support for enabling UBSan only for a few shared objects, so UBSan is now always linked into the main executable, so now we can use similar initialization as all the rest sanitizers. llvm-svn: 213983
This commit is contained in:
parent
ac4b69e40b
commit
6eb53d6460
|
@ -2,6 +2,7 @@
|
||||||
|
|
||||||
set(UBSAN_SOURCES
|
set(UBSAN_SOURCES
|
||||||
ubsan_diag.cc
|
ubsan_diag.cc
|
||||||
|
ubsan_init.cc
|
||||||
ubsan_flags.cc
|
ubsan_flags.cc
|
||||||
ubsan_handlers.cc
|
ubsan_handlers.cc
|
||||||
ubsan_value.cc
|
ubsan_value.cc
|
||||||
|
|
|
@ -12,10 +12,8 @@
|
||||||
//===----------------------------------------------------------------------===//
|
//===----------------------------------------------------------------------===//
|
||||||
|
|
||||||
#include "ubsan_diag.h"
|
#include "ubsan_diag.h"
|
||||||
|
#include "ubsan_init.h"
|
||||||
#include "ubsan_flags.h"
|
#include "ubsan_flags.h"
|
||||||
#include "sanitizer_common/sanitizer_common.h"
|
|
||||||
#include "sanitizer_common/sanitizer_flags.h"
|
|
||||||
#include "sanitizer_common/sanitizer_libc.h"
|
|
||||||
#include "sanitizer_common/sanitizer_report_decorator.h"
|
#include "sanitizer_common/sanitizer_report_decorator.h"
|
||||||
#include "sanitizer_common/sanitizer_stacktrace.h"
|
#include "sanitizer_common/sanitizer_stacktrace.h"
|
||||||
#include "sanitizer_common/sanitizer_symbolizer.h"
|
#include "sanitizer_common/sanitizer_symbolizer.h"
|
||||||
|
@ -23,28 +21,8 @@
|
||||||
|
|
||||||
using namespace __ubsan;
|
using namespace __ubsan;
|
||||||
|
|
||||||
static void InitializeSanitizerCommonAndFlags() {
|
|
||||||
static StaticSpinMutex init_mu;
|
|
||||||
SpinMutexLock l(&init_mu);
|
|
||||||
static bool initialized;
|
|
||||||
if (initialized)
|
|
||||||
return;
|
|
||||||
if (0 == internal_strcmp(SanitizerToolName, "SanitizerTool")) {
|
|
||||||
// UBSan is run in a standalone mode. Initialize it now.
|
|
||||||
SanitizerToolName = "UndefinedBehaviorSanitizer";
|
|
||||||
CommonFlags *cf = common_flags();
|
|
||||||
SetCommonFlagsDefaults(cf);
|
|
||||||
cf->print_summary = false;
|
|
||||||
// Common flags may only be modified via UBSAN_OPTIONS.
|
|
||||||
ParseCommonFlagsFromString(cf, GetEnv("UBSAN_OPTIONS"));
|
|
||||||
}
|
|
||||||
// Initialize UBSan-specific flags.
|
|
||||||
InitializeFlags();
|
|
||||||
initialized = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
void __ubsan::MaybePrintStackTrace(uptr pc, uptr bp) {
|
void __ubsan::MaybePrintStackTrace(uptr pc, uptr bp) {
|
||||||
// We assume that flags are already parsed: InitializeSanitizerCommonAndFlags
|
// We assume that flags are already parsed: InitIfNecessary
|
||||||
// will definitely be called when we print the first diagnostics message.
|
// will definitely be called when we print the first diagnostics message.
|
||||||
if (!flags()->print_stacktrace)
|
if (!flags()->print_stacktrace)
|
||||||
return;
|
return;
|
||||||
|
@ -82,7 +60,7 @@ Location __ubsan::getCallerLocation(uptr CallerLoc) {
|
||||||
Location __ubsan::getFunctionLocation(uptr Loc, const char **FName) {
|
Location __ubsan::getFunctionLocation(uptr Loc, const char **FName) {
|
||||||
if (!Loc)
|
if (!Loc)
|
||||||
return Location();
|
return Location();
|
||||||
InitializeSanitizerCommonAndFlags();
|
InitIfNecessary();
|
||||||
|
|
||||||
AddressInfo Info;
|
AddressInfo Info;
|
||||||
if (!Symbolizer::GetOrInit()->SymbolizePC(Loc, &Info, 1) ||
|
if (!Symbolizer::GetOrInit()->SymbolizePC(Loc, &Info, 1) ||
|
||||||
|
@ -296,7 +274,7 @@ static void renderMemorySnippet(const Decorator &Decor, MemoryLocation Loc,
|
||||||
}
|
}
|
||||||
|
|
||||||
Diag::~Diag() {
|
Diag::~Diag() {
|
||||||
InitializeSanitizerCommonAndFlags();
|
InitIfNecessary();
|
||||||
Decorator Decor;
|
Decorator Decor;
|
||||||
SpinMutexLock l(&CommonSanitizerReportMutex);
|
SpinMutexLock l(&CommonSanitizerReportMutex);
|
||||||
Printf(Decor.Bold());
|
Printf(Decor.Bold());
|
||||||
|
|
|
@ -0,0 +1,63 @@
|
||||||
|
//===-- ubsan_init.cc -----------------------------------------------------===//
|
||||||
|
//
|
||||||
|
// The LLVM Compiler Infrastructure
|
||||||
|
//
|
||||||
|
// This file is distributed under the University of Illinois Open Source
|
||||||
|
// License. See LICENSE.TXT for details.
|
||||||
|
//
|
||||||
|
//===----------------------------------------------------------------------===//
|
||||||
|
//
|
||||||
|
// Initialization of UBSan runtime.
|
||||||
|
//
|
||||||
|
//===----------------------------------------------------------------------===//
|
||||||
|
|
||||||
|
#include "ubsan_init.h"
|
||||||
|
#include "ubsan_flags.h"
|
||||||
|
#include "sanitizer_common/sanitizer_common.h"
|
||||||
|
#include "sanitizer_common/sanitizer_libc.h"
|
||||||
|
#include "sanitizer_common/sanitizer_flags.h"
|
||||||
|
#include "sanitizer_common/sanitizer_mutex.h"
|
||||||
|
|
||||||
|
using namespace __ubsan;
|
||||||
|
|
||||||
|
static bool ubsan_inited;
|
||||||
|
|
||||||
|
void __ubsan::InitIfNecessary() {
|
||||||
|
#if !SANITIZER_CAN_USE_PREINIT_ARRAY
|
||||||
|
// No need to lock mutex if we're initializing from preinit array.
|
||||||
|
static StaticSpinMutex init_mu;
|
||||||
|
SpinMutexLock l(&init_mu);
|
||||||
|
#endif
|
||||||
|
if (LIKELY(ubsan_inited))
|
||||||
|
return;
|
||||||
|
if (0 == internal_strcmp(SanitizerToolName, "SanitizerTool")) {
|
||||||
|
// WARNING: If this condition holds, then either UBSan runs in a standalone
|
||||||
|
// mode, or initializer for another sanitizer hasn't run yet. In a latter
|
||||||
|
// case, another sanitizer will overwrite "SanitizerToolName" and reparse
|
||||||
|
// common flags. It means, that we are not allowed to *use* common flags
|
||||||
|
// in this function.
|
||||||
|
SanitizerToolName = "UndefinedBehaviorSanitizer";
|
||||||
|
CommonFlags *cf = common_flags();
|
||||||
|
SetCommonFlagsDefaults(cf);
|
||||||
|
cf->print_summary = false;
|
||||||
|
// Common flags may only be modified via UBSAN_OPTIONS.
|
||||||
|
ParseCommonFlagsFromString(cf, GetEnv("UBSAN_OPTIONS"));
|
||||||
|
}
|
||||||
|
// Initialize UBSan-specific flags.
|
||||||
|
InitializeFlags();
|
||||||
|
ubsan_inited = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if SANITIZER_CAN_USE_PREINIT_ARRAY
|
||||||
|
__attribute__((section(".preinit_array"), used))
|
||||||
|
void (*__local_ubsan_preinit)(void) = __ubsan::InitIfNecessary;
|
||||||
|
#else
|
||||||
|
// Use a dynamic initializer.
|
||||||
|
class UbsanInitializer {
|
||||||
|
public:
|
||||||
|
UbsanInitializer() {
|
||||||
|
InitIfNecessary();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
static UbsanInitializer ubsan_initializer;
|
||||||
|
#endif // SANITIZER_CAN_USE_PREINIT_ARRAY
|
|
@ -0,0 +1,24 @@
|
||||||
|
//===-- ubsan_init.h --------------------------------------------*- C++ -*-===//
|
||||||
|
//
|
||||||
|
// The LLVM Compiler Infrastructure
|
||||||
|
//
|
||||||
|
// This file is distributed under the University of Illinois Open Source
|
||||||
|
// License. See LICENSE.TXT for details.
|
||||||
|
//
|
||||||
|
//===----------------------------------------------------------------------===//
|
||||||
|
//
|
||||||
|
// Initialization function for UBSan runtime.
|
||||||
|
//
|
||||||
|
//===----------------------------------------------------------------------===//
|
||||||
|
#ifndef UBSAN_INIT_H
|
||||||
|
#define UBSAN_INIT_H
|
||||||
|
|
||||||
|
namespace __ubsan {
|
||||||
|
|
||||||
|
// NOTE: This function might take a lock (if .preinit_array initialization is
|
||||||
|
// not used). It's generally a bad idea to call it on a fast path.
|
||||||
|
void InitIfNecessary();
|
||||||
|
|
||||||
|
} // namespace __ubsan
|
||||||
|
|
||||||
|
#endif // UBSAN_INIT_H
|
Loading…
Reference in New Issue