diff --git a/llvm/test/tools/llvm-readobj/XCOFF/relocations.test b/llvm/test/tools/llvm-readobj/XCOFF/relocations.test --- a/llvm/test/tools/llvm-readobj/XCOFF/relocations.test +++ b/llvm/test/tools/llvm-readobj/XCOFF/relocations.test @@ -2,40 +2,49 @@ # RUN: yaml2obj %s -o %t # RUN: llvm-readobj --relocs --expand-relocs %t | FileCheck %s --check-prefix=RELOCSEXP - -## TODO: Output of relocations not aligned. +# RUN: llvm-readobj --relocs %t | FileCheck %s --check-prefix=RELOCS # RELOCSEXP: Relocations [ # RELOCSEXP-NEXT: Section (index: 1) .text { -# RELOCSEXP-NEXT: Relocation { -# RELOCSEXP-NEXT: Virtual Address: 0x80 -# RELOCSEXP-NEXT: Symbol: .text (0) -# RELOCSEXP-NEXT: IsSigned: No -# RELOCSEXP-NEXT: FixupBitValue: 0 -# RELOCSEXP-NEXT: Length: 22 -# RELOCSEXP-NEXT: Type: R_POS (0x0) +# RELOCSEXP-NEXT: Relocation { +# RELOCSEXP-NEXT: Virtual Address: 0x80 +# RELOCSEXP-NEXT: Symbol: .text (0) +# RELOCSEXP-NEXT: IsSigned: No +# RELOCSEXP-NEXT: FixupBitValue: 0 +# RELOCSEXP-NEXT: Length: 22 +# RELOCSEXP-NEXT: Type: R_POS (0x0) +# RELOCSEXP-NEXT: } +# RELOCSEXP-NEXT: Relocation { +# RELOCSEXP-NEXT: Virtual Address: 0x100 +# RELOCSEXP-NEXT: Symbol: .text (0) +# RELOCSEXP-NEXT: IsSigned: No +# RELOCSEXP-NEXT: FixupBitValue: 0 +# RELOCSEXP-NEXT: Length: 21 +# RELOCSEXP-NEXT: Type: R_REL (0x2) +# RELOCSEXP-NEXT: } # RELOCSEXP-NEXT: } -# RELOCSEXP-NEXT: Relocation { -# RELOCSEXP-NEXT: Virtual Address: 0x100 -# RELOCSEXP-NEXT: Symbol: .text (0) -# RELOCSEXP-NEXT: IsSigned: No -# RELOCSEXP-NEXT: FixupBitValue: 0 -# RELOCSEXP-NEXT: Length: 21 -# RELOCSEXP-NEXT: Type: R_REL (0x2) +# RELOCSEXP-NEXT: Section (index: 2) .data { +# RELOCSEXP-NEXT: Relocation { +# RELOCSEXP-NEXT: Virtual Address: 0x200 +# RELOCSEXP-NEXT: Symbol: .data (1) +# RELOCSEXP-NEXT: IsSigned: No +# RELOCSEXP-NEXT: FixupBitValue: 0 +# RELOCSEXP-NEXT: Length: 20 +# RELOCSEXP-NEXT: Type: R_TOC (0x3) +# RELOCSEXP-NEXT: } # RELOCSEXP-NEXT: } -# RELOCSEXP-NEXT: } -# RELOCSEXP-NEXT: Section (index: 2) .data { -# RELOCSEXP-NEXT: Relocation { -# RELOCSEXP-NEXT: Virtual Address: 0x200 -# RELOCSEXP-NEXT: Symbol: .data (1) -# RELOCSEXP-NEXT: IsSigned: No -# RELOCSEXP-NEXT: FixupBitValue: 0 -# RELOCSEXP-NEXT: Length: 20 -# RELOCSEXP-NEXT: Type: R_TOC (0x3) -# RELOCSEXP-NEXT: } -# RELOCSEXP-NEXT: } # RELOCSEXP-NEXT: ] +# RELOCS: Relocations [ +# RELOCS-NEXT: Section (index: 1) .text { +# RELOCS-NEXT: 0x80 R_POS .text(0) +# RELOCS-NEXT: 0x100 R_REL .text(0) +# RELOCS-NEXT: } +# RELOCS-NEXT: Section (index: 2) .data { +# RELOCS-NEXT: 0x200 R_TOC .data(1) +# RELOCS-NEXT: } +# RELOCS-NEXT: ] + --- !XCOFF FileHeader: MagicNumber: 0x01DF diff --git a/llvm/tools/llvm-readobj/XCOFFDumper.cpp b/llvm/tools/llvm-readobj/XCOFFDumper.cpp --- a/llvm/tools/llvm-readobj/XCOFFDumper.cpp +++ b/llvm/tools/llvm-readobj/XCOFFDumper.cpp @@ -47,6 +47,7 @@ void printSymbol(const SymbolRef &); template void printRelocations(ArrayRef Sections); + template void printRelocation(RelTy Reloc); const XCOFFObjectFile &Obj; }; } // anonymous namespace @@ -124,11 +125,33 @@ #undef ECase }; +template void XCOFFDumper::printRelocation(RelTy Reloc) { + Expected ErrOrSymbolName = + Obj.getSymbolNameByIndex(Reloc.SymbolIndex); + if (Error E = ErrOrSymbolName.takeError()) { + reportUniqueWarning(std::move(E)); + return; + } + StringRef SymbolName = *ErrOrSymbolName; + StringRef RelocName = XCOFF::getRelocationTypeString(Reloc.Type); + if (opts::ExpandRelocs) { + DictScope Group(W, "Relocation"); + W.printHex("Virtual Address", Reloc.VirtualAddress); + W.printNumber("Symbol", SymbolName, Reloc.SymbolIndex); + W.printString("IsSigned", Reloc.isRelocationSigned() ? "Yes" : "No"); + W.printNumber("FixupBitValue", Reloc.isFixupIndicated() ? 1 : 0); + W.printNumber("Length", Reloc.getRelocatedLength()); + W.printEnum("Type", (uint8_t)Reloc.Type, + makeArrayRef(RelocationTypeNameclass)); + } else { + raw_ostream &OS = W.startLine(); + OS << W.hex(Reloc.VirtualAddress) << " " << RelocName << " " << SymbolName + << "(" << Reloc.SymbolIndex << ")\n"; + } +} + template void XCOFFDumper::printRelocations(ArrayRef Sections) { - if (!opts::ExpandRelocs) - report_fatal_error("Unexpanded relocation output not implemented."); - ListScope LS(W, "Relocations"); uint16_t Index = 0; for (const Shdr &Sec : Sections) { @@ -144,28 +167,17 @@ } const ArrayRef Relocations = *ErrOrRelocations; - if (Relocations.empty()) + size_t RelocCount = Relocations.size(); + if (!RelocCount) continue; - W.startLine() << "Section (index: " << Index << ") " << Sec.getName() - << " {\n"; - for (const RelTy Reloc : Relocations) { - Expected ErrOrSymbolName = - Obj.getSymbolNameByIndex(Reloc.SymbolIndex); - if (Error E = ErrOrSymbolName.takeError()) { - reportUniqueWarning(std::move(E)); - continue; + for (size_t I = 0; I < RelocCount; ++I) { + if (I == 0) { + W.startLine() << "Section (index: " << Index << ") " << Sec.getName() + << " {\n"; + W.indent(); } - - StringRef SymbolName = *ErrOrSymbolName; - DictScope RelocScope(W, "Relocation"); - W.printHex("Virtual Address", Reloc.VirtualAddress); - W.printNumber("Symbol", SymbolName, Reloc.SymbolIndex); - W.printString("IsSigned", Reloc.isRelocationSigned() ? "Yes" : "No"); - W.printNumber("FixupBitValue", Reloc.isFixupIndicated() ? 1 : 0); - W.printNumber("Length", Reloc.getRelocatedLength()); - W.printEnum("Type", (uint8_t)Reloc.Type, - makeArrayRef(RelocationTypeNameclass)); + printRelocation(Relocations[I]); } W.unindent(); W.startLine() << "}\n";