From: Matt Beaumont-Gay Date: Tue, 4 Dec 2012 21:18:26 +0000 (+0000) Subject: Currently, with -fsanitize=address, the driver appends libclang_rt.asan.a to X-Git-Url: http://review.tizen.org/git/?a=commitdiff_plain;h=35439dff76c4c6ccbed39182ba818ecbca179dac;p=platform%2Fupstream%2Fllvm.git Currently, with -fsanitize=address, the driver appends libclang_rt.asan.a to the link command. This all works fine when the driver is also responsible for adding -lstdc++ to the link command. But, if -lstdc++ (or libstdc++.a, etc) is passed explicitly to the driver, the ASan runtime will appear in the link command after the standard library, leading to multiple-definition errors for the global 'operator new' and 'operator delete'. Fix this in a painfully simple way, by inserting libclang_rt.asan.a at the start of the link command instead of the end. If we need to do something more clever, we can walk the link command looking for something that resembles libstdc++ and insert libclang_rt.asan.a as late as possible, but the simple solution works for now. llvm-svn: 169310 --- diff --git a/clang/lib/Driver/Tools.cpp b/clang/lib/Driver/Tools.cpp index ab00118..92f4106 100644 --- a/clang/lib/Driver/Tools.cpp +++ b/clang/lib/Driver/Tools.cpp @@ -1512,7 +1512,7 @@ static void addAsanRTLinux(const ToolChain &TC, const ArgList &Args, llvm::sys::path::append(LibAsan, "lib", "linux", (Twine("libclang_rt.asan-") + TC.getArchName() + "-android.so")); - CmdArgs.push_back(Args.MakeArgString(LibAsan)); + CmdArgs.insert(CmdArgs.begin(), Args.MakeArgString(LibAsan)); } else { if (!Args.hasArg(options::OPT_shared)) { // LibAsan is "libclang_rt.asan-.a" in the Linux library @@ -1521,7 +1521,11 @@ static void addAsanRTLinux(const ToolChain &TC, const ArgList &Args, llvm::sys::path::append(LibAsan, "lib", "linux", (Twine("libclang_rt.asan-") + TC.getArchName() + ".a")); - CmdArgs.push_back(Args.MakeArgString(LibAsan)); + // The ASan runtime needs to come before -lstdc++ (or -lc++, libstdc++.a, + // etc.) so that the linker picks ASan's versions of the global 'operator + // new' and 'operator delete' symbols. We take the extreme (but simple) + // strategy of inserting it at the front of the link command. + CmdArgs.insert(CmdArgs.begin(), Args.MakeArgString(LibAsan)); CmdArgs.push_back("-lpthread"); CmdArgs.push_back("-ldl"); CmdArgs.push_back("-export-dynamic"); diff --git a/clang/test/Driver/asan-ld.c b/clang/test/Driver/asan-ld.c index 9d8cbec..b90082a 100644 --- a/clang/test/Driver/asan-ld.c +++ b/clang/test/Driver/asan-ld.c @@ -24,7 +24,17 @@ // CHECK-LINUX-CXX: "-ldl" // CHECK-LINUX-CXX: "-export-dynamic" // CHECK-LINUX-CXX: stdc++ + +// RUN: %clang -no-canonical-prefixes %s -### -o /dev/null -fsanitize=address \ +// RUN: -target i386-unknown-linux --sysroot=%S/Inputs/basic_linux_tree \ +// RUN: -lstdc++ -static 2>&1 \ +// RUN: | FileCheck --check-prefix=CHECK-LINUX-CXX-STATIC %s // +// CHECK-LINUX-CXX-STATIC: "{{.*}}ld{{(.exe)?}}" +// CHECK-LINUX-CXX-STATIC-NOT: stdc++ +// CHECK-LINUX-CXX-STATIC: libclang_rt.asan-i386.a" +// CHECK-LINUX-CXX-STATIC: stdc++ + // RUN: %clang -no-canonical-prefixes %s -### -o %t.o 2>&1 \ // RUN: -target arm-linux-androideabi -fsanitize=address \ // RUN: --sysroot=%S/Inputs/basic_android_tree/sysroot \