Index: lib/asan/asan_globals.cc =================================================================== --- lib/asan/asan_globals.cc +++ lib/asan/asan_globals.cc @@ -284,6 +284,27 @@ // ---------------------- Interface ---------------- {{{1 using namespace __asan; // NOLINT + +// Apply __asan_(un)register_globals to all globals found in the same loaded +// shared object as `flag'. The flag tracks whether globals have already been +// registered or not for this shared object. +void __asan_apply_to_globals(globals_op_fptr op, uptr *flag) { + CHECK(op == __asan_register_globals || op == __asan_unregister_globals); + + // Filter out duplicate calls, which will happen if multiple linked + // translation units from the same library have constructors which invoke + // this function. + if ((op == __asan_register_globals && *flag) || + (op == __asan_unregister_globals && !*flag)) { + return; + } + + AsanApplyToGlobals(op, flag); + + // Update the flag to indicate whether globals have been registered + *flag = (op == __asan_register_globals); +} + // Register an array of globals. void __asan_register_globals(__asan_global *globals, uptr n) { if (!flags()->report_globals) return; Index: lib/asan/asan_init_version.h =================================================================== --- lib/asan/asan_init_version.h +++ lib/asan/asan_init_version.h @@ -22,14 +22,16 @@ // Changes between ABI versions: // v1=>v2: added 'module_name' to __asan_global // v2=>v3: stack frame description (created by the compiler) - // contains the function PC as the 3-rd field (see + // contains the function PC as the 3rd field (see // DescribeAddressIfStack). // v3=>v4: added '__asan_global_source_location' to __asan_global. // v4=>v5: changed the semantics and format of __asan_stack_malloc_ and // __asan_stack_free_ functions. // v5=>v6: changed the name of the version check symbol // v6=>v7: added 'odr_indicator' to __asan_global. - #define __asan_version_mismatch_check __asan_version_mismatch_check_v7 + // v7=>v8: added '__asan_apply_to_globals' function (for dead stripping + // support on Mach-O platforms) + #define __asan_version_mismatch_check __asan_version_mismatch_check_v8 } #endif // ASAN_INIT_VERSION_H Index: lib/asan/asan_interface_internal.h =================================================================== --- lib/asan/asan_interface_internal.h +++ lib/asan/asan_interface_internal.h @@ -57,6 +57,13 @@ uptr odr_indicator; // The address of the ODR indicator symbol. }; + // This function can be called on some platforms to find globals in the same + // loaded shared object as `flag' and apply __asan_(un)register_globals + // to them, filtering out redundant calls. + typedef void (*globals_op_fptr)(__asan_global *, uptr); + SANITIZER_INTERFACE_ATTRIBUTE + void __asan_apply_to_globals(globals_op_fptr op, uptr *flag); + // These two functions should be called by the instrumented code. // 'globals' is an array of structures describing 'n' globals. SANITIZER_INTERFACE_ATTRIBUTE Index: lib/asan/asan_internal.h =================================================================== --- lib/asan/asan_internal.h +++ lib/asan/asan_internal.h @@ -72,6 +72,7 @@ void *AsanDoesNotSupportStaticLinkage(); void AsanCheckDynamicRTPrereqs(); void AsanCheckIncompatibleRT(); +void AsanApplyToGlobals(globals_op_fptr op, uptr *flag); void AsanOnDeadlySignal(int, void *siginfo, void *context); Index: lib/asan/asan_linux.cc =================================================================== --- lib/asan/asan_linux.cc +++ lib/asan/asan_linux.cc @@ -75,6 +75,10 @@ return &_DYNAMIC; // defined in link.h } +void AsanApplyToGlobals(globals_op_fptr op, uptr *flag) { + UNIMPLEMENTED(); +} + #if SANITIZER_ANDROID // FIXME: should we do anything for Android? void AsanCheckDynamicRTPrereqs() {} Index: lib/asan/asan_mac.cc =================================================================== --- lib/asan/asan_mac.cc +++ lib/asan/asan_mac.cc @@ -24,9 +24,11 @@ #include "sanitizer_common/sanitizer_libc.h" #include "sanitizer_common/sanitizer_mac.h" +#include #include #include #include +#include #include #include #include // for free() @@ -66,6 +68,30 @@ // No-op. Mac does not support static linkage anyway. void AsanCheckIncompatibleRT() {} +void AsanApplyToGlobals(globals_op_fptr op, uptr *flag) { + // Find the Mach-O header for the image containing the flag + Dl_info info; + int err = dladdr(flag, &info); + if (err == 0) return; + +#if __LP64__ + const struct mach_header_64 *mh = (struct mach_header_64 *)info.dli_fbase; +#else + const struct mach_header *mh = (struct mach_header *)info.dli_fbase; +#endif + + // Look up the __asan_globals section in that image and register its globals + unsigned long size = 0; + __asan_global *globals = (__asan_global *)getsectiondata( + mh, + "__DATA", "__asan_globals", + &size); + + if (!globals) return; + if (size % sizeof(__asan_global) != 0) return; + op(globals, size / sizeof(__asan_global)); +} + void ReadContextStack(void *context, uptr *stack, uptr *ssize) { UNIMPLEMENTED(); } Index: lib/asan/asan_win.cc =================================================================== --- lib/asan/asan_win.cc +++ lib/asan/asan_win.cc @@ -149,6 +149,10 @@ (uptr *)&REAL(NtWaitForWorkViaWorkerFactory))); } +void AsanApplyToGlobals(globals_op_fptr op, uptr *flag) { + UNIMPLEMENTED(); +} + // ---------------------- TSD ---------------- {{{ static bool tsd_key_inited = false; Index: test/asan/TestCases/Darwin/dead-strip.c =================================================================== --- test/asan/TestCases/Darwin/dead-strip.c +++ test/asan/TestCases/Darwin/dead-strip.c @@ -0,0 +1,26 @@ +// Test that AddressSanitizer does not re-animate dead globals when dead +// stripping is turned on. +// +// This test verifies that an out-of-bounds access on a global variable is +// detected after dead stripping has been performed. This proves that the +// runtime is able to register globals in the __DATA,__asan_globals section. + +// RUN: %clang_asan -Xlinker -dead_strip -o %t %s +// RUN: llvm-nm -format=posix %t | FileCheck --check-prefix NM-CHECK %s +// RUN: not %run %t 2>&1 | FileCheck --check-prefix ASAN-CHECK %s + +#pragma clang diagnostic push +#pragma clang diagnostic ignored "-Warray-bounds" + +int alive[1] = {}; +int dead[1] = {}; +// NM-CHECK: {{^_alive }} +// NM-CHECK-NOT: {{^_dead }} + +int main() { + alive[1] = 0; + // ASAN-CHECK: {{0x.* is located 0 bytes to the right of global variable}} + return 0; +} + +#pragma clang diagnostic pop Index: test/asan/TestCases/initialization-bug.cc =================================================================== --- test/asan/TestCases/initialization-bug.cc +++ test/asan/TestCases/initialization-bug.cc @@ -6,7 +6,7 @@ // Do not test with optimization -- the error may be optimized away. // FIXME: https://code.google.com/p/address-sanitizer/issues/detail?id=186 -// XFAIL: darwin,win32 +// XFAIL: win32 #include