2012-07-09 21:21:39 +08:00
|
|
|
//===-- sanitizer_flags.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/AddressSanitizer runtime.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "sanitizer_flags.h"
|
|
|
|
|
|
|
|
#include "sanitizer_common.h"
|
|
|
|
#include "sanitizer_libc.h"
|
2014-03-20 21:49:21 +08:00
|
|
|
#include "sanitizer_list.h"
|
2012-07-09 21:21:39 +08:00
|
|
|
|
|
|
|
namespace __sanitizer {
|
|
|
|
|
2014-01-31 19:20:55 +08:00
|
|
|
CommonFlags common_flags_dont_use;
|
|
|
|
|
2014-03-20 21:49:21 +08:00
|
|
|
struct FlagDescription {
|
2014-03-20 20:52:52 +08:00
|
|
|
const char *name;
|
|
|
|
const char *description;
|
2014-03-20 21:49:21 +08:00
|
|
|
FlagDescription *next;
|
2014-03-20 20:52:52 +08:00
|
|
|
};
|
|
|
|
|
2014-03-20 21:49:21 +08:00
|
|
|
IntrusiveList<FlagDescription> flag_descriptions;
|
2014-03-20 20:52:52 +08:00
|
|
|
|
2014-07-10 07:59:40 +08:00
|
|
|
// If set, the tool will install its own SEGV signal handler by default.
|
|
|
|
#ifndef SANITIZER_NEEDS_SEGV
|
|
|
|
# define SANITIZER_NEEDS_SEGV 1
|
|
|
|
#endif
|
|
|
|
|
2014-12-20 05:40:04 +08:00
|
|
|
void CommonFlags::SetDefaults() {
|
|
|
|
symbolize = true;
|
|
|
|
external_symbolizer_path = 0;
|
|
|
|
allow_addr2line = false;
|
|
|
|
strip_path_prefix = "";
|
|
|
|
fast_unwind_on_check = false;
|
|
|
|
fast_unwind_on_fatal = false;
|
|
|
|
fast_unwind_on_malloc = true;
|
|
|
|
handle_ioctl = false;
|
|
|
|
malloc_context_size = 1;
|
|
|
|
log_path = "stderr";
|
|
|
|
verbosity = 0;
|
|
|
|
detect_leaks = true;
|
|
|
|
leak_check_at_exit = true;
|
|
|
|
allocator_may_return_null = false;
|
|
|
|
print_summary = true;
|
|
|
|
check_printf = true;
|
2014-12-20 07:09:59 +08:00
|
|
|
mmap_limit_mb = 0;
|
|
|
|
hard_rss_limit_mb = 0;
|
2014-01-28 17:28:57 +08:00
|
|
|
// TODO(glider): tools may want to set different defaults for handle_segv.
|
2014-12-20 05:40:04 +08:00
|
|
|
handle_segv = SANITIZER_NEEDS_SEGV;
|
|
|
|
allow_user_segv_handler = false;
|
|
|
|
use_sigaltstack = true;
|
|
|
|
detect_deadlocks = false;
|
|
|
|
clear_shadow_mmap_threshold = 64 * 1024;
|
|
|
|
color = "auto";
|
|
|
|
legacy_pthread_cond = false;
|
|
|
|
intercept_tls_get_addr = false;
|
|
|
|
coverage = false;
|
2014-12-24 06:32:17 +08:00
|
|
|
coverage_bitset = false;
|
2014-12-20 05:40:04 +08:00
|
|
|
coverage_direct = SANITIZER_ANDROID;
|
|
|
|
coverage_dir = ".";
|
|
|
|
full_address_space = false;
|
|
|
|
suppressions = "";
|
|
|
|
print_suppressions = true;
|
|
|
|
disable_coredump = (SANITIZER_WORDSIZE == 64);
|
|
|
|
symbolize_inline_frames = true;
|
|
|
|
stack_trace_format = "DEFAULT";
|
2013-11-12 21:59:08 +08:00
|
|
|
}
|
|
|
|
|
2014-12-20 05:40:04 +08:00
|
|
|
void CommonFlags::ParseFromString(const char *str) {
|
|
|
|
ParseFlag(str, &symbolize, "symbolize",
|
2014-03-22 01:28:12 +08:00
|
|
|
"If set, use the online symbolizer from common sanitizer runtime to turn "
|
|
|
|
"virtual addresses to file/line locations.");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &external_symbolizer_path, "external_symbolizer_path",
|
2014-03-22 01:28:12 +08:00
|
|
|
"Path to external symbolizer. If empty, the tool will search $PATH for "
|
|
|
|
"the symbolizer.");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &allow_addr2line, "allow_addr2line",
|
2014-03-22 01:28:12 +08:00
|
|
|
"If set, allows online symbolizer to run addr2line binary to symbolize "
|
|
|
|
"stack traces (addr2line will only be used if llvm-symbolizer binary is "
|
|
|
|
"unavailable.");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &strip_path_prefix, "strip_path_prefix",
|
2014-03-22 01:28:12 +08:00
|
|
|
"Strips this prefix from file paths in error reports.");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &fast_unwind_on_check, "fast_unwind_on_check",
|
2014-10-14 17:36:24 +08:00
|
|
|
"If available, use the fast frame-pointer-based unwinder on "
|
|
|
|
"internal CHECK failures.");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &fast_unwind_on_fatal, "fast_unwind_on_fatal",
|
2014-03-22 01:28:12 +08:00
|
|
|
"If available, use the fast frame-pointer-based unwinder on fatal "
|
|
|
|
"errors.");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &fast_unwind_on_malloc, "fast_unwind_on_malloc",
|
2014-03-22 01:28:12 +08:00
|
|
|
"If available, use the fast frame-pointer-based unwinder on "
|
|
|
|
"malloc/free.");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &handle_ioctl, "handle_ioctl",
|
2014-03-22 01:28:12 +08:00
|
|
|
"Intercept and handle ioctl requests.");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &malloc_context_size, "malloc_context_size",
|
2014-03-22 01:28:12 +08:00
|
|
|
"Max number of stack frames kept for each allocation/deallocation.");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &log_path, "log_path",
|
2014-11-14 03:37:30 +08:00
|
|
|
"Write logs to \"log_path.pid\". The special values are \"stdout\" and "
|
|
|
|
"\"stderr\". The default is \"stderr\".");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &verbosity, "verbosity",
|
2014-03-22 01:28:12 +08:00
|
|
|
"Verbosity level (0 - silent, 1 - a bit of output, 2+ - more output).");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &detect_leaks, "detect_leaks",
|
2014-03-22 01:28:12 +08:00
|
|
|
"Enable memory leak detection.");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &leak_check_at_exit, "leak_check_at_exit",
|
2014-03-22 01:28:12 +08:00
|
|
|
"Invoke leak checking in an atexit handler. Has no effect if "
|
|
|
|
"detect_leaks=false, or if __lsan_do_leak_check() is called before the "
|
|
|
|
"handler has a chance to run.");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &allocator_may_return_null, "allocator_may_return_null",
|
2014-03-22 01:28:12 +08:00
|
|
|
"If false, the allocator will crash instead of returning 0 on "
|
|
|
|
"out-of-memory.");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &print_summary, "print_summary",
|
2014-03-22 01:28:12 +08:00
|
|
|
"If false, disable printing error summaries in addition to error "
|
|
|
|
"reports.");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &check_printf, "check_printf",
|
2014-03-22 01:28:12 +08:00
|
|
|
"Check printf arguments.");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &handle_segv, "handle_segv",
|
2014-03-22 01:28:12 +08:00
|
|
|
"If set, registers the tool's custom SEGV handler (both SIGBUS and "
|
|
|
|
"SIGSEGV on OSX).");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &allow_user_segv_handler, "allow_user_segv_handler",
|
2014-03-22 01:28:12 +08:00
|
|
|
"If set, allows user to register a SEGV handler even if the tool "
|
|
|
|
"registers one.");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &use_sigaltstack, "use_sigaltstack",
|
2014-03-22 01:28:12 +08:00
|
|
|
"If set, uses alternate stack for signal handling.");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &detect_deadlocks, "detect_deadlocks",
|
2014-03-22 01:28:12 +08:00
|
|
|
"If set, deadlock detection is enabled.");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &clear_shadow_mmap_threshold,
|
2014-03-22 01:28:12 +08:00
|
|
|
"clear_shadow_mmap_threshold",
|
|
|
|
"Large shadow regions are zero-filled using mmap(NORESERVE) instead of "
|
|
|
|
"memset(). This is the threshold size in bytes.");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &color, "color",
|
2014-03-22 01:28:12 +08:00
|
|
|
"Colorize reports: (always|never|auto).");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &legacy_pthread_cond, "legacy_pthread_cond",
|
2014-03-22 01:28:12 +08:00
|
|
|
"Enables support for dynamic libraries linked with libpthread 2.2.5.");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &intercept_tls_get_addr, "intercept_tls_get_addr",
|
2014-04-04 17:10:58 +08:00
|
|
|
"Intercept __tls_get_addr.");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &help, "help", "Print the flag descriptions.");
|
|
|
|
ParseFlag(str, &mmap_limit_mb, "mmap_limit_mb",
|
2014-04-14 22:51:01 +08:00
|
|
|
"Limit the amount of mmap-ed memory (excluding shadow) in Mb; "
|
|
|
|
"not a user-facing flag, used mosly for testing the tools");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &hard_rss_limit_mb, "hard_rss_limit_mb",
|
2014-12-17 03:13:01 +08:00
|
|
|
"RSS limit in Mb."
|
|
|
|
" If non-zero, a background thread is spawned at startup"
|
|
|
|
" which periodically reads RSS and aborts the process if the"
|
|
|
|
" limit is reached");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &coverage, "coverage",
|
2014-05-12 22:27:36 +08:00
|
|
|
"If set, coverage information will be dumped at program shutdown (if the "
|
|
|
|
"coverage instrumentation was enabled at compile time).");
|
2014-12-24 06:32:17 +08:00
|
|
|
ParseFlag(str, &coverage_bitset, "coverage_bitset",
|
|
|
|
"If set (and if 'coverage' is set too), the coverage information "
|
|
|
|
"will also be dumped as a bitset to a separate file.");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &coverage_direct, "coverage_direct",
|
2014-05-27 20:37:52 +08:00
|
|
|
"If set, coverage information will be dumped directly to a memory "
|
|
|
|
"mapped file. This way data is not lost even if the process is "
|
|
|
|
"suddenly killed.");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &coverage_dir, "coverage_dir",
|
2014-05-29 22:33:16 +08:00
|
|
|
"Target directory for coverage dumps. Defaults to the current "
|
|
|
|
"directory.");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &full_address_space, "full_address_space",
|
2014-05-14 16:13:11 +08:00
|
|
|
"Sanitize complete address space; "
|
|
|
|
"by default kernel area on 32-bit platforms will not be sanitized");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &suppressions, "suppressions", "Suppressions file name.");
|
|
|
|
ParseFlag(str, &print_suppressions, "print_suppressions",
|
2014-07-31 05:33:04 +08:00
|
|
|
"Print matched suppressions at exit.");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &disable_coredump, "disable_coredump",
|
2014-08-13 06:07:48 +08:00
|
|
|
"Disable core dumping. By default, disable_core=1 on 64-bit to avoid "
|
|
|
|
"dumping a 16T+ core file. Ignored on OSes that don't dump core by"
|
|
|
|
"default and for sanitizers that don't reserve lots of virtual memory.");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &symbolize_inline_frames, "symbolize_inline_frames",
|
2014-10-25 02:34:43 +08:00
|
|
|
"Print inlined frames in stacktraces. Defaults to true.");
|
2014-12-20 05:40:04 +08:00
|
|
|
ParseFlag(str, &stack_trace_format, "stack_trace_format",
|
2014-11-07 02:43:45 +08:00
|
|
|
"Format string used to render stack frames. "
|
|
|
|
"See sanitizer_stacktrace_printer.h for the format description. "
|
|
|
|
"Use DEFAULT to get default format.");
|
2013-10-11 17:58:30 +08:00
|
|
|
|
|
|
|
// Do a sanity check for certain flags.
|
2014-12-20 05:40:04 +08:00
|
|
|
if (malloc_context_size < 1)
|
|
|
|
malloc_context_size = 1;
|
2013-05-06 19:27:58 +08:00
|
|
|
}
|
|
|
|
|
2012-08-27 22:04:54 +08:00
|
|
|
static bool GetFlagValue(const char *env, const char *name,
|
|
|
|
const char **value, int *value_length) {
|
2012-07-09 21:21:39 +08:00
|
|
|
if (env == 0)
|
2012-08-27 22:04:54 +08:00
|
|
|
return false;
|
2013-03-28 01:59:13 +08:00
|
|
|
const char *pos = 0;
|
|
|
|
for (;;) {
|
|
|
|
pos = internal_strstr(env, name);
|
|
|
|
if (pos == 0)
|
|
|
|
return false;
|
2014-05-29 22:32:32 +08:00
|
|
|
const char *name_end = pos + internal_strlen(name);
|
|
|
|
if ((pos != env &&
|
|
|
|
((pos[-1] >= 'a' && pos[-1] <= 'z') || pos[-1] == '_')) ||
|
|
|
|
*name_end != '=') {
|
2013-03-28 01:59:13 +08:00
|
|
|
// Seems to be middle of another flag name or value.
|
|
|
|
env = pos + 1;
|
|
|
|
continue;
|
|
|
|
}
|
2014-05-29 22:32:32 +08:00
|
|
|
pos = name_end;
|
2013-03-28 01:59:13 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
const char *end;
|
2012-07-09 21:21:39 +08:00
|
|
|
if (pos[0] != '=') {
|
|
|
|
end = pos;
|
|
|
|
} else {
|
|
|
|
pos += 1;
|
|
|
|
if (pos[0] == '"') {
|
|
|
|
pos += 1;
|
|
|
|
end = internal_strchr(pos, '"');
|
|
|
|
} else if (pos[0] == '\'') {
|
|
|
|
pos += 1;
|
|
|
|
end = internal_strchr(pos, '\'');
|
|
|
|
} else {
|
2013-02-19 21:03:37 +08:00
|
|
|
// Read until the next space or colon.
|
|
|
|
end = pos + internal_strcspn(pos, " :");
|
2012-07-09 21:21:39 +08:00
|
|
|
}
|
|
|
|
if (end == 0)
|
|
|
|
end = pos + internal_strlen(pos);
|
|
|
|
}
|
2012-08-27 22:04:54 +08:00
|
|
|
*value = pos;
|
|
|
|
*value_length = end - pos;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool StartsWith(const char *flag, int flag_length, const char *value) {
|
|
|
|
if (!flag || !value)
|
|
|
|
return false;
|
|
|
|
int value_length = internal_strlen(value);
|
|
|
|
return (flag_length >= value_length) &&
|
|
|
|
(0 == internal_strncmp(flag, value, value_length));
|
2012-07-09 21:21:39 +08:00
|
|
|
}
|
|
|
|
|
2014-03-20 20:52:52 +08:00
|
|
|
static LowLevelAllocator allocator_for_flags;
|
|
|
|
|
|
|
|
// The linear scan is suboptimal, but the number of flags is relatively small.
|
|
|
|
bool FlagInDescriptionList(const char *name) {
|
2014-03-20 21:49:21 +08:00
|
|
|
IntrusiveList<FlagDescription>::Iterator it(&flag_descriptions);
|
|
|
|
while (it.hasNext()) {
|
|
|
|
if (!internal_strcmp(it.next()->name, name)) return true;
|
2014-03-20 20:52:52 +08:00
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void AddFlagDescription(const char *name, const char *description) {
|
|
|
|
if (FlagInDescriptionList(name)) return;
|
2014-03-24 18:47:37 +08:00
|
|
|
FlagDescription *new_description = new(allocator_for_flags) FlagDescription;
|
2014-03-20 20:52:52 +08:00
|
|
|
new_description->name = name;
|
|
|
|
new_description->description = description;
|
2014-03-20 21:49:21 +08:00
|
|
|
flag_descriptions.push_back(new_description);
|
2014-03-20 20:52:52 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
// TODO(glider): put the descriptions inside CommonFlags.
|
|
|
|
void PrintFlagDescriptions() {
|
2014-03-20 21:49:21 +08:00
|
|
|
IntrusiveList<FlagDescription>::Iterator it(&flag_descriptions);
|
2014-03-20 20:52:52 +08:00
|
|
|
Printf("Available flags for %s:\n", SanitizerToolName);
|
2014-03-20 21:49:21 +08:00
|
|
|
while (it.hasNext()) {
|
|
|
|
FlagDescription *descr = it.next();
|
2014-03-22 01:28:12 +08:00
|
|
|
Printf("\t%s\n\t\t- %s\n", descr->name, descr->description);
|
2014-03-20 20:52:52 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void ParseFlag(const char *env, bool *flag,
|
|
|
|
const char *name, const char *descr) {
|
2012-08-27 22:04:54 +08:00
|
|
|
const char *value;
|
|
|
|
int value_length;
|
2014-03-20 20:52:52 +08:00
|
|
|
AddFlagDescription(name, descr);
|
2012-08-27 22:04:54 +08:00
|
|
|
if (!GetFlagValue(env, name, &value, &value_length))
|
2012-07-09 21:21:39 +08:00
|
|
|
return;
|
2012-08-27 22:04:54 +08:00
|
|
|
if (StartsWith(value, value_length, "0") ||
|
|
|
|
StartsWith(value, value_length, "no") ||
|
|
|
|
StartsWith(value, value_length, "false"))
|
2012-07-09 21:21:39 +08:00
|
|
|
*flag = false;
|
2012-08-27 22:04:54 +08:00
|
|
|
if (StartsWith(value, value_length, "1") ||
|
|
|
|
StartsWith(value, value_length, "yes") ||
|
|
|
|
StartsWith(value, value_length, "true"))
|
2012-07-09 21:21:39 +08:00
|
|
|
*flag = true;
|
|
|
|
}
|
|
|
|
|
2014-03-20 20:52:52 +08:00
|
|
|
void ParseFlag(const char *env, int *flag,
|
|
|
|
const char *name, const char *descr) {
|
2012-08-27 22:04:54 +08:00
|
|
|
const char *value;
|
|
|
|
int value_length;
|
2014-03-20 20:52:52 +08:00
|
|
|
AddFlagDescription(name, descr);
|
2012-08-27 22:04:54 +08:00
|
|
|
if (!GetFlagValue(env, name, &value, &value_length))
|
2012-07-09 21:21:39 +08:00
|
|
|
return;
|
2013-05-29 22:11:44 +08:00
|
|
|
*flag = static_cast<int>(internal_atoll(value));
|
2012-07-09 21:21:39 +08:00
|
|
|
}
|
|
|
|
|
2014-03-20 20:52:52 +08:00
|
|
|
void ParseFlag(const char *env, uptr *flag,
|
|
|
|
const char *name, const char *descr) {
|
2014-02-14 19:41:26 +08:00
|
|
|
const char *value;
|
|
|
|
int value_length;
|
2014-03-20 20:52:52 +08:00
|
|
|
AddFlagDescription(name, descr);
|
2014-02-14 19:41:26 +08:00
|
|
|
if (!GetFlagValue(env, name, &value, &value_length))
|
|
|
|
return;
|
|
|
|
*flag = static_cast<uptr>(internal_atoll(value));
|
|
|
|
}
|
|
|
|
|
2014-03-20 20:52:52 +08:00
|
|
|
void ParseFlag(const char *env, const char **flag,
|
|
|
|
const char *name, const char *descr) {
|
2012-08-27 22:04:54 +08:00
|
|
|
const char *value;
|
|
|
|
int value_length;
|
2014-03-20 20:52:52 +08:00
|
|
|
AddFlagDescription(name, descr);
|
2012-08-27 22:04:54 +08:00
|
|
|
if (!GetFlagValue(env, name, &value, &value_length))
|
2012-07-09 21:21:39 +08:00
|
|
|
return;
|
2012-08-27 22:51:36 +08:00
|
|
|
// Copy the flag value. Don't use locks here, as flags are parsed at
|
|
|
|
// tool startup.
|
2012-08-27 22:04:54 +08:00
|
|
|
char *value_copy = (char*)(allocator_for_flags.Allocate(value_length + 1));
|
|
|
|
internal_memcpy(value_copy, value, value_length);
|
|
|
|
value_copy[value_length] = '\0';
|
|
|
|
*flag = value_copy;
|
2012-07-09 21:21:39 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace __sanitizer
|