Index: test/tools/llvm-objdump/AArch64/plt.test =================================================================== --- /dev/null +++ test/tools/llvm-objdump/AArch64/plt.test @@ -0,0 +1,5 @@ +// RUN: llvm-objdump -d %p/Inputs/cfi.elf-aarch64 | FileCheck %s + +# CHECK: Disassembly of section .plt: +# CHECK: __cfi_slowpath@plt: +# CHECK: bl {{.*}} <__cfi_slowpath@plt> Index: test/tools/llvm-objdump/X86/plt.test =================================================================== --- /dev/null +++ test/tools/llvm-objdump/X86/plt.test @@ -0,0 +1,14 @@ +// RUN: llvm-objdump -d %p/Inputs/stripped-elf.so | FileCheck -check-prefix=64 %s +// RUN: llvm-objdump -d %p/Inputs/hello.exe.elf-i386 | FileCheck -check-prefix=32 %s +// RUN: llvm-objdump -d %p/Inputs/hello.exe.nopie.elf-i386 | FileCheck -check-prefix=32 %s + +# 64: Disassembly of section .plt: +# 64: __gmon_start__@plt: +# 64: __cxa_finalize@plt: +# 64: callq {{.*}} <__cxa_finalize@plt> + +# 32: Disassembly of section .plt: +# 32: puts@plt: +# 32: __libc_start_main@plt: +# 32: calll {{.*}} +# 32: calll {{.*}} <__libc_start_main@plt> Index: tools/llvm-objdump/llvm-objdump.cpp =================================================================== --- tools/llvm-objdump/llvm-objdump.cpp +++ tools/llvm-objdump/llvm-objdump.cpp @@ -56,6 +56,7 @@ #include "llvm/Support/InitLLVM.h" #include "llvm/Support/MemoryBuffer.h" #include "llvm/Support/SourceMgr.h" +#include "llvm/Support/StringSaver.h" #include "llvm/Support/TargetRegistry.h" #include "llvm/Support/TargetSelect.h" #include "llvm/Support/raw_ostream.h" @@ -1236,6 +1237,37 @@ llvm_unreachable("Unsupported binary format"); } +static void addPltEntries(const ObjectFile *Obj, + std::map &AllSymbols, + StringSaver &Saver) { + Optional Plt = None; + for (const SectionRef &Section : Obj->sections()) { + StringRef Name; + if (Section.getName(Name)) + continue; + if (Name == ".plt") + Plt = Section; + } + if (!Plt) + return; + if (auto *ElfObj = dyn_cast(Obj)) { + for (auto PltEntry : ElfObj->getPltAddresses()) { + SymbolRef Symbol(PltEntry.first, ElfObj); + + uint8_t SymbolType = getElfSymbolType(Obj, Symbol); + + Expected NameOrErr = Symbol.getName(); + if (!NameOrErr) + report_error(Obj->getFileName(), NameOrErr.takeError()); + if (NameOrErr->empty()) + continue; + StringRef Name = Saver.save((*NameOrErr + "@plt").str()); + + AllSymbols[*Plt].emplace_back(PltEntry.second, Name, SymbolType); + } + } +} + static void DisassembleObject(const ObjectFile *Obj, bool InlineRelocs) { if (StartAddress > StopAddress) error("Start address should be less than stop address"); @@ -1343,6 +1375,10 @@ if (AllSymbols.empty() && Obj->isELF()) addDynamicElfSymbols(Obj, AllSymbols); + BumpPtrAllocator A; + StringSaver Saver(A); + addPltEntries(Obj, AllSymbols, Saver); + // Create a mapping from virtual address to section. std::vector> SectionAddresses; for (SectionRef Sec : Obj->sections())