diff --git a/lld/ELF/Symbols.h b/lld/ELF/Symbols.h --- a/lld/ELF/Symbols.h +++ b/lld/ELF/Symbols.h @@ -21,6 +21,8 @@ #include "llvm/Object/ELF.h" namespace lld { +// Print the versioned name, optionally demangled. This function is used by +// diagnostics for better context. std::string toString(const elf::Symbol &); // There are two different ways to convert an Archive::Symbol to a string: diff --git a/lld/ELF/Symbols.cpp b/lld/ELF/Symbols.cpp --- a/lld/ELF/Symbols.cpp +++ b/lld/ELF/Symbols.cpp @@ -31,7 +31,17 @@ return std::string(symName); } -std::string toString(const elf::Symbol &b) { return demangle(b.getName()); } +std::string toString(const elf::Symbol &sym) { + StringRef name = sym.getName(); + std::string ret = demangle(name); + + // If sym comes from symtab, its name may have been truncated at '@' by + // SymbolTable::insert() or Symbol::parseSymbolVersion(). Add the trailing + // part. This check is safe because every symbol name ends with '\0'. + if (name.data()[name.size()] == '@') + ret += name.data() + name.size(); + return ret; +} std::string toELFString(const Archive::Symbol &b) { return demangle(b.getName()); } diff --git a/lld/test/ELF/undef-suggest-version.s b/lld/test/ELF/undef-suggest-version.s new file mode 100644 --- /dev/null +++ b/lld/test/ELF/undef-suggest-version.s @@ -0,0 +1,52 @@ +# REQUIRES: x86 +# RUN: llvm-mc -filetype=obj -triple=x86_64 %s -o %t.o +# RUN: echo 'v1 {bar;};' > %t.ver +# RUN: ld.lld -shared --version-script %t.ver %t.o -o %t.so + +## Suggest @@ for an unversioned reference. +# RUN: echo 'call bat' | llvm-mc -filetype=obj -triple=x86_64 - -o %tdef1.o +# RUN: not ld.lld %t.so %tdef1.o -o /dev/null 2>&1 | FileCheck --check-prefix=DEFAULT1 %s + +# DEFAULT1: error: undefined symbol: bat +# DEFAULT1-NEXT: >>> referenced by {{.*}}.o:(.text+0x1) +# DEFAULT1-NEXT: >>> did you mean: bar{{$}} +# DEFAULT1-NEXT: >>> defined in: {{.*}}.so + +## Suggest @@ for a versioned reference. +# RUN: echo '.symver bar.v2,bar@v2; call bar.v2' | llvm-mc -filetype=obj -triple=x86_64 - -o %tdef2.o +# RUN: not ld.lld %t.so %tdef2.o -o /dev/null 2>&1 | FileCheck --check-prefix=DEFAULT2 %s + +# DEFAULT2: error: undefined symbol: bar@v2 +# DEFAULT2-NEXT: >>> referenced by {{.*}}.o:(.text+0x1) +# DEFAULT2-NEXT: >>> did you mean: bar{{$}} +# DEFAULT2-NEXT: >>> defined in: {{.*}}.so + +## Suggest @ for unversioned references. +# RUN: echo 'call foo; call _Z3fooi' | llvm-mc -filetype=obj -triple=x86_64 - -o %thidden1.o +# RUN: not ld.lld %t.so %thidden1.o -o /dev/null 2>&1 | FileCheck --check-prefix=HIDDEN1 %s + +# HIDDEN1: error: undefined symbol: foo +# HIDDEN1-NEXT: >>> referenced by {{.*}}.o:(.text+0x1) +# HIDDEN1-NEXT: >>> did you mean: foo@v1 +# HIDDEN1-NEXT: >>> defined in: {{.*}}.so +# HIDDEN1-EMPTY: +# HIDDEN1-NEXT: error: undefined symbol: foo(int) +# HIDDEN1-NEXT: >>> referenced by {{.*}}.o:(.text+0x6) +# HIDDEN1-NEXT: >>> did you mean: foo(int)@v1 +# HIDDEN1-NEXT: >>> defined in: {{.*}}.so + +# RUN: echo '.symver foo.v2,foo@v2; call foo.v2' | llvm-mc -filetype=obj -triple=x86_64 - -o %thidden2.o +# RUN: not ld.lld %t.so %thidden2.o -o /dev/null 2>&1 | FileCheck --check-prefix=HIDDEN2 %s + +# HIDDEN2: error: undefined symbol: foo@v2 +# HIDDEN2-NEXT: >>> referenced by {{.*}}.o:(.text+0x1) +# HIDDEN2-NEXT: >>> did you mean: foo@v1 +# HIDDEN2-NEXT: >>> defined in: {{.*}}.so + +## %t.so exports bar@@v1 and two VERSYM_HIDDEN symbols: foo@v1 and _Z3fooi@v1. +.globl foo.v1, _Z3fooi.v1, bar +.symver foo.v1,foo@v1 +.symver _Z3fooi.v1,_Z3fooi@v1 +foo.v1: +_Z3fooi.v1: +bar: