| //===-- asan_activation.cpp -------------------------------------*- C++ -*-===// |
| // |
| // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
| // See https://llvm.org/LICENSE.txt for license information. |
| // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
| // |
| //===----------------------------------------------------------------------===// |
| // |
| // This file is a part of AddressSanitizer, an address sanity checker. |
| // |
| // ASan activation/deactivation logic. |
| //===----------------------------------------------------------------------===// |
| |
| #include "asan_activation.h" |
| #include "asan_allocator.h" |
| #include "asan_flags.h" |
| #include "asan_internal.h" |
| #include "asan_mapping.h" |
| #include "asan_poisoning.h" |
| #include "asan_stack.h" |
| #include "sanitizer_common/sanitizer_common.h" |
| #include "sanitizer_common/sanitizer_flags.h" |
| |
| namespace __asan { |
| |
| static struct AsanDeactivatedFlags { |
| AllocatorOptions allocator_options; |
| int malloc_context_size; |
| bool poison_heap; |
| bool coverage; |
| const char *coverage_dir; |
| |
| void RegisterActivationFlags(FlagParser *parser, Flags *f, CommonFlags *cf) { |
| #define ASAN_ACTIVATION_FLAG(Type, Name) \ |
| RegisterFlag(parser, #Name, "", &f->Name); |
| #define COMMON_ACTIVATION_FLAG(Type, Name) \ |
| RegisterFlag(parser, #Name, "", &cf->Name); |
| #include "asan_activation_flags.inc" |
| #undef ASAN_ACTIVATION_FLAG |
| #undef COMMON_ACTIVATION_FLAG |
| |
| RegisterIncludeFlags(parser, cf); |
| } |
| |
| void OverrideFromActivationFlags() { |
| Flags f; |
| CommonFlags cf; |
| FlagParser parser; |
| RegisterActivationFlags(&parser, &f, &cf); |
| |
| cf.SetDefaults(); |
| // Copy the current activation flags. |
| allocator_options.CopyTo(&f, &cf); |
| cf.malloc_context_size = malloc_context_size; |
| f.poison_heap = poison_heap; |
| cf.coverage = coverage; |
| cf.coverage_dir = coverage_dir; |
| cf.verbosity = Verbosity(); |
| cf.help = false; // this is activation-specific help |
| |
| // Check if activation flags need to be overriden. |
| if (const char *env = GetEnv("ASAN_ACTIVATION_OPTIONS")) { |
| parser.ParseString(env); |
| } |
| |
| InitializeCommonFlags(&cf); |
| |
| if (Verbosity()) ReportUnrecognizedFlags(); |
| |
| if (cf.help) parser.PrintFlagDescriptions(); |
| |
| allocator_options.SetFrom(&f, &cf); |
| malloc_context_size = cf.malloc_context_size; |
| poison_heap = f.poison_heap; |
| coverage = cf.coverage; |
| coverage_dir = cf.coverage_dir; |
| } |
| |
| void Print() { |
| Report( |
| "quarantine_size_mb %d, thread_local_quarantine_size_kb %d, " |
| "max_redzone %d, poison_heap %d, malloc_context_size %d, " |
| "alloc_dealloc_mismatch %d, allocator_may_return_null %d, coverage %d, " |
| "coverage_dir %s, allocator_release_to_os_interval_ms %d\n", |
| allocator_options.quarantine_size_mb, |
| allocator_options.thread_local_quarantine_size_kb, |
| allocator_options.max_redzone, poison_heap, malloc_context_size, |
| allocator_options.alloc_dealloc_mismatch, |
| allocator_options.may_return_null, coverage, coverage_dir, |
| allocator_options.release_to_os_interval_ms); |
| } |
| } asan_deactivated_flags; |
| |
| static bool asan_is_deactivated; |
| |
| void AsanDeactivate() { |
| CHECK(!asan_is_deactivated); |
| VReport(1, "Deactivating ASan\n"); |
| |
| // Stash runtime state. |
| GetAllocatorOptions(&asan_deactivated_flags.allocator_options); |
| asan_deactivated_flags.malloc_context_size = GetMallocContextSize(); |
| asan_deactivated_flags.poison_heap = CanPoisonMemory(); |
| asan_deactivated_flags.coverage = common_flags()->coverage; |
| asan_deactivated_flags.coverage_dir = common_flags()->coverage_dir; |
| |
| // Deactivate the runtime. |
| SetCanPoisonMemory(false); |
| SetMallocContextSize(1); |
| |
| AllocatorOptions disabled = asan_deactivated_flags.allocator_options; |
| disabled.quarantine_size_mb = 0; |
| disabled.thread_local_quarantine_size_kb = 0; |
| // Redzone must be at least Max(16, granularity) bytes long. |
| disabled.min_redzone = Max(16, (int)ASAN_SHADOW_GRANULARITY); |
| disabled.max_redzone = disabled.min_redzone; |
| disabled.alloc_dealloc_mismatch = false; |
| disabled.may_return_null = true; |
| ReInitializeAllocator(disabled); |
| |
| asan_is_deactivated = true; |
| } |
| |
| void AsanActivate() { |
| if (!asan_is_deactivated) return; |
| VReport(1, "Activating ASan\n"); |
| |
| UpdateProcessName(); |
| |
| asan_deactivated_flags.OverrideFromActivationFlags(); |
| |
| SetCanPoisonMemory(asan_deactivated_flags.poison_heap); |
| SetMallocContextSize(asan_deactivated_flags.malloc_context_size); |
| ReInitializeAllocator(asan_deactivated_flags.allocator_options); |
| |
| asan_is_deactivated = false; |
| if (Verbosity()) { |
| Report("Activated with flags:\n"); |
| asan_deactivated_flags.Print(); |
| } |
| } |
| |
| } // namespace __asan |