1*0b57cec5SDimitry Andric //===-- COFFDumper.cpp - COFF-specific dumper -------------------*- C++ -*-===// 2*0b57cec5SDimitry Andric // 3*0b57cec5SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4*0b57cec5SDimitry Andric // See https://llvm.org/LICENSE.txt for license information. 5*0b57cec5SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6*0b57cec5SDimitry Andric // 7*0b57cec5SDimitry Andric //===----------------------------------------------------------------------===// 8*0b57cec5SDimitry Andric /// 9*0b57cec5SDimitry Andric /// \file 10*0b57cec5SDimitry Andric /// This file implements the COFF-specific dumper for llvm-readobj. 11*0b57cec5SDimitry Andric /// 12*0b57cec5SDimitry Andric //===----------------------------------------------------------------------===// 13*0b57cec5SDimitry Andric 14*0b57cec5SDimitry Andric #include "ARMWinEHPrinter.h" 15*0b57cec5SDimitry Andric #include "Error.h" 16*0b57cec5SDimitry Andric #include "ObjDumper.h" 17*0b57cec5SDimitry Andric #include "StackMapPrinter.h" 18*0b57cec5SDimitry Andric #include "Win64EHDumper.h" 19*0b57cec5SDimitry Andric #include "llvm-readobj.h" 20*0b57cec5SDimitry Andric #include "llvm/ADT/DenseMap.h" 21*0b57cec5SDimitry Andric #include "llvm/ADT/SmallString.h" 22*0b57cec5SDimitry Andric #include "llvm/ADT/StringExtras.h" 23*0b57cec5SDimitry Andric #include "llvm/BinaryFormat/COFF.h" 24*0b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/CVTypeVisitor.h" 25*0b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/CodeView.h" 26*0b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/DebugChecksumsSubsection.h" 27*0b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/DebugFrameDataSubsection.h" 28*0b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/DebugInlineeLinesSubsection.h" 29*0b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/DebugLinesSubsection.h" 30*0b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/DebugStringTableSubsection.h" 31*0b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/LazyRandomTypeCollection.h" 32*0b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/Line.h" 33*0b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/MergingTypeTableBuilder.h" 34*0b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/RecordSerialization.h" 35*0b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/SymbolDumpDelegate.h" 36*0b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/SymbolDumper.h" 37*0b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/SymbolRecord.h" 38*0b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/TypeDumpVisitor.h" 39*0b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/TypeHashing.h" 40*0b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/TypeIndex.h" 41*0b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/TypeRecord.h" 42*0b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/TypeStreamMerger.h" 43*0b57cec5SDimitry Andric #include "llvm/DebugInfo/CodeView/TypeTableCollection.h" 44*0b57cec5SDimitry Andric #include "llvm/Object/COFF.h" 45*0b57cec5SDimitry Andric #include "llvm/Object/ObjectFile.h" 46*0b57cec5SDimitry Andric #include "llvm/Object/WindowsResource.h" 47*0b57cec5SDimitry Andric #include "llvm/Support/BinaryStreamReader.h" 48*0b57cec5SDimitry Andric #include "llvm/Support/Casting.h" 49*0b57cec5SDimitry Andric #include "llvm/Support/Compiler.h" 50*0b57cec5SDimitry Andric #include "llvm/Support/ConvertUTF.h" 51*0b57cec5SDimitry Andric #include "llvm/Support/FormatVariadic.h" 52*0b57cec5SDimitry Andric #include "llvm/Support/LEB128.h" 53*0b57cec5SDimitry Andric #include "llvm/Support/ScopedPrinter.h" 54*0b57cec5SDimitry Andric #include "llvm/Support/Win64EH.h" 55*0b57cec5SDimitry Andric #include "llvm/Support/raw_ostream.h" 56*0b57cec5SDimitry Andric 57*0b57cec5SDimitry Andric using namespace llvm; 58*0b57cec5SDimitry Andric using namespace llvm::object; 59*0b57cec5SDimitry Andric using namespace llvm::codeview; 60*0b57cec5SDimitry Andric using namespace llvm::support; 61*0b57cec5SDimitry Andric using namespace llvm::Win64EH; 62*0b57cec5SDimitry Andric 63*0b57cec5SDimitry Andric namespace { 64*0b57cec5SDimitry Andric 65*0b57cec5SDimitry Andric struct LoadConfigTables { 66*0b57cec5SDimitry Andric uint64_t SEHTableVA = 0; 67*0b57cec5SDimitry Andric uint64_t SEHTableCount = 0; 68*0b57cec5SDimitry Andric uint32_t GuardFlags = 0; 69*0b57cec5SDimitry Andric uint64_t GuardFidTableVA = 0; 70*0b57cec5SDimitry Andric uint64_t GuardFidTableCount = 0; 71*0b57cec5SDimitry Andric uint64_t GuardLJmpTableVA = 0; 72*0b57cec5SDimitry Andric uint64_t GuardLJmpTableCount = 0; 73*0b57cec5SDimitry Andric }; 74*0b57cec5SDimitry Andric 75*0b57cec5SDimitry Andric class COFFDumper : public ObjDumper { 76*0b57cec5SDimitry Andric public: 77*0b57cec5SDimitry Andric friend class COFFObjectDumpDelegate; 78*0b57cec5SDimitry Andric COFFDumper(const llvm::object::COFFObjectFile *Obj, ScopedPrinter &Writer) 79*0b57cec5SDimitry Andric : ObjDumper(Writer), Obj(Obj), Writer(Writer), Types(100) {} 80*0b57cec5SDimitry Andric 81*0b57cec5SDimitry Andric void printFileHeaders() override; 82*0b57cec5SDimitry Andric void printSectionHeaders() override; 83*0b57cec5SDimitry Andric void printRelocations() override; 84*0b57cec5SDimitry Andric void printUnwindInfo() override; 85*0b57cec5SDimitry Andric 86*0b57cec5SDimitry Andric void printNeededLibraries() override; 87*0b57cec5SDimitry Andric 88*0b57cec5SDimitry Andric void printCOFFImports() override; 89*0b57cec5SDimitry Andric void printCOFFExports() override; 90*0b57cec5SDimitry Andric void printCOFFDirectives() override; 91*0b57cec5SDimitry Andric void printCOFFBaseReloc() override; 92*0b57cec5SDimitry Andric void printCOFFDebugDirectory() override; 93*0b57cec5SDimitry Andric void printCOFFResources() override; 94*0b57cec5SDimitry Andric void printCOFFLoadConfig() override; 95*0b57cec5SDimitry Andric void printCodeViewDebugInfo() override; 96*0b57cec5SDimitry Andric void mergeCodeViewTypes(llvm::codeview::MergingTypeTableBuilder &CVIDs, 97*0b57cec5SDimitry Andric llvm::codeview::MergingTypeTableBuilder &CVTypes, 98*0b57cec5SDimitry Andric llvm::codeview::GlobalTypeTableBuilder &GlobalCVIDs, 99*0b57cec5SDimitry Andric llvm::codeview::GlobalTypeTableBuilder &GlobalCVTypes, 100*0b57cec5SDimitry Andric bool GHash) override; 101*0b57cec5SDimitry Andric void printStackMap() const override; 102*0b57cec5SDimitry Andric void printAddrsig() override; 103*0b57cec5SDimitry Andric private: 104*0b57cec5SDimitry Andric void printSymbols() override; 105*0b57cec5SDimitry Andric void printDynamicSymbols() override; 106*0b57cec5SDimitry Andric void printSymbol(const SymbolRef &Sym); 107*0b57cec5SDimitry Andric void printRelocation(const SectionRef &Section, const RelocationRef &Reloc, 108*0b57cec5SDimitry Andric uint64_t Bias = 0); 109*0b57cec5SDimitry Andric void printDataDirectory(uint32_t Index, const std::string &FieldName); 110*0b57cec5SDimitry Andric 111*0b57cec5SDimitry Andric void printDOSHeader(const dos_header *DH); 112*0b57cec5SDimitry Andric template <class PEHeader> void printPEHeader(const PEHeader *Hdr); 113*0b57cec5SDimitry Andric void printBaseOfDataField(const pe32_header *Hdr); 114*0b57cec5SDimitry Andric void printBaseOfDataField(const pe32plus_header *Hdr); 115*0b57cec5SDimitry Andric template <typename T> 116*0b57cec5SDimitry Andric void printCOFFLoadConfig(const T *Conf, LoadConfigTables &Tables); 117*0b57cec5SDimitry Andric typedef void (*PrintExtraCB)(raw_ostream &, const uint8_t *); 118*0b57cec5SDimitry Andric void printRVATable(uint64_t TableVA, uint64_t Count, uint64_t EntrySize, 119*0b57cec5SDimitry Andric PrintExtraCB PrintExtra = 0); 120*0b57cec5SDimitry Andric 121*0b57cec5SDimitry Andric void printCodeViewSymbolSection(StringRef SectionName, const SectionRef &Section); 122*0b57cec5SDimitry Andric void printCodeViewTypeSection(StringRef SectionName, const SectionRef &Section); 123*0b57cec5SDimitry Andric StringRef getTypeName(TypeIndex Ty); 124*0b57cec5SDimitry Andric StringRef getFileNameForFileOffset(uint32_t FileOffset); 125*0b57cec5SDimitry Andric void printFileNameForOffset(StringRef Label, uint32_t FileOffset); 126*0b57cec5SDimitry Andric void printTypeIndex(StringRef FieldName, TypeIndex TI) { 127*0b57cec5SDimitry Andric // Forward to CVTypeDumper for simplicity. 128*0b57cec5SDimitry Andric codeview::printTypeIndex(Writer, FieldName, TI, Types); 129*0b57cec5SDimitry Andric } 130*0b57cec5SDimitry Andric 131*0b57cec5SDimitry Andric void printCodeViewSymbolsSubsection(StringRef Subsection, 132*0b57cec5SDimitry Andric const SectionRef &Section, 133*0b57cec5SDimitry Andric StringRef SectionContents); 134*0b57cec5SDimitry Andric 135*0b57cec5SDimitry Andric void printCodeViewFileChecksums(StringRef Subsection); 136*0b57cec5SDimitry Andric 137*0b57cec5SDimitry Andric void printCodeViewInlineeLines(StringRef Subsection); 138*0b57cec5SDimitry Andric 139*0b57cec5SDimitry Andric void printRelocatedField(StringRef Label, const coff_section *Sec, 140*0b57cec5SDimitry Andric uint32_t RelocOffset, uint32_t Offset, 141*0b57cec5SDimitry Andric StringRef *RelocSym = nullptr); 142*0b57cec5SDimitry Andric 143*0b57cec5SDimitry Andric uint32_t countTotalTableEntries(ResourceSectionRef RSF, 144*0b57cec5SDimitry Andric const coff_resource_dir_table &Table, 145*0b57cec5SDimitry Andric StringRef Level); 146*0b57cec5SDimitry Andric 147*0b57cec5SDimitry Andric void printResourceDirectoryTable(ResourceSectionRef RSF, 148*0b57cec5SDimitry Andric const coff_resource_dir_table &Table, 149*0b57cec5SDimitry Andric StringRef Level); 150*0b57cec5SDimitry Andric 151*0b57cec5SDimitry Andric void printBinaryBlockWithRelocs(StringRef Label, const SectionRef &Sec, 152*0b57cec5SDimitry Andric StringRef SectionContents, StringRef Block); 153*0b57cec5SDimitry Andric 154*0b57cec5SDimitry Andric /// Given a .debug$S section, find the string table and file checksum table. 155*0b57cec5SDimitry Andric void initializeFileAndStringTables(BinaryStreamReader &Reader); 156*0b57cec5SDimitry Andric 157*0b57cec5SDimitry Andric void cacheRelocations(); 158*0b57cec5SDimitry Andric 159*0b57cec5SDimitry Andric std::error_code resolveSymbol(const coff_section *Section, uint64_t Offset, 160*0b57cec5SDimitry Andric SymbolRef &Sym); 161*0b57cec5SDimitry Andric std::error_code resolveSymbolName(const coff_section *Section, 162*0b57cec5SDimitry Andric uint64_t Offset, StringRef &Name); 163*0b57cec5SDimitry Andric std::error_code resolveSymbolName(const coff_section *Section, 164*0b57cec5SDimitry Andric StringRef SectionContents, 165*0b57cec5SDimitry Andric const void *RelocPtr, StringRef &Name); 166*0b57cec5SDimitry Andric void printImportedSymbols(iterator_range<imported_symbol_iterator> Range); 167*0b57cec5SDimitry Andric void printDelayImportedSymbols( 168*0b57cec5SDimitry Andric const DelayImportDirectoryEntryRef &I, 169*0b57cec5SDimitry Andric iterator_range<imported_symbol_iterator> Range); 170*0b57cec5SDimitry Andric ErrorOr<const coff_resource_dir_entry &> 171*0b57cec5SDimitry Andric getResourceDirectoryTableEntry(const coff_resource_dir_table &Table, 172*0b57cec5SDimitry Andric uint32_t Index); 173*0b57cec5SDimitry Andric 174*0b57cec5SDimitry Andric typedef DenseMap<const coff_section*, std::vector<RelocationRef> > RelocMapTy; 175*0b57cec5SDimitry Andric 176*0b57cec5SDimitry Andric const llvm::object::COFFObjectFile *Obj; 177*0b57cec5SDimitry Andric bool RelocCached = false; 178*0b57cec5SDimitry Andric RelocMapTy RelocMap; 179*0b57cec5SDimitry Andric 180*0b57cec5SDimitry Andric DebugChecksumsSubsectionRef CVFileChecksumTable; 181*0b57cec5SDimitry Andric 182*0b57cec5SDimitry Andric DebugStringTableSubsectionRef CVStringTable; 183*0b57cec5SDimitry Andric 184*0b57cec5SDimitry Andric /// Track the compilation CPU type. S_COMPILE3 symbol records typically come 185*0b57cec5SDimitry Andric /// first, but if we don't see one, just assume an X64 CPU type. It is common. 186*0b57cec5SDimitry Andric CPUType CompilationCPUType = CPUType::X64; 187*0b57cec5SDimitry Andric 188*0b57cec5SDimitry Andric ScopedPrinter &Writer; 189*0b57cec5SDimitry Andric BinaryByteStream TypeContents; 190*0b57cec5SDimitry Andric LazyRandomTypeCollection Types; 191*0b57cec5SDimitry Andric }; 192*0b57cec5SDimitry Andric 193*0b57cec5SDimitry Andric class COFFObjectDumpDelegate : public SymbolDumpDelegate { 194*0b57cec5SDimitry Andric public: 195*0b57cec5SDimitry Andric COFFObjectDumpDelegate(COFFDumper &CD, const SectionRef &SR, 196*0b57cec5SDimitry Andric const COFFObjectFile *Obj, StringRef SectionContents) 197*0b57cec5SDimitry Andric : CD(CD), SR(SR), SectionContents(SectionContents) { 198*0b57cec5SDimitry Andric Sec = Obj->getCOFFSection(SR); 199*0b57cec5SDimitry Andric } 200*0b57cec5SDimitry Andric 201*0b57cec5SDimitry Andric uint32_t getRecordOffset(BinaryStreamReader Reader) override { 202*0b57cec5SDimitry Andric ArrayRef<uint8_t> Data; 203*0b57cec5SDimitry Andric if (auto EC = Reader.readLongestContiguousChunk(Data)) { 204*0b57cec5SDimitry Andric llvm::consumeError(std::move(EC)); 205*0b57cec5SDimitry Andric return 0; 206*0b57cec5SDimitry Andric } 207*0b57cec5SDimitry Andric return Data.data() - SectionContents.bytes_begin(); 208*0b57cec5SDimitry Andric } 209*0b57cec5SDimitry Andric 210*0b57cec5SDimitry Andric void printRelocatedField(StringRef Label, uint32_t RelocOffset, 211*0b57cec5SDimitry Andric uint32_t Offset, StringRef *RelocSym) override { 212*0b57cec5SDimitry Andric CD.printRelocatedField(Label, Sec, RelocOffset, Offset, RelocSym); 213*0b57cec5SDimitry Andric } 214*0b57cec5SDimitry Andric 215*0b57cec5SDimitry Andric void printBinaryBlockWithRelocs(StringRef Label, 216*0b57cec5SDimitry Andric ArrayRef<uint8_t> Block) override { 217*0b57cec5SDimitry Andric StringRef SBlock(reinterpret_cast<const char *>(Block.data()), 218*0b57cec5SDimitry Andric Block.size()); 219*0b57cec5SDimitry Andric if (opts::CodeViewSubsectionBytes) 220*0b57cec5SDimitry Andric CD.printBinaryBlockWithRelocs(Label, SR, SectionContents, SBlock); 221*0b57cec5SDimitry Andric } 222*0b57cec5SDimitry Andric 223*0b57cec5SDimitry Andric StringRef getFileNameForFileOffset(uint32_t FileOffset) override { 224*0b57cec5SDimitry Andric return CD.getFileNameForFileOffset(FileOffset); 225*0b57cec5SDimitry Andric } 226*0b57cec5SDimitry Andric 227*0b57cec5SDimitry Andric DebugStringTableSubsectionRef getStringTable() override { 228*0b57cec5SDimitry Andric return CD.CVStringTable; 229*0b57cec5SDimitry Andric } 230*0b57cec5SDimitry Andric 231*0b57cec5SDimitry Andric private: 232*0b57cec5SDimitry Andric COFFDumper &CD; 233*0b57cec5SDimitry Andric const SectionRef &SR; 234*0b57cec5SDimitry Andric const coff_section *Sec; 235*0b57cec5SDimitry Andric StringRef SectionContents; 236*0b57cec5SDimitry Andric }; 237*0b57cec5SDimitry Andric 238*0b57cec5SDimitry Andric } // end namespace 239*0b57cec5SDimitry Andric 240*0b57cec5SDimitry Andric namespace llvm { 241*0b57cec5SDimitry Andric 242*0b57cec5SDimitry Andric std::error_code createCOFFDumper(const object::ObjectFile *Obj, 243*0b57cec5SDimitry Andric ScopedPrinter &Writer, 244*0b57cec5SDimitry Andric std::unique_ptr<ObjDumper> &Result) { 245*0b57cec5SDimitry Andric const COFFObjectFile *COFFObj = dyn_cast<COFFObjectFile>(Obj); 246*0b57cec5SDimitry Andric if (!COFFObj) 247*0b57cec5SDimitry Andric return readobj_error::unsupported_obj_file_format; 248*0b57cec5SDimitry Andric 249*0b57cec5SDimitry Andric Result.reset(new COFFDumper(COFFObj, Writer)); 250*0b57cec5SDimitry Andric return readobj_error::success; 251*0b57cec5SDimitry Andric } 252*0b57cec5SDimitry Andric 253*0b57cec5SDimitry Andric } // namespace llvm 254*0b57cec5SDimitry Andric 255*0b57cec5SDimitry Andric // Given a section and an offset into this section the function returns the 256*0b57cec5SDimitry Andric // symbol used for the relocation at the offset. 257*0b57cec5SDimitry Andric std::error_code COFFDumper::resolveSymbol(const coff_section *Section, 258*0b57cec5SDimitry Andric uint64_t Offset, SymbolRef &Sym) { 259*0b57cec5SDimitry Andric cacheRelocations(); 260*0b57cec5SDimitry Andric const auto &Relocations = RelocMap[Section]; 261*0b57cec5SDimitry Andric auto SymI = Obj->symbol_end(); 262*0b57cec5SDimitry Andric for (const auto &Relocation : Relocations) { 263*0b57cec5SDimitry Andric uint64_t RelocationOffset = Relocation.getOffset(); 264*0b57cec5SDimitry Andric 265*0b57cec5SDimitry Andric if (RelocationOffset == Offset) { 266*0b57cec5SDimitry Andric SymI = Relocation.getSymbol(); 267*0b57cec5SDimitry Andric break; 268*0b57cec5SDimitry Andric } 269*0b57cec5SDimitry Andric } 270*0b57cec5SDimitry Andric if (SymI == Obj->symbol_end()) 271*0b57cec5SDimitry Andric return readobj_error::unknown_symbol; 272*0b57cec5SDimitry Andric Sym = *SymI; 273*0b57cec5SDimitry Andric return readobj_error::success; 274*0b57cec5SDimitry Andric } 275*0b57cec5SDimitry Andric 276*0b57cec5SDimitry Andric // Given a section and an offset into this section the function returns the name 277*0b57cec5SDimitry Andric // of the symbol used for the relocation at the offset. 278*0b57cec5SDimitry Andric std::error_code COFFDumper::resolveSymbolName(const coff_section *Section, 279*0b57cec5SDimitry Andric uint64_t Offset, 280*0b57cec5SDimitry Andric StringRef &Name) { 281*0b57cec5SDimitry Andric SymbolRef Symbol; 282*0b57cec5SDimitry Andric if (std::error_code EC = resolveSymbol(Section, Offset, Symbol)) 283*0b57cec5SDimitry Andric return EC; 284*0b57cec5SDimitry Andric Expected<StringRef> NameOrErr = Symbol.getName(); 285*0b57cec5SDimitry Andric if (!NameOrErr) 286*0b57cec5SDimitry Andric return errorToErrorCode(NameOrErr.takeError()); 287*0b57cec5SDimitry Andric Name = *NameOrErr; 288*0b57cec5SDimitry Andric return std::error_code(); 289*0b57cec5SDimitry Andric } 290*0b57cec5SDimitry Andric 291*0b57cec5SDimitry Andric // Helper for when you have a pointer to real data and you want to know about 292*0b57cec5SDimitry Andric // relocations against it. 293*0b57cec5SDimitry Andric std::error_code COFFDumper::resolveSymbolName(const coff_section *Section, 294*0b57cec5SDimitry Andric StringRef SectionContents, 295*0b57cec5SDimitry Andric const void *RelocPtr, 296*0b57cec5SDimitry Andric StringRef &Name) { 297*0b57cec5SDimitry Andric assert(SectionContents.data() < RelocPtr && 298*0b57cec5SDimitry Andric RelocPtr < SectionContents.data() + SectionContents.size() && 299*0b57cec5SDimitry Andric "pointer to relocated object is not in section"); 300*0b57cec5SDimitry Andric uint64_t Offset = ptrdiff_t(reinterpret_cast<const char *>(RelocPtr) - 301*0b57cec5SDimitry Andric SectionContents.data()); 302*0b57cec5SDimitry Andric return resolveSymbolName(Section, Offset, Name); 303*0b57cec5SDimitry Andric } 304*0b57cec5SDimitry Andric 305*0b57cec5SDimitry Andric void COFFDumper::printRelocatedField(StringRef Label, const coff_section *Sec, 306*0b57cec5SDimitry Andric uint32_t RelocOffset, uint32_t Offset, 307*0b57cec5SDimitry Andric StringRef *RelocSym) { 308*0b57cec5SDimitry Andric StringRef SymStorage; 309*0b57cec5SDimitry Andric StringRef &Symbol = RelocSym ? *RelocSym : SymStorage; 310*0b57cec5SDimitry Andric if (!resolveSymbolName(Sec, RelocOffset, Symbol)) 311*0b57cec5SDimitry Andric W.printSymbolOffset(Label, Symbol, Offset); 312*0b57cec5SDimitry Andric else 313*0b57cec5SDimitry Andric W.printHex(Label, RelocOffset); 314*0b57cec5SDimitry Andric } 315*0b57cec5SDimitry Andric 316*0b57cec5SDimitry Andric void COFFDumper::printBinaryBlockWithRelocs(StringRef Label, 317*0b57cec5SDimitry Andric const SectionRef &Sec, 318*0b57cec5SDimitry Andric StringRef SectionContents, 319*0b57cec5SDimitry Andric StringRef Block) { 320*0b57cec5SDimitry Andric W.printBinaryBlock(Label, Block); 321*0b57cec5SDimitry Andric 322*0b57cec5SDimitry Andric assert(SectionContents.begin() < Block.begin() && 323*0b57cec5SDimitry Andric SectionContents.end() >= Block.end() && 324*0b57cec5SDimitry Andric "Block is not contained in SectionContents"); 325*0b57cec5SDimitry Andric uint64_t OffsetStart = Block.data() - SectionContents.data(); 326*0b57cec5SDimitry Andric uint64_t OffsetEnd = OffsetStart + Block.size(); 327*0b57cec5SDimitry Andric 328*0b57cec5SDimitry Andric W.flush(); 329*0b57cec5SDimitry Andric cacheRelocations(); 330*0b57cec5SDimitry Andric ListScope D(W, "BlockRelocations"); 331*0b57cec5SDimitry Andric const coff_section *Section = Obj->getCOFFSection(Sec); 332*0b57cec5SDimitry Andric const auto &Relocations = RelocMap[Section]; 333*0b57cec5SDimitry Andric for (const auto &Relocation : Relocations) { 334*0b57cec5SDimitry Andric uint64_t RelocationOffset = Relocation.getOffset(); 335*0b57cec5SDimitry Andric if (OffsetStart <= RelocationOffset && RelocationOffset < OffsetEnd) 336*0b57cec5SDimitry Andric printRelocation(Sec, Relocation, OffsetStart); 337*0b57cec5SDimitry Andric } 338*0b57cec5SDimitry Andric } 339*0b57cec5SDimitry Andric 340*0b57cec5SDimitry Andric static const EnumEntry<COFF::MachineTypes> ImageFileMachineType[] = { 341*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_MACHINE_UNKNOWN ), 342*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_MACHINE_AM33 ), 343*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_MACHINE_AMD64 ), 344*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_MACHINE_ARM ), 345*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_MACHINE_ARM64 ), 346*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_MACHINE_ARMNT ), 347*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_MACHINE_EBC ), 348*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_MACHINE_I386 ), 349*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_MACHINE_IA64 ), 350*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_MACHINE_M32R ), 351*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_MACHINE_MIPS16 ), 352*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_MACHINE_MIPSFPU ), 353*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_MACHINE_MIPSFPU16), 354*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_MACHINE_POWERPC ), 355*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_MACHINE_POWERPCFP), 356*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_MACHINE_R4000 ), 357*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_MACHINE_SH3 ), 358*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_MACHINE_SH3DSP ), 359*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_MACHINE_SH4 ), 360*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_MACHINE_SH5 ), 361*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_MACHINE_THUMB ), 362*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_MACHINE_WCEMIPSV2) 363*0b57cec5SDimitry Andric }; 364*0b57cec5SDimitry Andric 365*0b57cec5SDimitry Andric static const EnumEntry<COFF::Characteristics> ImageFileCharacteristics[] = { 366*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_RELOCS_STRIPPED ), 367*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_EXECUTABLE_IMAGE ), 368*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_LINE_NUMS_STRIPPED ), 369*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_LOCAL_SYMS_STRIPPED ), 370*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_AGGRESSIVE_WS_TRIM ), 371*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_LARGE_ADDRESS_AWARE ), 372*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_BYTES_REVERSED_LO ), 373*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_32BIT_MACHINE ), 374*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_DEBUG_STRIPPED ), 375*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_REMOVABLE_RUN_FROM_SWAP), 376*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_NET_RUN_FROM_SWAP ), 377*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_SYSTEM ), 378*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_DLL ), 379*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_UP_SYSTEM_ONLY ), 380*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_FILE_BYTES_REVERSED_HI ) 381*0b57cec5SDimitry Andric }; 382*0b57cec5SDimitry Andric 383*0b57cec5SDimitry Andric static const EnumEntry<COFF::WindowsSubsystem> PEWindowsSubsystem[] = { 384*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SUBSYSTEM_UNKNOWN ), 385*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SUBSYSTEM_NATIVE ), 386*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SUBSYSTEM_WINDOWS_GUI ), 387*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SUBSYSTEM_WINDOWS_CUI ), 388*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SUBSYSTEM_POSIX_CUI ), 389*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SUBSYSTEM_WINDOWS_CE_GUI ), 390*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SUBSYSTEM_EFI_APPLICATION ), 391*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SUBSYSTEM_EFI_BOOT_SERVICE_DRIVER), 392*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SUBSYSTEM_EFI_RUNTIME_DRIVER ), 393*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SUBSYSTEM_EFI_ROM ), 394*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SUBSYSTEM_XBOX ), 395*0b57cec5SDimitry Andric }; 396*0b57cec5SDimitry Andric 397*0b57cec5SDimitry Andric static const EnumEntry<COFF::DLLCharacteristics> PEDLLCharacteristics[] = { 398*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_DLL_CHARACTERISTICS_HIGH_ENTROPY_VA ), 399*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_DLL_CHARACTERISTICS_DYNAMIC_BASE ), 400*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_DLL_CHARACTERISTICS_FORCE_INTEGRITY ), 401*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_DLL_CHARACTERISTICS_NX_COMPAT ), 402*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_DLL_CHARACTERISTICS_NO_ISOLATION ), 403*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_DLL_CHARACTERISTICS_NO_SEH ), 404*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_DLL_CHARACTERISTICS_NO_BIND ), 405*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_DLL_CHARACTERISTICS_APPCONTAINER ), 406*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_DLL_CHARACTERISTICS_WDM_DRIVER ), 407*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_DLL_CHARACTERISTICS_GUARD_CF ), 408*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_DLL_CHARACTERISTICS_TERMINAL_SERVER_AWARE), 409*0b57cec5SDimitry Andric }; 410*0b57cec5SDimitry Andric 411*0b57cec5SDimitry Andric static const EnumEntry<COFF::SectionCharacteristics> 412*0b57cec5SDimitry Andric ImageSectionCharacteristics[] = { 413*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_TYPE_NOLOAD ), 414*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_TYPE_NO_PAD ), 415*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_CNT_CODE ), 416*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_CNT_INITIALIZED_DATA ), 417*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_CNT_UNINITIALIZED_DATA), 418*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_LNK_OTHER ), 419*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_LNK_INFO ), 420*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_LNK_REMOVE ), 421*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_LNK_COMDAT ), 422*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_GPREL ), 423*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_MEM_PURGEABLE ), 424*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_MEM_16BIT ), 425*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_MEM_LOCKED ), 426*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_MEM_PRELOAD ), 427*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_ALIGN_1BYTES ), 428*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_ALIGN_2BYTES ), 429*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_ALIGN_4BYTES ), 430*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_ALIGN_8BYTES ), 431*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_ALIGN_16BYTES ), 432*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_ALIGN_32BYTES ), 433*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_ALIGN_64BYTES ), 434*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_ALIGN_128BYTES ), 435*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_ALIGN_256BYTES ), 436*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_ALIGN_512BYTES ), 437*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_ALIGN_1024BYTES ), 438*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_ALIGN_2048BYTES ), 439*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_ALIGN_4096BYTES ), 440*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_ALIGN_8192BYTES ), 441*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_LNK_NRELOC_OVFL ), 442*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_MEM_DISCARDABLE ), 443*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_MEM_NOT_CACHED ), 444*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_MEM_NOT_PAGED ), 445*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_MEM_SHARED ), 446*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_MEM_EXECUTE ), 447*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_MEM_READ ), 448*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(COFF, IMAGE_SCN_MEM_WRITE ) 449*0b57cec5SDimitry Andric }; 450*0b57cec5SDimitry Andric 451*0b57cec5SDimitry Andric static const EnumEntry<COFF::SymbolBaseType> ImageSymType[] = { 452*0b57cec5SDimitry Andric { "Null" , COFF::IMAGE_SYM_TYPE_NULL }, 453*0b57cec5SDimitry Andric { "Void" , COFF::IMAGE_SYM_TYPE_VOID }, 454*0b57cec5SDimitry Andric { "Char" , COFF::IMAGE_SYM_TYPE_CHAR }, 455*0b57cec5SDimitry Andric { "Short" , COFF::IMAGE_SYM_TYPE_SHORT }, 456*0b57cec5SDimitry Andric { "Int" , COFF::IMAGE_SYM_TYPE_INT }, 457*0b57cec5SDimitry Andric { "Long" , COFF::IMAGE_SYM_TYPE_LONG }, 458*0b57cec5SDimitry Andric { "Float" , COFF::IMAGE_SYM_TYPE_FLOAT }, 459*0b57cec5SDimitry Andric { "Double", COFF::IMAGE_SYM_TYPE_DOUBLE }, 460*0b57cec5SDimitry Andric { "Struct", COFF::IMAGE_SYM_TYPE_STRUCT }, 461*0b57cec5SDimitry Andric { "Union" , COFF::IMAGE_SYM_TYPE_UNION }, 462*0b57cec5SDimitry Andric { "Enum" , COFF::IMAGE_SYM_TYPE_ENUM }, 463*0b57cec5SDimitry Andric { "MOE" , COFF::IMAGE_SYM_TYPE_MOE }, 464*0b57cec5SDimitry Andric { "Byte" , COFF::IMAGE_SYM_TYPE_BYTE }, 465*0b57cec5SDimitry Andric { "Word" , COFF::IMAGE_SYM_TYPE_WORD }, 466*0b57cec5SDimitry Andric { "UInt" , COFF::IMAGE_SYM_TYPE_UINT }, 467*0b57cec5SDimitry Andric { "DWord" , COFF::IMAGE_SYM_TYPE_DWORD } 468*0b57cec5SDimitry Andric }; 469*0b57cec5SDimitry Andric 470*0b57cec5SDimitry Andric static const EnumEntry<COFF::SymbolComplexType> ImageSymDType[] = { 471*0b57cec5SDimitry Andric { "Null" , COFF::IMAGE_SYM_DTYPE_NULL }, 472*0b57cec5SDimitry Andric { "Pointer" , COFF::IMAGE_SYM_DTYPE_POINTER }, 473*0b57cec5SDimitry Andric { "Function", COFF::IMAGE_SYM_DTYPE_FUNCTION }, 474*0b57cec5SDimitry Andric { "Array" , COFF::IMAGE_SYM_DTYPE_ARRAY } 475*0b57cec5SDimitry Andric }; 476*0b57cec5SDimitry Andric 477*0b57cec5SDimitry Andric static const EnumEntry<COFF::SymbolStorageClass> ImageSymClass[] = { 478*0b57cec5SDimitry Andric { "EndOfFunction" , COFF::IMAGE_SYM_CLASS_END_OF_FUNCTION }, 479*0b57cec5SDimitry Andric { "Null" , COFF::IMAGE_SYM_CLASS_NULL }, 480*0b57cec5SDimitry Andric { "Automatic" , COFF::IMAGE_SYM_CLASS_AUTOMATIC }, 481*0b57cec5SDimitry Andric { "External" , COFF::IMAGE_SYM_CLASS_EXTERNAL }, 482*0b57cec5SDimitry Andric { "Static" , COFF::IMAGE_SYM_CLASS_STATIC }, 483*0b57cec5SDimitry Andric { "Register" , COFF::IMAGE_SYM_CLASS_REGISTER }, 484*0b57cec5SDimitry Andric { "ExternalDef" , COFF::IMAGE_SYM_CLASS_EXTERNAL_DEF }, 485*0b57cec5SDimitry Andric { "Label" , COFF::IMAGE_SYM_CLASS_LABEL }, 486*0b57cec5SDimitry Andric { "UndefinedLabel" , COFF::IMAGE_SYM_CLASS_UNDEFINED_LABEL }, 487*0b57cec5SDimitry Andric { "MemberOfStruct" , COFF::IMAGE_SYM_CLASS_MEMBER_OF_STRUCT }, 488*0b57cec5SDimitry Andric { "Argument" , COFF::IMAGE_SYM_CLASS_ARGUMENT }, 489*0b57cec5SDimitry Andric { "StructTag" , COFF::IMAGE_SYM_CLASS_STRUCT_TAG }, 490*0b57cec5SDimitry Andric { "MemberOfUnion" , COFF::IMAGE_SYM_CLASS_MEMBER_OF_UNION }, 491*0b57cec5SDimitry Andric { "UnionTag" , COFF::IMAGE_SYM_CLASS_UNION_TAG }, 492*0b57cec5SDimitry Andric { "TypeDefinition" , COFF::IMAGE_SYM_CLASS_TYPE_DEFINITION }, 493*0b57cec5SDimitry Andric { "UndefinedStatic", COFF::IMAGE_SYM_CLASS_UNDEFINED_STATIC }, 494*0b57cec5SDimitry Andric { "EnumTag" , COFF::IMAGE_SYM_CLASS_ENUM_TAG }, 495*0b57cec5SDimitry Andric { "MemberOfEnum" , COFF::IMAGE_SYM_CLASS_MEMBER_OF_ENUM }, 496*0b57cec5SDimitry Andric { "RegisterParam" , COFF::IMAGE_SYM_CLASS_REGISTER_PARAM }, 497*0b57cec5SDimitry Andric { "BitField" , COFF::IMAGE_SYM_CLASS_BIT_FIELD }, 498*0b57cec5SDimitry Andric { "Block" , COFF::IMAGE_SYM_CLASS_BLOCK }, 499*0b57cec5SDimitry Andric { "Function" , COFF::IMAGE_SYM_CLASS_FUNCTION }, 500*0b57cec5SDimitry Andric { "EndOfStruct" , COFF::IMAGE_SYM_CLASS_END_OF_STRUCT }, 501*0b57cec5SDimitry Andric { "File" , COFF::IMAGE_SYM_CLASS_FILE }, 502*0b57cec5SDimitry Andric { "Section" , COFF::IMAGE_SYM_CLASS_SECTION }, 503*0b57cec5SDimitry Andric { "WeakExternal" , COFF::IMAGE_SYM_CLASS_WEAK_EXTERNAL }, 504*0b57cec5SDimitry Andric { "CLRToken" , COFF::IMAGE_SYM_CLASS_CLR_TOKEN } 505*0b57cec5SDimitry Andric }; 506*0b57cec5SDimitry Andric 507*0b57cec5SDimitry Andric static const EnumEntry<COFF::COMDATType> ImageCOMDATSelect[] = { 508*0b57cec5SDimitry Andric { "NoDuplicates", COFF::IMAGE_COMDAT_SELECT_NODUPLICATES }, 509*0b57cec5SDimitry Andric { "Any" , COFF::IMAGE_COMDAT_SELECT_ANY }, 510*0b57cec5SDimitry Andric { "SameSize" , COFF::IMAGE_COMDAT_SELECT_SAME_SIZE }, 511*0b57cec5SDimitry Andric { "ExactMatch" , COFF::IMAGE_COMDAT_SELECT_EXACT_MATCH }, 512*0b57cec5SDimitry Andric { "Associative" , COFF::IMAGE_COMDAT_SELECT_ASSOCIATIVE }, 513*0b57cec5SDimitry Andric { "Largest" , COFF::IMAGE_COMDAT_SELECT_LARGEST }, 514*0b57cec5SDimitry Andric { "Newest" , COFF::IMAGE_COMDAT_SELECT_NEWEST } 515*0b57cec5SDimitry Andric }; 516*0b57cec5SDimitry Andric 517*0b57cec5SDimitry Andric static const EnumEntry<COFF::DebugType> ImageDebugType[] = { 518*0b57cec5SDimitry Andric { "Unknown" , COFF::IMAGE_DEBUG_TYPE_UNKNOWN }, 519*0b57cec5SDimitry Andric { "COFF" , COFF::IMAGE_DEBUG_TYPE_COFF }, 520*0b57cec5SDimitry Andric { "CodeView" , COFF::IMAGE_DEBUG_TYPE_CODEVIEW }, 521*0b57cec5SDimitry Andric { "FPO" , COFF::IMAGE_DEBUG_TYPE_FPO }, 522*0b57cec5SDimitry Andric { "Misc" , COFF::IMAGE_DEBUG_TYPE_MISC }, 523*0b57cec5SDimitry Andric { "Exception" , COFF::IMAGE_DEBUG_TYPE_EXCEPTION }, 524*0b57cec5SDimitry Andric { "Fixup" , COFF::IMAGE_DEBUG_TYPE_FIXUP }, 525*0b57cec5SDimitry Andric { "OmapToSrc" , COFF::IMAGE_DEBUG_TYPE_OMAP_TO_SRC }, 526*0b57cec5SDimitry Andric { "OmapFromSrc", COFF::IMAGE_DEBUG_TYPE_OMAP_FROM_SRC }, 527*0b57cec5SDimitry Andric { "Borland" , COFF::IMAGE_DEBUG_TYPE_BORLAND }, 528*0b57cec5SDimitry Andric { "Reserved10" , COFF::IMAGE_DEBUG_TYPE_RESERVED10 }, 529*0b57cec5SDimitry Andric { "CLSID" , COFF::IMAGE_DEBUG_TYPE_CLSID }, 530*0b57cec5SDimitry Andric { "VCFeature" , COFF::IMAGE_DEBUG_TYPE_VC_FEATURE }, 531*0b57cec5SDimitry Andric { "POGO" , COFF::IMAGE_DEBUG_TYPE_POGO }, 532*0b57cec5SDimitry Andric { "ILTCG" , COFF::IMAGE_DEBUG_TYPE_ILTCG }, 533*0b57cec5SDimitry Andric { "MPX" , COFF::IMAGE_DEBUG_TYPE_MPX }, 534*0b57cec5SDimitry Andric { "Repro" , COFF::IMAGE_DEBUG_TYPE_REPRO }, 535*0b57cec5SDimitry Andric }; 536*0b57cec5SDimitry Andric 537*0b57cec5SDimitry Andric static const EnumEntry<COFF::WeakExternalCharacteristics> 538*0b57cec5SDimitry Andric WeakExternalCharacteristics[] = { 539*0b57cec5SDimitry Andric { "NoLibrary", COFF::IMAGE_WEAK_EXTERN_SEARCH_NOLIBRARY }, 540*0b57cec5SDimitry Andric { "Library" , COFF::IMAGE_WEAK_EXTERN_SEARCH_LIBRARY }, 541*0b57cec5SDimitry Andric { "Alias" , COFF::IMAGE_WEAK_EXTERN_SEARCH_ALIAS } 542*0b57cec5SDimitry Andric }; 543*0b57cec5SDimitry Andric 544*0b57cec5SDimitry Andric static const EnumEntry<uint32_t> SubSectionTypes[] = { 545*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_CLASS_ENT(DebugSubsectionKind, Symbols), 546*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_CLASS_ENT(DebugSubsectionKind, Lines), 547*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_CLASS_ENT(DebugSubsectionKind, StringTable), 548*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_CLASS_ENT(DebugSubsectionKind, FileChecksums), 549*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_CLASS_ENT(DebugSubsectionKind, FrameData), 550*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_CLASS_ENT(DebugSubsectionKind, InlineeLines), 551*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_CLASS_ENT(DebugSubsectionKind, CrossScopeImports), 552*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_CLASS_ENT(DebugSubsectionKind, CrossScopeExports), 553*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_CLASS_ENT(DebugSubsectionKind, ILLines), 554*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_CLASS_ENT(DebugSubsectionKind, FuncMDTokenMap), 555*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_CLASS_ENT(DebugSubsectionKind, TypeMDTokenMap), 556*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_CLASS_ENT(DebugSubsectionKind, MergedAssemblyInput), 557*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_CLASS_ENT(DebugSubsectionKind, CoffSymbolRVA), 558*0b57cec5SDimitry Andric }; 559*0b57cec5SDimitry Andric 560*0b57cec5SDimitry Andric static const EnumEntry<uint32_t> FrameDataFlags[] = { 561*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(FrameData, HasSEH), 562*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(FrameData, HasEH), 563*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_ENT(FrameData, IsFunctionStart), 564*0b57cec5SDimitry Andric }; 565*0b57cec5SDimitry Andric 566*0b57cec5SDimitry Andric static const EnumEntry<uint8_t> FileChecksumKindNames[] = { 567*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_CLASS_ENT(FileChecksumKind, None), 568*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_CLASS_ENT(FileChecksumKind, MD5), 569*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_CLASS_ENT(FileChecksumKind, SHA1), 570*0b57cec5SDimitry Andric LLVM_READOBJ_ENUM_CLASS_ENT(FileChecksumKind, SHA256), 571*0b57cec5SDimitry Andric }; 572*0b57cec5SDimitry Andric 573*0b57cec5SDimitry Andric template <typename T> 574*0b57cec5SDimitry Andric static std::error_code getSymbolAuxData(const COFFObjectFile *Obj, 575*0b57cec5SDimitry Andric COFFSymbolRef Symbol, 576*0b57cec5SDimitry Andric uint8_t AuxSymbolIdx, const T *&Aux) { 577*0b57cec5SDimitry Andric ArrayRef<uint8_t> AuxData = Obj->getSymbolAuxData(Symbol); 578*0b57cec5SDimitry Andric AuxData = AuxData.slice(AuxSymbolIdx * Obj->getSymbolTableEntrySize()); 579*0b57cec5SDimitry Andric Aux = reinterpret_cast<const T*>(AuxData.data()); 580*0b57cec5SDimitry Andric return readobj_error::success; 581*0b57cec5SDimitry Andric } 582*0b57cec5SDimitry Andric 583*0b57cec5SDimitry Andric void COFFDumper::cacheRelocations() { 584*0b57cec5SDimitry Andric if (RelocCached) 585*0b57cec5SDimitry Andric return; 586*0b57cec5SDimitry Andric RelocCached = true; 587*0b57cec5SDimitry Andric 588*0b57cec5SDimitry Andric for (const SectionRef &S : Obj->sections()) { 589*0b57cec5SDimitry Andric const coff_section *Section = Obj->getCOFFSection(S); 590*0b57cec5SDimitry Andric 591*0b57cec5SDimitry Andric for (const RelocationRef &Reloc : S.relocations()) 592*0b57cec5SDimitry Andric RelocMap[Section].push_back(Reloc); 593*0b57cec5SDimitry Andric 594*0b57cec5SDimitry Andric // Sort relocations by address. 595*0b57cec5SDimitry Andric llvm::sort(RelocMap[Section], [](RelocationRef L, RelocationRef R) { 596*0b57cec5SDimitry Andric return L.getOffset() < R.getOffset(); 597*0b57cec5SDimitry Andric }); 598*0b57cec5SDimitry Andric } 599*0b57cec5SDimitry Andric } 600*0b57cec5SDimitry Andric 601*0b57cec5SDimitry Andric void COFFDumper::printDataDirectory(uint32_t Index, 602*0b57cec5SDimitry Andric const std::string &FieldName) { 603*0b57cec5SDimitry Andric const data_directory *Data; 604*0b57cec5SDimitry Andric if (Obj->getDataDirectory(Index, Data)) 605*0b57cec5SDimitry Andric return; 606*0b57cec5SDimitry Andric W.printHex(FieldName + "RVA", Data->RelativeVirtualAddress); 607*0b57cec5SDimitry Andric W.printHex(FieldName + "Size", Data->Size); 608*0b57cec5SDimitry Andric } 609*0b57cec5SDimitry Andric 610*0b57cec5SDimitry Andric void COFFDumper::printFileHeaders() { 611*0b57cec5SDimitry Andric time_t TDS = Obj->getTimeDateStamp(); 612*0b57cec5SDimitry Andric char FormattedTime[20] = { }; 613*0b57cec5SDimitry Andric strftime(FormattedTime, 20, "%Y-%m-%d %H:%M:%S", gmtime(&TDS)); 614*0b57cec5SDimitry Andric 615*0b57cec5SDimitry Andric { 616*0b57cec5SDimitry Andric DictScope D(W, "ImageFileHeader"); 617*0b57cec5SDimitry Andric W.printEnum ("Machine", Obj->getMachine(), 618*0b57cec5SDimitry Andric makeArrayRef(ImageFileMachineType)); 619*0b57cec5SDimitry Andric W.printNumber("SectionCount", Obj->getNumberOfSections()); 620*0b57cec5SDimitry Andric W.printHex ("TimeDateStamp", FormattedTime, Obj->getTimeDateStamp()); 621*0b57cec5SDimitry Andric W.printHex ("PointerToSymbolTable", Obj->getPointerToSymbolTable()); 622*0b57cec5SDimitry Andric W.printNumber("SymbolCount", Obj->getNumberOfSymbols()); 623*0b57cec5SDimitry Andric W.printNumber("OptionalHeaderSize", Obj->getSizeOfOptionalHeader()); 624*0b57cec5SDimitry Andric W.printFlags ("Characteristics", Obj->getCharacteristics(), 625*0b57cec5SDimitry Andric makeArrayRef(ImageFileCharacteristics)); 626*0b57cec5SDimitry Andric } 627*0b57cec5SDimitry Andric 628*0b57cec5SDimitry Andric // Print PE header. This header does not exist if this is an object file and 629*0b57cec5SDimitry Andric // not an executable. 630*0b57cec5SDimitry Andric const pe32_header *PEHeader = nullptr; 631*0b57cec5SDimitry Andric error(Obj->getPE32Header(PEHeader)); 632*0b57cec5SDimitry Andric if (PEHeader) 633*0b57cec5SDimitry Andric printPEHeader<pe32_header>(PEHeader); 634*0b57cec5SDimitry Andric 635*0b57cec5SDimitry Andric const pe32plus_header *PEPlusHeader = nullptr; 636*0b57cec5SDimitry Andric error(Obj->getPE32PlusHeader(PEPlusHeader)); 637*0b57cec5SDimitry Andric if (PEPlusHeader) 638*0b57cec5SDimitry Andric printPEHeader<pe32plus_header>(PEPlusHeader); 639*0b57cec5SDimitry Andric 640*0b57cec5SDimitry Andric if (const dos_header *DH = Obj->getDOSHeader()) 641*0b57cec5SDimitry Andric printDOSHeader(DH); 642*0b57cec5SDimitry Andric } 643*0b57cec5SDimitry Andric 644*0b57cec5SDimitry Andric void COFFDumper::printDOSHeader(const dos_header *DH) { 645*0b57cec5SDimitry Andric DictScope D(W, "DOSHeader"); 646*0b57cec5SDimitry Andric W.printString("Magic", StringRef(DH->Magic, sizeof(DH->Magic))); 647*0b57cec5SDimitry Andric W.printNumber("UsedBytesInTheLastPage", DH->UsedBytesInTheLastPage); 648*0b57cec5SDimitry Andric W.printNumber("FileSizeInPages", DH->FileSizeInPages); 649*0b57cec5SDimitry Andric W.printNumber("NumberOfRelocationItems", DH->NumberOfRelocationItems); 650*0b57cec5SDimitry Andric W.printNumber("HeaderSizeInParagraphs", DH->HeaderSizeInParagraphs); 651*0b57cec5SDimitry Andric W.printNumber("MinimumExtraParagraphs", DH->MinimumExtraParagraphs); 652*0b57cec5SDimitry Andric W.printNumber("MaximumExtraParagraphs", DH->MaximumExtraParagraphs); 653*0b57cec5SDimitry Andric W.printNumber("InitialRelativeSS", DH->InitialRelativeSS); 654*0b57cec5SDimitry Andric W.printNumber("InitialSP", DH->InitialSP); 655*0b57cec5SDimitry Andric W.printNumber("Checksum", DH->Checksum); 656*0b57cec5SDimitry Andric W.printNumber("InitialIP", DH->InitialIP); 657*0b57cec5SDimitry Andric W.printNumber("InitialRelativeCS", DH->InitialRelativeCS); 658*0b57cec5SDimitry Andric W.printNumber("AddressOfRelocationTable", DH->AddressOfRelocationTable); 659*0b57cec5SDimitry Andric W.printNumber("OverlayNumber", DH->OverlayNumber); 660*0b57cec5SDimitry Andric W.printNumber("OEMid", DH->OEMid); 661*0b57cec5SDimitry Andric W.printNumber("OEMinfo", DH->OEMinfo); 662*0b57cec5SDimitry Andric W.printNumber("AddressOfNewExeHeader", DH->AddressOfNewExeHeader); 663*0b57cec5SDimitry Andric } 664*0b57cec5SDimitry Andric 665*0b57cec5SDimitry Andric template <class PEHeader> 666*0b57cec5SDimitry Andric void COFFDumper::printPEHeader(const PEHeader *Hdr) { 667*0b57cec5SDimitry Andric DictScope D(W, "ImageOptionalHeader"); 668*0b57cec5SDimitry Andric W.printHex ("Magic", Hdr->Magic); 669*0b57cec5SDimitry Andric W.printNumber("MajorLinkerVersion", Hdr->MajorLinkerVersion); 670*0b57cec5SDimitry Andric W.printNumber("MinorLinkerVersion", Hdr->MinorLinkerVersion); 671*0b57cec5SDimitry Andric W.printNumber("SizeOfCode", Hdr->SizeOfCode); 672*0b57cec5SDimitry Andric W.printNumber("SizeOfInitializedData", Hdr->SizeOfInitializedData); 673*0b57cec5SDimitry Andric W.printNumber("SizeOfUninitializedData", Hdr->SizeOfUninitializedData); 674*0b57cec5SDimitry Andric W.printHex ("AddressOfEntryPoint", Hdr->AddressOfEntryPoint); 675*0b57cec5SDimitry Andric W.printHex ("BaseOfCode", Hdr->BaseOfCode); 676*0b57cec5SDimitry Andric printBaseOfDataField(Hdr); 677*0b57cec5SDimitry Andric W.printHex ("ImageBase", Hdr->ImageBase); 678*0b57cec5SDimitry Andric W.printNumber("SectionAlignment", Hdr->SectionAlignment); 679*0b57cec5SDimitry Andric W.printNumber("FileAlignment", Hdr->FileAlignment); 680*0b57cec5SDimitry Andric W.printNumber("MajorOperatingSystemVersion", 681*0b57cec5SDimitry Andric Hdr->MajorOperatingSystemVersion); 682*0b57cec5SDimitry Andric W.printNumber("MinorOperatingSystemVersion", 683*0b57cec5SDimitry Andric Hdr->MinorOperatingSystemVersion); 684*0b57cec5SDimitry Andric W.printNumber("MajorImageVersion", Hdr->MajorImageVersion); 685*0b57cec5SDimitry Andric W.printNumber("MinorImageVersion", Hdr->MinorImageVersion); 686*0b57cec5SDimitry Andric W.printNumber("MajorSubsystemVersion", Hdr->MajorSubsystemVersion); 687*0b57cec5SDimitry Andric W.printNumber("MinorSubsystemVersion", Hdr->MinorSubsystemVersion); 688*0b57cec5SDimitry Andric W.printNumber("SizeOfImage", Hdr->SizeOfImage); 689*0b57cec5SDimitry Andric W.printNumber("SizeOfHeaders", Hdr->SizeOfHeaders); 690*0b57cec5SDimitry Andric W.printEnum ("Subsystem", Hdr->Subsystem, makeArrayRef(PEWindowsSubsystem)); 691*0b57cec5SDimitry Andric W.printFlags ("Characteristics", Hdr->DLLCharacteristics, 692*0b57cec5SDimitry Andric makeArrayRef(PEDLLCharacteristics)); 693*0b57cec5SDimitry Andric W.printNumber("SizeOfStackReserve", Hdr->SizeOfStackReserve); 694*0b57cec5SDimitry Andric W.printNumber("SizeOfStackCommit", Hdr->SizeOfStackCommit); 695*0b57cec5SDimitry Andric W.printNumber("SizeOfHeapReserve", Hdr->SizeOfHeapReserve); 696*0b57cec5SDimitry Andric W.printNumber("SizeOfHeapCommit", Hdr->SizeOfHeapCommit); 697*0b57cec5SDimitry Andric W.printNumber("NumberOfRvaAndSize", Hdr->NumberOfRvaAndSize); 698*0b57cec5SDimitry Andric 699*0b57cec5SDimitry Andric if (Hdr->NumberOfRvaAndSize > 0) { 700*0b57cec5SDimitry Andric DictScope D(W, "DataDirectory"); 701*0b57cec5SDimitry Andric static const char * const directory[] = { 702*0b57cec5SDimitry Andric "ExportTable", "ImportTable", "ResourceTable", "ExceptionTable", 703*0b57cec5SDimitry Andric "CertificateTable", "BaseRelocationTable", "Debug", "Architecture", 704*0b57cec5SDimitry Andric "GlobalPtr", "TLSTable", "LoadConfigTable", "BoundImport", "IAT", 705*0b57cec5SDimitry Andric "DelayImportDescriptor", "CLRRuntimeHeader", "Reserved" 706*0b57cec5SDimitry Andric }; 707*0b57cec5SDimitry Andric 708*0b57cec5SDimitry Andric for (uint32_t i = 0; i < Hdr->NumberOfRvaAndSize; ++i) 709*0b57cec5SDimitry Andric printDataDirectory(i, directory[i]); 710*0b57cec5SDimitry Andric } 711*0b57cec5SDimitry Andric } 712*0b57cec5SDimitry Andric 713*0b57cec5SDimitry Andric void COFFDumper::printCOFFDebugDirectory() { 714*0b57cec5SDimitry Andric ListScope LS(W, "DebugDirectory"); 715*0b57cec5SDimitry Andric for (const debug_directory &D : Obj->debug_directories()) { 716*0b57cec5SDimitry Andric char FormattedTime[20] = {}; 717*0b57cec5SDimitry Andric time_t TDS = D.TimeDateStamp; 718*0b57cec5SDimitry Andric strftime(FormattedTime, 20, "%Y-%m-%d %H:%M:%S", gmtime(&TDS)); 719*0b57cec5SDimitry Andric DictScope S(W, "DebugEntry"); 720*0b57cec5SDimitry Andric W.printHex("Characteristics", D.Characteristics); 721*0b57cec5SDimitry Andric W.printHex("TimeDateStamp", FormattedTime, D.TimeDateStamp); 722*0b57cec5SDimitry Andric W.printHex("MajorVersion", D.MajorVersion); 723*0b57cec5SDimitry Andric W.printHex("MinorVersion", D.MinorVersion); 724*0b57cec5SDimitry Andric W.printEnum("Type", D.Type, makeArrayRef(ImageDebugType)); 725*0b57cec5SDimitry Andric W.printHex("SizeOfData", D.SizeOfData); 726*0b57cec5SDimitry Andric W.printHex("AddressOfRawData", D.AddressOfRawData); 727*0b57cec5SDimitry Andric W.printHex("PointerToRawData", D.PointerToRawData); 728*0b57cec5SDimitry Andric if (D.Type == COFF::IMAGE_DEBUG_TYPE_CODEVIEW) { 729*0b57cec5SDimitry Andric const codeview::DebugInfo *DebugInfo; 730*0b57cec5SDimitry Andric StringRef PDBFileName; 731*0b57cec5SDimitry Andric error(Obj->getDebugPDBInfo(&D, DebugInfo, PDBFileName)); 732*0b57cec5SDimitry Andric DictScope PDBScope(W, "PDBInfo"); 733*0b57cec5SDimitry Andric W.printHex("PDBSignature", DebugInfo->Signature.CVSignature); 734*0b57cec5SDimitry Andric if (DebugInfo->Signature.CVSignature == OMF::Signature::PDB70) { 735*0b57cec5SDimitry Andric W.printBinary("PDBGUID", makeArrayRef(DebugInfo->PDB70.Signature)); 736*0b57cec5SDimitry Andric W.printNumber("PDBAge", DebugInfo->PDB70.Age); 737*0b57cec5SDimitry Andric W.printString("PDBFileName", PDBFileName); 738*0b57cec5SDimitry Andric } 739*0b57cec5SDimitry Andric } else if (D.SizeOfData != 0) { 740*0b57cec5SDimitry Andric // FIXME: Type values of 12 and 13 are commonly observed but are not in 741*0b57cec5SDimitry Andric // the documented type enum. Figure out what they mean. 742*0b57cec5SDimitry Andric ArrayRef<uint8_t> RawData; 743*0b57cec5SDimitry Andric error( 744*0b57cec5SDimitry Andric Obj->getRvaAndSizeAsBytes(D.AddressOfRawData, D.SizeOfData, RawData)); 745*0b57cec5SDimitry Andric W.printBinaryBlock("RawData", RawData); 746*0b57cec5SDimitry Andric } 747*0b57cec5SDimitry Andric } 748*0b57cec5SDimitry Andric } 749*0b57cec5SDimitry Andric 750*0b57cec5SDimitry Andric void COFFDumper::printRVATable(uint64_t TableVA, uint64_t Count, 751*0b57cec5SDimitry Andric uint64_t EntrySize, PrintExtraCB PrintExtra) { 752*0b57cec5SDimitry Andric uintptr_t TableStart, TableEnd; 753*0b57cec5SDimitry Andric error(Obj->getVaPtr(TableVA, TableStart)); 754*0b57cec5SDimitry Andric error(Obj->getVaPtr(TableVA + Count * EntrySize - 1, TableEnd)); 755*0b57cec5SDimitry Andric TableEnd++; 756*0b57cec5SDimitry Andric for (uintptr_t I = TableStart; I < TableEnd; I += EntrySize) { 757*0b57cec5SDimitry Andric uint32_t RVA = *reinterpret_cast<const ulittle32_t *>(I); 758*0b57cec5SDimitry Andric raw_ostream &OS = W.startLine(); 759*0b57cec5SDimitry Andric OS << W.hex(Obj->getImageBase() + RVA); 760*0b57cec5SDimitry Andric if (PrintExtra) 761*0b57cec5SDimitry Andric PrintExtra(OS, reinterpret_cast<const uint8_t *>(I)); 762*0b57cec5SDimitry Andric OS << '\n'; 763*0b57cec5SDimitry Andric } 764*0b57cec5SDimitry Andric } 765*0b57cec5SDimitry Andric 766*0b57cec5SDimitry Andric void COFFDumper::printCOFFLoadConfig() { 767*0b57cec5SDimitry Andric LoadConfigTables Tables; 768*0b57cec5SDimitry Andric if (Obj->is64()) 769*0b57cec5SDimitry Andric printCOFFLoadConfig(Obj->getLoadConfig64(), Tables); 770*0b57cec5SDimitry Andric else 771*0b57cec5SDimitry Andric printCOFFLoadConfig(Obj->getLoadConfig32(), Tables); 772*0b57cec5SDimitry Andric 773*0b57cec5SDimitry Andric if (Tables.SEHTableVA) { 774*0b57cec5SDimitry Andric ListScope LS(W, "SEHTable"); 775*0b57cec5SDimitry Andric printRVATable(Tables.SEHTableVA, Tables.SEHTableCount, 4); 776*0b57cec5SDimitry Andric } 777*0b57cec5SDimitry Andric 778*0b57cec5SDimitry Andric if (Tables.GuardFidTableVA) { 779*0b57cec5SDimitry Andric ListScope LS(W, "GuardFidTable"); 780*0b57cec5SDimitry Andric if (Tables.GuardFlags & uint32_t(coff_guard_flags::FidTableHasFlags)) { 781*0b57cec5SDimitry Andric auto PrintGuardFlags = [](raw_ostream &OS, const uint8_t *Entry) { 782*0b57cec5SDimitry Andric uint8_t Flags = *reinterpret_cast<const uint8_t *>(Entry + 4); 783*0b57cec5SDimitry Andric if (Flags) 784*0b57cec5SDimitry Andric OS << " flags " << utohexstr(Flags); 785*0b57cec5SDimitry Andric }; 786*0b57cec5SDimitry Andric printRVATable(Tables.GuardFidTableVA, Tables.GuardFidTableCount, 5, 787*0b57cec5SDimitry Andric PrintGuardFlags); 788*0b57cec5SDimitry Andric } else { 789*0b57cec5SDimitry Andric printRVATable(Tables.GuardFidTableVA, Tables.GuardFidTableCount, 4); 790*0b57cec5SDimitry Andric } 791*0b57cec5SDimitry Andric } 792*0b57cec5SDimitry Andric 793*0b57cec5SDimitry Andric if (Tables.GuardLJmpTableVA) { 794*0b57cec5SDimitry Andric ListScope LS(W, "GuardLJmpTable"); 795*0b57cec5SDimitry Andric printRVATable(Tables.GuardLJmpTableVA, Tables.GuardLJmpTableCount, 4); 796*0b57cec5SDimitry Andric } 797*0b57cec5SDimitry Andric } 798*0b57cec5SDimitry Andric 799*0b57cec5SDimitry Andric template <typename T> 800*0b57cec5SDimitry Andric void COFFDumper::printCOFFLoadConfig(const T *Conf, LoadConfigTables &Tables) { 801*0b57cec5SDimitry Andric if (!Conf) 802*0b57cec5SDimitry Andric return; 803*0b57cec5SDimitry Andric 804*0b57cec5SDimitry Andric ListScope LS(W, "LoadConfig"); 805*0b57cec5SDimitry Andric char FormattedTime[20] = {}; 806*0b57cec5SDimitry Andric time_t TDS = Conf->TimeDateStamp; 807*0b57cec5SDimitry Andric strftime(FormattedTime, 20, "%Y-%m-%d %H:%M:%S", gmtime(&TDS)); 808*0b57cec5SDimitry Andric W.printHex("Size", Conf->Size); 809*0b57cec5SDimitry Andric 810*0b57cec5SDimitry Andric // Print everything before SecurityCookie. The vast majority of images today 811*0b57cec5SDimitry Andric // have all these fields. 812*0b57cec5SDimitry Andric if (Conf->Size < offsetof(T, SEHandlerTable)) 813*0b57cec5SDimitry Andric return; 814*0b57cec5SDimitry Andric W.printHex("TimeDateStamp", FormattedTime, TDS); 815*0b57cec5SDimitry Andric W.printHex("MajorVersion", Conf->MajorVersion); 816*0b57cec5SDimitry Andric W.printHex("MinorVersion", Conf->MinorVersion); 817*0b57cec5SDimitry Andric W.printHex("GlobalFlagsClear", Conf->GlobalFlagsClear); 818*0b57cec5SDimitry Andric W.printHex("GlobalFlagsSet", Conf->GlobalFlagsSet); 819*0b57cec5SDimitry Andric W.printHex("CriticalSectionDefaultTimeout", 820*0b57cec5SDimitry Andric Conf->CriticalSectionDefaultTimeout); 821*0b57cec5SDimitry Andric W.printHex("DeCommitFreeBlockThreshold", Conf->DeCommitFreeBlockThreshold); 822*0b57cec5SDimitry Andric W.printHex("DeCommitTotalFreeThreshold", Conf->DeCommitTotalFreeThreshold); 823*0b57cec5SDimitry Andric W.printHex("LockPrefixTable", Conf->LockPrefixTable); 824*0b57cec5SDimitry Andric W.printHex("MaximumAllocationSize", Conf->MaximumAllocationSize); 825*0b57cec5SDimitry Andric W.printHex("VirtualMemoryThreshold", Conf->VirtualMemoryThreshold); 826*0b57cec5SDimitry Andric W.printHex("ProcessHeapFlags", Conf->ProcessHeapFlags); 827*0b57cec5SDimitry Andric W.printHex("ProcessAffinityMask", Conf->ProcessAffinityMask); 828*0b57cec5SDimitry Andric W.printHex("CSDVersion", Conf->CSDVersion); 829*0b57cec5SDimitry Andric W.printHex("DependentLoadFlags", Conf->DependentLoadFlags); 830*0b57cec5SDimitry Andric W.printHex("EditList", Conf->EditList); 831*0b57cec5SDimitry Andric W.printHex("SecurityCookie", Conf->SecurityCookie); 832*0b57cec5SDimitry Andric 833*0b57cec5SDimitry Andric // Print the safe SEH table if present. 834*0b57cec5SDimitry Andric if (Conf->Size < offsetof(coff_load_configuration32, GuardCFCheckFunction)) 835*0b57cec5SDimitry Andric return; 836*0b57cec5SDimitry Andric W.printHex("SEHandlerTable", Conf->SEHandlerTable); 837*0b57cec5SDimitry Andric W.printNumber("SEHandlerCount", Conf->SEHandlerCount); 838*0b57cec5SDimitry Andric 839*0b57cec5SDimitry Andric Tables.SEHTableVA = Conf->SEHandlerTable; 840*0b57cec5SDimitry Andric Tables.SEHTableCount = Conf->SEHandlerCount; 841*0b57cec5SDimitry Andric 842*0b57cec5SDimitry Andric // Print everything before CodeIntegrity. (2015) 843*0b57cec5SDimitry Andric if (Conf->Size < offsetof(T, CodeIntegrity)) 844*0b57cec5SDimitry Andric return; 845*0b57cec5SDimitry Andric W.printHex("GuardCFCheckFunction", Conf->GuardCFCheckFunction); 846*0b57cec5SDimitry Andric W.printHex("GuardCFCheckDispatch", Conf->GuardCFCheckDispatch); 847*0b57cec5SDimitry Andric W.printHex("GuardCFFunctionTable", Conf->GuardCFFunctionTable); 848*0b57cec5SDimitry Andric W.printNumber("GuardCFFunctionCount", Conf->GuardCFFunctionCount); 849*0b57cec5SDimitry Andric W.printHex("GuardFlags", Conf->GuardFlags); 850*0b57cec5SDimitry Andric 851*0b57cec5SDimitry Andric Tables.GuardFidTableVA = Conf->GuardCFFunctionTable; 852*0b57cec5SDimitry Andric Tables.GuardFidTableCount = Conf->GuardCFFunctionCount; 853*0b57cec5SDimitry Andric Tables.GuardFlags = Conf->GuardFlags; 854*0b57cec5SDimitry Andric 855*0b57cec5SDimitry Andric // Print the rest. (2017) 856*0b57cec5SDimitry Andric if (Conf->Size < sizeof(T)) 857*0b57cec5SDimitry Andric return; 858*0b57cec5SDimitry Andric W.printHex("GuardAddressTakenIatEntryTable", 859*0b57cec5SDimitry Andric Conf->GuardAddressTakenIatEntryTable); 860*0b57cec5SDimitry Andric W.printNumber("GuardAddressTakenIatEntryCount", 861*0b57cec5SDimitry Andric Conf->GuardAddressTakenIatEntryCount); 862*0b57cec5SDimitry Andric W.printHex("GuardLongJumpTargetTable", Conf->GuardLongJumpTargetTable); 863*0b57cec5SDimitry Andric W.printNumber("GuardLongJumpTargetCount", Conf->GuardLongJumpTargetCount); 864*0b57cec5SDimitry Andric W.printHex("DynamicValueRelocTable", Conf->DynamicValueRelocTable); 865*0b57cec5SDimitry Andric W.printHex("CHPEMetadataPointer", Conf->CHPEMetadataPointer); 866*0b57cec5SDimitry Andric W.printHex("GuardRFFailureRoutine", Conf->GuardRFFailureRoutine); 867*0b57cec5SDimitry Andric W.printHex("GuardRFFailureRoutineFunctionPointer", 868*0b57cec5SDimitry Andric Conf->GuardRFFailureRoutineFunctionPointer); 869*0b57cec5SDimitry Andric W.printHex("DynamicValueRelocTableOffset", 870*0b57cec5SDimitry Andric Conf->DynamicValueRelocTableOffset); 871*0b57cec5SDimitry Andric W.printNumber("DynamicValueRelocTableSection", 872*0b57cec5SDimitry Andric Conf->DynamicValueRelocTableSection); 873*0b57cec5SDimitry Andric W.printHex("GuardRFVerifyStackPointerFunctionPointer", 874*0b57cec5SDimitry Andric Conf->GuardRFVerifyStackPointerFunctionPointer); 875*0b57cec5SDimitry Andric W.printHex("HotPatchTableOffset", Conf->HotPatchTableOffset); 876*0b57cec5SDimitry Andric 877*0b57cec5SDimitry Andric Tables.GuardLJmpTableVA = Conf->GuardLongJumpTargetTable; 878*0b57cec5SDimitry Andric Tables.GuardLJmpTableCount = Conf->GuardLongJumpTargetCount; 879*0b57cec5SDimitry Andric } 880*0b57cec5SDimitry Andric 881*0b57cec5SDimitry Andric void COFFDumper::printBaseOfDataField(const pe32_header *Hdr) { 882*0b57cec5SDimitry Andric W.printHex("BaseOfData", Hdr->BaseOfData); 883*0b57cec5SDimitry Andric } 884*0b57cec5SDimitry Andric 885*0b57cec5SDimitry Andric void COFFDumper::printBaseOfDataField(const pe32plus_header *) {} 886*0b57cec5SDimitry Andric 887*0b57cec5SDimitry Andric void COFFDumper::printCodeViewDebugInfo() { 888*0b57cec5SDimitry Andric // Print types first to build CVUDTNames, then print symbols. 889*0b57cec5SDimitry Andric for (const SectionRef &S : Obj->sections()) { 890*0b57cec5SDimitry Andric StringRef SectionName; 891*0b57cec5SDimitry Andric error(S.getName(SectionName)); 892*0b57cec5SDimitry Andric // .debug$T is a standard CodeView type section, while .debug$P is the same 893*0b57cec5SDimitry Andric // format but used for MSVC precompiled header object files. 894*0b57cec5SDimitry Andric if (SectionName == ".debug$T" || SectionName == ".debug$P") 895*0b57cec5SDimitry Andric printCodeViewTypeSection(SectionName, S); 896*0b57cec5SDimitry Andric } 897*0b57cec5SDimitry Andric for (const SectionRef &S : Obj->sections()) { 898*0b57cec5SDimitry Andric StringRef SectionName; 899*0b57cec5SDimitry Andric error(S.getName(SectionName)); 900*0b57cec5SDimitry Andric if (SectionName == ".debug$S") 901*0b57cec5SDimitry Andric printCodeViewSymbolSection(SectionName, S); 902*0b57cec5SDimitry Andric } 903*0b57cec5SDimitry Andric } 904*0b57cec5SDimitry Andric 905*0b57cec5SDimitry Andric void COFFDumper::initializeFileAndStringTables(BinaryStreamReader &Reader) { 906*0b57cec5SDimitry Andric while (Reader.bytesRemaining() > 0 && 907*0b57cec5SDimitry Andric (!CVFileChecksumTable.valid() || !CVStringTable.valid())) { 908*0b57cec5SDimitry Andric // The section consists of a number of subsection in the following format: 909*0b57cec5SDimitry Andric // |SubSectionType|SubSectionSize|Contents...| 910*0b57cec5SDimitry Andric uint32_t SubType, SubSectionSize; 911*0b57cec5SDimitry Andric error(Reader.readInteger(SubType)); 912*0b57cec5SDimitry Andric error(Reader.readInteger(SubSectionSize)); 913*0b57cec5SDimitry Andric 914*0b57cec5SDimitry Andric StringRef Contents; 915*0b57cec5SDimitry Andric error(Reader.readFixedString(Contents, SubSectionSize)); 916*0b57cec5SDimitry Andric 917*0b57cec5SDimitry Andric BinaryStreamRef ST(Contents, support::little); 918*0b57cec5SDimitry Andric switch (DebugSubsectionKind(SubType)) { 919*0b57cec5SDimitry Andric case DebugSubsectionKind::FileChecksums: 920*0b57cec5SDimitry Andric error(CVFileChecksumTable.initialize(ST)); 921*0b57cec5SDimitry Andric break; 922*0b57cec5SDimitry Andric case DebugSubsectionKind::StringTable: 923*0b57cec5SDimitry Andric error(CVStringTable.initialize(ST)); 924*0b57cec5SDimitry Andric break; 925*0b57cec5SDimitry Andric default: 926*0b57cec5SDimitry Andric break; 927*0b57cec5SDimitry Andric } 928*0b57cec5SDimitry Andric 929*0b57cec5SDimitry Andric uint32_t PaddedSize = alignTo(SubSectionSize, 4); 930*0b57cec5SDimitry Andric error(Reader.skip(PaddedSize - SubSectionSize)); 931*0b57cec5SDimitry Andric } 932*0b57cec5SDimitry Andric } 933*0b57cec5SDimitry Andric 934*0b57cec5SDimitry Andric void COFFDumper::printCodeViewSymbolSection(StringRef SectionName, 935*0b57cec5SDimitry Andric const SectionRef &Section) { 936*0b57cec5SDimitry Andric StringRef SectionContents = unwrapOrError(Section.getContents()); 937*0b57cec5SDimitry Andric StringRef Data = SectionContents; 938*0b57cec5SDimitry Andric 939*0b57cec5SDimitry Andric SmallVector<StringRef, 10> FunctionNames; 940*0b57cec5SDimitry Andric StringMap<StringRef> FunctionLineTables; 941*0b57cec5SDimitry Andric 942*0b57cec5SDimitry Andric ListScope D(W, "CodeViewDebugInfo"); 943*0b57cec5SDimitry Andric // Print the section to allow correlation with printSectionHeaders. 944*0b57cec5SDimitry Andric W.printNumber("Section", SectionName, Obj->getSectionID(Section)); 945*0b57cec5SDimitry Andric 946*0b57cec5SDimitry Andric uint32_t Magic; 947*0b57cec5SDimitry Andric error(consume(Data, Magic)); 948*0b57cec5SDimitry Andric W.printHex("Magic", Magic); 949*0b57cec5SDimitry Andric if (Magic != COFF::DEBUG_SECTION_MAGIC) 950*0b57cec5SDimitry Andric return error(object_error::parse_failed); 951*0b57cec5SDimitry Andric 952*0b57cec5SDimitry Andric BinaryStreamReader FSReader(Data, support::little); 953*0b57cec5SDimitry Andric initializeFileAndStringTables(FSReader); 954*0b57cec5SDimitry Andric 955*0b57cec5SDimitry Andric // TODO: Convert this over to using ModuleSubstreamVisitor. 956*0b57cec5SDimitry Andric while (!Data.empty()) { 957*0b57cec5SDimitry Andric // The section consists of a number of subsection in the following format: 958*0b57cec5SDimitry Andric // |SubSectionType|SubSectionSize|Contents...| 959*0b57cec5SDimitry Andric uint32_t SubType, SubSectionSize; 960*0b57cec5SDimitry Andric error(consume(Data, SubType)); 961*0b57cec5SDimitry Andric error(consume(Data, SubSectionSize)); 962*0b57cec5SDimitry Andric 963*0b57cec5SDimitry Andric ListScope S(W, "Subsection"); 964*0b57cec5SDimitry Andric // Dump the subsection as normal even if the ignore bit is set. 965*0b57cec5SDimitry Andric if (SubType & SubsectionIgnoreFlag) { 966*0b57cec5SDimitry Andric W.printHex("IgnoredSubsectionKind", SubType); 967*0b57cec5SDimitry Andric SubType &= ~SubsectionIgnoreFlag; 968*0b57cec5SDimitry Andric } 969*0b57cec5SDimitry Andric W.printEnum("SubSectionType", SubType, makeArrayRef(SubSectionTypes)); 970*0b57cec5SDimitry Andric W.printHex("SubSectionSize", SubSectionSize); 971*0b57cec5SDimitry Andric 972*0b57cec5SDimitry Andric // Get the contents of the subsection. 973*0b57cec5SDimitry Andric if (SubSectionSize > Data.size()) 974*0b57cec5SDimitry Andric return error(object_error::parse_failed); 975*0b57cec5SDimitry Andric StringRef Contents = Data.substr(0, SubSectionSize); 976*0b57cec5SDimitry Andric 977*0b57cec5SDimitry Andric // Add SubSectionSize to the current offset and align that offset to find 978*0b57cec5SDimitry Andric // the next subsection. 979*0b57cec5SDimitry Andric size_t SectionOffset = Data.data() - SectionContents.data(); 980*0b57cec5SDimitry Andric size_t NextOffset = SectionOffset + SubSectionSize; 981*0b57cec5SDimitry Andric NextOffset = alignTo(NextOffset, 4); 982*0b57cec5SDimitry Andric if (NextOffset > SectionContents.size()) 983*0b57cec5SDimitry Andric return error(object_error::parse_failed); 984*0b57cec5SDimitry Andric Data = SectionContents.drop_front(NextOffset); 985*0b57cec5SDimitry Andric 986*0b57cec5SDimitry Andric // Optionally print the subsection bytes in case our parsing gets confused 987*0b57cec5SDimitry Andric // later. 988*0b57cec5SDimitry Andric if (opts::CodeViewSubsectionBytes) 989*0b57cec5SDimitry Andric printBinaryBlockWithRelocs("SubSectionContents", Section, SectionContents, 990*0b57cec5SDimitry Andric Contents); 991*0b57cec5SDimitry Andric 992*0b57cec5SDimitry Andric switch (DebugSubsectionKind(SubType)) { 993*0b57cec5SDimitry Andric case DebugSubsectionKind::Symbols: 994*0b57cec5SDimitry Andric printCodeViewSymbolsSubsection(Contents, Section, SectionContents); 995*0b57cec5SDimitry Andric break; 996*0b57cec5SDimitry Andric 997*0b57cec5SDimitry Andric case DebugSubsectionKind::InlineeLines: 998*0b57cec5SDimitry Andric printCodeViewInlineeLines(Contents); 999*0b57cec5SDimitry Andric break; 1000*0b57cec5SDimitry Andric 1001*0b57cec5SDimitry Andric case DebugSubsectionKind::FileChecksums: 1002*0b57cec5SDimitry Andric printCodeViewFileChecksums(Contents); 1003*0b57cec5SDimitry Andric break; 1004*0b57cec5SDimitry Andric 1005*0b57cec5SDimitry Andric case DebugSubsectionKind::Lines: { 1006*0b57cec5SDimitry Andric // Holds a PC to file:line table. Some data to parse this subsection is 1007*0b57cec5SDimitry Andric // stored in the other subsections, so just check sanity and store the 1008*0b57cec5SDimitry Andric // pointers for deferred processing. 1009*0b57cec5SDimitry Andric 1010*0b57cec5SDimitry Andric if (SubSectionSize < 12) { 1011*0b57cec5SDimitry Andric // There should be at least three words to store two function 1012*0b57cec5SDimitry Andric // relocations and size of the code. 1013*0b57cec5SDimitry Andric error(object_error::parse_failed); 1014*0b57cec5SDimitry Andric return; 1015*0b57cec5SDimitry Andric } 1016*0b57cec5SDimitry Andric 1017*0b57cec5SDimitry Andric StringRef LinkageName; 1018*0b57cec5SDimitry Andric error(resolveSymbolName(Obj->getCOFFSection(Section), SectionOffset, 1019*0b57cec5SDimitry Andric LinkageName)); 1020*0b57cec5SDimitry Andric W.printString("LinkageName", LinkageName); 1021*0b57cec5SDimitry Andric if (FunctionLineTables.count(LinkageName) != 0) { 1022*0b57cec5SDimitry Andric // Saw debug info for this function already? 1023*0b57cec5SDimitry Andric error(object_error::parse_failed); 1024*0b57cec5SDimitry Andric return; 1025*0b57cec5SDimitry Andric } 1026*0b57cec5SDimitry Andric 1027*0b57cec5SDimitry Andric FunctionLineTables[LinkageName] = Contents; 1028*0b57cec5SDimitry Andric FunctionNames.push_back(LinkageName); 1029*0b57cec5SDimitry Andric break; 1030*0b57cec5SDimitry Andric } 1031*0b57cec5SDimitry Andric case DebugSubsectionKind::FrameData: { 1032*0b57cec5SDimitry Andric // First four bytes is a relocation against the function. 1033*0b57cec5SDimitry Andric BinaryStreamReader SR(Contents, llvm::support::little); 1034*0b57cec5SDimitry Andric 1035*0b57cec5SDimitry Andric DebugFrameDataSubsectionRef FrameData; 1036*0b57cec5SDimitry Andric error(FrameData.initialize(SR)); 1037*0b57cec5SDimitry Andric 1038*0b57cec5SDimitry Andric StringRef LinkageName; 1039*0b57cec5SDimitry Andric error(resolveSymbolName(Obj->getCOFFSection(Section), SectionContents, 1040*0b57cec5SDimitry Andric FrameData.getRelocPtr(), LinkageName)); 1041*0b57cec5SDimitry Andric W.printString("LinkageName", LinkageName); 1042*0b57cec5SDimitry Andric 1043*0b57cec5SDimitry Andric // To find the active frame description, search this array for the 1044*0b57cec5SDimitry Andric // smallest PC range that includes the current PC. 1045*0b57cec5SDimitry Andric for (const auto &FD : FrameData) { 1046*0b57cec5SDimitry Andric StringRef FrameFunc = error(CVStringTable.getString(FD.FrameFunc)); 1047*0b57cec5SDimitry Andric 1048*0b57cec5SDimitry Andric DictScope S(W, "FrameData"); 1049*0b57cec5SDimitry Andric W.printHex("RvaStart", FD.RvaStart); 1050*0b57cec5SDimitry Andric W.printHex("CodeSize", FD.CodeSize); 1051*0b57cec5SDimitry Andric W.printHex("LocalSize", FD.LocalSize); 1052*0b57cec5SDimitry Andric W.printHex("ParamsSize", FD.ParamsSize); 1053*0b57cec5SDimitry Andric W.printHex("MaxStackSize", FD.MaxStackSize); 1054*0b57cec5SDimitry Andric W.printHex("PrologSize", FD.PrologSize); 1055*0b57cec5SDimitry Andric W.printHex("SavedRegsSize", FD.SavedRegsSize); 1056*0b57cec5SDimitry Andric W.printFlags("Flags", FD.Flags, makeArrayRef(FrameDataFlags)); 1057*0b57cec5SDimitry Andric 1058*0b57cec5SDimitry Andric // The FrameFunc string is a small RPN program. It can be broken up into 1059*0b57cec5SDimitry Andric // statements that end in the '=' operator, which assigns the value on 1060*0b57cec5SDimitry Andric // the top of the stack to the previously pushed variable. Variables can 1061*0b57cec5SDimitry Andric // be temporary values ($T0) or physical registers ($esp). Print each 1062*0b57cec5SDimitry Andric // assignment on its own line to make these programs easier to read. 1063*0b57cec5SDimitry Andric { 1064*0b57cec5SDimitry Andric ListScope FFS(W, "FrameFunc"); 1065*0b57cec5SDimitry Andric while (!FrameFunc.empty()) { 1066*0b57cec5SDimitry Andric size_t EqOrEnd = FrameFunc.find('='); 1067*0b57cec5SDimitry Andric if (EqOrEnd == StringRef::npos) 1068*0b57cec5SDimitry Andric EqOrEnd = FrameFunc.size(); 1069*0b57cec5SDimitry Andric else 1070*0b57cec5SDimitry Andric ++EqOrEnd; 1071*0b57cec5SDimitry Andric StringRef Stmt = FrameFunc.substr(0, EqOrEnd); 1072*0b57cec5SDimitry Andric W.printString(Stmt); 1073*0b57cec5SDimitry Andric FrameFunc = FrameFunc.drop_front(EqOrEnd).trim(); 1074*0b57cec5SDimitry Andric } 1075*0b57cec5SDimitry Andric } 1076*0b57cec5SDimitry Andric } 1077*0b57cec5SDimitry Andric break; 1078*0b57cec5SDimitry Andric } 1079*0b57cec5SDimitry Andric 1080*0b57cec5SDimitry Andric // Do nothing for unrecognized subsections. 1081*0b57cec5SDimitry Andric default: 1082*0b57cec5SDimitry Andric break; 1083*0b57cec5SDimitry Andric } 1084*0b57cec5SDimitry Andric W.flush(); 1085*0b57cec5SDimitry Andric } 1086*0b57cec5SDimitry Andric 1087*0b57cec5SDimitry Andric // Dump the line tables now that we've read all the subsections and know all 1088*0b57cec5SDimitry Andric // the required information. 1089*0b57cec5SDimitry Andric for (unsigned I = 0, E = FunctionNames.size(); I != E; ++I) { 1090*0b57cec5SDimitry Andric StringRef Name = FunctionNames[I]; 1091*0b57cec5SDimitry Andric ListScope S(W, "FunctionLineTable"); 1092*0b57cec5SDimitry Andric W.printString("LinkageName", Name); 1093*0b57cec5SDimitry Andric 1094*0b57cec5SDimitry Andric BinaryStreamReader Reader(FunctionLineTables[Name], support::little); 1095*0b57cec5SDimitry Andric 1096*0b57cec5SDimitry Andric DebugLinesSubsectionRef LineInfo; 1097*0b57cec5SDimitry Andric error(LineInfo.initialize(Reader)); 1098*0b57cec5SDimitry Andric 1099*0b57cec5SDimitry Andric W.printHex("Flags", LineInfo.header()->Flags); 1100*0b57cec5SDimitry Andric W.printHex("CodeSize", LineInfo.header()->CodeSize); 1101*0b57cec5SDimitry Andric for (const auto &Entry : LineInfo) { 1102*0b57cec5SDimitry Andric 1103*0b57cec5SDimitry Andric ListScope S(W, "FilenameSegment"); 1104*0b57cec5SDimitry Andric printFileNameForOffset("Filename", Entry.NameIndex); 1105*0b57cec5SDimitry Andric uint32_t ColumnIndex = 0; 1106*0b57cec5SDimitry Andric for (const auto &Line : Entry.LineNumbers) { 1107*0b57cec5SDimitry Andric if (Line.Offset >= LineInfo.header()->CodeSize) { 1108*0b57cec5SDimitry Andric error(object_error::parse_failed); 1109*0b57cec5SDimitry Andric return; 1110*0b57cec5SDimitry Andric } 1111*0b57cec5SDimitry Andric 1112*0b57cec5SDimitry Andric std::string PC = formatv("+{0:X}", uint32_t(Line.Offset)); 1113*0b57cec5SDimitry Andric ListScope PCScope(W, PC); 1114*0b57cec5SDimitry Andric codeview::LineInfo LI(Line.Flags); 1115*0b57cec5SDimitry Andric 1116*0b57cec5SDimitry Andric if (LI.isAlwaysStepInto()) 1117*0b57cec5SDimitry Andric W.printString("StepInto", StringRef("Always")); 1118*0b57cec5SDimitry Andric else if (LI.isNeverStepInto()) 1119*0b57cec5SDimitry Andric W.printString("StepInto", StringRef("Never")); 1120*0b57cec5SDimitry Andric else 1121*0b57cec5SDimitry Andric W.printNumber("LineNumberStart", LI.getStartLine()); 1122*0b57cec5SDimitry Andric W.printNumber("LineNumberEndDelta", LI.getLineDelta()); 1123*0b57cec5SDimitry Andric W.printBoolean("IsStatement", LI.isStatement()); 1124*0b57cec5SDimitry Andric if (LineInfo.hasColumnInfo()) { 1125*0b57cec5SDimitry Andric W.printNumber("ColStart", Entry.Columns[ColumnIndex].StartColumn); 1126*0b57cec5SDimitry Andric W.printNumber("ColEnd", Entry.Columns[ColumnIndex].EndColumn); 1127*0b57cec5SDimitry Andric ++ColumnIndex; 1128*0b57cec5SDimitry Andric } 1129*0b57cec5SDimitry Andric } 1130*0b57cec5SDimitry Andric } 1131*0b57cec5SDimitry Andric } 1132*0b57cec5SDimitry Andric } 1133*0b57cec5SDimitry Andric 1134*0b57cec5SDimitry Andric void COFFDumper::printCodeViewSymbolsSubsection(StringRef Subsection, 1135*0b57cec5SDimitry Andric const SectionRef &Section, 1136*0b57cec5SDimitry Andric StringRef SectionContents) { 1137*0b57cec5SDimitry Andric ArrayRef<uint8_t> BinaryData(Subsection.bytes_begin(), 1138*0b57cec5SDimitry Andric Subsection.bytes_end()); 1139*0b57cec5SDimitry Andric auto CODD = llvm::make_unique<COFFObjectDumpDelegate>(*this, Section, Obj, 1140*0b57cec5SDimitry Andric SectionContents); 1141*0b57cec5SDimitry Andric CVSymbolDumper CVSD(W, Types, CodeViewContainer::ObjectFile, std::move(CODD), 1142*0b57cec5SDimitry Andric CompilationCPUType, opts::CodeViewSubsectionBytes); 1143*0b57cec5SDimitry Andric CVSymbolArray Symbols; 1144*0b57cec5SDimitry Andric BinaryStreamReader Reader(BinaryData, llvm::support::little); 1145*0b57cec5SDimitry Andric if (auto EC = Reader.readArray(Symbols, Reader.getLength())) { 1146*0b57cec5SDimitry Andric consumeError(std::move(EC)); 1147*0b57cec5SDimitry Andric W.flush(); 1148*0b57cec5SDimitry Andric error(object_error::parse_failed); 1149*0b57cec5SDimitry Andric } 1150*0b57cec5SDimitry Andric 1151*0b57cec5SDimitry Andric if (auto EC = CVSD.dump(Symbols)) { 1152*0b57cec5SDimitry Andric W.flush(); 1153*0b57cec5SDimitry Andric error(std::move(EC)); 1154*0b57cec5SDimitry Andric } 1155*0b57cec5SDimitry Andric CompilationCPUType = CVSD.getCompilationCPUType(); 1156*0b57cec5SDimitry Andric W.flush(); 1157*0b57cec5SDimitry Andric } 1158*0b57cec5SDimitry Andric 1159*0b57cec5SDimitry Andric void COFFDumper::printCodeViewFileChecksums(StringRef Subsection) { 1160*0b57cec5SDimitry Andric BinaryStreamRef Stream(Subsection, llvm::support::little); 1161*0b57cec5SDimitry Andric DebugChecksumsSubsectionRef Checksums; 1162*0b57cec5SDimitry Andric error(Checksums.initialize(Stream)); 1163*0b57cec5SDimitry Andric 1164*0b57cec5SDimitry Andric for (auto &FC : Checksums) { 1165*0b57cec5SDimitry Andric DictScope S(W, "FileChecksum"); 1166*0b57cec5SDimitry Andric 1167*0b57cec5SDimitry Andric StringRef Filename = error(CVStringTable.getString(FC.FileNameOffset)); 1168*0b57cec5SDimitry Andric W.printHex("Filename", Filename, FC.FileNameOffset); 1169*0b57cec5SDimitry Andric W.printHex("ChecksumSize", FC.Checksum.size()); 1170*0b57cec5SDimitry Andric W.printEnum("ChecksumKind", uint8_t(FC.Kind), 1171*0b57cec5SDimitry Andric makeArrayRef(FileChecksumKindNames)); 1172*0b57cec5SDimitry Andric 1173*0b57cec5SDimitry Andric W.printBinary("ChecksumBytes", FC.Checksum); 1174*0b57cec5SDimitry Andric } 1175*0b57cec5SDimitry Andric } 1176*0b57cec5SDimitry Andric 1177*0b57cec5SDimitry Andric void COFFDumper::printCodeViewInlineeLines(StringRef Subsection) { 1178*0b57cec5SDimitry Andric BinaryStreamReader SR(Subsection, llvm::support::little); 1179*0b57cec5SDimitry Andric DebugInlineeLinesSubsectionRef Lines; 1180*0b57cec5SDimitry Andric error(Lines.initialize(SR)); 1181*0b57cec5SDimitry Andric 1182*0b57cec5SDimitry Andric for (auto &Line : Lines) { 1183*0b57cec5SDimitry Andric DictScope S(W, "InlineeSourceLine"); 1184*0b57cec5SDimitry Andric printTypeIndex("Inlinee", Line.Header->Inlinee); 1185*0b57cec5SDimitry Andric printFileNameForOffset("FileID", Line.Header->FileID); 1186*0b57cec5SDimitry Andric W.printNumber("SourceLineNum", Line.Header->SourceLineNum); 1187*0b57cec5SDimitry Andric 1188*0b57cec5SDimitry Andric if (Lines.hasExtraFiles()) { 1189*0b57cec5SDimitry Andric W.printNumber("ExtraFileCount", Line.ExtraFiles.size()); 1190*0b57cec5SDimitry Andric ListScope ExtraFiles(W, "ExtraFiles"); 1191*0b57cec5SDimitry Andric for (const auto &FID : Line.ExtraFiles) { 1192*0b57cec5SDimitry Andric printFileNameForOffset("FileID", FID); 1193*0b57cec5SDimitry Andric } 1194*0b57cec5SDimitry Andric } 1195*0b57cec5SDimitry Andric } 1196*0b57cec5SDimitry Andric } 1197*0b57cec5SDimitry Andric 1198*0b57cec5SDimitry Andric StringRef COFFDumper::getFileNameForFileOffset(uint32_t FileOffset) { 1199*0b57cec5SDimitry Andric // The file checksum subsection should precede all references to it. 1200*0b57cec5SDimitry Andric if (!CVFileChecksumTable.valid() || !CVStringTable.valid()) 1201*0b57cec5SDimitry Andric error(object_error::parse_failed); 1202*0b57cec5SDimitry Andric 1203*0b57cec5SDimitry Andric auto Iter = CVFileChecksumTable.getArray().at(FileOffset); 1204*0b57cec5SDimitry Andric 1205*0b57cec5SDimitry Andric // Check if the file checksum table offset is valid. 1206*0b57cec5SDimitry Andric if (Iter == CVFileChecksumTable.end()) 1207*0b57cec5SDimitry Andric error(object_error::parse_failed); 1208*0b57cec5SDimitry Andric 1209*0b57cec5SDimitry Andric return error(CVStringTable.getString(Iter->FileNameOffset)); 1210*0b57cec5SDimitry Andric } 1211*0b57cec5SDimitry Andric 1212*0b57cec5SDimitry Andric void COFFDumper::printFileNameForOffset(StringRef Label, uint32_t FileOffset) { 1213*0b57cec5SDimitry Andric W.printHex(Label, getFileNameForFileOffset(FileOffset), FileOffset); 1214*0b57cec5SDimitry Andric } 1215*0b57cec5SDimitry Andric 1216*0b57cec5SDimitry Andric void COFFDumper::mergeCodeViewTypes(MergingTypeTableBuilder &CVIDs, 1217*0b57cec5SDimitry Andric MergingTypeTableBuilder &CVTypes, 1218*0b57cec5SDimitry Andric GlobalTypeTableBuilder &GlobalCVIDs, 1219*0b57cec5SDimitry Andric GlobalTypeTableBuilder &GlobalCVTypes, 1220*0b57cec5SDimitry Andric bool GHash) { 1221*0b57cec5SDimitry Andric for (const SectionRef &S : Obj->sections()) { 1222*0b57cec5SDimitry Andric StringRef SectionName; 1223*0b57cec5SDimitry Andric error(S.getName(SectionName)); 1224*0b57cec5SDimitry Andric if (SectionName == ".debug$T") { 1225*0b57cec5SDimitry Andric StringRef Data = unwrapOrError(S.getContents()); 1226*0b57cec5SDimitry Andric uint32_t Magic; 1227*0b57cec5SDimitry Andric error(consume(Data, Magic)); 1228*0b57cec5SDimitry Andric if (Magic != 4) 1229*0b57cec5SDimitry Andric error(object_error::parse_failed); 1230*0b57cec5SDimitry Andric 1231*0b57cec5SDimitry Andric CVTypeArray Types; 1232*0b57cec5SDimitry Andric BinaryStreamReader Reader(Data, llvm::support::little); 1233*0b57cec5SDimitry Andric if (auto EC = Reader.readArray(Types, Reader.getLength())) { 1234*0b57cec5SDimitry Andric consumeError(std::move(EC)); 1235*0b57cec5SDimitry Andric W.flush(); 1236*0b57cec5SDimitry Andric error(object_error::parse_failed); 1237*0b57cec5SDimitry Andric } 1238*0b57cec5SDimitry Andric SmallVector<TypeIndex, 128> SourceToDest; 1239*0b57cec5SDimitry Andric Optional<uint32_t> PCHSignature; 1240*0b57cec5SDimitry Andric if (GHash) { 1241*0b57cec5SDimitry Andric std::vector<GloballyHashedType> Hashes = 1242*0b57cec5SDimitry Andric GloballyHashedType::hashTypes(Types); 1243*0b57cec5SDimitry Andric if (auto EC = 1244*0b57cec5SDimitry Andric mergeTypeAndIdRecords(GlobalCVIDs, GlobalCVTypes, SourceToDest, 1245*0b57cec5SDimitry Andric Types, Hashes, PCHSignature)) 1246*0b57cec5SDimitry Andric return error(std::move(EC)); 1247*0b57cec5SDimitry Andric } else { 1248*0b57cec5SDimitry Andric if (auto EC = mergeTypeAndIdRecords(CVIDs, CVTypes, SourceToDest, Types, 1249*0b57cec5SDimitry Andric PCHSignature)) 1250*0b57cec5SDimitry Andric return error(std::move(EC)); 1251*0b57cec5SDimitry Andric } 1252*0b57cec5SDimitry Andric } 1253*0b57cec5SDimitry Andric } 1254*0b57cec5SDimitry Andric } 1255*0b57cec5SDimitry Andric 1256*0b57cec5SDimitry Andric void COFFDumper::printCodeViewTypeSection(StringRef SectionName, 1257*0b57cec5SDimitry Andric const SectionRef &Section) { 1258*0b57cec5SDimitry Andric ListScope D(W, "CodeViewTypes"); 1259*0b57cec5SDimitry Andric W.printNumber("Section", SectionName, Obj->getSectionID(Section)); 1260*0b57cec5SDimitry Andric 1261*0b57cec5SDimitry Andric StringRef Data = unwrapOrError(Section.getContents()); 1262*0b57cec5SDimitry Andric if (opts::CodeViewSubsectionBytes) 1263*0b57cec5SDimitry Andric W.printBinaryBlock("Data", Data); 1264*0b57cec5SDimitry Andric 1265*0b57cec5SDimitry Andric uint32_t Magic; 1266*0b57cec5SDimitry Andric error(consume(Data, Magic)); 1267*0b57cec5SDimitry Andric W.printHex("Magic", Magic); 1268*0b57cec5SDimitry Andric if (Magic != COFF::DEBUG_SECTION_MAGIC) 1269*0b57cec5SDimitry Andric return error(object_error::parse_failed); 1270*0b57cec5SDimitry Andric 1271*0b57cec5SDimitry Andric Types.reset(Data, 100); 1272*0b57cec5SDimitry Andric 1273*0b57cec5SDimitry Andric TypeDumpVisitor TDV(Types, &W, opts::CodeViewSubsectionBytes); 1274*0b57cec5SDimitry Andric error(codeview::visitTypeStream(Types, TDV)); 1275*0b57cec5SDimitry Andric W.flush(); 1276*0b57cec5SDimitry Andric } 1277*0b57cec5SDimitry Andric 1278*0b57cec5SDimitry Andric void COFFDumper::printSectionHeaders() { 1279*0b57cec5SDimitry Andric ListScope SectionsD(W, "Sections"); 1280*0b57cec5SDimitry Andric int SectionNumber = 0; 1281*0b57cec5SDimitry Andric for (const SectionRef &Sec : Obj->sections()) { 1282*0b57cec5SDimitry Andric ++SectionNumber; 1283*0b57cec5SDimitry Andric const coff_section *Section = Obj->getCOFFSection(Sec); 1284*0b57cec5SDimitry Andric 1285*0b57cec5SDimitry Andric StringRef Name; 1286*0b57cec5SDimitry Andric error(Sec.getName(Name)); 1287*0b57cec5SDimitry Andric 1288*0b57cec5SDimitry Andric DictScope D(W, "Section"); 1289*0b57cec5SDimitry Andric W.printNumber("Number", SectionNumber); 1290*0b57cec5SDimitry Andric W.printBinary("Name", Name, Section->Name); 1291*0b57cec5SDimitry Andric W.printHex ("VirtualSize", Section->VirtualSize); 1292*0b57cec5SDimitry Andric W.printHex ("VirtualAddress", Section->VirtualAddress); 1293*0b57cec5SDimitry Andric W.printNumber("RawDataSize", Section->SizeOfRawData); 1294*0b57cec5SDimitry Andric W.printHex ("PointerToRawData", Section->PointerToRawData); 1295*0b57cec5SDimitry Andric W.printHex ("PointerToRelocations", Section->PointerToRelocations); 1296*0b57cec5SDimitry Andric W.printHex ("PointerToLineNumbers", Section->PointerToLinenumbers); 1297*0b57cec5SDimitry Andric W.printNumber("RelocationCount", Section->NumberOfRelocations); 1298*0b57cec5SDimitry Andric W.printNumber("LineNumberCount", Section->NumberOfLinenumbers); 1299*0b57cec5SDimitry Andric W.printFlags ("Characteristics", Section->Characteristics, 1300*0b57cec5SDimitry Andric makeArrayRef(ImageSectionCharacteristics), 1301*0b57cec5SDimitry Andric COFF::SectionCharacteristics(0x00F00000)); 1302*0b57cec5SDimitry Andric 1303*0b57cec5SDimitry Andric if (opts::SectionRelocations) { 1304*0b57cec5SDimitry Andric ListScope D(W, "Relocations"); 1305*0b57cec5SDimitry Andric for (const RelocationRef &Reloc : Sec.relocations()) 1306*0b57cec5SDimitry Andric printRelocation(Sec, Reloc); 1307*0b57cec5SDimitry Andric } 1308*0b57cec5SDimitry Andric 1309*0b57cec5SDimitry Andric if (opts::SectionSymbols) { 1310*0b57cec5SDimitry Andric ListScope D(W, "Symbols"); 1311*0b57cec5SDimitry Andric for (const SymbolRef &Symbol : Obj->symbols()) { 1312*0b57cec5SDimitry Andric if (!Sec.containsSymbol(Symbol)) 1313*0b57cec5SDimitry Andric continue; 1314*0b57cec5SDimitry Andric 1315*0b57cec5SDimitry Andric printSymbol(Symbol); 1316*0b57cec5SDimitry Andric } 1317*0b57cec5SDimitry Andric } 1318*0b57cec5SDimitry Andric 1319*0b57cec5SDimitry Andric if (opts::SectionData && 1320*0b57cec5SDimitry Andric !(Section->Characteristics & COFF::IMAGE_SCN_CNT_UNINITIALIZED_DATA)) { 1321*0b57cec5SDimitry Andric StringRef Data = unwrapOrError(Sec.getContents()); 1322*0b57cec5SDimitry Andric W.printBinaryBlock("SectionData", Data); 1323*0b57cec5SDimitry Andric } 1324*0b57cec5SDimitry Andric } 1325*0b57cec5SDimitry Andric } 1326*0b57cec5SDimitry Andric 1327*0b57cec5SDimitry Andric void COFFDumper::printRelocations() { 1328*0b57cec5SDimitry Andric ListScope D(W, "Relocations"); 1329*0b57cec5SDimitry Andric 1330*0b57cec5SDimitry Andric int SectionNumber = 0; 1331*0b57cec5SDimitry Andric for (const SectionRef &Section : Obj->sections()) { 1332*0b57cec5SDimitry Andric ++SectionNumber; 1333*0b57cec5SDimitry Andric StringRef Name; 1334*0b57cec5SDimitry Andric error(Section.getName(Name)); 1335*0b57cec5SDimitry Andric 1336*0b57cec5SDimitry Andric bool PrintedGroup = false; 1337*0b57cec5SDimitry Andric for (const RelocationRef &Reloc : Section.relocations()) { 1338*0b57cec5SDimitry Andric if (!PrintedGroup) { 1339*0b57cec5SDimitry Andric W.startLine() << "Section (" << SectionNumber << ") " << Name << " {\n"; 1340*0b57cec5SDimitry Andric W.indent(); 1341*0b57cec5SDimitry Andric PrintedGroup = true; 1342*0b57cec5SDimitry Andric } 1343*0b57cec5SDimitry Andric 1344*0b57cec5SDimitry Andric printRelocation(Section, Reloc); 1345*0b57cec5SDimitry Andric } 1346*0b57cec5SDimitry Andric 1347*0b57cec5SDimitry Andric if (PrintedGroup) { 1348*0b57cec5SDimitry Andric W.unindent(); 1349*0b57cec5SDimitry Andric W.startLine() << "}\n"; 1350*0b57cec5SDimitry Andric } 1351*0b57cec5SDimitry Andric } 1352*0b57cec5SDimitry Andric } 1353*0b57cec5SDimitry Andric 1354*0b57cec5SDimitry Andric void COFFDumper::printRelocation(const SectionRef &Section, 1355*0b57cec5SDimitry Andric const RelocationRef &Reloc, uint64_t Bias) { 1356*0b57cec5SDimitry Andric uint64_t Offset = Reloc.getOffset() - Bias; 1357*0b57cec5SDimitry Andric uint64_t RelocType = Reloc.getType(); 1358*0b57cec5SDimitry Andric SmallString<32> RelocName; 1359*0b57cec5SDimitry Andric StringRef SymbolName; 1360*0b57cec5SDimitry Andric Reloc.getTypeName(RelocName); 1361*0b57cec5SDimitry Andric symbol_iterator Symbol = Reloc.getSymbol(); 1362*0b57cec5SDimitry Andric int64_t SymbolIndex = -1; 1363*0b57cec5SDimitry Andric if (Symbol != Obj->symbol_end()) { 1364*0b57cec5SDimitry Andric Expected<StringRef> SymbolNameOrErr = Symbol->getName(); 1365*0b57cec5SDimitry Andric error(errorToErrorCode(SymbolNameOrErr.takeError())); 1366*0b57cec5SDimitry Andric SymbolName = *SymbolNameOrErr; 1367*0b57cec5SDimitry Andric SymbolIndex = Obj->getSymbolIndex(Obj->getCOFFSymbol(*Symbol)); 1368*0b57cec5SDimitry Andric } 1369*0b57cec5SDimitry Andric 1370*0b57cec5SDimitry Andric if (opts::ExpandRelocs) { 1371*0b57cec5SDimitry Andric DictScope Group(W, "Relocation"); 1372*0b57cec5SDimitry Andric W.printHex("Offset", Offset); 1373*0b57cec5SDimitry Andric W.printNumber("Type", RelocName, RelocType); 1374*0b57cec5SDimitry Andric W.printString("Symbol", SymbolName.empty() ? "-" : SymbolName); 1375*0b57cec5SDimitry Andric W.printNumber("SymbolIndex", SymbolIndex); 1376*0b57cec5SDimitry Andric } else { 1377*0b57cec5SDimitry Andric raw_ostream& OS = W.startLine(); 1378*0b57cec5SDimitry Andric OS << W.hex(Offset) 1379*0b57cec5SDimitry Andric << " " << RelocName 1380*0b57cec5SDimitry Andric << " " << (SymbolName.empty() ? "-" : SymbolName) 1381*0b57cec5SDimitry Andric << " (" << SymbolIndex << ")" 1382*0b57cec5SDimitry Andric << "\n"; 1383*0b57cec5SDimitry Andric } 1384*0b57cec5SDimitry Andric } 1385*0b57cec5SDimitry Andric 1386*0b57cec5SDimitry Andric void COFFDumper::printSymbols() { 1387*0b57cec5SDimitry Andric ListScope Group(W, "Symbols"); 1388*0b57cec5SDimitry Andric 1389*0b57cec5SDimitry Andric for (const SymbolRef &Symbol : Obj->symbols()) 1390*0b57cec5SDimitry Andric printSymbol(Symbol); 1391*0b57cec5SDimitry Andric } 1392*0b57cec5SDimitry Andric 1393*0b57cec5SDimitry Andric void COFFDumper::printDynamicSymbols() { ListScope Group(W, "DynamicSymbols"); } 1394*0b57cec5SDimitry Andric 1395*0b57cec5SDimitry Andric static Expected<StringRef> 1396*0b57cec5SDimitry Andric getSectionName(const llvm::object::COFFObjectFile *Obj, int32_t SectionNumber, 1397*0b57cec5SDimitry Andric const coff_section *Section) { 1398*0b57cec5SDimitry Andric if (Section) 1399*0b57cec5SDimitry Andric return Obj->getSectionName(Section); 1400*0b57cec5SDimitry Andric if (SectionNumber == llvm::COFF::IMAGE_SYM_DEBUG) 1401*0b57cec5SDimitry Andric return StringRef("IMAGE_SYM_DEBUG"); 1402*0b57cec5SDimitry Andric if (SectionNumber == llvm::COFF::IMAGE_SYM_ABSOLUTE) 1403*0b57cec5SDimitry Andric return StringRef("IMAGE_SYM_ABSOLUTE"); 1404*0b57cec5SDimitry Andric if (SectionNumber == llvm::COFF::IMAGE_SYM_UNDEFINED) 1405*0b57cec5SDimitry Andric return StringRef("IMAGE_SYM_UNDEFINED"); 1406*0b57cec5SDimitry Andric return StringRef(""); 1407*0b57cec5SDimitry Andric } 1408*0b57cec5SDimitry Andric 1409*0b57cec5SDimitry Andric void COFFDumper::printSymbol(const SymbolRef &Sym) { 1410*0b57cec5SDimitry Andric DictScope D(W, "Symbol"); 1411*0b57cec5SDimitry Andric 1412*0b57cec5SDimitry Andric COFFSymbolRef Symbol = Obj->getCOFFSymbol(Sym); 1413*0b57cec5SDimitry Andric const coff_section *Section; 1414*0b57cec5SDimitry Andric if (std::error_code EC = Obj->getSection(Symbol.getSectionNumber(), Section)) { 1415*0b57cec5SDimitry Andric W.startLine() << "Invalid section number: " << EC.message() << "\n"; 1416*0b57cec5SDimitry Andric W.flush(); 1417*0b57cec5SDimitry Andric return; 1418*0b57cec5SDimitry Andric } 1419*0b57cec5SDimitry Andric 1420*0b57cec5SDimitry Andric StringRef SymbolName; 1421*0b57cec5SDimitry Andric if (Obj->getSymbolName(Symbol, SymbolName)) 1422*0b57cec5SDimitry Andric SymbolName = ""; 1423*0b57cec5SDimitry Andric 1424*0b57cec5SDimitry Andric StringRef SectionName; 1425*0b57cec5SDimitry Andric if (Expected<StringRef> NameOrErr = 1426*0b57cec5SDimitry Andric getSectionName(Obj, Symbol.getSectionNumber(), Section)) 1427*0b57cec5SDimitry Andric SectionName = *NameOrErr; 1428*0b57cec5SDimitry Andric 1429*0b57cec5SDimitry Andric W.printString("Name", SymbolName); 1430*0b57cec5SDimitry Andric W.printNumber("Value", Symbol.getValue()); 1431*0b57cec5SDimitry Andric W.printNumber("Section", SectionName, Symbol.getSectionNumber()); 1432*0b57cec5SDimitry Andric W.printEnum ("BaseType", Symbol.getBaseType(), makeArrayRef(ImageSymType)); 1433*0b57cec5SDimitry Andric W.printEnum ("ComplexType", Symbol.getComplexType(), 1434*0b57cec5SDimitry Andric makeArrayRef(ImageSymDType)); 1435*0b57cec5SDimitry Andric W.printEnum ("StorageClass", Symbol.getStorageClass(), 1436*0b57cec5SDimitry Andric makeArrayRef(ImageSymClass)); 1437*0b57cec5SDimitry Andric W.printNumber("AuxSymbolCount", Symbol.getNumberOfAuxSymbols()); 1438*0b57cec5SDimitry Andric 1439*0b57cec5SDimitry Andric for (uint8_t I = 0; I < Symbol.getNumberOfAuxSymbols(); ++I) { 1440*0b57cec5SDimitry Andric if (Symbol.isFunctionDefinition()) { 1441*0b57cec5SDimitry Andric const coff_aux_function_definition *Aux; 1442*0b57cec5SDimitry Andric error(getSymbolAuxData(Obj, Symbol, I, Aux)); 1443*0b57cec5SDimitry Andric 1444*0b57cec5SDimitry Andric DictScope AS(W, "AuxFunctionDef"); 1445*0b57cec5SDimitry Andric W.printNumber("TagIndex", Aux->TagIndex); 1446*0b57cec5SDimitry Andric W.printNumber("TotalSize", Aux->TotalSize); 1447*0b57cec5SDimitry Andric W.printHex("PointerToLineNumber", Aux->PointerToLinenumber); 1448*0b57cec5SDimitry Andric W.printHex("PointerToNextFunction", Aux->PointerToNextFunction); 1449*0b57cec5SDimitry Andric 1450*0b57cec5SDimitry Andric } else if (Symbol.isAnyUndefined()) { 1451*0b57cec5SDimitry Andric const coff_aux_weak_external *Aux; 1452*0b57cec5SDimitry Andric error(getSymbolAuxData(Obj, Symbol, I, Aux)); 1453*0b57cec5SDimitry Andric 1454*0b57cec5SDimitry Andric Expected<COFFSymbolRef> Linked = Obj->getSymbol(Aux->TagIndex); 1455*0b57cec5SDimitry Andric StringRef LinkedName; 1456*0b57cec5SDimitry Andric std::error_code EC = errorToErrorCode(Linked.takeError()); 1457*0b57cec5SDimitry Andric if (EC || (EC = Obj->getSymbolName(*Linked, LinkedName))) { 1458*0b57cec5SDimitry Andric LinkedName = ""; 1459*0b57cec5SDimitry Andric error(EC); 1460*0b57cec5SDimitry Andric } 1461*0b57cec5SDimitry Andric 1462*0b57cec5SDimitry Andric DictScope AS(W, "AuxWeakExternal"); 1463*0b57cec5SDimitry Andric W.printNumber("Linked", LinkedName, Aux->TagIndex); 1464*0b57cec5SDimitry Andric W.printEnum ("Search", Aux->Characteristics, 1465*0b57cec5SDimitry Andric makeArrayRef(WeakExternalCharacteristics)); 1466*0b57cec5SDimitry Andric 1467*0b57cec5SDimitry Andric } else if (Symbol.isFileRecord()) { 1468*0b57cec5SDimitry Andric const char *FileName; 1469*0b57cec5SDimitry Andric error(getSymbolAuxData(Obj, Symbol, I, FileName)); 1470*0b57cec5SDimitry Andric 1471*0b57cec5SDimitry Andric DictScope AS(W, "AuxFileRecord"); 1472*0b57cec5SDimitry Andric 1473*0b57cec5SDimitry Andric StringRef Name(FileName, Symbol.getNumberOfAuxSymbols() * 1474*0b57cec5SDimitry Andric Obj->getSymbolTableEntrySize()); 1475*0b57cec5SDimitry Andric W.printString("FileName", Name.rtrim(StringRef("\0", 1))); 1476*0b57cec5SDimitry Andric break; 1477*0b57cec5SDimitry Andric } else if (Symbol.isSectionDefinition()) { 1478*0b57cec5SDimitry Andric const coff_aux_section_definition *Aux; 1479*0b57cec5SDimitry Andric error(getSymbolAuxData(Obj, Symbol, I, Aux)); 1480*0b57cec5SDimitry Andric 1481*0b57cec5SDimitry Andric int32_t AuxNumber = Aux->getNumber(Symbol.isBigObj()); 1482*0b57cec5SDimitry Andric 1483*0b57cec5SDimitry Andric DictScope AS(W, "AuxSectionDef"); 1484*0b57cec5SDimitry Andric W.printNumber("Length", Aux->Length); 1485*0b57cec5SDimitry Andric W.printNumber("RelocationCount", Aux->NumberOfRelocations); 1486*0b57cec5SDimitry Andric W.printNumber("LineNumberCount", Aux->NumberOfLinenumbers); 1487*0b57cec5SDimitry Andric W.printHex("Checksum", Aux->CheckSum); 1488*0b57cec5SDimitry Andric W.printNumber("Number", AuxNumber); 1489*0b57cec5SDimitry Andric W.printEnum("Selection", Aux->Selection, makeArrayRef(ImageCOMDATSelect)); 1490*0b57cec5SDimitry Andric 1491*0b57cec5SDimitry Andric if (Section && Section->Characteristics & COFF::IMAGE_SCN_LNK_COMDAT 1492*0b57cec5SDimitry Andric && Aux->Selection == COFF::IMAGE_COMDAT_SELECT_ASSOCIATIVE) { 1493*0b57cec5SDimitry Andric const coff_section *Assoc; 1494*0b57cec5SDimitry Andric StringRef AssocName = ""; 1495*0b57cec5SDimitry Andric if (std::error_code EC = Obj->getSection(AuxNumber, Assoc)) 1496*0b57cec5SDimitry Andric error(EC); 1497*0b57cec5SDimitry Andric Expected<StringRef> Res = getSectionName(Obj, AuxNumber, Assoc); 1498*0b57cec5SDimitry Andric if (!Res) 1499*0b57cec5SDimitry Andric error(Res.takeError()); 1500*0b57cec5SDimitry Andric AssocName = *Res; 1501*0b57cec5SDimitry Andric 1502*0b57cec5SDimitry Andric W.printNumber("AssocSection", AssocName, AuxNumber); 1503*0b57cec5SDimitry Andric } 1504*0b57cec5SDimitry Andric } else if (Symbol.isCLRToken()) { 1505*0b57cec5SDimitry Andric const coff_aux_clr_token *Aux; 1506*0b57cec5SDimitry Andric error(getSymbolAuxData(Obj, Symbol, I, Aux)); 1507*0b57cec5SDimitry Andric 1508*0b57cec5SDimitry Andric Expected<COFFSymbolRef> ReferredSym = 1509*0b57cec5SDimitry Andric Obj->getSymbol(Aux->SymbolTableIndex); 1510*0b57cec5SDimitry Andric StringRef ReferredName; 1511*0b57cec5SDimitry Andric std::error_code EC = errorToErrorCode(ReferredSym.takeError()); 1512*0b57cec5SDimitry Andric if (EC || (EC = Obj->getSymbolName(*ReferredSym, ReferredName))) { 1513*0b57cec5SDimitry Andric ReferredName = ""; 1514*0b57cec5SDimitry Andric error(EC); 1515*0b57cec5SDimitry Andric } 1516*0b57cec5SDimitry Andric 1517*0b57cec5SDimitry Andric DictScope AS(W, "AuxCLRToken"); 1518*0b57cec5SDimitry Andric W.printNumber("AuxType", Aux->AuxType); 1519*0b57cec5SDimitry Andric W.printNumber("Reserved", Aux->Reserved); 1520*0b57cec5SDimitry Andric W.printNumber("SymbolTableIndex", ReferredName, Aux->SymbolTableIndex); 1521*0b57cec5SDimitry Andric 1522*0b57cec5SDimitry Andric } else { 1523*0b57cec5SDimitry Andric W.startLine() << "<unhandled auxiliary record>\n"; 1524*0b57cec5SDimitry Andric } 1525*0b57cec5SDimitry Andric } 1526*0b57cec5SDimitry Andric } 1527*0b57cec5SDimitry Andric 1528*0b57cec5SDimitry Andric void COFFDumper::printUnwindInfo() { 1529*0b57cec5SDimitry Andric ListScope D(W, "UnwindInformation"); 1530*0b57cec5SDimitry Andric switch (Obj->getMachine()) { 1531*0b57cec5SDimitry Andric case COFF::IMAGE_FILE_MACHINE_AMD64: { 1532*0b57cec5SDimitry Andric Win64EH::Dumper Dumper(W); 1533*0b57cec5SDimitry Andric Win64EH::Dumper::SymbolResolver 1534*0b57cec5SDimitry Andric Resolver = [](const object::coff_section *Section, uint64_t Offset, 1535*0b57cec5SDimitry Andric SymbolRef &Symbol, void *user_data) -> std::error_code { 1536*0b57cec5SDimitry Andric COFFDumper *Dumper = reinterpret_cast<COFFDumper *>(user_data); 1537*0b57cec5SDimitry Andric return Dumper->resolveSymbol(Section, Offset, Symbol); 1538*0b57cec5SDimitry Andric }; 1539*0b57cec5SDimitry Andric Win64EH::Dumper::Context Ctx(*Obj, Resolver, this); 1540*0b57cec5SDimitry Andric Dumper.printData(Ctx); 1541*0b57cec5SDimitry Andric break; 1542*0b57cec5SDimitry Andric } 1543*0b57cec5SDimitry Andric case COFF::IMAGE_FILE_MACHINE_ARM64: 1544*0b57cec5SDimitry Andric case COFF::IMAGE_FILE_MACHINE_ARMNT: { 1545*0b57cec5SDimitry Andric ARM::WinEH::Decoder Decoder(W, Obj->getMachine() == 1546*0b57cec5SDimitry Andric COFF::IMAGE_FILE_MACHINE_ARM64); 1547*0b57cec5SDimitry Andric // TODO Propagate the error. 1548*0b57cec5SDimitry Andric consumeError(Decoder.dumpProcedureData(*Obj)); 1549*0b57cec5SDimitry Andric break; 1550*0b57cec5SDimitry Andric } 1551*0b57cec5SDimitry Andric default: 1552*0b57cec5SDimitry Andric W.printEnum("unsupported Image Machine", Obj->getMachine(), 1553*0b57cec5SDimitry Andric makeArrayRef(ImageFileMachineType)); 1554*0b57cec5SDimitry Andric break; 1555*0b57cec5SDimitry Andric } 1556*0b57cec5SDimitry Andric } 1557*0b57cec5SDimitry Andric 1558*0b57cec5SDimitry Andric void COFFDumper::printNeededLibraries() { 1559*0b57cec5SDimitry Andric ListScope D(W, "NeededLibraries"); 1560*0b57cec5SDimitry Andric 1561*0b57cec5SDimitry Andric using LibsTy = std::vector<StringRef>; 1562*0b57cec5SDimitry Andric LibsTy Libs; 1563*0b57cec5SDimitry Andric 1564*0b57cec5SDimitry Andric for (const ImportDirectoryEntryRef &DirRef : Obj->import_directories()) { 1565*0b57cec5SDimitry Andric StringRef Name; 1566*0b57cec5SDimitry Andric if (!DirRef.getName(Name)) 1567*0b57cec5SDimitry Andric Libs.push_back(Name); 1568*0b57cec5SDimitry Andric } 1569*0b57cec5SDimitry Andric 1570*0b57cec5SDimitry Andric llvm::stable_sort(Libs); 1571*0b57cec5SDimitry Andric 1572*0b57cec5SDimitry Andric for (const auto &L : Libs) { 1573*0b57cec5SDimitry Andric W.startLine() << L << "\n"; 1574*0b57cec5SDimitry Andric } 1575*0b57cec5SDimitry Andric } 1576*0b57cec5SDimitry Andric 1577*0b57cec5SDimitry Andric void COFFDumper::printImportedSymbols( 1578*0b57cec5SDimitry Andric iterator_range<imported_symbol_iterator> Range) { 1579*0b57cec5SDimitry Andric for (const ImportedSymbolRef &I : Range) { 1580*0b57cec5SDimitry Andric StringRef Sym; 1581*0b57cec5SDimitry Andric error(I.getSymbolName(Sym)); 1582*0b57cec5SDimitry Andric uint16_t Ordinal; 1583*0b57cec5SDimitry Andric error(I.getOrdinal(Ordinal)); 1584*0b57cec5SDimitry Andric W.printNumber("Symbol", Sym, Ordinal); 1585*0b57cec5SDimitry Andric } 1586*0b57cec5SDimitry Andric } 1587*0b57cec5SDimitry Andric 1588*0b57cec5SDimitry Andric void COFFDumper::printDelayImportedSymbols( 1589*0b57cec5SDimitry Andric const DelayImportDirectoryEntryRef &I, 1590*0b57cec5SDimitry Andric iterator_range<imported_symbol_iterator> Range) { 1591*0b57cec5SDimitry Andric int Index = 0; 1592*0b57cec5SDimitry Andric for (const ImportedSymbolRef &S : Range) { 1593*0b57cec5SDimitry Andric DictScope Import(W, "Import"); 1594*0b57cec5SDimitry Andric StringRef Sym; 1595*0b57cec5SDimitry Andric error(S.getSymbolName(Sym)); 1596*0b57cec5SDimitry Andric uint16_t Ordinal; 1597*0b57cec5SDimitry Andric error(S.getOrdinal(Ordinal)); 1598*0b57cec5SDimitry Andric W.printNumber("Symbol", Sym, Ordinal); 1599*0b57cec5SDimitry Andric uint64_t Addr; 1600*0b57cec5SDimitry Andric error(I.getImportAddress(Index++, Addr)); 1601*0b57cec5SDimitry Andric W.printHex("Address", Addr); 1602*0b57cec5SDimitry Andric } 1603*0b57cec5SDimitry Andric } 1604*0b57cec5SDimitry Andric 1605*0b57cec5SDimitry Andric void COFFDumper::printCOFFImports() { 1606*0b57cec5SDimitry Andric // Regular imports 1607*0b57cec5SDimitry Andric for (const ImportDirectoryEntryRef &I : Obj->import_directories()) { 1608*0b57cec5SDimitry Andric DictScope Import(W, "Import"); 1609*0b57cec5SDimitry Andric StringRef Name; 1610*0b57cec5SDimitry Andric error(I.getName(Name)); 1611*0b57cec5SDimitry Andric W.printString("Name", Name); 1612*0b57cec5SDimitry Andric uint32_t ILTAddr; 1613*0b57cec5SDimitry Andric error(I.getImportLookupTableRVA(ILTAddr)); 1614*0b57cec5SDimitry Andric W.printHex("ImportLookupTableRVA", ILTAddr); 1615*0b57cec5SDimitry Andric uint32_t IATAddr; 1616*0b57cec5SDimitry Andric error(I.getImportAddressTableRVA(IATAddr)); 1617*0b57cec5SDimitry Andric W.printHex("ImportAddressTableRVA", IATAddr); 1618*0b57cec5SDimitry Andric // The import lookup table can be missing with certain older linkers, so 1619*0b57cec5SDimitry Andric // fall back to the import address table in that case. 1620*0b57cec5SDimitry Andric if (ILTAddr) 1621*0b57cec5SDimitry Andric printImportedSymbols(I.lookup_table_symbols()); 1622*0b57cec5SDimitry Andric else 1623*0b57cec5SDimitry Andric printImportedSymbols(I.imported_symbols()); 1624*0b57cec5SDimitry Andric } 1625*0b57cec5SDimitry Andric 1626*0b57cec5SDimitry Andric // Delay imports 1627*0b57cec5SDimitry Andric for (const DelayImportDirectoryEntryRef &I : Obj->delay_import_directories()) { 1628*0b57cec5SDimitry Andric DictScope Import(W, "DelayImport"); 1629*0b57cec5SDimitry Andric StringRef Name; 1630*0b57cec5SDimitry Andric error(I.getName(Name)); 1631*0b57cec5SDimitry Andric W.printString("Name", Name); 1632*0b57cec5SDimitry Andric const delay_import_directory_table_entry *Table; 1633*0b57cec5SDimitry Andric error(I.getDelayImportTable(Table)); 1634*0b57cec5SDimitry Andric W.printHex("Attributes", Table->Attributes); 1635*0b57cec5SDimitry Andric W.printHex("ModuleHandle", Table->ModuleHandle); 1636*0b57cec5SDimitry Andric W.printHex("ImportAddressTable", Table->DelayImportAddressTable); 1637*0b57cec5SDimitry Andric W.printHex("ImportNameTable", Table->DelayImportNameTable); 1638*0b57cec5SDimitry Andric W.printHex("BoundDelayImportTable", Table->BoundDelayImportTable); 1639*0b57cec5SDimitry Andric W.printHex("UnloadDelayImportTable", Table->UnloadDelayImportTable); 1640*0b57cec5SDimitry Andric printDelayImportedSymbols(I, I.imported_symbols()); 1641*0b57cec5SDimitry Andric } 1642*0b57cec5SDimitry Andric } 1643*0b57cec5SDimitry Andric 1644*0b57cec5SDimitry Andric void COFFDumper::printCOFFExports() { 1645*0b57cec5SDimitry Andric for (const ExportDirectoryEntryRef &E : Obj->export_directories()) { 1646*0b57cec5SDimitry Andric DictScope Export(W, "Export"); 1647*0b57cec5SDimitry Andric 1648*0b57cec5SDimitry Andric StringRef Name; 1649*0b57cec5SDimitry Andric uint32_t Ordinal, RVA; 1650*0b57cec5SDimitry Andric 1651*0b57cec5SDimitry Andric error(E.getSymbolName(Name)); 1652*0b57cec5SDimitry Andric error(E.getOrdinal(Ordinal)); 1653*0b57cec5SDimitry Andric error(E.getExportRVA(RVA)); 1654*0b57cec5SDimitry Andric 1655*0b57cec5SDimitry Andric W.printNumber("Ordinal", Ordinal); 1656*0b57cec5SDimitry Andric W.printString("Name", Name); 1657*0b57cec5SDimitry Andric W.printHex("RVA", RVA); 1658*0b57cec5SDimitry Andric } 1659*0b57cec5SDimitry Andric } 1660*0b57cec5SDimitry Andric 1661*0b57cec5SDimitry Andric void COFFDumper::printCOFFDirectives() { 1662*0b57cec5SDimitry Andric for (const SectionRef &Section : Obj->sections()) { 1663*0b57cec5SDimitry Andric StringRef Name; 1664*0b57cec5SDimitry Andric 1665*0b57cec5SDimitry Andric error(Section.getName(Name)); 1666*0b57cec5SDimitry Andric if (Name != ".drectve") 1667*0b57cec5SDimitry Andric continue; 1668*0b57cec5SDimitry Andric 1669*0b57cec5SDimitry Andric StringRef Contents = unwrapOrError(Section.getContents()); 1670*0b57cec5SDimitry Andric W.printString("Directive(s)", Contents); 1671*0b57cec5SDimitry Andric } 1672*0b57cec5SDimitry Andric } 1673*0b57cec5SDimitry Andric 1674*0b57cec5SDimitry Andric static std::string getBaseRelocTypeName(uint8_t Type) { 1675*0b57cec5SDimitry Andric switch (Type) { 1676*0b57cec5SDimitry Andric case COFF::IMAGE_REL_BASED_ABSOLUTE: return "ABSOLUTE"; 1677*0b57cec5SDimitry Andric case COFF::IMAGE_REL_BASED_HIGH: return "HIGH"; 1678*0b57cec5SDimitry Andric case COFF::IMAGE_REL_BASED_LOW: return "LOW"; 1679*0b57cec5SDimitry Andric case COFF::IMAGE_REL_BASED_HIGHLOW: return "HIGHLOW"; 1680*0b57cec5SDimitry Andric case COFF::IMAGE_REL_BASED_HIGHADJ: return "HIGHADJ"; 1681*0b57cec5SDimitry Andric case COFF::IMAGE_REL_BASED_ARM_MOV32T: return "ARM_MOV32(T)"; 1682*0b57cec5SDimitry Andric case COFF::IMAGE_REL_BASED_DIR64: return "DIR64"; 1683*0b57cec5SDimitry Andric default: return "unknown (" + llvm::utostr(Type) + ")"; 1684*0b57cec5SDimitry Andric } 1685*0b57cec5SDimitry Andric } 1686*0b57cec5SDimitry Andric 1687*0b57cec5SDimitry Andric void COFFDumper::printCOFFBaseReloc() { 1688*0b57cec5SDimitry Andric ListScope D(W, "BaseReloc"); 1689*0b57cec5SDimitry Andric for (const BaseRelocRef &I : Obj->base_relocs()) { 1690*0b57cec5SDimitry Andric uint8_t Type; 1691*0b57cec5SDimitry Andric uint32_t RVA; 1692*0b57cec5SDimitry Andric error(I.getRVA(RVA)); 1693*0b57cec5SDimitry Andric error(I.getType(Type)); 1694*0b57cec5SDimitry Andric DictScope Import(W, "Entry"); 1695*0b57cec5SDimitry Andric W.printString("Type", getBaseRelocTypeName(Type)); 1696*0b57cec5SDimitry Andric W.printHex("Address", RVA); 1697*0b57cec5SDimitry Andric } 1698*0b57cec5SDimitry Andric } 1699*0b57cec5SDimitry Andric 1700*0b57cec5SDimitry Andric void COFFDumper::printCOFFResources() { 1701*0b57cec5SDimitry Andric ListScope ResourcesD(W, "Resources"); 1702*0b57cec5SDimitry Andric for (const SectionRef &S : Obj->sections()) { 1703*0b57cec5SDimitry Andric StringRef Name; 1704*0b57cec5SDimitry Andric error(S.getName(Name)); 1705*0b57cec5SDimitry Andric if (!Name.startswith(".rsrc")) 1706*0b57cec5SDimitry Andric continue; 1707*0b57cec5SDimitry Andric 1708*0b57cec5SDimitry Andric StringRef Ref = unwrapOrError(S.getContents()); 1709*0b57cec5SDimitry Andric 1710*0b57cec5SDimitry Andric if ((Name == ".rsrc") || (Name == ".rsrc$01")) { 1711*0b57cec5SDimitry Andric ResourceSectionRef RSF(Ref); 1712*0b57cec5SDimitry Andric auto &BaseTable = unwrapOrError(RSF.getBaseTable()); 1713*0b57cec5SDimitry Andric W.printNumber("Total Number of Resources", 1714*0b57cec5SDimitry Andric countTotalTableEntries(RSF, BaseTable, "Type")); 1715*0b57cec5SDimitry Andric W.printHex("Base Table Address", 1716*0b57cec5SDimitry Andric Obj->getCOFFSection(S)->PointerToRawData); 1717*0b57cec5SDimitry Andric W.startLine() << "\n"; 1718*0b57cec5SDimitry Andric printResourceDirectoryTable(RSF, BaseTable, "Type"); 1719*0b57cec5SDimitry Andric } 1720*0b57cec5SDimitry Andric if (opts::SectionData) 1721*0b57cec5SDimitry Andric W.printBinaryBlock(Name.str() + " Data", Ref); 1722*0b57cec5SDimitry Andric } 1723*0b57cec5SDimitry Andric } 1724*0b57cec5SDimitry Andric 1725*0b57cec5SDimitry Andric uint32_t 1726*0b57cec5SDimitry Andric COFFDumper::countTotalTableEntries(ResourceSectionRef RSF, 1727*0b57cec5SDimitry Andric const coff_resource_dir_table &Table, 1728*0b57cec5SDimitry Andric StringRef Level) { 1729*0b57cec5SDimitry Andric uint32_t TotalEntries = 0; 1730*0b57cec5SDimitry Andric for (int i = 0; i < Table.NumberOfNameEntries + Table.NumberOfIDEntries; 1731*0b57cec5SDimitry Andric i++) { 1732*0b57cec5SDimitry Andric auto Entry = unwrapOrError(getResourceDirectoryTableEntry(Table, i)); 1733*0b57cec5SDimitry Andric if (Entry.Offset.isSubDir()) { 1734*0b57cec5SDimitry Andric StringRef NextLevel; 1735*0b57cec5SDimitry Andric if (Level == "Name") 1736*0b57cec5SDimitry Andric NextLevel = "Language"; 1737*0b57cec5SDimitry Andric else 1738*0b57cec5SDimitry Andric NextLevel = "Name"; 1739*0b57cec5SDimitry Andric auto &NextTable = unwrapOrError(RSF.getEntrySubDir(Entry)); 1740*0b57cec5SDimitry Andric TotalEntries += countTotalTableEntries(RSF, NextTable, NextLevel); 1741*0b57cec5SDimitry Andric } else { 1742*0b57cec5SDimitry Andric TotalEntries += 1; 1743*0b57cec5SDimitry Andric } 1744*0b57cec5SDimitry Andric } 1745*0b57cec5SDimitry Andric return TotalEntries; 1746*0b57cec5SDimitry Andric } 1747*0b57cec5SDimitry Andric 1748*0b57cec5SDimitry Andric void COFFDumper::printResourceDirectoryTable( 1749*0b57cec5SDimitry Andric ResourceSectionRef RSF, const coff_resource_dir_table &Table, 1750*0b57cec5SDimitry Andric StringRef Level) { 1751*0b57cec5SDimitry Andric 1752*0b57cec5SDimitry Andric W.printNumber("Number of String Entries", Table.NumberOfNameEntries); 1753*0b57cec5SDimitry Andric W.printNumber("Number of ID Entries", Table.NumberOfIDEntries); 1754*0b57cec5SDimitry Andric 1755*0b57cec5SDimitry Andric // Iterate through level in resource directory tree. 1756*0b57cec5SDimitry Andric for (int i = 0; i < Table.NumberOfNameEntries + Table.NumberOfIDEntries; 1757*0b57cec5SDimitry Andric i++) { 1758*0b57cec5SDimitry Andric auto Entry = unwrapOrError(getResourceDirectoryTableEntry(Table, i)); 1759*0b57cec5SDimitry Andric StringRef Name; 1760*0b57cec5SDimitry Andric SmallString<20> IDStr; 1761*0b57cec5SDimitry Andric raw_svector_ostream OS(IDStr); 1762*0b57cec5SDimitry Andric if (i < Table.NumberOfNameEntries) { 1763*0b57cec5SDimitry Andric ArrayRef<UTF16> RawEntryNameString = 1764*0b57cec5SDimitry Andric unwrapOrError(RSF.getEntryNameString(Entry)); 1765*0b57cec5SDimitry Andric std::vector<UTF16> EndianCorrectedNameString; 1766*0b57cec5SDimitry Andric if (llvm::sys::IsBigEndianHost) { 1767*0b57cec5SDimitry Andric EndianCorrectedNameString.resize(RawEntryNameString.size() + 1); 1768*0b57cec5SDimitry Andric std::copy(RawEntryNameString.begin(), RawEntryNameString.end(), 1769*0b57cec5SDimitry Andric EndianCorrectedNameString.begin() + 1); 1770*0b57cec5SDimitry Andric EndianCorrectedNameString[0] = UNI_UTF16_BYTE_ORDER_MARK_SWAPPED; 1771*0b57cec5SDimitry Andric RawEntryNameString = makeArrayRef(EndianCorrectedNameString); 1772*0b57cec5SDimitry Andric } 1773*0b57cec5SDimitry Andric std::string EntryNameString; 1774*0b57cec5SDimitry Andric if (!llvm::convertUTF16ToUTF8String(RawEntryNameString, EntryNameString)) 1775*0b57cec5SDimitry Andric error(object_error::parse_failed); 1776*0b57cec5SDimitry Andric OS << ": "; 1777*0b57cec5SDimitry Andric OS << EntryNameString; 1778*0b57cec5SDimitry Andric } else { 1779*0b57cec5SDimitry Andric if (Level == "Type") { 1780*0b57cec5SDimitry Andric OS << ": "; 1781*0b57cec5SDimitry Andric printResourceTypeName(Entry.Identifier.ID, OS); 1782*0b57cec5SDimitry Andric IDStr = IDStr.slice(0, IDStr.find_first_of(")", 0) + 1); 1783*0b57cec5SDimitry Andric } else { 1784*0b57cec5SDimitry Andric OS << ": (ID " << Entry.Identifier.ID << ")"; 1785*0b57cec5SDimitry Andric } 1786*0b57cec5SDimitry Andric } 1787*0b57cec5SDimitry Andric Name = StringRef(IDStr); 1788*0b57cec5SDimitry Andric ListScope ResourceType(W, Level.str() + Name.str()); 1789*0b57cec5SDimitry Andric if (Entry.Offset.isSubDir()) { 1790*0b57cec5SDimitry Andric W.printHex("Table Offset", Entry.Offset.value()); 1791*0b57cec5SDimitry Andric StringRef NextLevel; 1792*0b57cec5SDimitry Andric if (Level == "Name") 1793*0b57cec5SDimitry Andric NextLevel = "Language"; 1794*0b57cec5SDimitry Andric else 1795*0b57cec5SDimitry Andric NextLevel = "Name"; 1796*0b57cec5SDimitry Andric auto &NextTable = unwrapOrError(RSF.getEntrySubDir(Entry)); 1797*0b57cec5SDimitry Andric printResourceDirectoryTable(RSF, NextTable, NextLevel); 1798*0b57cec5SDimitry Andric } else { 1799*0b57cec5SDimitry Andric W.printHex("Entry Offset", Entry.Offset.value()); 1800*0b57cec5SDimitry Andric char FormattedTime[20] = {}; 1801*0b57cec5SDimitry Andric time_t TDS = time_t(Table.TimeDateStamp); 1802*0b57cec5SDimitry Andric strftime(FormattedTime, 20, "%Y-%m-%d %H:%M:%S", gmtime(&TDS)); 1803*0b57cec5SDimitry Andric W.printHex("Time/Date Stamp", FormattedTime, Table.TimeDateStamp); 1804*0b57cec5SDimitry Andric W.printNumber("Major Version", Table.MajorVersion); 1805*0b57cec5SDimitry Andric W.printNumber("Minor Version", Table.MinorVersion); 1806*0b57cec5SDimitry Andric W.printNumber("Characteristics", Table.Characteristics); 1807*0b57cec5SDimitry Andric } 1808*0b57cec5SDimitry Andric } 1809*0b57cec5SDimitry Andric } 1810*0b57cec5SDimitry Andric 1811*0b57cec5SDimitry Andric ErrorOr<const coff_resource_dir_entry &> 1812*0b57cec5SDimitry Andric COFFDumper::getResourceDirectoryTableEntry(const coff_resource_dir_table &Table, 1813*0b57cec5SDimitry Andric uint32_t Index) { 1814*0b57cec5SDimitry Andric if (Index >= (uint32_t)(Table.NumberOfNameEntries + Table.NumberOfIDEntries)) 1815*0b57cec5SDimitry Andric return object_error::parse_failed; 1816*0b57cec5SDimitry Andric auto TablePtr = reinterpret_cast<const coff_resource_dir_entry *>(&Table + 1); 1817*0b57cec5SDimitry Andric return TablePtr[Index]; 1818*0b57cec5SDimitry Andric } 1819*0b57cec5SDimitry Andric 1820*0b57cec5SDimitry Andric void COFFDumper::printStackMap() const { 1821*0b57cec5SDimitry Andric object::SectionRef StackMapSection; 1822*0b57cec5SDimitry Andric for (auto Sec : Obj->sections()) { 1823*0b57cec5SDimitry Andric StringRef Name; 1824*0b57cec5SDimitry Andric Sec.getName(Name); 1825*0b57cec5SDimitry Andric if (Name == ".llvm_stackmaps") { 1826*0b57cec5SDimitry Andric StackMapSection = Sec; 1827*0b57cec5SDimitry Andric break; 1828*0b57cec5SDimitry Andric } 1829*0b57cec5SDimitry Andric } 1830*0b57cec5SDimitry Andric 1831*0b57cec5SDimitry Andric if (StackMapSection == object::SectionRef()) 1832*0b57cec5SDimitry Andric return; 1833*0b57cec5SDimitry Andric 1834*0b57cec5SDimitry Andric StringRef StackMapContents = unwrapOrError(StackMapSection.getContents()); 1835*0b57cec5SDimitry Andric ArrayRef<uint8_t> StackMapContentsArray = 1836*0b57cec5SDimitry Andric arrayRefFromStringRef(StackMapContents); 1837*0b57cec5SDimitry Andric 1838*0b57cec5SDimitry Andric if (Obj->isLittleEndian()) 1839*0b57cec5SDimitry Andric prettyPrintStackMap( 1840*0b57cec5SDimitry Andric W, StackMapParser<support::little>(StackMapContentsArray)); 1841*0b57cec5SDimitry Andric else 1842*0b57cec5SDimitry Andric prettyPrintStackMap( 1843*0b57cec5SDimitry Andric W, StackMapParser<support::big>(StackMapContentsArray)); 1844*0b57cec5SDimitry Andric } 1845*0b57cec5SDimitry Andric 1846*0b57cec5SDimitry Andric void COFFDumper::printAddrsig() { 1847*0b57cec5SDimitry Andric object::SectionRef AddrsigSection; 1848*0b57cec5SDimitry Andric for (auto Sec : Obj->sections()) { 1849*0b57cec5SDimitry Andric StringRef Name; 1850*0b57cec5SDimitry Andric Sec.getName(Name); 1851*0b57cec5SDimitry Andric if (Name == ".llvm_addrsig") { 1852*0b57cec5SDimitry Andric AddrsigSection = Sec; 1853*0b57cec5SDimitry Andric break; 1854*0b57cec5SDimitry Andric } 1855*0b57cec5SDimitry Andric } 1856*0b57cec5SDimitry Andric 1857*0b57cec5SDimitry Andric if (AddrsigSection == object::SectionRef()) 1858*0b57cec5SDimitry Andric return; 1859*0b57cec5SDimitry Andric 1860*0b57cec5SDimitry Andric StringRef AddrsigContents = unwrapOrError(AddrsigSection.getContents()); 1861*0b57cec5SDimitry Andric ArrayRef<uint8_t> AddrsigContentsArray(AddrsigContents.bytes_begin(), 1862*0b57cec5SDimitry Andric AddrsigContents.size()); 1863*0b57cec5SDimitry Andric 1864*0b57cec5SDimitry Andric ListScope L(W, "Addrsig"); 1865*0b57cec5SDimitry Andric const uint8_t *Cur = AddrsigContents.bytes_begin(); 1866*0b57cec5SDimitry Andric const uint8_t *End = AddrsigContents.bytes_end(); 1867*0b57cec5SDimitry Andric while (Cur != End) { 1868*0b57cec5SDimitry Andric unsigned Size; 1869*0b57cec5SDimitry Andric const char *Err; 1870*0b57cec5SDimitry Andric uint64_t SymIndex = decodeULEB128(Cur, &Size, End, &Err); 1871*0b57cec5SDimitry Andric if (Err) 1872*0b57cec5SDimitry Andric reportError(Err); 1873*0b57cec5SDimitry Andric 1874*0b57cec5SDimitry Andric Expected<COFFSymbolRef> Sym = Obj->getSymbol(SymIndex); 1875*0b57cec5SDimitry Andric StringRef SymName; 1876*0b57cec5SDimitry Andric std::error_code EC = errorToErrorCode(Sym.takeError()); 1877*0b57cec5SDimitry Andric if (EC || (EC = Obj->getSymbolName(*Sym, SymName))) { 1878*0b57cec5SDimitry Andric SymName = ""; 1879*0b57cec5SDimitry Andric error(EC); 1880*0b57cec5SDimitry Andric } 1881*0b57cec5SDimitry Andric 1882*0b57cec5SDimitry Andric W.printNumber("Sym", SymName, SymIndex); 1883*0b57cec5SDimitry Andric Cur += Size; 1884*0b57cec5SDimitry Andric } 1885*0b57cec5SDimitry Andric } 1886*0b57cec5SDimitry Andric 1887*0b57cec5SDimitry Andric void llvm::dumpCodeViewMergedTypes(ScopedPrinter &Writer, 1888*0b57cec5SDimitry Andric ArrayRef<ArrayRef<uint8_t>> IpiRecords, 1889*0b57cec5SDimitry Andric ArrayRef<ArrayRef<uint8_t>> TpiRecords) { 1890*0b57cec5SDimitry Andric TypeTableCollection TpiTypes(TpiRecords); 1891*0b57cec5SDimitry Andric { 1892*0b57cec5SDimitry Andric ListScope S(Writer, "MergedTypeStream"); 1893*0b57cec5SDimitry Andric TypeDumpVisitor TDV(TpiTypes, &Writer, opts::CodeViewSubsectionBytes); 1894*0b57cec5SDimitry Andric error(codeview::visitTypeStream(TpiTypes, TDV)); 1895*0b57cec5SDimitry Andric Writer.flush(); 1896*0b57cec5SDimitry Andric } 1897*0b57cec5SDimitry Andric 1898*0b57cec5SDimitry Andric // Flatten the id stream and print it next. The ID stream refers to names from 1899*0b57cec5SDimitry Andric // the type stream. 1900*0b57cec5SDimitry Andric TypeTableCollection IpiTypes(IpiRecords); 1901*0b57cec5SDimitry Andric { 1902*0b57cec5SDimitry Andric ListScope S(Writer, "MergedIDStream"); 1903*0b57cec5SDimitry Andric TypeDumpVisitor TDV(TpiTypes, &Writer, opts::CodeViewSubsectionBytes); 1904*0b57cec5SDimitry Andric TDV.setIpiTypes(IpiTypes); 1905*0b57cec5SDimitry Andric error(codeview::visitTypeStream(IpiTypes, TDV)); 1906*0b57cec5SDimitry Andric Writer.flush(); 1907*0b57cec5SDimitry Andric } 1908*0b57cec5SDimitry Andric } 1909