summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--lld/ELF/LTO.cpp24
-rw-r--r--lld/ELF/LTO.h2
-rw-r--r--lld/ELF/Symbols.h3
-rw-r--r--lld/test/ELF/lto/internalize-basic.ll21
4 files changed, 49 insertions, 1 deletions
diff --git a/lld/ELF/LTO.cpp b/lld/ELF/LTO.cpp
index cbdd49ab3d1..528a914b6a1 100644
--- a/lld/ELF/LTO.cpp
+++ b/lld/ELF/LTO.cpp
@@ -97,6 +97,18 @@ void BitcodeCompiler::add(BitcodeFile &F) {
GV->setLinkage(GlobalValue::WeakODRLinkage);
break;
}
+
+ // We collect the set of symbols we want to internalize here
+ // and change the linkage after the IRMover executed, i.e. after
+ // we imported the symbols and satisfied undefined references
+ // to it. We can't just change linkage here because otherwise
+ // the IRMover will just rename the symbol.
+ // Shared libraries need to be handled slightly differently.
+ // For now, let's be conservative and just never internalize
+ // symbols when creating a shared library.
+ if (!Config->Shared && !B->isUsedInRegularObj())
+ InternalizedSyms.insert(GV->getName());
+
Keep.push_back(GV);
}
@@ -104,10 +116,22 @@ void BitcodeCompiler::add(BitcodeFile &F) {
[](GlobalValue &, IRMover::ValueAdder) {});
}
+static void internalize(GlobalValue &GV) {
+ assert(!GV.hasLocalLinkage() &&
+ "Trying to internalize a symbol with local linkage!") ;
+ GV.setLinkage(GlobalValue::InternalLinkage);
+}
+
// Merge all the bitcode files we have seen, codegen the result
// and return the resulting ObjectFile.
template <class ELFT>
std::unique_ptr<elf::ObjectFile<ELFT>> BitcodeCompiler::compile() {
+ for (const auto &Name : InternalizedSyms) {
+ GlobalValue *GV = Combined.getNamedValue(Name.first());
+ assert(GV);
+ internalize(*GV);
+ }
+
if (Config->SaveTemps)
saveBCFile(Combined, ".lto.bc");
diff --git a/lld/ELF/LTO.h b/lld/ELF/LTO.h
index bc4d5b053b3..7a90e97f1a4 100644
--- a/lld/ELF/LTO.h
+++ b/lld/ELF/LTO.h
@@ -23,6 +23,7 @@
#include "lld/Core/LLVM.h"
#include "llvm/ADT/SmallString.h"
+#include "llvm/ADT/StringSet.h"
#include "llvm/IR/LLVMContext.h"
#include "llvm/IR/Module.h"
#include "llvm/Linker/IRMover.h"
@@ -46,6 +47,7 @@ private:
llvm::IRMover Mover{Combined};
SmallString<0> OwningData;
std::unique_ptr<MemoryBuffer> MB;
+ llvm::StringSet<> InternalizedSyms;
};
}
}
diff --git a/lld/ELF/Symbols.h b/lld/ELF/Symbols.h
index c3143575788..f316f83e504 100644
--- a/lld/ELF/Symbols.h
+++ b/lld/ELF/Symbols.h
@@ -120,7 +120,8 @@ protected:
IsFunc = Type == llvm::ELF::STT_FUNC;
IsTls = Type == llvm::ELF::STT_TLS;
IsGnuIFunc = Type == llvm::ELF::STT_GNU_IFUNC;
- IsUsedInRegularObj = K != SharedKind && K != LazyKind;
+ IsUsedInRegularObj =
+ K != SharedKind && K != LazyKind && K != DefinedBitcodeKind;
}
const unsigned SymbolKind : 8;
diff --git a/lld/test/ELF/lto/internalize-basic.ll b/lld/test/ELF/lto/internalize-basic.ll
new file mode 100644
index 00000000000..396b9cb60f1
--- /dev/null
+++ b/lld/test/ELF/lto/internalize-basic.ll
@@ -0,0 +1,21 @@
+; REQUIRES: x86
+; RUN: llvm-as %s -o %t.o
+; RUN: ld.lld -m elf_x86_64 %t.o -o %t2 -save-temps
+; RUN: llvm-dis < %t2.lto.bc | FileCheck %s
+
+target triple = "x86_64-unknown-linux-gnu"
+target datalayout = "e-m:e-i64:64-f80:128-n8:16:32:64-S128"
+
+define void @_start() {
+ ret void
+}
+
+define hidden void @foo() {
+ ret void
+}
+
+; Check that _start is not internalized.
+; CHECK: define void @_start()
+
+; Check that foo function is correctly internalized.
+; CHECK: define internal void @foo()
OpenPOWER on IntegriCloud