summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--lld/COFF/Driver.cpp21
-rw-r--r--lld/COFF/SymbolTable.cpp9
-rw-r--r--lld/COFF/SymbolTable.h3
-rw-r--r--lld/test/COFF/alternatename.test4
4 files changed, 37 insertions, 0 deletions
diff --git a/lld/COFF/Driver.cpp b/lld/COFF/Driver.cpp
index 93b86bfc78a..9e2c89c0fc5 100644
--- a/lld/COFF/Driver.cpp
+++ b/lld/COFF/Driver.cpp
@@ -320,6 +320,27 @@ bool LinkerDriver::link(int Argc, const char *Argv[]) {
}
}
+ // Add weak aliases. Weak aliases is a mechanism to give remaining
+ // undefined symbols final chance to be resolved successfully.
+ // This is symbol renaming.
+ for (auto *Arg : Args->filtered(OPT_alternatename)) {
+ StringRef From, To;
+ std::tie(From, To) = StringRef(Arg->getValue()).split('=');
+ if (From.empty() || To.empty()) {
+ llvm::errs() << "/alternatename: invalid argument: "
+ << Arg->getValue() << "\n";
+ return false;
+ }
+ // If it's already resolved as some Defined type, do nothing.
+ // Otherwise, rename it to see if To can be resolved successfully.
+ if (Symtab.find(From))
+ continue;
+ if (auto EC = Symtab.rename(From, To)) {
+ llvm::errs() << EC.message() << "\n";
+ return false;
+ }
+ }
+
// Windows specific -- If entry point name is not given, we need to
// infer that from user-defined entry name. The symbol table takes
// care of details.
diff --git a/lld/COFF/SymbolTable.cpp b/lld/COFF/SymbolTable.cpp
index e0dfbc38196..da1aa845a74 100644
--- a/lld/COFF/SymbolTable.cpp
+++ b/lld/COFF/SymbolTable.cpp
@@ -187,6 +187,15 @@ std::error_code SymbolTable::addUndefined(StringRef Name) {
return addSymbol(new Undefined(Name));
}
+// Resolve To, and make From an alias to To.
+std::error_code SymbolTable::rename(StringRef From, StringRef To) {
+ SymbolBody *Body = new (Alloc) Undefined(To);
+ if (auto EC = resolve(Body))
+ return EC;
+ Symtab[From]->Body = Body->getReplacement();
+ return std::error_code();
+}
+
std::error_code SymbolTable::addSymbol(SymbolBody *Body) {
OwningSymbols.push_back(std::unique_ptr<SymbolBody>(Body));
return resolve(Body);
diff --git a/lld/COFF/SymbolTable.h b/lld/COFF/SymbolTable.h
index ed3969b9514..3be6e7691a5 100644
--- a/lld/COFF/SymbolTable.h
+++ b/lld/COFF/SymbolTable.h
@@ -68,6 +68,9 @@ public:
// Creates an Undefined symbol for a given name.
std::error_code addUndefined(StringRef Name);
+ // Rename From -> To in the symbol table.
+ std::error_code rename(StringRef From, StringRef To);
+
private:
std::error_code addObject(ObjectFile *File);
std::error_code addArchive(ArchiveFile *File);
diff --git a/lld/test/COFF/alternatename.test b/lld/test/COFF/alternatename.test
new file mode 100644
index 00000000000..0e8a6045726
--- /dev/null
+++ b/lld/test/COFF/alternatename.test
@@ -0,0 +1,4 @@
+# RUN: yaml2obj < %p/Inputs/ret42.yaml > %t.obj
+
+# RUN: lld -flavor link2 /entry:foo /subsystem:console \
+# RUN: /alternatename:foo=mainCRTStartup /out:%t.exe %t.obj
OpenPOWER on IntegriCloud