1 //===-- ELFDumper.cpp - ELF-specific dumper ---------------------*- C++ -*-===// 2 // 3 // The LLVM Compiler Infrastructure 4 // 5 // This file is distributed under the University of Illinois Open Source 6 // License. See LICENSE.TXT for details. 7 // 8 //===----------------------------------------------------------------------===// 9 /// 10 /// \file 11 /// \brief This file implements the ELF-specific dumper for llvm-readobj. 12 /// 13 //===----------------------------------------------------------------------===// 14 15 #include "ARMAttributeParser.h" 16 #include "ARMEHABIPrinter.h" 17 #include "Error.h" 18 #include "ObjDumper.h" 19 #include "StackMapPrinter.h" 20 #include "llvm-readobj.h" 21 #include "llvm/ADT/Optional.h" 22 #include "llvm/ADT/SmallString.h" 23 #include "llvm/ADT/StringExtras.h" 24 #include "llvm/Object/ELFObjectFile.h" 25 #include "llvm/Support/ARMBuildAttributes.h" 26 #include "llvm/Support/Compiler.h" 27 #include "llvm/Support/Format.h" 28 #include "llvm/Support/FormattedStream.h" 29 #include "llvm/Support/MathExtras.h" 30 #include "llvm/Support/MipsABIFlags.h" 31 #include "llvm/Support/ScopedPrinter.h" 32 #include "llvm/Support/raw_ostream.h" 33 34 using namespace llvm; 35 using namespace llvm::object; 36 using namespace ELF; 37 38 #define LLVM_READOBJ_ENUM_CASE(ns, enum) \ 39 case ns::enum: return #enum; 40 41 #define ENUM_ENT(enum, altName) \ 42 { #enum, altName, ELF::enum } 43 44 #define ENUM_ENT_1(enum) \ 45 { #enum, #enum, ELF::enum } 46 47 #define LLVM_READOBJ_PHDR_ENUM(ns, enum) \ 48 case ns::enum: \ 49 return std::string(#enum).substr(3); 50 51 #define TYPEDEF_ELF_TYPES(ELFT) \ 52 typedef ELFFile<ELFT> ELFO; \ 53 typedef typename ELFO::Elf_Shdr Elf_Shdr; \ 54 typedef typename ELFO::Elf_Sym Elf_Sym; \ 55 typedef typename ELFO::Elf_Dyn Elf_Dyn; \ 56 typedef typename ELFO::Elf_Dyn_Range Elf_Dyn_Range; \ 57 typedef typename ELFO::Elf_Rel Elf_Rel; \ 58 typedef typename ELFO::Elf_Rela Elf_Rela; \ 59 typedef typename ELFO::Elf_Rel_Range Elf_Rel_Range; \ 60 typedef typename ELFO::Elf_Rela_Range Elf_Rela_Range; \ 61 typedef typename ELFO::Elf_Phdr Elf_Phdr; \ 62 typedef typename ELFO::Elf_Half Elf_Half; \ 63 typedef typename ELFO::Elf_Ehdr Elf_Ehdr; \ 64 typedef typename ELFO::Elf_Word Elf_Word; \ 65 typedef typename ELFO::Elf_Hash Elf_Hash; \ 66 typedef typename ELFO::Elf_GnuHash Elf_GnuHash; \ 67 typedef typename ELFO::Elf_Sym_Range Elf_Sym_Range; \ 68 typedef typename ELFO::Elf_Versym Elf_Versym; \ 69 typedef typename ELFO::Elf_Verneed Elf_Verneed; \ 70 typedef typename ELFO::Elf_Vernaux Elf_Vernaux; \ 71 typedef typename ELFO::Elf_Verdef Elf_Verdef; \ 72 typedef typename ELFO::Elf_Verdaux Elf_Verdaux; \ 73 typedef typename ELFO::uintX_t uintX_t; 74 75 namespace { 76 77 template <class ELFT> class DumpStyle; 78 79 /// Represents a contiguous uniform range in the file. We cannot just create a 80 /// range directly because when creating one of these from the .dynamic table 81 /// the size, entity size and virtual address are different entries in arbitrary 82 /// order (DT_REL, DT_RELSZ, DT_RELENT for example). 83 struct DynRegionInfo { 84 DynRegionInfo() : Addr(nullptr), Size(0), EntSize(0) {} 85 DynRegionInfo(const void *A, uint64_t S, uint64_t ES) 86 : Addr(A), Size(S), EntSize(ES) {} 87 /// \brief Address in current address space. 88 const void *Addr; 89 /// \brief Size in bytes of the region. 90 uint64_t Size; 91 /// \brief Size of each entity in the region. 92 uint64_t EntSize; 93 94 template <typename Type> ArrayRef<Type> getAsArrayRef() const { 95 const Type *Start = reinterpret_cast<const Type *>(Addr); 96 if (!Start) 97 return {Start, Start}; 98 if (EntSize != sizeof(Type) || Size % EntSize) 99 reportError("Invalid entity size"); 100 return {Start, Start + (Size / EntSize)}; 101 } 102 }; 103 104 template<typename ELFT> 105 class ELFDumper : public ObjDumper { 106 public: 107 ELFDumper(const ELFFile<ELFT> *Obj, ScopedPrinter &Writer); 108 109 void printFileHeaders() override; 110 void printSections() override; 111 void printRelocations() override; 112 void printDynamicRelocations() override; 113 void printSymbols() override; 114 void printDynamicSymbols() override; 115 void printUnwindInfo() override; 116 117 void printDynamicTable() override; 118 void printNeededLibraries() override; 119 void printProgramHeaders() override; 120 void printHashTable() override; 121 void printGnuHashTable() override; 122 void printLoadName() override; 123 void printVersionInfo() override; 124 void printGroupSections() override; 125 126 void printAttributes() override; 127 void printMipsPLTGOT() override; 128 void printMipsABIFlags() override; 129 void printMipsReginfo() override; 130 void printMipsOptions() override; 131 132 void printStackMap() const override; 133 134 void printHashHistogram() override; 135 136 void printNotes() override; 137 138 private: 139 std::unique_ptr<DumpStyle<ELFT>> ELFDumperStyle; 140 TYPEDEF_ELF_TYPES(ELFT) 141 142 DynRegionInfo checkDRI(DynRegionInfo DRI) { 143 if (DRI.Addr < Obj->base() || 144 (const uint8_t *)DRI.Addr + DRI.Size > Obj->base() + Obj->getBufSize()) 145 error(llvm::object::object_error::parse_failed); 146 return DRI; 147 } 148 149 DynRegionInfo createDRIFrom(const Elf_Phdr *P, uintX_t EntSize) { 150 return checkDRI({Obj->base() + P->p_offset, P->p_filesz, EntSize}); 151 } 152 153 DynRegionInfo createDRIFrom(const Elf_Shdr *S) { 154 return checkDRI({Obj->base() + S->sh_offset, S->sh_size, S->sh_entsize}); 155 } 156 157 void parseDynamicTable(ArrayRef<const Elf_Phdr *> LoadSegments); 158 159 void printValue(uint64_t Type, uint64_t Value); 160 161 StringRef getDynamicString(uint64_t Offset) const; 162 StringRef getSymbolVersion(StringRef StrTab, const Elf_Sym *symb, 163 bool &IsDefault) const; 164 void LoadVersionMap() const; 165 void LoadVersionNeeds(const Elf_Shdr *ec) const; 166 void LoadVersionDefs(const Elf_Shdr *sec) const; 167 168 const ELFO *Obj; 169 DynRegionInfo DynRelRegion; 170 DynRegionInfo DynRelaRegion; 171 DynRegionInfo DynPLTRelRegion; 172 DynRegionInfo DynSymRegion; 173 DynRegionInfo DynamicTable; 174 StringRef DynamicStringTable; 175 StringRef SOName; 176 const Elf_Hash *HashTable = nullptr; 177 const Elf_GnuHash *GnuHashTable = nullptr; 178 const Elf_Shdr *DotSymtabSec = nullptr; 179 StringRef DynSymtabName; 180 ArrayRef<Elf_Word> ShndxTable; 181 182 const Elf_Shdr *dot_gnu_version_sec = nullptr; // .gnu.version 183 const Elf_Shdr *dot_gnu_version_r_sec = nullptr; // .gnu.version_r 184 const Elf_Shdr *dot_gnu_version_d_sec = nullptr; // .gnu.version_d 185 186 // Records for each version index the corresponding Verdef or Vernaux entry. 187 // This is filled the first time LoadVersionMap() is called. 188 class VersionMapEntry : public PointerIntPair<const void *, 1> { 189 public: 190 // If the integer is 0, this is an Elf_Verdef*. 191 // If the integer is 1, this is an Elf_Vernaux*. 192 VersionMapEntry() : PointerIntPair<const void *, 1>(nullptr, 0) {} 193 VersionMapEntry(const Elf_Verdef *verdef) 194 : PointerIntPair<const void *, 1>(verdef, 0) {} 195 VersionMapEntry(const Elf_Vernaux *vernaux) 196 : PointerIntPair<const void *, 1>(vernaux, 1) {} 197 bool isNull() const { return getPointer() == nullptr; } 198 bool isVerdef() const { return !isNull() && getInt() == 0; } 199 bool isVernaux() const { return !isNull() && getInt() == 1; } 200 const Elf_Verdef *getVerdef() const { 201 return isVerdef() ? (const Elf_Verdef *)getPointer() : nullptr; 202 } 203 const Elf_Vernaux *getVernaux() const { 204 return isVernaux() ? (const Elf_Vernaux *)getPointer() : nullptr; 205 } 206 }; 207 mutable SmallVector<VersionMapEntry, 16> VersionMap; 208 209 public: 210 Elf_Dyn_Range dynamic_table() const { 211 return DynamicTable.getAsArrayRef<Elf_Dyn>(); 212 } 213 214 Elf_Sym_Range dynamic_symbols() const { 215 return DynSymRegion.getAsArrayRef<Elf_Sym>(); 216 } 217 218 Elf_Rel_Range dyn_rels() const; 219 Elf_Rela_Range dyn_relas() const; 220 std::string getFullSymbolName(const Elf_Sym *Symbol, StringRef StrTable, 221 bool IsDynamic) const; 222 223 void printSymbolsHelper(bool IsDynamic) const; 224 const Elf_Shdr *getDotSymtabSec() const { return DotSymtabSec; } 225 ArrayRef<Elf_Word> getShndxTable() const { return ShndxTable; } 226 StringRef getDynamicStringTable() const { return DynamicStringTable; } 227 const DynRegionInfo &getDynRelRegion() const { return DynRelRegion; } 228 const DynRegionInfo &getDynRelaRegion() const { return DynRelaRegion; } 229 const DynRegionInfo &getDynPLTRelRegion() const { return DynPLTRelRegion; } 230 const Elf_Hash *getHashTable() const { return HashTable; } 231 const Elf_GnuHash *getGnuHashTable() const { return GnuHashTable; } 232 }; 233 234 template <class ELFT> 235 void ELFDumper<ELFT>::printSymbolsHelper(bool IsDynamic) const { 236 StringRef StrTable, SymtabName; 237 size_t Entries = 0; 238 Elf_Sym_Range Syms(nullptr, nullptr); 239 if (IsDynamic) { 240 StrTable = DynamicStringTable; 241 Syms = dynamic_symbols(); 242 SymtabName = DynSymtabName; 243 if (DynSymRegion.Addr) 244 Entries = DynSymRegion.Size / DynSymRegion.EntSize; 245 } else { 246 if (!DotSymtabSec) 247 return; 248 StrTable = unwrapOrError(Obj->getStringTableForSymtab(*DotSymtabSec)); 249 Syms = Obj->symbols(DotSymtabSec); 250 SymtabName = unwrapOrError(Obj->getSectionName(DotSymtabSec)); 251 Entries = DotSymtabSec->getEntityCount(); 252 } 253 if (Syms.begin() == Syms.end()) 254 return; 255 ELFDumperStyle->printSymtabMessage(Obj, SymtabName, Entries); 256 for (const auto &Sym : Syms) 257 ELFDumperStyle->printSymbol(Obj, &Sym, Syms.begin(), StrTable, IsDynamic); 258 } 259 260 template <typename ELFT> class DumpStyle { 261 public: 262 using Elf_Shdr = typename ELFFile<ELFT>::Elf_Shdr; 263 using Elf_Sym = typename ELFFile<ELFT>::Elf_Sym; 264 265 DumpStyle(ELFDumper<ELFT> *Dumper) : Dumper(Dumper) {} 266 virtual ~DumpStyle() {} 267 virtual void printFileHeaders(const ELFFile<ELFT> *Obj) = 0; 268 virtual void printGroupSections(const ELFFile<ELFT> *Obj) = 0; 269 virtual void printRelocations(const ELFFile<ELFT> *Obj) = 0; 270 virtual void printSections(const ELFFile<ELFT> *Obj) = 0; 271 virtual void printSymbols(const ELFFile<ELFT> *Obj) = 0; 272 virtual void printDynamicSymbols(const ELFFile<ELFT> *Obj) = 0; 273 virtual void printDynamicRelocations(const ELFFile<ELFT> *Obj) = 0; 274 virtual void printSymtabMessage(const ELFFile<ELFT> *obj, StringRef Name, 275 size_t Offset) { 276 return; 277 } 278 virtual void printSymbol(const ELFFile<ELFT> *Obj, const Elf_Sym *Symbol, 279 const Elf_Sym *FirstSym, StringRef StrTable, 280 bool IsDynamic) = 0; 281 virtual void printProgramHeaders(const ELFFile<ELFT> *Obj) = 0; 282 virtual void printHashHistogram(const ELFFile<ELFT> *Obj) = 0; 283 virtual void printNotes(const ELFFile<ELFT> *Obj) = 0; 284 const ELFDumper<ELFT> *dumper() const { return Dumper; } 285 private: 286 const ELFDumper<ELFT> *Dumper; 287 }; 288 289 template <typename ELFT> class GNUStyle : public DumpStyle<ELFT> { 290 formatted_raw_ostream OS; 291 public: 292 TYPEDEF_ELF_TYPES(ELFT) 293 GNUStyle(ScopedPrinter &W, ELFDumper<ELFT> *Dumper) 294 : DumpStyle<ELFT>(Dumper), OS(W.getOStream()) {} 295 void printFileHeaders(const ELFO *Obj) override; 296 void printGroupSections(const ELFFile<ELFT> *Obj) override; 297 void printRelocations(const ELFO *Obj) override; 298 void printSections(const ELFO *Obj) override; 299 void printSymbols(const ELFO *Obj) override; 300 void printDynamicSymbols(const ELFO *Obj) override; 301 void printDynamicRelocations(const ELFO *Obj) override; 302 virtual void printSymtabMessage(const ELFO *Obj, StringRef Name, 303 size_t Offset) override; 304 void printProgramHeaders(const ELFO *Obj) override; 305 void printHashHistogram(const ELFFile<ELFT> *Obj) override; 306 void printNotes(const ELFFile<ELFT> *Obj) override; 307 308 private: 309 struct Field { 310 StringRef Str; 311 unsigned Column; 312 Field(StringRef S, unsigned Col) : Str(S), Column(Col) {} 313 Field(unsigned Col) : Str(""), Column(Col) {} 314 }; 315 316 template <typename T, typename TEnum> 317 std::string printEnum(T Value, ArrayRef<EnumEntry<TEnum>> EnumValues) { 318 for (const auto &EnumItem : EnumValues) 319 if (EnumItem.Value == Value) 320 return EnumItem.AltName; 321 return to_hexString(Value, false); 322 } 323 324 formatted_raw_ostream &printField(struct Field F) { 325 if (F.Column != 0) 326 OS.PadToColumn(F.Column); 327 OS << F.Str; 328 OS.flush(); 329 return OS; 330 } 331 void printRelocation(const ELFO *Obj, const Elf_Shdr *SymTab, 332 const Elf_Rela &R, bool IsRela); 333 void printSymbol(const ELFO *Obj, const Elf_Sym *Symbol, const Elf_Sym *First, 334 StringRef StrTable, bool IsDynamic) override; 335 std::string getSymbolSectionNdx(const ELFO *Obj, const Elf_Sym *Symbol, 336 const Elf_Sym *FirstSym); 337 void printDynamicRelocation(const ELFO *Obj, Elf_Rela R, bool IsRela); 338 bool checkTLSSections(const Elf_Phdr &Phdr, const Elf_Shdr &Sec); 339 bool checkoffsets(const Elf_Phdr &Phdr, const Elf_Shdr &Sec); 340 bool checkVMA(const Elf_Phdr &Phdr, const Elf_Shdr &Sec); 341 bool checkPTDynamic(const Elf_Phdr &Phdr, const Elf_Shdr &Sec); 342 }; 343 344 template <typename ELFT> class LLVMStyle : public DumpStyle<ELFT> { 345 public: 346 TYPEDEF_ELF_TYPES(ELFT) 347 LLVMStyle(ScopedPrinter &W, ELFDumper<ELFT> *Dumper) 348 : DumpStyle<ELFT>(Dumper), W(W) {} 349 350 void printFileHeaders(const ELFO *Obj) override; 351 void printGroupSections(const ELFFile<ELFT> *Obj) override; 352 void printRelocations(const ELFO *Obj) override; 353 void printRelocations(const Elf_Shdr *Sec, const ELFO *Obj); 354 void printSections(const ELFO *Obj) override; 355 void printSymbols(const ELFO *Obj) override; 356 void printDynamicSymbols(const ELFO *Obj) override; 357 void printDynamicRelocations(const ELFO *Obj) override; 358 void printProgramHeaders(const ELFO *Obj) override; 359 void printHashHistogram(const ELFFile<ELFT> *Obj) override; 360 void printNotes(const ELFFile<ELFT> *Obj) override; 361 362 private: 363 void printRelocation(const ELFO *Obj, Elf_Rela Rel, const Elf_Shdr *SymTab); 364 void printDynamicRelocation(const ELFO *Obj, Elf_Rela Rel); 365 void printSymbol(const ELFO *Obj, const Elf_Sym *Symbol, const Elf_Sym *First, 366 StringRef StrTable, bool IsDynamic) override; 367 ScopedPrinter &W; 368 }; 369 370 } // namespace 371 372 namespace llvm { 373 374 template <class ELFT> 375 static std::error_code createELFDumper(const ELFFile<ELFT> *Obj, 376 ScopedPrinter &Writer, 377 std::unique_ptr<ObjDumper> &Result) { 378 Result.reset(new ELFDumper<ELFT>(Obj, Writer)); 379 return readobj_error::success; 380 } 381 382 std::error_code createELFDumper(const object::ObjectFile *Obj, 383 ScopedPrinter &Writer, 384 std::unique_ptr<ObjDumper> &Result) { 385 // Little-endian 32-bit 386 if (const ELF32LEObjectFile *ELFObj = dyn_cast<ELF32LEObjectFile>(Obj)) 387 return createELFDumper(ELFObj->getELFFile(), Writer, Result); 388 389 // Big-endian 32-bit 390 if (const ELF32BEObjectFile *ELFObj = dyn_cast<ELF32BEObjectFile>(Obj)) 391 return createELFDumper(ELFObj->getELFFile(), Writer, Result); 392 393 // Little-endian 64-bit 394 if (const ELF64LEObjectFile *ELFObj = dyn_cast<ELF64LEObjectFile>(Obj)) 395 return createELFDumper(ELFObj->getELFFile(), Writer, Result); 396 397 // Big-endian 64-bit 398 if (const ELF64BEObjectFile *ELFObj = dyn_cast<ELF64BEObjectFile>(Obj)) 399 return createELFDumper(ELFObj->getELFFile(), Writer, Result); 400 401 return readobj_error::unsupported_obj_file_format; 402 } 403 404 } // namespace llvm 405 406 // Iterate through the versions needed section, and place each Elf_Vernaux 407 // in the VersionMap according to its index. 408 template <class ELFT> 409 void ELFDumper<ELFT>::LoadVersionNeeds(const Elf_Shdr *sec) const { 410 unsigned vn_size = sec->sh_size; // Size of section in bytes 411 unsigned vn_count = sec->sh_info; // Number of Verneed entries 412 const char *sec_start = (const char *)Obj->base() + sec->sh_offset; 413 const char *sec_end = sec_start + vn_size; 414 // The first Verneed entry is at the start of the section. 415 const char *p = sec_start; 416 for (unsigned i = 0; i < vn_count; i++) { 417 if (p + sizeof(Elf_Verneed) > sec_end) 418 report_fatal_error("Section ended unexpectedly while scanning " 419 "version needed records."); 420 const Elf_Verneed *vn = reinterpret_cast<const Elf_Verneed *>(p); 421 if (vn->vn_version != ELF::VER_NEED_CURRENT) 422 report_fatal_error("Unexpected verneed version"); 423 // Iterate through the Vernaux entries 424 const char *paux = p + vn->vn_aux; 425 for (unsigned j = 0; j < vn->vn_cnt; j++) { 426 if (paux + sizeof(Elf_Vernaux) > sec_end) 427 report_fatal_error("Section ended unexpected while scanning auxiliary " 428 "version needed records."); 429 const Elf_Vernaux *vna = reinterpret_cast<const Elf_Vernaux *>(paux); 430 size_t index = vna->vna_other & ELF::VERSYM_VERSION; 431 if (index >= VersionMap.size()) 432 VersionMap.resize(index + 1); 433 VersionMap[index] = VersionMapEntry(vna); 434 paux += vna->vna_next; 435 } 436 p += vn->vn_next; 437 } 438 } 439 440 // Iterate through the version definitions, and place each Elf_Verdef 441 // in the VersionMap according to its index. 442 template <class ELFT> 443 void ELFDumper<ELFT>::LoadVersionDefs(const Elf_Shdr *sec) const { 444 unsigned vd_size = sec->sh_size; // Size of section in bytes 445 unsigned vd_count = sec->sh_info; // Number of Verdef entries 446 const char *sec_start = (const char *)Obj->base() + sec->sh_offset; 447 const char *sec_end = sec_start + vd_size; 448 // The first Verdef entry is at the start of the section. 449 const char *p = sec_start; 450 for (unsigned i = 0; i < vd_count; i++) { 451 if (p + sizeof(Elf_Verdef) > sec_end) 452 report_fatal_error("Section ended unexpectedly while scanning " 453 "version definitions."); 454 const Elf_Verdef *vd = reinterpret_cast<const Elf_Verdef *>(p); 455 if (vd->vd_version != ELF::VER_DEF_CURRENT) 456 report_fatal_error("Unexpected verdef version"); 457 size_t index = vd->vd_ndx & ELF::VERSYM_VERSION; 458 if (index >= VersionMap.size()) 459 VersionMap.resize(index + 1); 460 VersionMap[index] = VersionMapEntry(vd); 461 p += vd->vd_next; 462 } 463 } 464 465 template <class ELFT> void ELFDumper<ELFT>::LoadVersionMap() const { 466 // If there is no dynamic symtab or version table, there is nothing to do. 467 if (!DynSymRegion.Addr || !dot_gnu_version_sec) 468 return; 469 470 // Has the VersionMap already been loaded? 471 if (VersionMap.size() > 0) 472 return; 473 474 // The first two version indexes are reserved. 475 // Index 0 is LOCAL, index 1 is GLOBAL. 476 VersionMap.push_back(VersionMapEntry()); 477 VersionMap.push_back(VersionMapEntry()); 478 479 if (dot_gnu_version_d_sec) 480 LoadVersionDefs(dot_gnu_version_d_sec); 481 482 if (dot_gnu_version_r_sec) 483 LoadVersionNeeds(dot_gnu_version_r_sec); 484 } 485 486 template <typename ELFO, class ELFT> 487 static void printVersionSymbolSection(ELFDumper<ELFT> *Dumper, const ELFO *Obj, 488 const typename ELFO::Elf_Shdr *Sec, 489 ScopedPrinter &W) { 490 DictScope SS(W, "Version symbols"); 491 if (!Sec) 492 return; 493 StringRef Name = unwrapOrError(Obj->getSectionName(Sec)); 494 W.printNumber("Section Name", Name, Sec->sh_name); 495 W.printHex("Address", Sec->sh_addr); 496 W.printHex("Offset", Sec->sh_offset); 497 W.printNumber("Link", Sec->sh_link); 498 499 const uint8_t *P = (const uint8_t *)Obj->base() + Sec->sh_offset; 500 StringRef StrTable = Dumper->getDynamicStringTable(); 501 502 // Same number of entries in the dynamic symbol table (DT_SYMTAB). 503 ListScope Syms(W, "Symbols"); 504 for (const typename ELFO::Elf_Sym &Sym : Dumper->dynamic_symbols()) { 505 DictScope S(W, "Symbol"); 506 std::string FullSymbolName = 507 Dumper->getFullSymbolName(&Sym, StrTable, true /* IsDynamic */); 508 W.printNumber("Version", *P); 509 W.printString("Name", FullSymbolName); 510 P += sizeof(typename ELFO::Elf_Half); 511 } 512 } 513 514 static const EnumEntry<unsigned> SymVersionFlags[] = { 515 {"Base", "BASE", VER_FLG_BASE}, 516 {"Weak", "WEAK", VER_FLG_WEAK}, 517 {"Info", "INFO", VER_FLG_INFO}}; 518 519 template <typename ELFO, class ELFT> 520 static void printVersionDefinitionSection(ELFDumper<ELFT> *Dumper, 521 const ELFO *Obj, 522 const typename ELFO::Elf_Shdr *Sec, 523 ScopedPrinter &W) { 524 typedef typename ELFO::Elf_Verdef VerDef; 525 typedef typename ELFO::Elf_Verdaux VerdAux; 526 527 DictScope SD(W, "SHT_GNU_verdef"); 528 if (!Sec) 529 return; 530 531 // The number of entries in the section SHT_GNU_verdef 532 // is determined by DT_VERDEFNUM tag. 533 unsigned VerDefsNum = 0; 534 for (const typename ELFO::Elf_Dyn &Dyn : Dumper->dynamic_table()) { 535 if (Dyn.d_tag == DT_VERDEFNUM) 536 VerDefsNum = Dyn.d_un.d_val; 537 } 538 const uint8_t *SecStartAddress = 539 (const uint8_t *)Obj->base() + Sec->sh_offset; 540 const uint8_t *SecEndAddress = SecStartAddress + Sec->sh_size; 541 const uint8_t *P = SecStartAddress; 542 const typename ELFO::Elf_Shdr *StrTab = 543 unwrapOrError(Obj->getSection(Sec->sh_link)); 544 545 while (VerDefsNum--) { 546 if (P + sizeof(VerDef) > SecEndAddress) 547 report_fatal_error("invalid offset in the section"); 548 549 auto *VD = reinterpret_cast<const VerDef *>(P); 550 DictScope Def(W, "Definition"); 551 W.printNumber("Version", VD->vd_version); 552 W.printEnum("Flags", VD->vd_flags, makeArrayRef(SymVersionFlags)); 553 W.printNumber("Index", VD->vd_ndx); 554 W.printNumber("Hash", VD->vd_hash); 555 W.printString("Name", 556 StringRef((const char *)(Obj->base() + StrTab->sh_offset + 557 VD->getAux()->vda_name))); 558 if (!VD->vd_cnt) 559 report_fatal_error("at least one definition string must exist"); 560 if (VD->vd_cnt > 2) 561 report_fatal_error("more than one predecessor is not expected"); 562 563 if (VD->vd_cnt == 2) { 564 const uint8_t *PAux = P + VD->vd_aux + VD->getAux()->vda_next; 565 const VerdAux *Aux = reinterpret_cast<const VerdAux *>(PAux); 566 W.printString("Predecessor", 567 StringRef((const char *)(Obj->base() + StrTab->sh_offset + 568 Aux->vda_name))); 569 } 570 571 P += VD->vd_next; 572 } 573 } 574 575 template <typename ELFO, class ELFT> 576 static void printVersionDependencySection(ELFDumper<ELFT> *Dumper, 577 const ELFO *Obj, 578 const typename ELFO::Elf_Shdr *Sec, 579 ScopedPrinter &W) { 580 typedef typename ELFO::Elf_Verneed VerNeed; 581 typedef typename ELFO::Elf_Vernaux VernAux; 582 583 DictScope SD(W, "SHT_GNU_verneed"); 584 if (!Sec) 585 return; 586 587 unsigned VerNeedNum = 0; 588 for (const typename ELFO::Elf_Dyn &Dyn : Dumper->dynamic_table()) 589 if (Dyn.d_tag == DT_VERNEEDNUM) 590 VerNeedNum = Dyn.d_un.d_val; 591 592 const uint8_t *SecData = (const uint8_t *)Obj->base() + Sec->sh_offset; 593 const typename ELFO::Elf_Shdr *StrTab = 594 unwrapOrError(Obj->getSection(Sec->sh_link)); 595 596 const uint8_t *P = SecData; 597 for (unsigned I = 0; I < VerNeedNum; ++I) { 598 const VerNeed *Need = reinterpret_cast<const VerNeed *>(P); 599 DictScope Entry(W, "Dependency"); 600 W.printNumber("Version", Need->vn_version); 601 W.printNumber("Count", Need->vn_cnt); 602 W.printString("FileName", 603 StringRef((const char *)(Obj->base() + StrTab->sh_offset + 604 Need->vn_file))); 605 606 const uint8_t *PAux = P + Need->vn_aux; 607 for (unsigned J = 0; J < Need->vn_cnt; ++J) { 608 const VernAux *Aux = reinterpret_cast<const VernAux *>(PAux); 609 DictScope Entry(W, "Entry"); 610 W.printNumber("Hash", Aux->vna_hash); 611 W.printEnum("Flags", Aux->vna_flags, makeArrayRef(SymVersionFlags)); 612 W.printNumber("Index", Aux->vna_other); 613 W.printString("Name", 614 StringRef((const char *)(Obj->base() + StrTab->sh_offset + 615 Aux->vna_name))); 616 PAux += Aux->vna_next; 617 } 618 P += Need->vn_next; 619 } 620 } 621 622 template <typename ELFT> void ELFDumper<ELFT>::printVersionInfo() { 623 // Dump version symbol section. 624 printVersionSymbolSection(this, Obj, dot_gnu_version_sec, W); 625 626 // Dump version definition section. 627 printVersionDefinitionSection(this, Obj, dot_gnu_version_d_sec, W); 628 629 // Dump version dependency section. 630 printVersionDependencySection(this, Obj, dot_gnu_version_r_sec, W); 631 } 632 633 template <typename ELFT> 634 StringRef ELFDumper<ELFT>::getSymbolVersion(StringRef StrTab, 635 const Elf_Sym *symb, 636 bool &IsDefault) const { 637 // This is a dynamic symbol. Look in the GNU symbol version table. 638 if (!dot_gnu_version_sec) { 639 // No version table. 640 IsDefault = false; 641 return StringRef(""); 642 } 643 644 // Determine the position in the symbol table of this entry. 645 size_t entry_index = (reinterpret_cast<uintptr_t>(symb) - 646 reinterpret_cast<uintptr_t>(DynSymRegion.Addr)) / 647 sizeof(Elf_Sym); 648 649 // Get the corresponding version index entry 650 const Elf_Versym *vs = 651 Obj->template getEntry<Elf_Versym>(dot_gnu_version_sec, entry_index); 652 size_t version_index = vs->vs_index & ELF::VERSYM_VERSION; 653 654 // Special markers for unversioned symbols. 655 if (version_index == ELF::VER_NDX_LOCAL || 656 version_index == ELF::VER_NDX_GLOBAL) { 657 IsDefault = false; 658 return StringRef(""); 659 } 660 661 // Lookup this symbol in the version table 662 LoadVersionMap(); 663 if (version_index >= VersionMap.size() || VersionMap[version_index].isNull()) 664 reportError("Invalid version entry"); 665 const VersionMapEntry &entry = VersionMap[version_index]; 666 667 // Get the version name string 668 size_t name_offset; 669 if (entry.isVerdef()) { 670 // The first Verdaux entry holds the name. 671 name_offset = entry.getVerdef()->getAux()->vda_name; 672 IsDefault = !(vs->vs_index & ELF::VERSYM_HIDDEN); 673 } else { 674 name_offset = entry.getVernaux()->vna_name; 675 IsDefault = false; 676 } 677 if (name_offset >= StrTab.size()) 678 reportError("Invalid string offset"); 679 return StringRef(StrTab.data() + name_offset); 680 } 681 682 template <typename ELFT> 683 std::string ELFDumper<ELFT>::getFullSymbolName(const Elf_Sym *Symbol, 684 StringRef StrTable, 685 bool IsDynamic) const { 686 StringRef SymbolName = unwrapOrError(Symbol->getName(StrTable)); 687 if (!IsDynamic) 688 return SymbolName; 689 690 std::string FullSymbolName(SymbolName); 691 692 bool IsDefault; 693 StringRef Version = getSymbolVersion(StrTable, &*Symbol, IsDefault); 694 FullSymbolName += (IsDefault ? "@@" : "@"); 695 FullSymbolName += Version; 696 return FullSymbolName; 697 } 698 699 template <typename ELFO> 700 static void 701 getSectionNameIndex(const ELFO &Obj, const typename ELFO::Elf_Sym *Symbol, 702 const typename ELFO::Elf_Sym *FirstSym, 703 ArrayRef<typename ELFO::Elf_Word> ShndxTable, 704 StringRef &SectionName, unsigned &SectionIndex) { 705 SectionIndex = Symbol->st_shndx; 706 if (Symbol->isUndefined()) 707 SectionName = "Undefined"; 708 else if (Symbol->isProcessorSpecific()) 709 SectionName = "Processor Specific"; 710 else if (Symbol->isOSSpecific()) 711 SectionName = "Operating System Specific"; 712 else if (Symbol->isAbsolute()) 713 SectionName = "Absolute"; 714 else if (Symbol->isCommon()) 715 SectionName = "Common"; 716 else if (Symbol->isReserved() && SectionIndex != SHN_XINDEX) 717 SectionName = "Reserved"; 718 else { 719 if (SectionIndex == SHN_XINDEX) 720 SectionIndex = 721 Obj.getExtendedSymbolTableIndex(Symbol, FirstSym, ShndxTable); 722 const typename ELFO::Elf_Shdr *Sec = 723 unwrapOrError(Obj.getSection(SectionIndex)); 724 SectionName = unwrapOrError(Obj.getSectionName(Sec)); 725 } 726 } 727 728 template <class ELFO> 729 static const typename ELFO::Elf_Shdr * 730 findNotEmptySectionByAddress(const ELFO *Obj, uint64_t Addr) { 731 for (const auto &Shdr : Obj->sections()) 732 if (Shdr.sh_addr == Addr && Shdr.sh_size > 0) 733 return &Shdr; 734 return nullptr; 735 } 736 737 template <class ELFO> 738 static const typename ELFO::Elf_Shdr *findSectionByName(const ELFO &Obj, 739 StringRef Name) { 740 for (const auto &Shdr : Obj.sections()) { 741 if (Name == unwrapOrError(Obj.getSectionName(&Shdr))) 742 return &Shdr; 743 } 744 return nullptr; 745 } 746 747 static const EnumEntry<unsigned> ElfClass[] = { 748 {"None", "none", ELF::ELFCLASSNONE}, 749 {"32-bit", "ELF32", ELF::ELFCLASS32}, 750 {"64-bit", "ELF64", ELF::ELFCLASS64}, 751 }; 752 753 static const EnumEntry<unsigned> ElfDataEncoding[] = { 754 {"None", "none", ELF::ELFDATANONE}, 755 {"LittleEndian", "2's complement, little endian", ELF::ELFDATA2LSB}, 756 {"BigEndian", "2's complement, big endian", ELF::ELFDATA2MSB}, 757 }; 758 759 static const EnumEntry<unsigned> ElfObjectFileType[] = { 760 {"None", "NONE (none)", ELF::ET_NONE}, 761 {"Relocatable", "REL (Relocatable file)", ELF::ET_REL}, 762 {"Executable", "EXEC (Executable file)", ELF::ET_EXEC}, 763 {"SharedObject", "DYN (Shared object file)", ELF::ET_DYN}, 764 {"Core", "CORE (Core file)", ELF::ET_CORE}, 765 }; 766 767 static const EnumEntry<unsigned> ElfOSABI[] = { 768 {"SystemV", "UNIX - System V", ELF::ELFOSABI_NONE}, 769 {"HPUX", "UNIX - HP-UX", ELF::ELFOSABI_HPUX}, 770 {"NetBSD", "UNIX - NetBSD", ELF::ELFOSABI_NETBSD}, 771 {"GNU/Linux", "UNIX - GNU", ELF::ELFOSABI_LINUX}, 772 {"GNU/Hurd", "GNU/Hurd", ELF::ELFOSABI_HURD}, 773 {"Solaris", "UNIX - Solaris", ELF::ELFOSABI_SOLARIS}, 774 {"AIX", "UNIX - AIX", ELF::ELFOSABI_AIX}, 775 {"IRIX", "UNIX - IRIX", ELF::ELFOSABI_IRIX}, 776 {"FreeBSD", "UNIX - FreeBSD", ELF::ELFOSABI_FREEBSD}, 777 {"TRU64", "UNIX - TRU64", ELF::ELFOSABI_TRU64}, 778 {"Modesto", "Novell - Modesto", ELF::ELFOSABI_MODESTO}, 779 {"OpenBSD", "UNIX - OpenBSD", ELF::ELFOSABI_OPENBSD}, 780 {"OpenVMS", "VMS - OpenVMS", ELF::ELFOSABI_OPENVMS}, 781 {"NSK", "HP - Non-Stop Kernel", ELF::ELFOSABI_NSK}, 782 {"AROS", "AROS", ELF::ELFOSABI_AROS}, 783 {"FenixOS", "FenixOS", ELF::ELFOSABI_FENIXOS}, 784 {"CloudABI", "CloudABI", ELF::ELFOSABI_CLOUDABI}, 785 {"C6000_ELFABI", "Bare-metal C6000", ELF::ELFOSABI_C6000_ELFABI}, 786 {"C6000_LINUX", "Linux C6000", ELF::ELFOSABI_C6000_LINUX}, 787 {"ARM", "ARM", ELF::ELFOSABI_ARM}, 788 {"Standalone", "Standalone App", ELF::ELFOSABI_STANDALONE} 789 }; 790 791 static const EnumEntry<unsigned> ElfMachineType[] = { 792 ENUM_ENT(EM_NONE, "None"), 793 ENUM_ENT(EM_M32, "WE32100"), 794 ENUM_ENT(EM_SPARC, "Sparc"), 795 ENUM_ENT(EM_386, "Intel 80386"), 796 ENUM_ENT(EM_68K, "MC68000"), 797 ENUM_ENT(EM_88K, "MC88000"), 798 ENUM_ENT(EM_IAMCU, "EM_IAMCU"), 799 ENUM_ENT(EM_860, "Intel 80860"), 800 ENUM_ENT(EM_MIPS, "MIPS R3000"), 801 ENUM_ENT(EM_S370, "IBM System/370"), 802 ENUM_ENT(EM_MIPS_RS3_LE, "MIPS R3000 little-endian"), 803 ENUM_ENT(EM_PARISC, "HPPA"), 804 ENUM_ENT(EM_VPP500, "Fujitsu VPP500"), 805 ENUM_ENT(EM_SPARC32PLUS, "Sparc v8+"), 806 ENUM_ENT(EM_960, "Intel 80960"), 807 ENUM_ENT(EM_PPC, "PowerPC"), 808 ENUM_ENT(EM_PPC64, "PowerPC64"), 809 ENUM_ENT(EM_S390, "IBM S/390"), 810 ENUM_ENT(EM_SPU, "SPU"), 811 ENUM_ENT(EM_V800, "NEC V800 series"), 812 ENUM_ENT(EM_FR20, "Fujistsu FR20"), 813 ENUM_ENT(EM_RH32, "TRW RH-32"), 814 ENUM_ENT(EM_RCE, "Motorola RCE"), 815 ENUM_ENT(EM_ARM, "ARM"), 816 ENUM_ENT(EM_ALPHA, "EM_ALPHA"), 817 ENUM_ENT(EM_SH, "Hitachi SH"), 818 ENUM_ENT(EM_SPARCV9, "Sparc v9"), 819 ENUM_ENT(EM_TRICORE, "Siemens Tricore"), 820 ENUM_ENT(EM_ARC, "ARC"), 821 ENUM_ENT(EM_H8_300, "Hitachi H8/300"), 822 ENUM_ENT(EM_H8_300H, "Hitachi H8/300H"), 823 ENUM_ENT(EM_H8S, "Hitachi H8S"), 824 ENUM_ENT(EM_H8_500, "Hitachi H8/500"), 825 ENUM_ENT(EM_IA_64, "Intel IA-64"), 826 ENUM_ENT(EM_MIPS_X, "Stanford MIPS-X"), 827 ENUM_ENT(EM_COLDFIRE, "Motorola Coldfire"), 828 ENUM_ENT(EM_68HC12, "Motorola MC68HC12 Microcontroller"), 829 ENUM_ENT(EM_MMA, "Fujitsu Multimedia Accelerator"), 830 ENUM_ENT(EM_PCP, "Siemens PCP"), 831 ENUM_ENT(EM_NCPU, "Sony nCPU embedded RISC processor"), 832 ENUM_ENT(EM_NDR1, "Denso NDR1 microprocesspr"), 833 ENUM_ENT(EM_STARCORE, "Motorola Star*Core processor"), 834 ENUM_ENT(EM_ME16, "Toyota ME16 processor"), 835 ENUM_ENT(EM_ST100, "STMicroelectronics ST100 processor"), 836 ENUM_ENT(EM_TINYJ, "Advanced Logic Corp. TinyJ embedded processor"), 837 ENUM_ENT(EM_X86_64, "Advanced Micro Devices X86-64"), 838 ENUM_ENT(EM_PDSP, "Sony DSP processor"), 839 ENUM_ENT(EM_PDP10, "Digital Equipment Corp. PDP-10"), 840 ENUM_ENT(EM_PDP11, "Digital Equipment Corp. PDP-11"), 841 ENUM_ENT(EM_FX66, "Siemens FX66 microcontroller"), 842 ENUM_ENT(EM_ST9PLUS, "STMicroelectronics ST9+ 8/16 bit microcontroller"), 843 ENUM_ENT(EM_ST7, "STMicroelectronics ST7 8-bit microcontroller"), 844 ENUM_ENT(EM_68HC16, "Motorola MC68HC16 Microcontroller"), 845 ENUM_ENT(EM_68HC11, "Motorola MC68HC11 Microcontroller"), 846 ENUM_ENT(EM_68HC08, "Motorola MC68HC08 Microcontroller"), 847 ENUM_ENT(EM_68HC05, "Motorola MC68HC05 Microcontroller"), 848 ENUM_ENT(EM_SVX, "Silicon Graphics SVx"), 849 ENUM_ENT(EM_ST19, "STMicroelectronics ST19 8-bit microcontroller"), 850 ENUM_ENT(EM_VAX, "Digital VAX"), 851 ENUM_ENT(EM_CRIS, "Axis Communications 32-bit embedded processor"), 852 ENUM_ENT(EM_JAVELIN, "Infineon Technologies 32-bit embedded cpu"), 853 ENUM_ENT(EM_FIREPATH, "Element 14 64-bit DSP processor"), 854 ENUM_ENT(EM_ZSP, "LSI Logic's 16-bit DSP processor"), 855 ENUM_ENT(EM_MMIX, "Donald Knuth's educational 64-bit processor"), 856 ENUM_ENT(EM_HUANY, "Harvard Universitys's machine-independent object format"), 857 ENUM_ENT(EM_PRISM, "Vitesse Prism"), 858 ENUM_ENT(EM_AVR, "Atmel AVR 8-bit microcontroller"), 859 ENUM_ENT(EM_FR30, "Fujitsu FR30"), 860 ENUM_ENT(EM_D10V, "Mitsubishi D10V"), 861 ENUM_ENT(EM_D30V, "Mitsubishi D30V"), 862 ENUM_ENT(EM_V850, "NEC v850"), 863 ENUM_ENT(EM_M32R, "Renesas M32R (formerly Mitsubishi M32r)"), 864 ENUM_ENT(EM_MN10300, "Matsushita MN10300"), 865 ENUM_ENT(EM_MN10200, "Matsushita MN10200"), 866 ENUM_ENT(EM_PJ, "picoJava"), 867 ENUM_ENT(EM_OPENRISC, "OpenRISC 32-bit embedded processor"), 868 ENUM_ENT(EM_ARC_COMPACT, "EM_ARC_COMPACT"), 869 ENUM_ENT(EM_XTENSA, "Tensilica Xtensa Processor"), 870 ENUM_ENT(EM_VIDEOCORE, "Alphamosaic VideoCore processor"), 871 ENUM_ENT(EM_TMM_GPP, "Thompson Multimedia General Purpose Processor"), 872 ENUM_ENT(EM_NS32K, "National Semiconductor 32000 series"), 873 ENUM_ENT(EM_TPC, "Tenor Network TPC processor"), 874 ENUM_ENT(EM_SNP1K, "EM_SNP1K"), 875 ENUM_ENT(EM_ST200, "STMicroelectronics ST200 microcontroller"), 876 ENUM_ENT(EM_IP2K, "Ubicom IP2xxx 8-bit microcontrollers"), 877 ENUM_ENT(EM_MAX, "MAX Processor"), 878 ENUM_ENT(EM_CR, "National Semiconductor CompactRISC"), 879 ENUM_ENT(EM_F2MC16, "Fujitsu F2MC16"), 880 ENUM_ENT(EM_MSP430, "Texas Instruments msp430 microcontroller"), 881 ENUM_ENT(EM_BLACKFIN, "Analog Devices Blackfin"), 882 ENUM_ENT(EM_SE_C33, "S1C33 Family of Seiko Epson processors"), 883 ENUM_ENT(EM_SEP, "Sharp embedded microprocessor"), 884 ENUM_ENT(EM_ARCA, "Arca RISC microprocessor"), 885 ENUM_ENT(EM_UNICORE, "Unicore"), 886 ENUM_ENT(EM_EXCESS, "eXcess 16/32/64-bit configurable embedded CPU"), 887 ENUM_ENT(EM_DXP, "Icera Semiconductor Inc. Deep Execution Processor"), 888 ENUM_ENT(EM_ALTERA_NIOS2, "Altera Nios"), 889 ENUM_ENT(EM_CRX, "National Semiconductor CRX microprocessor"), 890 ENUM_ENT(EM_XGATE, "Motorola XGATE embedded processor"), 891 ENUM_ENT(EM_C166, "Infineon Technologies xc16x"), 892 ENUM_ENT(EM_M16C, "Renesas M16C"), 893 ENUM_ENT(EM_DSPIC30F, "Microchip Technology dsPIC30F Digital Signal Controller"), 894 ENUM_ENT(EM_CE, "Freescale Communication Engine RISC core"), 895 ENUM_ENT(EM_M32C, "Renesas M32C"), 896 ENUM_ENT(EM_TSK3000, "Altium TSK3000 core"), 897 ENUM_ENT(EM_RS08, "Freescale RS08 embedded processor"), 898 ENUM_ENT(EM_SHARC, "EM_SHARC"), 899 ENUM_ENT(EM_ECOG2, "Cyan Technology eCOG2 microprocessor"), 900 ENUM_ENT(EM_SCORE7, "SUNPLUS S+Core"), 901 ENUM_ENT(EM_DSP24, "New Japan Radio (NJR) 24-bit DSP Processor"), 902 ENUM_ENT(EM_VIDEOCORE3, "Broadcom VideoCore III processor"), 903 ENUM_ENT(EM_LATTICEMICO32, "Lattice Mico32"), 904 ENUM_ENT(EM_SE_C17, "Seiko Epson C17 family"), 905 ENUM_ENT(EM_TI_C6000, "Texas Instruments TMS320C6000 DSP family"), 906 ENUM_ENT(EM_TI_C2000, "Texas Instruments TMS320C2000 DSP family"), 907 ENUM_ENT(EM_TI_C5500, "Texas Instruments TMS320C55x DSP family"), 908 ENUM_ENT(EM_MMDSP_PLUS, "STMicroelectronics 64bit VLIW Data Signal Processor"), 909 ENUM_ENT(EM_CYPRESS_M8C, "Cypress M8C microprocessor"), 910 ENUM_ENT(EM_R32C, "Renesas R32C series microprocessors"), 911 ENUM_ENT(EM_TRIMEDIA, "NXP Semiconductors TriMedia architecture family"), 912 ENUM_ENT(EM_HEXAGON, "Qualcomm Hexagon"), 913 ENUM_ENT(EM_8051, "Intel 8051 and variants"), 914 ENUM_ENT(EM_STXP7X, "STMicroelectronics STxP7x family"), 915 ENUM_ENT(EM_NDS32, "Andes Technology compact code size embedded RISC processor family"), 916 ENUM_ENT(EM_ECOG1, "Cyan Technology eCOG1 microprocessor"), 917 ENUM_ENT(EM_ECOG1X, "Cyan Technology eCOG1X family"), 918 ENUM_ENT(EM_MAXQ30, "Dallas Semiconductor MAXQ30 Core microcontrollers"), 919 ENUM_ENT(EM_XIMO16, "New Japan Radio (NJR) 16-bit DSP Processor"), 920 ENUM_ENT(EM_MANIK, "M2000 Reconfigurable RISC Microprocessor"), 921 ENUM_ENT(EM_CRAYNV2, "Cray Inc. NV2 vector architecture"), 922 ENUM_ENT(EM_RX, "Renesas RX"), 923 ENUM_ENT(EM_METAG, "Imagination Technologies Meta processor architecture"), 924 ENUM_ENT(EM_MCST_ELBRUS, "MCST Elbrus general purpose hardware architecture"), 925 ENUM_ENT(EM_ECOG16, "Cyan Technology eCOG16 family"), 926 ENUM_ENT(EM_CR16, "Xilinx MicroBlaze"), 927 ENUM_ENT(EM_ETPU, "Freescale Extended Time Processing Unit"), 928 ENUM_ENT(EM_SLE9X, "Infineon Technologies SLE9X core"), 929 ENUM_ENT(EM_L10M, "EM_L10M"), 930 ENUM_ENT(EM_K10M, "EM_K10M"), 931 ENUM_ENT(EM_AARCH64, "AArch64"), 932 ENUM_ENT(EM_AVR32, "Atmel AVR 8-bit microcontroller"), 933 ENUM_ENT(EM_STM8, "STMicroeletronics STM8 8-bit microcontroller"), 934 ENUM_ENT(EM_TILE64, "Tilera TILE64 multicore architecture family"), 935 ENUM_ENT(EM_TILEPRO, "Tilera TILEPro multicore architecture family"), 936 ENUM_ENT(EM_CUDA, "NVIDIA CUDA architecture"), 937 ENUM_ENT(EM_TILEGX, "Tilera TILE-Gx multicore architecture family"), 938 ENUM_ENT(EM_CLOUDSHIELD, "EM_CLOUDSHIELD"), 939 ENUM_ENT(EM_COREA_1ST, "EM_COREA_1ST"), 940 ENUM_ENT(EM_COREA_2ND, "EM_COREA_2ND"), 941 ENUM_ENT(EM_ARC_COMPACT2, "EM_ARC_COMPACT2"), 942 ENUM_ENT(EM_OPEN8, "EM_OPEN8"), 943 ENUM_ENT(EM_RL78, "Renesas RL78"), 944 ENUM_ENT(EM_VIDEOCORE5, "Broadcom VideoCore V processor"), 945 ENUM_ENT(EM_78KOR, "EM_78KOR"), 946 ENUM_ENT(EM_56800EX, "EM_56800EX"), 947 ENUM_ENT(EM_AMDGPU, "EM_AMDGPU"), 948 ENUM_ENT(EM_WEBASSEMBLY, "EM_WEBASSEMBLY"), 949 ENUM_ENT(EM_LANAI, "EM_LANAI"), 950 ENUM_ENT(EM_BPF, "EM_BPF"), 951 }; 952 953 static const EnumEntry<unsigned> ElfSymbolBindings[] = { 954 {"Local", "LOCAL", ELF::STB_LOCAL}, 955 {"Global", "GLOBAL", ELF::STB_GLOBAL}, 956 {"Weak", "WEAK", ELF::STB_WEAK}, 957 {"Unique", "UNIQUE", ELF::STB_GNU_UNIQUE}}; 958 959 static const EnumEntry<unsigned> ElfSymbolVisibilities[] = { 960 {"DEFAULT", "DEFAULT", ELF::STV_DEFAULT}, 961 {"INTERNAL", "INTERNAL", ELF::STV_INTERNAL}, 962 {"HIDDEN", "HIDDEN", ELF::STV_HIDDEN}, 963 {"PROTECTED", "PROTECTED", ELF::STV_PROTECTED}}; 964 965 static const EnumEntry<unsigned> ElfSymbolTypes[] = { 966 {"None", "NOTYPE", ELF::STT_NOTYPE}, 967 {"Object", "OBJECT", ELF::STT_OBJECT}, 968 {"Function", "FUNC", ELF::STT_FUNC}, 969 {"Section", "SECTION", ELF::STT_SECTION}, 970 {"File", "FILE", ELF::STT_FILE}, 971 {"Common", "COMMON", ELF::STT_COMMON}, 972 {"TLS", "TLS", ELF::STT_TLS}, 973 {"GNU_IFunc", "IFUNC", ELF::STT_GNU_IFUNC}}; 974 975 static const EnumEntry<unsigned> AMDGPUSymbolTypes[] = { 976 { "AMDGPU_HSA_KERNEL", ELF::STT_AMDGPU_HSA_KERNEL }, 977 { "AMDGPU_HSA_INDIRECT_FUNCTION", ELF::STT_AMDGPU_HSA_INDIRECT_FUNCTION }, 978 { "AMDGPU_HSA_METADATA", ELF::STT_AMDGPU_HSA_METADATA } 979 }; 980 981 static const char *getElfSectionType(unsigned Arch, unsigned Type) { 982 switch (Arch) { 983 case ELF::EM_ARM: 984 switch (Type) { 985 LLVM_READOBJ_ENUM_CASE(ELF, SHT_ARM_EXIDX); 986 LLVM_READOBJ_ENUM_CASE(ELF, SHT_ARM_PREEMPTMAP); 987 LLVM_READOBJ_ENUM_CASE(ELF, SHT_ARM_ATTRIBUTES); 988 LLVM_READOBJ_ENUM_CASE(ELF, SHT_ARM_DEBUGOVERLAY); 989 LLVM_READOBJ_ENUM_CASE(ELF, SHT_ARM_OVERLAYSECTION); 990 } 991 case ELF::EM_HEXAGON: 992 switch (Type) { LLVM_READOBJ_ENUM_CASE(ELF, SHT_HEX_ORDERED); } 993 case ELF::EM_X86_64: 994 switch (Type) { LLVM_READOBJ_ENUM_CASE(ELF, SHT_X86_64_UNWIND); } 995 case ELF::EM_MIPS: 996 case ELF::EM_MIPS_RS3_LE: 997 switch (Type) { 998 LLVM_READOBJ_ENUM_CASE(ELF, SHT_MIPS_REGINFO); 999 LLVM_READOBJ_ENUM_CASE(ELF, SHT_MIPS_OPTIONS); 1000 LLVM_READOBJ_ENUM_CASE(ELF, SHT_MIPS_ABIFLAGS); 1001 } 1002 } 1003 1004 switch (Type) { 1005 LLVM_READOBJ_ENUM_CASE(ELF, SHT_NULL ); 1006 LLVM_READOBJ_ENUM_CASE(ELF, SHT_PROGBITS ); 1007 LLVM_READOBJ_ENUM_CASE(ELF, SHT_SYMTAB ); 1008 LLVM_READOBJ_ENUM_CASE(ELF, SHT_STRTAB ); 1009 LLVM_READOBJ_ENUM_CASE(ELF, SHT_RELA ); 1010 LLVM_READOBJ_ENUM_CASE(ELF, SHT_HASH ); 1011 LLVM_READOBJ_ENUM_CASE(ELF, SHT_DYNAMIC ); 1012 LLVM_READOBJ_ENUM_CASE(ELF, SHT_NOTE ); 1013 LLVM_READOBJ_ENUM_CASE(ELF, SHT_NOBITS ); 1014 LLVM_READOBJ_ENUM_CASE(ELF, SHT_REL ); 1015 LLVM_READOBJ_ENUM_CASE(ELF, SHT_SHLIB ); 1016 LLVM_READOBJ_ENUM_CASE(ELF, SHT_DYNSYM ); 1017 LLVM_READOBJ_ENUM_CASE(ELF, SHT_INIT_ARRAY ); 1018 LLVM_READOBJ_ENUM_CASE(ELF, SHT_FINI_ARRAY ); 1019 LLVM_READOBJ_ENUM_CASE(ELF, SHT_PREINIT_ARRAY ); 1020 LLVM_READOBJ_ENUM_CASE(ELF, SHT_GROUP ); 1021 LLVM_READOBJ_ENUM_CASE(ELF, SHT_SYMTAB_SHNDX ); 1022 LLVM_READOBJ_ENUM_CASE(ELF, SHT_GNU_ATTRIBUTES ); 1023 LLVM_READOBJ_ENUM_CASE(ELF, SHT_GNU_HASH ); 1024 LLVM_READOBJ_ENUM_CASE(ELF, SHT_GNU_verdef ); 1025 LLVM_READOBJ_ENUM_CASE(ELF, SHT_GNU_verneed ); 1026 LLVM_READOBJ_ENUM_CASE(ELF, SHT_GNU_versym ); 1027 default: return ""; 1028 } 1029 } 1030 1031 static const char *getGroupType(uint32_t Flag) { 1032 if (Flag & ELF::GRP_COMDAT) 1033 return "COMDAT"; 1034 else 1035 return "(unknown)"; 1036 } 1037 1038 static const EnumEntry<unsigned> ElfSectionFlags[] = { 1039 ENUM_ENT(SHF_WRITE, "W"), 1040 ENUM_ENT(SHF_ALLOC, "A"), 1041 ENUM_ENT(SHF_EXCLUDE, "E"), 1042 ENUM_ENT(SHF_EXECINSTR, "X"), 1043 ENUM_ENT(SHF_MERGE, "M"), 1044 ENUM_ENT(SHF_STRINGS, "S"), 1045 ENUM_ENT(SHF_INFO_LINK, "I"), 1046 ENUM_ENT(SHF_LINK_ORDER, "L"), 1047 ENUM_ENT(SHF_OS_NONCONFORMING, "o"), 1048 ENUM_ENT(SHF_GROUP, "G"), 1049 ENUM_ENT(SHF_TLS, "T"), 1050 ENUM_ENT(SHF_MASKOS, "o"), 1051 ENUM_ENT(SHF_MASKPROC, "p"), 1052 ENUM_ENT_1(SHF_COMPRESSED), 1053 }; 1054 1055 static const EnumEntry<unsigned> ElfXCoreSectionFlags[] = { 1056 LLVM_READOBJ_ENUM_ENT(ELF, XCORE_SHF_CP_SECTION), 1057 LLVM_READOBJ_ENUM_ENT(ELF, XCORE_SHF_DP_SECTION) 1058 }; 1059 1060 static const EnumEntry<unsigned> ElfAMDGPUSectionFlags[] = { 1061 LLVM_READOBJ_ENUM_ENT(ELF, SHF_AMDGPU_HSA_GLOBAL), 1062 LLVM_READOBJ_ENUM_ENT(ELF, SHF_AMDGPU_HSA_READONLY), 1063 LLVM_READOBJ_ENUM_ENT(ELF, SHF_AMDGPU_HSA_CODE), 1064 LLVM_READOBJ_ENUM_ENT(ELF, SHF_AMDGPU_HSA_AGENT) 1065 }; 1066 1067 static const EnumEntry<unsigned> ElfHexagonSectionFlags[] = { 1068 LLVM_READOBJ_ENUM_ENT(ELF, SHF_HEX_GPREL) 1069 }; 1070 1071 static const EnumEntry<unsigned> ElfMipsSectionFlags[] = { 1072 LLVM_READOBJ_ENUM_ENT(ELF, SHF_MIPS_NODUPES), 1073 LLVM_READOBJ_ENUM_ENT(ELF, SHF_MIPS_NAMES ), 1074 LLVM_READOBJ_ENUM_ENT(ELF, SHF_MIPS_LOCAL ), 1075 LLVM_READOBJ_ENUM_ENT(ELF, SHF_MIPS_NOSTRIP), 1076 LLVM_READOBJ_ENUM_ENT(ELF, SHF_MIPS_GPREL ), 1077 LLVM_READOBJ_ENUM_ENT(ELF, SHF_MIPS_MERGE ), 1078 LLVM_READOBJ_ENUM_ENT(ELF, SHF_MIPS_ADDR ), 1079 LLVM_READOBJ_ENUM_ENT(ELF, SHF_MIPS_STRING ) 1080 }; 1081 1082 static const EnumEntry<unsigned> ElfX86_64SectionFlags[] = { 1083 LLVM_READOBJ_ENUM_ENT(ELF, SHF_X86_64_LARGE) 1084 }; 1085 1086 static std::string getGNUFlags(uint64_t Flags) { 1087 std::string Str; 1088 for (auto Entry : ElfSectionFlags) { 1089 uint64_t Flag = Entry.Value & Flags; 1090 Flags &= ~Entry.Value; 1091 switch (Flag) { 1092 case ELF::SHF_WRITE: 1093 case ELF::SHF_ALLOC: 1094 case ELF::SHF_EXECINSTR: 1095 case ELF::SHF_MERGE: 1096 case ELF::SHF_STRINGS: 1097 case ELF::SHF_INFO_LINK: 1098 case ELF::SHF_LINK_ORDER: 1099 case ELF::SHF_OS_NONCONFORMING: 1100 case ELF::SHF_GROUP: 1101 case ELF::SHF_TLS: 1102 case ELF::SHF_EXCLUDE: 1103 Str += Entry.AltName; 1104 break; 1105 default: 1106 if (Flag & ELF::SHF_MASKOS) 1107 Str += "o"; 1108 else if (Flag & ELF::SHF_MASKPROC) 1109 Str += "p"; 1110 else if (Flag) 1111 Str += "x"; 1112 } 1113 } 1114 return Str; 1115 } 1116 1117 static const char *getElfSegmentType(unsigned Arch, unsigned Type) { 1118 // Check potentially overlapped processor-specific 1119 // program header type. 1120 switch (Arch) { 1121 case ELF::EM_AMDGPU: 1122 switch (Type) { 1123 LLVM_READOBJ_ENUM_CASE(ELF, PT_AMDGPU_HSA_LOAD_GLOBAL_PROGRAM); 1124 LLVM_READOBJ_ENUM_CASE(ELF, PT_AMDGPU_HSA_LOAD_GLOBAL_AGENT); 1125 LLVM_READOBJ_ENUM_CASE(ELF, PT_AMDGPU_HSA_LOAD_READONLY_AGENT); 1126 LLVM_READOBJ_ENUM_CASE(ELF, PT_AMDGPU_HSA_LOAD_CODE_AGENT); 1127 } 1128 case ELF::EM_ARM: 1129 switch (Type) { 1130 LLVM_READOBJ_ENUM_CASE(ELF, PT_ARM_EXIDX); 1131 } 1132 case ELF::EM_MIPS: 1133 case ELF::EM_MIPS_RS3_LE: 1134 switch (Type) { 1135 LLVM_READOBJ_ENUM_CASE(ELF, PT_MIPS_REGINFO); 1136 LLVM_READOBJ_ENUM_CASE(ELF, PT_MIPS_RTPROC); 1137 LLVM_READOBJ_ENUM_CASE(ELF, PT_MIPS_OPTIONS); 1138 LLVM_READOBJ_ENUM_CASE(ELF, PT_MIPS_ABIFLAGS); 1139 } 1140 } 1141 1142 switch (Type) { 1143 LLVM_READOBJ_ENUM_CASE(ELF, PT_NULL ); 1144 LLVM_READOBJ_ENUM_CASE(ELF, PT_LOAD ); 1145 LLVM_READOBJ_ENUM_CASE(ELF, PT_DYNAMIC); 1146 LLVM_READOBJ_ENUM_CASE(ELF, PT_INTERP ); 1147 LLVM_READOBJ_ENUM_CASE(ELF, PT_NOTE ); 1148 LLVM_READOBJ_ENUM_CASE(ELF, PT_SHLIB ); 1149 LLVM_READOBJ_ENUM_CASE(ELF, PT_PHDR ); 1150 LLVM_READOBJ_ENUM_CASE(ELF, PT_TLS ); 1151 1152 LLVM_READOBJ_ENUM_CASE(ELF, PT_GNU_EH_FRAME); 1153 LLVM_READOBJ_ENUM_CASE(ELF, PT_SUNW_UNWIND); 1154 1155 LLVM_READOBJ_ENUM_CASE(ELF, PT_GNU_STACK); 1156 LLVM_READOBJ_ENUM_CASE(ELF, PT_GNU_RELRO); 1157 default: return ""; 1158 } 1159 } 1160 1161 static std::string getElfPtType(unsigned Arch, unsigned Type) { 1162 switch (Type) { 1163 LLVM_READOBJ_PHDR_ENUM(ELF, PT_NULL) 1164 LLVM_READOBJ_PHDR_ENUM(ELF, PT_LOAD) 1165 LLVM_READOBJ_PHDR_ENUM(ELF, PT_DYNAMIC) 1166 LLVM_READOBJ_PHDR_ENUM(ELF, PT_INTERP) 1167 LLVM_READOBJ_PHDR_ENUM(ELF, PT_NOTE) 1168 LLVM_READOBJ_PHDR_ENUM(ELF, PT_SHLIB) 1169 LLVM_READOBJ_PHDR_ENUM(ELF, PT_PHDR) 1170 LLVM_READOBJ_PHDR_ENUM(ELF, PT_TLS) 1171 LLVM_READOBJ_PHDR_ENUM(ELF, PT_GNU_EH_FRAME) 1172 LLVM_READOBJ_PHDR_ENUM(ELF, PT_SUNW_UNWIND) 1173 LLVM_READOBJ_PHDR_ENUM(ELF, PT_GNU_STACK) 1174 LLVM_READOBJ_PHDR_ENUM(ELF, PT_GNU_RELRO) 1175 default: 1176 // All machine specific PT_* types 1177 switch (Arch) { 1178 case ELF::EM_AMDGPU: 1179 switch (Type) { 1180 LLVM_READOBJ_ENUM_CASE(ELF, PT_AMDGPU_HSA_LOAD_GLOBAL_PROGRAM); 1181 LLVM_READOBJ_ENUM_CASE(ELF, PT_AMDGPU_HSA_LOAD_GLOBAL_AGENT); 1182 LLVM_READOBJ_ENUM_CASE(ELF, PT_AMDGPU_HSA_LOAD_READONLY_AGENT); 1183 LLVM_READOBJ_ENUM_CASE(ELF, PT_AMDGPU_HSA_LOAD_CODE_AGENT); 1184 } 1185 return ""; 1186 case ELF::EM_ARM: 1187 if (Type == ELF::PT_ARM_EXIDX) 1188 return "EXIDX"; 1189 return ""; 1190 case ELF::EM_MIPS: 1191 case ELF::EM_MIPS_RS3_LE: 1192 switch (Type) { 1193 case PT_MIPS_REGINFO: 1194 return "REGINFO"; 1195 case PT_MIPS_RTPROC: 1196 return "RTPROC"; 1197 case PT_MIPS_OPTIONS: 1198 return "OPTIONS"; 1199 case PT_MIPS_ABIFLAGS: 1200 return "ABIFLAGS"; 1201 } 1202 return ""; 1203 } 1204 } 1205 return std::string("<unknown>: ") + to_string(format_hex(Type, 1)); 1206 } 1207 1208 static const EnumEntry<unsigned> ElfSegmentFlags[] = { 1209 LLVM_READOBJ_ENUM_ENT(ELF, PF_X), 1210 LLVM_READOBJ_ENUM_ENT(ELF, PF_W), 1211 LLVM_READOBJ_ENUM_ENT(ELF, PF_R) 1212 }; 1213 1214 static const EnumEntry<unsigned> ElfHeaderMipsFlags[] = { 1215 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_NOREORDER), 1216 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_PIC), 1217 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_CPIC), 1218 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_ABI2), 1219 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_32BITMODE), 1220 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_FP64), 1221 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_NAN2008), 1222 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_ABI_O32), 1223 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_ABI_O64), 1224 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_ABI_EABI32), 1225 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_ABI_EABI64), 1226 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_MACH_3900), 1227 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_MACH_4010), 1228 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_MACH_4100), 1229 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_MACH_4650), 1230 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_MACH_4120), 1231 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_MACH_4111), 1232 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_MACH_SB1), 1233 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_MACH_OCTEON), 1234 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_MACH_XLR), 1235 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_MACH_OCTEON2), 1236 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_MACH_OCTEON3), 1237 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_MACH_5400), 1238 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_MACH_5900), 1239 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_MACH_5500), 1240 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_MACH_9000), 1241 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_MACH_LS2E), 1242 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_MACH_LS2F), 1243 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_MACH_LS3A), 1244 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_MICROMIPS), 1245 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_ARCH_ASE_M16), 1246 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_ARCH_ASE_MDMX), 1247 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_ARCH_1), 1248 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_ARCH_2), 1249 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_ARCH_3), 1250 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_ARCH_4), 1251 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_ARCH_5), 1252 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_ARCH_32), 1253 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_ARCH_64), 1254 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_ARCH_32R2), 1255 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_ARCH_64R2), 1256 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_ARCH_32R6), 1257 LLVM_READOBJ_ENUM_ENT(ELF, EF_MIPS_ARCH_64R6) 1258 }; 1259 1260 static const EnumEntry<unsigned> ElfSymOtherFlags[] = { 1261 LLVM_READOBJ_ENUM_ENT(ELF, STV_INTERNAL), 1262 LLVM_READOBJ_ENUM_ENT(ELF, STV_HIDDEN), 1263 LLVM_READOBJ_ENUM_ENT(ELF, STV_PROTECTED) 1264 }; 1265 1266 static const EnumEntry<unsigned> ElfMipsSymOtherFlags[] = { 1267 LLVM_READOBJ_ENUM_ENT(ELF, STO_MIPS_OPTIONAL), 1268 LLVM_READOBJ_ENUM_ENT(ELF, STO_MIPS_PLT), 1269 LLVM_READOBJ_ENUM_ENT(ELF, STO_MIPS_PIC), 1270 LLVM_READOBJ_ENUM_ENT(ELF, STO_MIPS_MICROMIPS) 1271 }; 1272 1273 static const EnumEntry<unsigned> ElfMips16SymOtherFlags[] = { 1274 LLVM_READOBJ_ENUM_ENT(ELF, STO_MIPS_OPTIONAL), 1275 LLVM_READOBJ_ENUM_ENT(ELF, STO_MIPS_PLT), 1276 LLVM_READOBJ_ENUM_ENT(ELF, STO_MIPS_MIPS16) 1277 }; 1278 1279 static const char *getElfMipsOptionsOdkType(unsigned Odk) { 1280 switch (Odk) { 1281 LLVM_READOBJ_ENUM_CASE(ELF, ODK_NULL); 1282 LLVM_READOBJ_ENUM_CASE(ELF, ODK_REGINFO); 1283 LLVM_READOBJ_ENUM_CASE(ELF, ODK_EXCEPTIONS); 1284 LLVM_READOBJ_ENUM_CASE(ELF, ODK_PAD); 1285 LLVM_READOBJ_ENUM_CASE(ELF, ODK_HWPATCH); 1286 LLVM_READOBJ_ENUM_CASE(ELF, ODK_FILL); 1287 LLVM_READOBJ_ENUM_CASE(ELF, ODK_TAGS); 1288 LLVM_READOBJ_ENUM_CASE(ELF, ODK_HWAND); 1289 LLVM_READOBJ_ENUM_CASE(ELF, ODK_HWOR); 1290 LLVM_READOBJ_ENUM_CASE(ELF, ODK_GP_GROUP); 1291 LLVM_READOBJ_ENUM_CASE(ELF, ODK_IDENT); 1292 LLVM_READOBJ_ENUM_CASE(ELF, ODK_PAGESIZE); 1293 default: 1294 return "Unknown"; 1295 } 1296 } 1297 1298 template <typename ELFT> 1299 ELFDumper<ELFT>::ELFDumper(const ELFFile<ELFT> *Obj, ScopedPrinter &Writer) 1300 : ObjDumper(Writer), Obj(Obj) { 1301 1302 SmallVector<const Elf_Phdr *, 4> LoadSegments; 1303 for (const Elf_Phdr &Phdr : Obj->program_headers()) { 1304 if (Phdr.p_type == ELF::PT_DYNAMIC) { 1305 DynamicTable = createDRIFrom(&Phdr, sizeof(Elf_Dyn)); 1306 continue; 1307 } 1308 if (Phdr.p_type != ELF::PT_LOAD || Phdr.p_filesz == 0) 1309 continue; 1310 LoadSegments.push_back(&Phdr); 1311 } 1312 1313 for (const Elf_Shdr &Sec : Obj->sections()) { 1314 switch (Sec.sh_type) { 1315 case ELF::SHT_SYMTAB: 1316 if (DotSymtabSec != nullptr) 1317 reportError("Multilpe SHT_SYMTAB"); 1318 DotSymtabSec = &Sec; 1319 break; 1320 case ELF::SHT_DYNSYM: 1321 if (DynSymRegion.Size) 1322 reportError("Multilpe SHT_DYNSYM"); 1323 DynSymRegion = createDRIFrom(&Sec); 1324 // This is only used (if Elf_Shdr present)for naming section in GNU style 1325 DynSymtabName = unwrapOrError(Obj->getSectionName(&Sec)); 1326 break; 1327 case ELF::SHT_SYMTAB_SHNDX: 1328 ShndxTable = unwrapOrError(Obj->getSHNDXTable(Sec)); 1329 break; 1330 case ELF::SHT_GNU_versym: 1331 if (dot_gnu_version_sec != nullptr) 1332 reportError("Multiple SHT_GNU_versym"); 1333 dot_gnu_version_sec = &Sec; 1334 break; 1335 case ELF::SHT_GNU_verdef: 1336 if (dot_gnu_version_d_sec != nullptr) 1337 reportError("Multiple SHT_GNU_verdef"); 1338 dot_gnu_version_d_sec = &Sec; 1339 break; 1340 case ELF::SHT_GNU_verneed: 1341 if (dot_gnu_version_r_sec != nullptr) 1342 reportError("Multilpe SHT_GNU_verneed"); 1343 dot_gnu_version_r_sec = &Sec; 1344 break; 1345 } 1346 } 1347 1348 parseDynamicTable(LoadSegments); 1349 1350 if (opts::Output == opts::GNU) 1351 ELFDumperStyle.reset(new GNUStyle<ELFT>(Writer, this)); 1352 else 1353 ELFDumperStyle.reset(new LLVMStyle<ELFT>(Writer, this)); 1354 } 1355 1356 template <typename ELFT> 1357 void ELFDumper<ELFT>::parseDynamicTable( 1358 ArrayRef<const Elf_Phdr *> LoadSegments) { 1359 auto toMappedAddr = [&](uint64_t VAddr) -> const uint8_t * { 1360 const Elf_Phdr *const *I = std::upper_bound( 1361 LoadSegments.begin(), LoadSegments.end(), VAddr, compareAddr<ELFT>); 1362 if (I == LoadSegments.begin()) 1363 report_fatal_error("Virtual address is not in any segment"); 1364 --I; 1365 const Elf_Phdr &Phdr = **I; 1366 uint64_t Delta = VAddr - Phdr.p_vaddr; 1367 if (Delta >= Phdr.p_filesz) 1368 report_fatal_error("Virtual address is not in any segment"); 1369 return Obj->base() + Phdr.p_offset + Delta; 1370 }; 1371 1372 uint64_t SONameOffset = 0; 1373 const char *StringTableBegin = nullptr; 1374 uint64_t StringTableSize = 0; 1375 for (const Elf_Dyn &Dyn : dynamic_table()) { 1376 switch (Dyn.d_tag) { 1377 case ELF::DT_HASH: 1378 HashTable = 1379 reinterpret_cast<const Elf_Hash *>(toMappedAddr(Dyn.getPtr())); 1380 break; 1381 case ELF::DT_GNU_HASH: 1382 GnuHashTable = 1383 reinterpret_cast<const Elf_GnuHash *>(toMappedAddr(Dyn.getPtr())); 1384 break; 1385 case ELF::DT_STRTAB: 1386 StringTableBegin = (const char *)toMappedAddr(Dyn.getPtr()); 1387 break; 1388 case ELF::DT_STRSZ: 1389 StringTableSize = Dyn.getVal(); 1390 break; 1391 case ELF::DT_SYMTAB: 1392 DynSymRegion.Addr = toMappedAddr(Dyn.getPtr()); 1393 DynSymRegion.EntSize = sizeof(Elf_Sym); 1394 break; 1395 case ELF::DT_RELA: 1396 DynRelaRegion.Addr = toMappedAddr(Dyn.getPtr()); 1397 break; 1398 case ELF::DT_RELASZ: 1399 DynRelaRegion.Size = Dyn.getVal(); 1400 break; 1401 case ELF::DT_RELAENT: 1402 DynRelaRegion.EntSize = Dyn.getVal(); 1403 break; 1404 case ELF::DT_SONAME: 1405 SONameOffset = Dyn.getVal(); 1406 break; 1407 case ELF::DT_REL: 1408 DynRelRegion.Addr = toMappedAddr(Dyn.getPtr()); 1409 break; 1410 case ELF::DT_RELSZ: 1411 DynRelRegion.Size = Dyn.getVal(); 1412 break; 1413 case ELF::DT_RELENT: 1414 DynRelRegion.EntSize = Dyn.getVal(); 1415 break; 1416 case ELF::DT_PLTREL: 1417 if (Dyn.getVal() == DT_REL) 1418 DynPLTRelRegion.EntSize = sizeof(Elf_Rel); 1419 else if (Dyn.getVal() == DT_RELA) 1420 DynPLTRelRegion.EntSize = sizeof(Elf_Rela); 1421 else 1422 reportError(Twine("unknown DT_PLTREL value of ") + 1423 Twine((uint64_t)Dyn.getVal())); 1424 break; 1425 case ELF::DT_JMPREL: 1426 DynPLTRelRegion.Addr = toMappedAddr(Dyn.getPtr()); 1427 break; 1428 case ELF::DT_PLTRELSZ: 1429 DynPLTRelRegion.Size = Dyn.getVal(); 1430 break; 1431 } 1432 } 1433 if (StringTableBegin) 1434 DynamicStringTable = StringRef(StringTableBegin, StringTableSize); 1435 if (SONameOffset) 1436 SOName = getDynamicString(SONameOffset); 1437 } 1438 1439 template <typename ELFT> 1440 typename ELFDumper<ELFT>::Elf_Rel_Range ELFDumper<ELFT>::dyn_rels() const { 1441 return DynRelRegion.getAsArrayRef<Elf_Rel>(); 1442 } 1443 1444 template <typename ELFT> 1445 typename ELFDumper<ELFT>::Elf_Rela_Range ELFDumper<ELFT>::dyn_relas() const { 1446 return DynRelaRegion.getAsArrayRef<Elf_Rela>(); 1447 } 1448 1449 template<class ELFT> 1450 void ELFDumper<ELFT>::printFileHeaders() { 1451 ELFDumperStyle->printFileHeaders(Obj); 1452 } 1453 1454 template<class ELFT> 1455 void ELFDumper<ELFT>::printSections() { 1456 ELFDumperStyle->printSections(Obj); 1457 } 1458 1459 template<class ELFT> 1460 void ELFDumper<ELFT>::printRelocations() { 1461 ELFDumperStyle->printRelocations(Obj); 1462 } 1463 1464 template <class ELFT> void ELFDumper<ELFT>::printProgramHeaders() { 1465 ELFDumperStyle->printProgramHeaders(Obj); 1466 } 1467 1468 template <class ELFT> void ELFDumper<ELFT>::printDynamicRelocations() { 1469 ELFDumperStyle->printDynamicRelocations(Obj); 1470 } 1471 1472 template<class ELFT> 1473 void ELFDumper<ELFT>::printSymbols() { 1474 ELFDumperStyle->printSymbols(Obj); 1475 } 1476 1477 template<class ELFT> 1478 void ELFDumper<ELFT>::printDynamicSymbols() { 1479 ELFDumperStyle->printDynamicSymbols(Obj); 1480 } 1481 1482 template <class ELFT> void ELFDumper<ELFT>::printHashHistogram() { 1483 ELFDumperStyle->printHashHistogram(Obj); 1484 } 1485 1486 template <class ELFT> void ELFDumper<ELFT>::printNotes() { 1487 ELFDumperStyle->printNotes(Obj); 1488 } 1489 1490 #define LLVM_READOBJ_TYPE_CASE(name) \ 1491 case DT_##name: return #name 1492 1493 static const char *getTypeString(uint64_t Type) { 1494 switch (Type) { 1495 LLVM_READOBJ_TYPE_CASE(BIND_NOW); 1496 LLVM_READOBJ_TYPE_CASE(DEBUG); 1497 LLVM_READOBJ_TYPE_CASE(FINI); 1498 LLVM_READOBJ_TYPE_CASE(FINI_ARRAY); 1499 LLVM_READOBJ_TYPE_CASE(FINI_ARRAYSZ); 1500 LLVM_READOBJ_TYPE_CASE(FLAGS); 1501 LLVM_READOBJ_TYPE_CASE(FLAGS_1); 1502 LLVM_READOBJ_TYPE_CASE(HASH); 1503 LLVM_READOBJ_TYPE_CASE(INIT); 1504 LLVM_READOBJ_TYPE_CASE(INIT_ARRAY); 1505 LLVM_READOBJ_TYPE_CASE(INIT_ARRAYSZ); 1506 LLVM_READOBJ_TYPE_CASE(PREINIT_ARRAY); 1507 LLVM_READOBJ_TYPE_CASE(PREINIT_ARRAYSZ); 1508 LLVM_READOBJ_TYPE_CASE(JMPREL); 1509 LLVM_READOBJ_TYPE_CASE(NEEDED); 1510 LLVM_READOBJ_TYPE_CASE(NULL); 1511 LLVM_READOBJ_TYPE_CASE(PLTGOT); 1512 LLVM_READOBJ_TYPE_CASE(PLTREL); 1513 LLVM_READOBJ_TYPE_CASE(PLTRELSZ); 1514 LLVM_READOBJ_TYPE_CASE(REL); 1515 LLVM_READOBJ_TYPE_CASE(RELA); 1516 LLVM_READOBJ_TYPE_CASE(RELENT); 1517 LLVM_READOBJ_TYPE_CASE(RELSZ); 1518 LLVM_READOBJ_TYPE_CASE(RELAENT); 1519 LLVM_READOBJ_TYPE_CASE(RELASZ); 1520 LLVM_READOBJ_TYPE_CASE(RPATH); 1521 LLVM_READOBJ_TYPE_CASE(RUNPATH); 1522 LLVM_READOBJ_TYPE_CASE(SONAME); 1523 LLVM_READOBJ_TYPE_CASE(STRSZ); 1524 LLVM_READOBJ_TYPE_CASE(STRTAB); 1525 LLVM_READOBJ_TYPE_CASE(SYMBOLIC); 1526 LLVM_READOBJ_TYPE_CASE(SYMENT); 1527 LLVM_READOBJ_TYPE_CASE(SYMTAB); 1528 LLVM_READOBJ_TYPE_CASE(TEXTREL); 1529 LLVM_READOBJ_TYPE_CASE(VERDEF); 1530 LLVM_READOBJ_TYPE_CASE(VERDEFNUM); 1531 LLVM_READOBJ_TYPE_CASE(VERNEED); 1532 LLVM_READOBJ_TYPE_CASE(VERNEEDNUM); 1533 LLVM_READOBJ_TYPE_CASE(VERSYM); 1534 LLVM_READOBJ_TYPE_CASE(RELACOUNT); 1535 LLVM_READOBJ_TYPE_CASE(RELCOUNT); 1536 LLVM_READOBJ_TYPE_CASE(GNU_HASH); 1537 LLVM_READOBJ_TYPE_CASE(TLSDESC_PLT); 1538 LLVM_READOBJ_TYPE_CASE(TLSDESC_GOT); 1539 LLVM_READOBJ_TYPE_CASE(MIPS_RLD_VERSION); 1540 LLVM_READOBJ_TYPE_CASE(MIPS_RLD_MAP_REL); 1541 LLVM_READOBJ_TYPE_CASE(MIPS_FLAGS); 1542 LLVM_READOBJ_TYPE_CASE(MIPS_BASE_ADDRESS); 1543 LLVM_READOBJ_TYPE_CASE(MIPS_LOCAL_GOTNO); 1544 LLVM_READOBJ_TYPE_CASE(MIPS_SYMTABNO); 1545 LLVM_READOBJ_TYPE_CASE(MIPS_UNREFEXTNO); 1546 LLVM_READOBJ_TYPE_CASE(MIPS_GOTSYM); 1547 LLVM_READOBJ_TYPE_CASE(MIPS_RLD_MAP); 1548 LLVM_READOBJ_TYPE_CASE(MIPS_PLTGOT); 1549 LLVM_READOBJ_TYPE_CASE(MIPS_OPTIONS); 1550 LLVM_READOBJ_TYPE_CASE(AUXILIARY); 1551 default: return "unknown"; 1552 } 1553 } 1554 1555 #undef LLVM_READOBJ_TYPE_CASE 1556 1557 #define LLVM_READOBJ_DT_FLAG_ENT(prefix, enum) \ 1558 { #enum, prefix##_##enum } 1559 1560 static const EnumEntry<unsigned> ElfDynamicDTFlags[] = { 1561 LLVM_READOBJ_DT_FLAG_ENT(DF, ORIGIN), 1562 LLVM_READOBJ_DT_FLAG_ENT(DF, SYMBOLIC), 1563 LLVM_READOBJ_DT_FLAG_ENT(DF, TEXTREL), 1564 LLVM_READOBJ_DT_FLAG_ENT(DF, BIND_NOW), 1565 LLVM_READOBJ_DT_FLAG_ENT(DF, STATIC_TLS) 1566 }; 1567 1568 static const EnumEntry<unsigned> ElfDynamicDTFlags1[] = { 1569 LLVM_READOBJ_DT_FLAG_ENT(DF_1, NOW), 1570 LLVM_READOBJ_DT_FLAG_ENT(DF_1, GLOBAL), 1571 LLVM_READOBJ_DT_FLAG_ENT(DF_1, GROUP), 1572 LLVM_READOBJ_DT_FLAG_ENT(DF_1, NODELETE), 1573 LLVM_READOBJ_DT_FLAG_ENT(DF_1, LOADFLTR), 1574 LLVM_READOBJ_DT_FLAG_ENT(DF_1, INITFIRST), 1575 LLVM_READOBJ_DT_FLAG_ENT(DF_1, NOOPEN), 1576 LLVM_READOBJ_DT_FLAG_ENT(DF_1, ORIGIN), 1577 LLVM_READOBJ_DT_FLAG_ENT(DF_1, DIRECT), 1578 LLVM_READOBJ_DT_FLAG_ENT(DF_1, TRANS), 1579 LLVM_READOBJ_DT_FLAG_ENT(DF_1, INTERPOSE), 1580 LLVM_READOBJ_DT_FLAG_ENT(DF_1, NODEFLIB), 1581 LLVM_READOBJ_DT_FLAG_ENT(DF_1, NODUMP), 1582 LLVM_READOBJ_DT_FLAG_ENT(DF_1, CONFALT), 1583 LLVM_READOBJ_DT_FLAG_ENT(DF_1, ENDFILTEE), 1584 LLVM_READOBJ_DT_FLAG_ENT(DF_1, DISPRELDNE), 1585 LLVM_READOBJ_DT_FLAG_ENT(DF_1, NODIRECT), 1586 LLVM_READOBJ_DT_FLAG_ENT(DF_1, IGNMULDEF), 1587 LLVM_READOBJ_DT_FLAG_ENT(DF_1, NOKSYMS), 1588 LLVM_READOBJ_DT_FLAG_ENT(DF_1, NOHDR), 1589 LLVM_READOBJ_DT_FLAG_ENT(DF_1, EDITED), 1590 LLVM_READOBJ_DT_FLAG_ENT(DF_1, NORELOC), 1591 LLVM_READOBJ_DT_FLAG_ENT(DF_1, SYMINTPOSE), 1592 LLVM_READOBJ_DT_FLAG_ENT(DF_1, GLOBAUDIT), 1593 LLVM_READOBJ_DT_FLAG_ENT(DF_1, SINGLETON) 1594 }; 1595 1596 static const EnumEntry<unsigned> ElfDynamicDTMipsFlags[] = { 1597 LLVM_READOBJ_DT_FLAG_ENT(RHF, NONE), 1598 LLVM_READOBJ_DT_FLAG_ENT(RHF, QUICKSTART), 1599 LLVM_READOBJ_DT_FLAG_ENT(RHF, NOTPOT), 1600 LLVM_READOBJ_DT_FLAG_ENT(RHS, NO_LIBRARY_REPLACEMENT), 1601 LLVM_READOBJ_DT_FLAG_ENT(RHF, NO_MOVE), 1602 LLVM_READOBJ_DT_FLAG_ENT(RHF, SGI_ONLY), 1603 LLVM_READOBJ_DT_FLAG_ENT(RHF, GUARANTEE_INIT), 1604 LLVM_READOBJ_DT_FLAG_ENT(RHF, DELTA_C_PLUS_PLUS), 1605 LLVM_READOBJ_DT_FLAG_ENT(RHF, GUARANTEE_START_INIT), 1606 LLVM_READOBJ_DT_FLAG_ENT(RHF, PIXIE), 1607 LLVM_READOBJ_DT_FLAG_ENT(RHF, DEFAULT_DELAY_LOAD), 1608 LLVM_READOBJ_DT_FLAG_ENT(RHF, REQUICKSTART), 1609 LLVM_READOBJ_DT_FLAG_ENT(RHF, REQUICKSTARTED), 1610 LLVM_READOBJ_DT_FLAG_ENT(RHF, CORD), 1611 LLVM_READOBJ_DT_FLAG_ENT(RHF, NO_UNRES_UNDEF), 1612 LLVM_READOBJ_DT_FLAG_ENT(RHF, RLD_ORDER_SAFE) 1613 }; 1614 1615 #undef LLVM_READOBJ_DT_FLAG_ENT 1616 1617 template <typename T, typename TFlag> 1618 void printFlags(T Value, ArrayRef<EnumEntry<TFlag>> Flags, raw_ostream &OS) { 1619 typedef EnumEntry<TFlag> FlagEntry; 1620 typedef SmallVector<FlagEntry, 10> FlagVector; 1621 FlagVector SetFlags; 1622 1623 for (const auto &Flag : Flags) { 1624 if (Flag.Value == 0) 1625 continue; 1626 1627 if ((Value & Flag.Value) == Flag.Value) 1628 SetFlags.push_back(Flag); 1629 } 1630 1631 for (const auto &Flag : SetFlags) { 1632 OS << Flag.Name << " "; 1633 } 1634 } 1635 1636 template <class ELFT> 1637 StringRef ELFDumper<ELFT>::getDynamicString(uint64_t Value) const { 1638 if (Value >= DynamicStringTable.size()) 1639 reportError("Invalid dynamic string table reference"); 1640 return StringRef(DynamicStringTable.data() + Value); 1641 } 1642 1643 template <class ELFT> 1644 void ELFDumper<ELFT>::printValue(uint64_t Type, uint64_t Value) { 1645 raw_ostream &OS = W.getOStream(); 1646 const char* ConvChar = (opts::Output == opts::GNU) ? "0x%" PRIx64 : "0x%" PRIX64; 1647 switch (Type) { 1648 case DT_PLTREL: 1649 if (Value == DT_REL) { 1650 OS << "REL"; 1651 break; 1652 } else if (Value == DT_RELA) { 1653 OS << "RELA"; 1654 break; 1655 } 1656 LLVM_FALLTHROUGH; 1657 case DT_PLTGOT: 1658 case DT_HASH: 1659 case DT_STRTAB: 1660 case DT_SYMTAB: 1661 case DT_RELA: 1662 case DT_INIT: 1663 case DT_FINI: 1664 case DT_REL: 1665 case DT_JMPREL: 1666 case DT_INIT_ARRAY: 1667 case DT_FINI_ARRAY: 1668 case DT_PREINIT_ARRAY: 1669 case DT_DEBUG: 1670 case DT_VERDEF: 1671 case DT_VERNEED: 1672 case DT_VERSYM: 1673 case DT_GNU_HASH: 1674 case DT_NULL: 1675 case DT_MIPS_BASE_ADDRESS: 1676 case DT_MIPS_GOTSYM: 1677 case DT_MIPS_RLD_MAP: 1678 case DT_MIPS_RLD_MAP_REL: 1679 case DT_MIPS_PLTGOT: 1680 case DT_MIPS_OPTIONS: 1681 OS << format(ConvChar, Value); 1682 break; 1683 case DT_RELACOUNT: 1684 case DT_RELCOUNT: 1685 case DT_VERDEFNUM: 1686 case DT_VERNEEDNUM: 1687 case DT_MIPS_RLD_VERSION: 1688 case DT_MIPS_LOCAL_GOTNO: 1689 case DT_MIPS_SYMTABNO: 1690 case DT_MIPS_UNREFEXTNO: 1691 OS << Value; 1692 break; 1693 case DT_PLTRELSZ: 1694 case DT_RELASZ: 1695 case DT_RELAENT: 1696 case DT_STRSZ: 1697 case DT_SYMENT: 1698 case DT_RELSZ: 1699 case DT_RELENT: 1700 case DT_INIT_ARRAYSZ: 1701 case DT_FINI_ARRAYSZ: 1702 case DT_PREINIT_ARRAYSZ: 1703 OS << Value << " (bytes)"; 1704 break; 1705 case DT_NEEDED: 1706 OS << "SharedLibrary (" << getDynamicString(Value) << ")"; 1707 break; 1708 case DT_SONAME: 1709 OS << "LibrarySoname (" << getDynamicString(Value) << ")"; 1710 break; 1711 case DT_AUXILIARY: 1712 OS << "Auxiliary library: [" << getDynamicString(Value) << "]"; 1713 break; 1714 case DT_RPATH: 1715 case DT_RUNPATH: 1716 OS << getDynamicString(Value); 1717 break; 1718 case DT_MIPS_FLAGS: 1719 printFlags(Value, makeArrayRef(ElfDynamicDTMipsFlags), OS); 1720 break; 1721 case DT_FLAGS: 1722 printFlags(Value, makeArrayRef(ElfDynamicDTFlags), OS); 1723 break; 1724 case DT_FLAGS_1: 1725 printFlags(Value, makeArrayRef(ElfDynamicDTFlags1), OS); 1726 break; 1727 default: 1728 OS << format(ConvChar, Value); 1729 break; 1730 } 1731 } 1732 1733 template<class ELFT> 1734 void ELFDumper<ELFT>::printUnwindInfo() { 1735 W.startLine() << "UnwindInfo not implemented.\n"; 1736 } 1737 1738 namespace { 1739 template <> void ELFDumper<ELFType<support::little, false>>::printUnwindInfo() { 1740 const unsigned Machine = Obj->getHeader()->e_machine; 1741 if (Machine == EM_ARM) { 1742 ARM::EHABI::PrinterContext<ELFType<support::little, false>> Ctx( 1743 W, Obj, DotSymtabSec); 1744 return Ctx.PrintUnwindInformation(); 1745 } 1746 W.startLine() << "UnwindInfo not implemented.\n"; 1747 } 1748 } 1749 1750 template<class ELFT> 1751 void ELFDumper<ELFT>::printDynamicTable() { 1752 auto I = dynamic_table().begin(); 1753 auto E = dynamic_table().end(); 1754 1755 if (I == E) 1756 return; 1757 1758 --E; 1759 while (I != E && E->getTag() == ELF::DT_NULL) 1760 --E; 1761 if (E->getTag() != ELF::DT_NULL) 1762 ++E; 1763 ++E; 1764 1765 ptrdiff_t Total = std::distance(I, E); 1766 if (Total == 0) 1767 return; 1768 1769 raw_ostream &OS = W.getOStream(); 1770 W.startLine() << "DynamicSection [ (" << Total << " entries)\n"; 1771 1772 bool Is64 = ELFT::Is64Bits; 1773 1774 W.startLine() 1775 << " Tag" << (Is64 ? " " : " ") << "Type" 1776 << " " << "Name/Value\n"; 1777 while (I != E) { 1778 const Elf_Dyn &Entry = *I; 1779 uintX_t Tag = Entry.getTag(); 1780 ++I; 1781 W.startLine() << " " << format_hex(Tag, Is64 ? 18 : 10, opts::Output != opts::GNU) << " " 1782 << format("%-21s", getTypeString(Tag)); 1783 printValue(Tag, Entry.getVal()); 1784 OS << "\n"; 1785 } 1786 1787 W.startLine() << "]\n"; 1788 } 1789 1790 template<class ELFT> 1791 void ELFDumper<ELFT>::printNeededLibraries() { 1792 ListScope D(W, "NeededLibraries"); 1793 1794 typedef std::vector<StringRef> LibsTy; 1795 LibsTy Libs; 1796 1797 for (const auto &Entry : dynamic_table()) 1798 if (Entry.d_tag == ELF::DT_NEEDED) 1799 Libs.push_back(getDynamicString(Entry.d_un.d_val)); 1800 1801 std::stable_sort(Libs.begin(), Libs.end()); 1802 1803 for (const auto &L : Libs) { 1804 outs() << " " << L << "\n"; 1805 } 1806 } 1807 1808 1809 template <typename ELFT> 1810 void ELFDumper<ELFT>::printHashTable() { 1811 DictScope D(W, "HashTable"); 1812 if (!HashTable) 1813 return; 1814 W.printNumber("Num Buckets", HashTable->nbucket); 1815 W.printNumber("Num Chains", HashTable->nchain); 1816 W.printList("Buckets", HashTable->buckets()); 1817 W.printList("Chains", HashTable->chains()); 1818 } 1819 1820 template <typename ELFT> 1821 void ELFDumper<ELFT>::printGnuHashTable() { 1822 DictScope D(W, "GnuHashTable"); 1823 if (!GnuHashTable) 1824 return; 1825 W.printNumber("Num Buckets", GnuHashTable->nbuckets); 1826 W.printNumber("First Hashed Symbol Index", GnuHashTable->symndx); 1827 W.printNumber("Num Mask Words", GnuHashTable->maskwords); 1828 W.printNumber("Shift Count", GnuHashTable->shift2); 1829 W.printHexList("Bloom Filter", GnuHashTable->filter()); 1830 W.printList("Buckets", GnuHashTable->buckets()); 1831 Elf_Sym_Range Syms = dynamic_symbols(); 1832 unsigned NumSyms = std::distance(Syms.begin(), Syms.end()); 1833 if (!NumSyms) 1834 reportError("No dynamic symbol section"); 1835 W.printHexList("Values", GnuHashTable->values(NumSyms)); 1836 } 1837 1838 template <typename ELFT> void ELFDumper<ELFT>::printLoadName() { 1839 outs() << "LoadName: " << SOName << '\n'; 1840 } 1841 1842 template <class ELFT> 1843 void ELFDumper<ELFT>::printAttributes() { 1844 W.startLine() << "Attributes not implemented.\n"; 1845 } 1846 1847 namespace { 1848 template <> void ELFDumper<ELFType<support::little, false>>::printAttributes() { 1849 if (Obj->getHeader()->e_machine != EM_ARM) { 1850 W.startLine() << "Attributes not implemented.\n"; 1851 return; 1852 } 1853 1854 DictScope BA(W, "BuildAttributes"); 1855 for (const ELFO::Elf_Shdr &Sec : Obj->sections()) { 1856 if (Sec.sh_type != ELF::SHT_ARM_ATTRIBUTES) 1857 continue; 1858 1859 ArrayRef<uint8_t> Contents = unwrapOrError(Obj->getSectionContents(&Sec)); 1860 if (Contents[0] != ARMBuildAttrs::Format_Version) { 1861 errs() << "unrecognised FormatVersion: 0x" << utohexstr(Contents[0]) 1862 << '\n'; 1863 continue; 1864 } 1865 1866 W.printHex("FormatVersion", Contents[0]); 1867 if (Contents.size() == 1) 1868 continue; 1869 1870 ARMAttributeParser(W).Parse(Contents); 1871 } 1872 } 1873 } 1874 1875 namespace { 1876 template <class ELFT> class MipsGOTParser { 1877 public: 1878 TYPEDEF_ELF_TYPES(ELFT) 1879 typedef typename ELFO::Elf_Addr GOTEntry; 1880 MipsGOTParser(ELFDumper<ELFT> *Dumper, const ELFO *Obj, 1881 Elf_Dyn_Range DynTable, ScopedPrinter &W); 1882 1883 void parseGOT(); 1884 void parsePLT(); 1885 1886 private: 1887 ELFDumper<ELFT> *Dumper; 1888 const ELFO *Obj; 1889 ScopedPrinter &W; 1890 llvm::Optional<uint64_t> DtPltGot; 1891 llvm::Optional<uint64_t> DtLocalGotNum; 1892 llvm::Optional<uint64_t> DtGotSym; 1893 llvm::Optional<uint64_t> DtMipsPltGot; 1894 llvm::Optional<uint64_t> DtJmpRel; 1895 1896 std::size_t getGOTTotal(ArrayRef<uint8_t> GOT) const; 1897 const GOTEntry *makeGOTIter(ArrayRef<uint8_t> GOT, std::size_t EntryNum); 1898 1899 void printGotEntry(uint64_t GotAddr, const GOTEntry *BeginIt, 1900 const GOTEntry *It); 1901 void printGlobalGotEntry(uint64_t GotAddr, const GOTEntry *BeginIt, 1902 const GOTEntry *It, const Elf_Sym *Sym, 1903 StringRef StrTable, bool IsDynamic); 1904 void printPLTEntry(uint64_t PLTAddr, const GOTEntry *BeginIt, 1905 const GOTEntry *It, StringRef Purpose); 1906 void printPLTEntry(uint64_t PLTAddr, const GOTEntry *BeginIt, 1907 const GOTEntry *It, StringRef StrTable, 1908 const Elf_Sym *Sym); 1909 }; 1910 } 1911 1912 template <class ELFT> 1913 MipsGOTParser<ELFT>::MipsGOTParser(ELFDumper<ELFT> *Dumper, const ELFO *Obj, 1914 Elf_Dyn_Range DynTable, ScopedPrinter &W) 1915 : Dumper(Dumper), Obj(Obj), W(W) { 1916 for (const auto &Entry : DynTable) { 1917 switch (Entry.getTag()) { 1918 case ELF::DT_PLTGOT: 1919 DtPltGot = Entry.getVal(); 1920 break; 1921 case ELF::DT_MIPS_LOCAL_GOTNO: 1922 DtLocalGotNum = Entry.getVal(); 1923 break; 1924 case ELF::DT_MIPS_GOTSYM: 1925 DtGotSym = Entry.getVal(); 1926 break; 1927 case ELF::DT_MIPS_PLTGOT: 1928 DtMipsPltGot = Entry.getVal(); 1929 break; 1930 case ELF::DT_JMPREL: 1931 DtJmpRel = Entry.getVal(); 1932 break; 1933 } 1934 } 1935 } 1936 1937 template <class ELFT> void MipsGOTParser<ELFT>::parseGOT() { 1938 // See "Global Offset Table" in Chapter 5 in the following document 1939 // for detailed GOT description. 1940 // ftp://www.linux-mips.org/pub/linux/mips/doc/ABI/mipsabi.pdf 1941 if (!DtPltGot) { 1942 W.startLine() << "Cannot find PLTGOT dynamic table tag.\n"; 1943 return; 1944 } 1945 if (!DtLocalGotNum) { 1946 W.startLine() << "Cannot find MIPS_LOCAL_GOTNO dynamic table tag.\n"; 1947 return; 1948 } 1949 if (!DtGotSym) { 1950 W.startLine() << "Cannot find MIPS_GOTSYM dynamic table tag.\n"; 1951 return; 1952 } 1953 1954 StringRef StrTable = Dumper->getDynamicStringTable(); 1955 const Elf_Sym *DynSymBegin = Dumper->dynamic_symbols().begin(); 1956 const Elf_Sym *DynSymEnd = Dumper->dynamic_symbols().end(); 1957 std::size_t DynSymTotal = std::size_t(std::distance(DynSymBegin, DynSymEnd)); 1958 1959 if (*DtGotSym > DynSymTotal) 1960 report_fatal_error("MIPS_GOTSYM exceeds a number of dynamic symbols"); 1961 1962 std::size_t GlobalGotNum = DynSymTotal - *DtGotSym; 1963 1964 if (*DtLocalGotNum + GlobalGotNum == 0) { 1965 W.startLine() << "GOT is empty.\n"; 1966 return; 1967 } 1968 1969 const Elf_Shdr *GOTShdr = findNotEmptySectionByAddress(Obj, *DtPltGot); 1970 if (!GOTShdr) 1971 report_fatal_error("There is no not empty GOT section at 0x" + 1972 Twine::utohexstr(*DtPltGot)); 1973 1974 ArrayRef<uint8_t> GOT = unwrapOrError(Obj->getSectionContents(GOTShdr)); 1975 1976 if (*DtLocalGotNum + GlobalGotNum > getGOTTotal(GOT)) 1977 report_fatal_error("Number of GOT entries exceeds the size of GOT section"); 1978 1979 const GOTEntry *GotBegin = makeGOTIter(GOT, 0); 1980 const GOTEntry *GotLocalEnd = makeGOTIter(GOT, *DtLocalGotNum); 1981 const GOTEntry *It = GotBegin; 1982 1983 DictScope GS(W, "Primary GOT"); 1984 1985 W.printHex("Canonical gp value", GOTShdr->sh_addr + 0x7ff0); 1986 { 1987 ListScope RS(W, "Reserved entries"); 1988 1989 { 1990 DictScope D(W, "Entry"); 1991 printGotEntry(GOTShdr->sh_addr, GotBegin, It++); 1992 W.printString("Purpose", StringRef("Lazy resolver")); 1993 } 1994 1995 if (It != GotLocalEnd && (*It >> (sizeof(GOTEntry) * 8 - 1)) != 0) { 1996 DictScope D(W, "Entry"); 1997 printGotEntry(GOTShdr->sh_addr, GotBegin, It++); 1998 W.printString("Purpose", StringRef("Module pointer (GNU extension)")); 1999 } 2000 } 2001 { 2002 ListScope LS(W, "Local entries"); 2003 for (; It != GotLocalEnd; ++It) { 2004 DictScope D(W, "Entry"); 2005 printGotEntry(GOTShdr->sh_addr, GotBegin, It); 2006 } 2007 } 2008 { 2009 ListScope GS(W, "Global entries"); 2010 2011 const GOTEntry *GotGlobalEnd = 2012 makeGOTIter(GOT, *DtLocalGotNum + GlobalGotNum); 2013 const Elf_Sym *GotDynSym = DynSymBegin + *DtGotSym; 2014 for (; It != GotGlobalEnd; ++It) { 2015 DictScope D(W, "Entry"); 2016 printGlobalGotEntry(GOTShdr->sh_addr, GotBegin, It, GotDynSym++, StrTable, 2017 true); 2018 } 2019 } 2020 2021 std::size_t SpecGotNum = getGOTTotal(GOT) - *DtLocalGotNum - GlobalGotNum; 2022 W.printNumber("Number of TLS and multi-GOT entries", uint64_t(SpecGotNum)); 2023 } 2024 2025 template <class ELFT> void MipsGOTParser<ELFT>::parsePLT() { 2026 if (!DtMipsPltGot) { 2027 W.startLine() << "Cannot find MIPS_PLTGOT dynamic table tag.\n"; 2028 return; 2029 } 2030 if (!DtJmpRel) { 2031 W.startLine() << "Cannot find JMPREL dynamic table tag.\n"; 2032 return; 2033 } 2034 2035 const Elf_Shdr *PLTShdr = findNotEmptySectionByAddress(Obj, *DtMipsPltGot); 2036 if (!PLTShdr) 2037 report_fatal_error("There is no not empty PLTGOT section at 0x " + 2038 Twine::utohexstr(*DtMipsPltGot)); 2039 ArrayRef<uint8_t> PLT = unwrapOrError(Obj->getSectionContents(PLTShdr)); 2040 2041 const Elf_Shdr *PLTRelShdr = findNotEmptySectionByAddress(Obj, *DtJmpRel); 2042 if (!PLTRelShdr) 2043 report_fatal_error("There is no not empty RELPLT section at 0x" + 2044 Twine::utohexstr(*DtJmpRel)); 2045 const Elf_Shdr *SymTable = 2046 unwrapOrError(Obj->getSection(PLTRelShdr->sh_link)); 2047 StringRef StrTable = unwrapOrError(Obj->getStringTableForSymtab(*SymTable)); 2048 2049 const GOTEntry *PLTBegin = makeGOTIter(PLT, 0); 2050 const GOTEntry *PLTEnd = makeGOTIter(PLT, getGOTTotal(PLT)); 2051 const GOTEntry *It = PLTBegin; 2052 2053 DictScope GS(W, "PLT GOT"); 2054 { 2055 ListScope RS(W, "Reserved entries"); 2056 printPLTEntry(PLTShdr->sh_addr, PLTBegin, It++, "PLT lazy resolver"); 2057 if (It != PLTEnd) 2058 printPLTEntry(PLTShdr->sh_addr, PLTBegin, It++, "Module pointer"); 2059 } 2060 { 2061 ListScope GS(W, "Entries"); 2062 2063 switch (PLTRelShdr->sh_type) { 2064 case ELF::SHT_REL: 2065 for (const Elf_Rel &Rel : Obj->rels(PLTRelShdr)) { 2066 const Elf_Sym *Sym = Obj->getRelocationSymbol(&Rel, SymTable); 2067 printPLTEntry(PLTShdr->sh_addr, PLTBegin, It, StrTable, Sym); 2068 if (++It == PLTEnd) 2069 break; 2070 } 2071 break; 2072 case ELF::SHT_RELA: 2073 for (const Elf_Rela &Rel : Obj->relas(PLTRelShdr)) { 2074 const Elf_Sym *Sym = Obj->getRelocationSymbol(&Rel, SymTable); 2075 printPLTEntry(PLTShdr->sh_addr, PLTBegin, It, StrTable, Sym); 2076 if (++It == PLTEnd) 2077 break; 2078 } 2079 break; 2080 } 2081 } 2082 } 2083 2084 template <class ELFT> 2085 std::size_t MipsGOTParser<ELFT>::getGOTTotal(ArrayRef<uint8_t> GOT) const { 2086 return GOT.size() / sizeof(GOTEntry); 2087 } 2088 2089 template <class ELFT> 2090 const typename MipsGOTParser<ELFT>::GOTEntry * 2091 MipsGOTParser<ELFT>::makeGOTIter(ArrayRef<uint8_t> GOT, std::size_t EntryNum) { 2092 const char *Data = reinterpret_cast<const char *>(GOT.data()); 2093 return reinterpret_cast<const GOTEntry *>(Data + EntryNum * sizeof(GOTEntry)); 2094 } 2095 2096 template <class ELFT> 2097 void MipsGOTParser<ELFT>::printGotEntry(uint64_t GotAddr, 2098 const GOTEntry *BeginIt, 2099 const GOTEntry *It) { 2100 int64_t Offset = std::distance(BeginIt, It) * sizeof(GOTEntry); 2101 W.printHex("Address", GotAddr + Offset); 2102 W.printNumber("Access", Offset - 0x7ff0); 2103 W.printHex("Initial", *It); 2104 } 2105 2106 template <class ELFT> 2107 void MipsGOTParser<ELFT>::printGlobalGotEntry( 2108 uint64_t GotAddr, const GOTEntry *BeginIt, const GOTEntry *It, 2109 const Elf_Sym *Sym, StringRef StrTable, bool IsDynamic) { 2110 printGotEntry(GotAddr, BeginIt, It); 2111 2112 W.printHex("Value", Sym->st_value); 2113 W.printEnum("Type", Sym->getType(), makeArrayRef(ElfSymbolTypes)); 2114 2115 unsigned SectionIndex = 0; 2116 StringRef SectionName; 2117 getSectionNameIndex(*Obj, Sym, Dumper->dynamic_symbols().begin(), 2118 Dumper->getShndxTable(), SectionName, SectionIndex); 2119 W.printHex("Section", SectionName, SectionIndex); 2120 2121 std::string FullSymbolName = 2122 Dumper->getFullSymbolName(Sym, StrTable, IsDynamic); 2123 W.printNumber("Name", FullSymbolName, Sym->st_name); 2124 } 2125 2126 template <class ELFT> 2127 void MipsGOTParser<ELFT>::printPLTEntry(uint64_t PLTAddr, 2128 const GOTEntry *BeginIt, 2129 const GOTEntry *It, StringRef Purpose) { 2130 DictScope D(W, "Entry"); 2131 int64_t Offset = std::distance(BeginIt, It) * sizeof(GOTEntry); 2132 W.printHex("Address", PLTAddr + Offset); 2133 W.printHex("Initial", *It); 2134 W.printString("Purpose", Purpose); 2135 } 2136 2137 template <class ELFT> 2138 void MipsGOTParser<ELFT>::printPLTEntry(uint64_t PLTAddr, 2139 const GOTEntry *BeginIt, 2140 const GOTEntry *It, StringRef StrTable, 2141 const Elf_Sym *Sym) { 2142 DictScope D(W, "Entry"); 2143 int64_t Offset = std::distance(BeginIt, It) * sizeof(GOTEntry); 2144 W.printHex("Address", PLTAddr + Offset); 2145 W.printHex("Initial", *It); 2146 W.printHex("Value", Sym->st_value); 2147 W.printEnum("Type", Sym->getType(), makeArrayRef(ElfSymbolTypes)); 2148 2149 unsigned SectionIndex = 0; 2150 StringRef SectionName; 2151 getSectionNameIndex(*Obj, Sym, Dumper->dynamic_symbols().begin(), 2152 Dumper->getShndxTable(), SectionName, SectionIndex); 2153 W.printHex("Section", SectionName, SectionIndex); 2154 2155 std::string FullSymbolName = Dumper->getFullSymbolName(Sym, StrTable, true); 2156 W.printNumber("Name", FullSymbolName, Sym->st_name); 2157 } 2158 2159 template <class ELFT> void ELFDumper<ELFT>::printMipsPLTGOT() { 2160 if (Obj->getHeader()->e_machine != EM_MIPS) { 2161 W.startLine() << "MIPS PLT GOT is available for MIPS targets only.\n"; 2162 return; 2163 } 2164 2165 MipsGOTParser<ELFT> GOTParser(this, Obj, dynamic_table(), W); 2166 GOTParser.parseGOT(); 2167 GOTParser.parsePLT(); 2168 } 2169 2170 static const EnumEntry<unsigned> ElfMipsISAExtType[] = { 2171 {"None", Mips::AFL_EXT_NONE}, 2172 {"Broadcom SB-1", Mips::AFL_EXT_SB1}, 2173 {"Cavium Networks Octeon", Mips::AFL_EXT_OCTEON}, 2174 {"Cavium Networks Octeon2", Mips::AFL_EXT_OCTEON2}, 2175 {"Cavium Networks OcteonP", Mips::AFL_EXT_OCTEONP}, 2176 {"Cavium Networks Octeon3", Mips::AFL_EXT_OCTEON3}, 2177 {"LSI R4010", Mips::AFL_EXT_4010}, 2178 {"Loongson 2E", Mips::AFL_EXT_LOONGSON_2E}, 2179 {"Loongson 2F", Mips::AFL_EXT_LOONGSON_2F}, 2180 {"Loongson 3A", Mips::AFL_EXT_LOONGSON_3A}, 2181 {"MIPS R4650", Mips::AFL_EXT_4650}, 2182 {"MIPS R5900", Mips::AFL_EXT_5900}, 2183 {"MIPS R10000", Mips::AFL_EXT_10000}, 2184 {"NEC VR4100", Mips::AFL_EXT_4100}, 2185 {"NEC VR4111/VR4181", Mips::AFL_EXT_4111}, 2186 {"NEC VR4120", Mips::AFL_EXT_4120}, 2187 {"NEC VR5400", Mips::AFL_EXT_5400}, 2188 {"NEC VR5500", Mips::AFL_EXT_5500}, 2189 {"RMI Xlr", Mips::AFL_EXT_XLR}, 2190 {"Toshiba R3900", Mips::AFL_EXT_3900} 2191 }; 2192 2193 static const EnumEntry<unsigned> ElfMipsASEFlags[] = { 2194 {"DSP", Mips::AFL_ASE_DSP}, 2195 {"DSPR2", Mips::AFL_ASE_DSPR2}, 2196 {"Enhanced VA Scheme", Mips::AFL_ASE_EVA}, 2197 {"MCU", Mips::AFL_ASE_MCU}, 2198 {"MDMX", Mips::AFL_ASE_MDMX}, 2199 {"MIPS-3D", Mips::AFL_ASE_MIPS3D}, 2200 {"MT", Mips::AFL_ASE_MT}, 2201 {"SmartMIPS", Mips::AFL_ASE_SMARTMIPS}, 2202 {"VZ", Mips::AFL_ASE_VIRT}, 2203 {"MSA", Mips::AFL_ASE_MSA}, 2204 {"MIPS16", Mips::AFL_ASE_MIPS16}, 2205 {"microMIPS", Mips::AFL_ASE_MICROMIPS}, 2206 {"XPA", Mips::AFL_ASE_XPA} 2207 }; 2208 2209 static const EnumEntry<unsigned> ElfMipsFpABIType[] = { 2210 {"Hard or soft float", Mips::Val_GNU_MIPS_ABI_FP_ANY}, 2211 {"Hard float (double precision)", Mips::Val_GNU_MIPS_ABI_FP_DOUBLE}, 2212 {"Hard float (single precision)", Mips::Val_GNU_MIPS_ABI_FP_SINGLE}, 2213 {"Soft float", Mips::Val_GNU_MIPS_ABI_FP_SOFT}, 2214 {"Hard float (MIPS32r2 64-bit FPU 12 callee-saved)", 2215 Mips::Val_GNU_MIPS_ABI_FP_OLD_64}, 2216 {"Hard float (32-bit CPU, Any FPU)", Mips::Val_GNU_MIPS_ABI_FP_XX}, 2217 {"Hard float (32-bit CPU, 64-bit FPU)", Mips::Val_GNU_MIPS_ABI_FP_64}, 2218 {"Hard float compat (32-bit CPU, 64-bit FPU)", 2219 Mips::Val_GNU_MIPS_ABI_FP_64A} 2220 }; 2221 2222 static const EnumEntry<unsigned> ElfMipsFlags1[] { 2223 {"ODDSPREG", Mips::AFL_FLAGS1_ODDSPREG}, 2224 }; 2225 2226 static int getMipsRegisterSize(uint8_t Flag) { 2227 switch (Flag) { 2228 case Mips::AFL_REG_NONE: 2229 return 0; 2230 case Mips::AFL_REG_32: 2231 return 32; 2232 case Mips::AFL_REG_64: 2233 return 64; 2234 case Mips::AFL_REG_128: 2235 return 128; 2236 default: 2237 return -1; 2238 } 2239 } 2240 2241 template <class ELFT> void ELFDumper<ELFT>::printMipsABIFlags() { 2242 const Elf_Shdr *Shdr = findSectionByName(*Obj, ".MIPS.abiflags"); 2243 if (!Shdr) { 2244 W.startLine() << "There is no .MIPS.abiflags section in the file.\n"; 2245 return; 2246 } 2247 ArrayRef<uint8_t> Sec = unwrapOrError(Obj->getSectionContents(Shdr)); 2248 if (Sec.size() != sizeof(Elf_Mips_ABIFlags<ELFT>)) { 2249 W.startLine() << "The .MIPS.abiflags section has a wrong size.\n"; 2250 return; 2251 } 2252 2253 auto *Flags = reinterpret_cast<const Elf_Mips_ABIFlags<ELFT> *>(Sec.data()); 2254 2255 raw_ostream &OS = W.getOStream(); 2256 DictScope GS(W, "MIPS ABI Flags"); 2257 2258 W.printNumber("Version", Flags->version); 2259 W.startLine() << "ISA: "; 2260 if (Flags->isa_rev <= 1) 2261 OS << format("MIPS%u", Flags->isa_level); 2262 else 2263 OS << format("MIPS%ur%u", Flags->isa_level, Flags->isa_rev); 2264 OS << "\n"; 2265 W.printEnum("ISA Extension", Flags->isa_ext, makeArrayRef(ElfMipsISAExtType)); 2266 W.printFlags("ASEs", Flags->ases, makeArrayRef(ElfMipsASEFlags)); 2267 W.printEnum("FP ABI", Flags->fp_abi, makeArrayRef(ElfMipsFpABIType)); 2268 W.printNumber("GPR size", getMipsRegisterSize(Flags->gpr_size)); 2269 W.printNumber("CPR1 size", getMipsRegisterSize(Flags->cpr1_size)); 2270 W.printNumber("CPR2 size", getMipsRegisterSize(Flags->cpr2_size)); 2271 W.printFlags("Flags 1", Flags->flags1, makeArrayRef(ElfMipsFlags1)); 2272 W.printHex("Flags 2", Flags->flags2); 2273 } 2274 2275 template <class ELFT> 2276 static void printMipsReginfoData(ScopedPrinter &W, 2277 const Elf_Mips_RegInfo<ELFT> &Reginfo) { 2278 W.printHex("GP", Reginfo.ri_gp_value); 2279 W.printHex("General Mask", Reginfo.ri_gprmask); 2280 W.printHex("Co-Proc Mask0", Reginfo.ri_cprmask[0]); 2281 W.printHex("Co-Proc Mask1", Reginfo.ri_cprmask[1]); 2282 W.printHex("Co-Proc Mask2", Reginfo.ri_cprmask[2]); 2283 W.printHex("Co-Proc Mask3", Reginfo.ri_cprmask[3]); 2284 } 2285 2286 template <class ELFT> void ELFDumper<ELFT>::printMipsReginfo() { 2287 const Elf_Shdr *Shdr = findSectionByName(*Obj, ".reginfo"); 2288 if (!Shdr) { 2289 W.startLine() << "There is no .reginfo section in the file.\n"; 2290 return; 2291 } 2292 ArrayRef<uint8_t> Sec = unwrapOrError(Obj->getSectionContents(Shdr)); 2293 if (Sec.size() != sizeof(Elf_Mips_RegInfo<ELFT>)) { 2294 W.startLine() << "The .reginfo section has a wrong size.\n"; 2295 return; 2296 } 2297 2298 DictScope GS(W, "MIPS RegInfo"); 2299 auto *Reginfo = reinterpret_cast<const Elf_Mips_RegInfo<ELFT> *>(Sec.data()); 2300 printMipsReginfoData(W, *Reginfo); 2301 } 2302 2303 template <class ELFT> void ELFDumper<ELFT>::printMipsOptions() { 2304 const Elf_Shdr *Shdr = findSectionByName(*Obj, ".MIPS.options"); 2305 if (!Shdr) { 2306 W.startLine() << "There is no .MIPS.options section in the file.\n"; 2307 return; 2308 } 2309 2310 DictScope GS(W, "MIPS Options"); 2311 2312 ArrayRef<uint8_t> Sec = unwrapOrError(Obj->getSectionContents(Shdr)); 2313 while (!Sec.empty()) { 2314 if (Sec.size() < sizeof(Elf_Mips_Options<ELFT>)) { 2315 W.startLine() << "The .MIPS.options section has a wrong size.\n"; 2316 return; 2317 } 2318 auto *O = reinterpret_cast<const Elf_Mips_Options<ELFT> *>(Sec.data()); 2319 DictScope GS(W, getElfMipsOptionsOdkType(O->kind)); 2320 switch (O->kind) { 2321 case ODK_REGINFO: 2322 printMipsReginfoData(W, O->getRegInfo()); 2323 break; 2324 default: 2325 W.startLine() << "Unsupported MIPS options tag.\n"; 2326 break; 2327 } 2328 Sec = Sec.slice(O->size); 2329 } 2330 } 2331 2332 template <class ELFT> void ELFDumper<ELFT>::printStackMap() const { 2333 const Elf_Shdr *StackMapSection = nullptr; 2334 for (const auto &Sec : Obj->sections()) { 2335 StringRef Name = unwrapOrError(Obj->getSectionName(&Sec)); 2336 if (Name == ".llvm_stackmaps") { 2337 StackMapSection = &Sec; 2338 break; 2339 } 2340 } 2341 2342 if (!StackMapSection) 2343 return; 2344 2345 StringRef StackMapContents; 2346 ArrayRef<uint8_t> StackMapContentsArray = 2347 unwrapOrError(Obj->getSectionContents(StackMapSection)); 2348 2349 prettyPrintStackMap(llvm::outs(), StackMapV2Parser<ELFT::TargetEndianness>( 2350 StackMapContentsArray)); 2351 } 2352 2353 template <class ELFT> void ELFDumper<ELFT>::printGroupSections() { 2354 ELFDumperStyle->printGroupSections(Obj); 2355 } 2356 2357 static inline void printFields(formatted_raw_ostream &OS, StringRef Str1, 2358 StringRef Str2) { 2359 OS.PadToColumn(2u); 2360 OS << Str1; 2361 OS.PadToColumn(37u); 2362 OS << Str2 << "\n"; 2363 OS.flush(); 2364 } 2365 2366 template <class ELFT> void GNUStyle<ELFT>::printFileHeaders(const ELFO *Obj) { 2367 const Elf_Ehdr *e = Obj->getHeader(); 2368 OS << "ELF Header:\n"; 2369 OS << " Magic: "; 2370 std::string Str; 2371 for (int i = 0; i < ELF::EI_NIDENT; i++) 2372 OS << format(" %02x", static_cast<int>(e->e_ident[i])); 2373 OS << "\n"; 2374 Str = printEnum(e->e_ident[ELF::EI_CLASS], makeArrayRef(ElfClass)); 2375 printFields(OS, "Class:", Str); 2376 Str = printEnum(e->e_ident[ELF::EI_DATA], makeArrayRef(ElfDataEncoding)); 2377 printFields(OS, "Data:", Str); 2378 OS.PadToColumn(2u); 2379 OS << "Version:"; 2380 OS.PadToColumn(37u); 2381 OS << to_hexString(e->e_ident[ELF::EI_VERSION]); 2382 if (e->e_version == ELF::EV_CURRENT) 2383 OS << " (current)"; 2384 OS << "\n"; 2385 Str = printEnum(e->e_ident[ELF::EI_OSABI], makeArrayRef(ElfOSABI)); 2386 printFields(OS, "OS/ABI:", Str); 2387 Str = "0x" + to_hexString(e->e_version); 2388 Str = to_hexString(e->e_ident[ELF::EI_ABIVERSION]); 2389 printFields(OS, "ABI Version:", Str); 2390 Str = printEnum(e->e_type, makeArrayRef(ElfObjectFileType)); 2391 printFields(OS, "Type:", Str); 2392 Str = printEnum(e->e_machine, makeArrayRef(ElfMachineType)); 2393 printFields(OS, "Machine:", Str); 2394 Str = "0x" + to_hexString(e->e_version); 2395 printFields(OS, "Version:", Str); 2396 Str = "0x" + to_hexString(e->e_entry); 2397 printFields(OS, "Entry point address:", Str); 2398 Str = to_string(e->e_phoff) + " (bytes into file)"; 2399 printFields(OS, "Start of program headers:", Str); 2400 Str = to_string(e->e_shoff) + " (bytes into file)"; 2401 printFields(OS, "Start of section headers:", Str); 2402 Str = "0x" + to_hexString(e->e_flags); 2403 printFields(OS, "Flags:", Str); 2404 Str = to_string(e->e_ehsize) + " (bytes)"; 2405 printFields(OS, "Size of this header:", Str); 2406 Str = to_string(e->e_phentsize) + " (bytes)"; 2407 printFields(OS, "Size of program headers:", Str); 2408 Str = to_string(e->e_phnum); 2409 printFields(OS, "Number of program headers:", Str); 2410 Str = to_string(e->e_shentsize) + " (bytes)"; 2411 printFields(OS, "Size of section headers:", Str); 2412 Str = to_string(e->e_shnum); 2413 printFields(OS, "Number of section headers:", Str); 2414 Str = to_string(e->e_shstrndx); 2415 printFields(OS, "Section header string table index:", Str); 2416 } 2417 2418 template <class ELFT> void GNUStyle<ELFT>::printGroupSections(const ELFO *Obj) { 2419 uint32_t SectionIndex = 0; 2420 bool HasGroups = false; 2421 for (const Elf_Shdr &Sec : Obj->sections()) { 2422 if (Sec.sh_type == ELF::SHT_GROUP) { 2423 HasGroups = true; 2424 const Elf_Shdr *Symtab = unwrapOrError(Obj->getSection(Sec.sh_link)); 2425 StringRef StrTable = unwrapOrError(Obj->getStringTableForSymtab(*Symtab)); 2426 const Elf_Sym *Signature = 2427 Obj->template getEntry<Elf_Sym>(Symtab, Sec.sh_info); 2428 ArrayRef<Elf_Word> Data = unwrapOrError( 2429 Obj->template getSectionContentsAsArray<Elf_Word>(&Sec)); 2430 StringRef Name = unwrapOrError(Obj->getSectionName(&Sec)); 2431 OS << "\n" << getGroupType(Data[0]) << " group section [" 2432 << format_decimal(SectionIndex, 5) << "] `" << Name << "' [" 2433 << StrTable.data() + Signature->st_name << "] contains " 2434 << (Data.size() - 1) << " sections:\n" 2435 << " [Index] Name\n"; 2436 for (auto &Ndx : Data.slice(1)) { 2437 auto Sec = unwrapOrError(Obj->getSection(Ndx)); 2438 const StringRef Name = unwrapOrError(Obj->getSectionName(Sec)); 2439 OS << " [" << format_decimal(Ndx, 5) << "] " << Name 2440 << "\n"; 2441 } 2442 } 2443 ++SectionIndex; 2444 } 2445 if (!HasGroups) 2446 OS << "There are no section groups in this file.\n"; 2447 } 2448 2449 template <class ELFT> 2450 void GNUStyle<ELFT>::printRelocation(const ELFO *Obj, const Elf_Shdr *SymTab, 2451 const Elf_Rela &R, bool IsRela) { 2452 std::string Offset, Info, Addend = "", Value; 2453 SmallString<32> RelocName; 2454 StringRef StrTable = unwrapOrError(Obj->getStringTableForSymtab(*SymTab)); 2455 StringRef TargetName; 2456 const Elf_Sym *Sym = nullptr; 2457 unsigned Width = ELFT::Is64Bits ? 16 : 8; 2458 unsigned Bias = ELFT::Is64Bits ? 8 : 0; 2459 2460 // First two fields are bit width dependent. The rest of them are after are 2461 // fixed width. 2462 Field Fields[5] = {0, 10 + Bias, 19 + 2 * Bias, 42 + 2 * Bias, 53 + 2 * Bias}; 2463 Obj->getRelocationTypeName(R.getType(Obj->isMips64EL()), RelocName); 2464 Sym = Obj->getRelocationSymbol(&R, SymTab); 2465 if (Sym && Sym->getType() == ELF::STT_SECTION) { 2466 const Elf_Shdr *Sec = unwrapOrError( 2467 Obj->getSection(Sym, SymTab, this->dumper()->getShndxTable())); 2468 TargetName = unwrapOrError(Obj->getSectionName(Sec)); 2469 } else if (Sym) { 2470 TargetName = unwrapOrError(Sym->getName(StrTable)); 2471 } 2472 2473 if (Sym && IsRela) { 2474 if (R.r_addend < 0) 2475 Addend = " - "; 2476 else 2477 Addend = " + "; 2478 } 2479 2480 Offset = to_string(format_hex_no_prefix(R.r_offset, Width)); 2481 Info = to_string(format_hex_no_prefix(R.r_info, Width)); 2482 2483 int64_t RelAddend = R.r_addend; 2484 if (IsRela) 2485 Addend += to_hexString(std::abs(RelAddend), false); 2486 2487 if (Sym) 2488 Value = to_string(format_hex_no_prefix(Sym->getValue(), Width)); 2489 2490 Fields[0].Str = Offset; 2491 Fields[1].Str = Info; 2492 Fields[2].Str = RelocName; 2493 Fields[3].Str = Value; 2494 Fields[4].Str = TargetName; 2495 for (auto &field : Fields) 2496 printField(field); 2497 OS << Addend; 2498 OS << "\n"; 2499 } 2500 2501 static inline void printRelocHeader(raw_ostream &OS, bool Is64, bool IsRela) { 2502 if (Is64) 2503 OS << " Offset Info Type" 2504 << " Symbol's Value Symbol's Name"; 2505 else 2506 OS << " Offset Info Type Sym. Value " 2507 << "Symbol's Name"; 2508 if (IsRela) 2509 OS << (IsRela ? " + Addend" : ""); 2510 OS << "\n"; 2511 } 2512 2513 template <class ELFT> void GNUStyle<ELFT>::printRelocations(const ELFO *Obj) { 2514 bool HasRelocSections = false; 2515 for (const Elf_Shdr &Sec : Obj->sections()) { 2516 if (Sec.sh_type != ELF::SHT_REL && Sec.sh_type != ELF::SHT_RELA) 2517 continue; 2518 HasRelocSections = true; 2519 StringRef Name = unwrapOrError(Obj->getSectionName(&Sec)); 2520 unsigned Entries = Sec.getEntityCount(); 2521 uintX_t Offset = Sec.sh_offset; 2522 OS << "\nRelocation section '" << Name << "' at offset 0x" 2523 << to_hexString(Offset, false) << " contains " << Entries 2524 << " entries:\n"; 2525 printRelocHeader(OS, ELFT::Is64Bits, (Sec.sh_type == ELF::SHT_RELA)); 2526 const Elf_Shdr *SymTab = unwrapOrError(Obj->getSection(Sec.sh_link)); 2527 if (Sec.sh_type == ELF::SHT_REL) { 2528 for (const auto &R : Obj->rels(&Sec)) { 2529 Elf_Rela Rela; 2530 Rela.r_offset = R.r_offset; 2531 Rela.r_info = R.r_info; 2532 Rela.r_addend = 0; 2533 printRelocation(Obj, SymTab, Rela, false); 2534 } 2535 } else { 2536 for (const auto &R : Obj->relas(&Sec)) 2537 printRelocation(Obj, SymTab, R, true); 2538 } 2539 } 2540 if (!HasRelocSections) 2541 OS << "\nThere are no relocations in this file.\n"; 2542 } 2543 2544 std::string getSectionTypeString(unsigned Arch, unsigned Type) { 2545 using namespace ELF; 2546 switch (Arch) { 2547 case EM_ARM: 2548 switch (Type) { 2549 case SHT_ARM_EXIDX: 2550 return "ARM_EXIDX"; 2551 case SHT_ARM_PREEMPTMAP: 2552 return "ARM_PREEMPTMAP"; 2553 case SHT_ARM_ATTRIBUTES: 2554 return "ARM_ATTRIBUTES"; 2555 case SHT_ARM_DEBUGOVERLAY: 2556 return "ARM_DEBUGOVERLAY"; 2557 case SHT_ARM_OVERLAYSECTION: 2558 return "ARM_OVERLAYSECTION"; 2559 } 2560 case EM_X86_64: 2561 switch (Type) { 2562 case SHT_X86_64_UNWIND: 2563 return "X86_64_UNWIND"; 2564 } 2565 case EM_MIPS: 2566 case EM_MIPS_RS3_LE: 2567 switch (Type) { 2568 case SHT_MIPS_REGINFO: 2569 return "MIPS_REGINFO"; 2570 case SHT_MIPS_OPTIONS: 2571 return "MIPS_OPTIONS"; 2572 case SHT_MIPS_ABIFLAGS: 2573 return "MIPS_ABIFLAGS"; 2574 } 2575 } 2576 switch (Type) { 2577 case SHT_NULL: 2578 return "NULL"; 2579 case SHT_PROGBITS: 2580 return "PROGBITS"; 2581 case SHT_SYMTAB: 2582 return "SYMTAB"; 2583 case SHT_STRTAB: 2584 return "STRTAB"; 2585 case SHT_RELA: 2586 return "RELA"; 2587 case SHT_HASH: 2588 return "HASH"; 2589 case SHT_DYNAMIC: 2590 return "DYNAMIC"; 2591 case SHT_NOTE: 2592 return "NOTE"; 2593 case SHT_NOBITS: 2594 return "NOBITS"; 2595 case SHT_REL: 2596 return "REL"; 2597 case SHT_SHLIB: 2598 return "SHLIB"; 2599 case SHT_DYNSYM: 2600 return "DYNSYM"; 2601 case SHT_INIT_ARRAY: 2602 return "INIT_ARRAY"; 2603 case SHT_FINI_ARRAY: 2604 return "FINI_ARRAY"; 2605 case SHT_PREINIT_ARRAY: 2606 return "PREINIT_ARRAY"; 2607 case SHT_GROUP: 2608 return "GROUP"; 2609 case SHT_SYMTAB_SHNDX: 2610 return "SYMTAB SECTION INDICES"; 2611 // FIXME: Parse processor specific GNU attributes 2612 case SHT_GNU_ATTRIBUTES: 2613 return "ATTRIBUTES"; 2614 case SHT_GNU_HASH: 2615 return "GNU_HASH"; 2616 case SHT_GNU_verdef: 2617 return "VERDEF"; 2618 case SHT_GNU_verneed: 2619 return "VERNEED"; 2620 case SHT_GNU_versym: 2621 return "VERSYM"; 2622 default: 2623 return ""; 2624 } 2625 return ""; 2626 } 2627 2628 template <class ELFT> void GNUStyle<ELFT>::printSections(const ELFO *Obj) { 2629 size_t SectionIndex = 0; 2630 std::string Number, Type, Size, Address, Offset, Flags, Link, Info, EntrySize, 2631 Alignment; 2632 unsigned Bias; 2633 unsigned Width; 2634 2635 if (ELFT::Is64Bits) { 2636 Bias = 0; 2637 Width = 16; 2638 } else { 2639 Bias = 8; 2640 Width = 8; 2641 } 2642 OS << "There are " << to_string(Obj->getHeader()->e_shnum) 2643 << " section headers, starting at offset " 2644 << "0x" << to_hexString(Obj->getHeader()->e_shoff, false) << ":\n\n"; 2645 OS << "Section Headers:\n"; 2646 Field Fields[11] = {{"[Nr]", 2}, 2647 {"Name", 7}, 2648 {"Type", 25}, 2649 {"Address", 41}, 2650 {"Off", 58 - Bias}, 2651 {"Size", 65 - Bias}, 2652 {"ES", 72 - Bias}, 2653 {"Flg", 75 - Bias}, 2654 {"Lk", 79 - Bias}, 2655 {"Inf", 82 - Bias}, 2656 {"Al", 86 - Bias}}; 2657 for (auto &f : Fields) 2658 printField(f); 2659 OS << "\n"; 2660 2661 for (const Elf_Shdr &Sec : Obj->sections()) { 2662 Number = to_string(SectionIndex); 2663 Fields[0].Str = Number; 2664 Fields[1].Str = unwrapOrError(Obj->getSectionName(&Sec)); 2665 Type = getSectionTypeString(Obj->getHeader()->e_machine, Sec.sh_type); 2666 Fields[2].Str = Type; 2667 Address = to_string(format_hex_no_prefix(Sec.sh_addr, Width)); 2668 Fields[3].Str = Address; 2669 Offset = to_string(format_hex_no_prefix(Sec.sh_offset, 6)); 2670 Fields[4].Str = Offset; 2671 Size = to_string(format_hex_no_prefix(Sec.sh_size, 6)); 2672 Fields[5].Str = Size; 2673 EntrySize = to_string(format_hex_no_prefix(Sec.sh_entsize, 2)); 2674 Fields[6].Str = EntrySize; 2675 Flags = getGNUFlags(Sec.sh_flags); 2676 Fields[7].Str = Flags; 2677 Link = to_string(Sec.sh_link); 2678 Fields[8].Str = Link; 2679 Info = to_string(Sec.sh_info); 2680 Fields[9].Str = Info; 2681 Alignment = to_string(Sec.sh_addralign); 2682 Fields[10].Str = Alignment; 2683 OS.PadToColumn(Fields[0].Column); 2684 OS << "[" << right_justify(Fields[0].Str, 2) << "]"; 2685 for (int i = 1; i < 7; i++) 2686 printField(Fields[i]); 2687 OS.PadToColumn(Fields[7].Column); 2688 OS << right_justify(Fields[7].Str, 3); 2689 OS.PadToColumn(Fields[8].Column); 2690 OS << right_justify(Fields[8].Str, 2); 2691 OS.PadToColumn(Fields[9].Column); 2692 OS << right_justify(Fields[9].Str, 3); 2693 OS.PadToColumn(Fields[10].Column); 2694 OS << right_justify(Fields[10].Str, 2); 2695 OS << "\n"; 2696 ++SectionIndex; 2697 } 2698 OS << "Key to Flags:\n" 2699 << " W (write), A (alloc), X (execute), M (merge), S (strings), l " 2700 "(large)\n" 2701 << " I (info), L (link order), G (group), T (TLS), E (exclude),\ 2702 x (unknown)\n" 2703 << " O (extra OS processing required) o (OS specific),\ 2704 p (processor specific)\n"; 2705 } 2706 2707 template <class ELFT> 2708 void GNUStyle<ELFT>::printSymtabMessage(const ELFO *Obj, StringRef Name, 2709 size_t Entries) { 2710 if (Name.size()) 2711 OS << "\nSymbol table '" << Name << "' contains " << Entries 2712 << " entries:\n"; 2713 else 2714 OS << "\n Symbol table for image:\n"; 2715 2716 if (ELFT::Is64Bits) 2717 OS << " Num: Value Size Type Bind Vis Ndx Name\n"; 2718 else 2719 OS << " Num: Value Size Type Bind Vis Ndx Name\n"; 2720 } 2721 2722 template <class ELFT> 2723 std::string GNUStyle<ELFT>::getSymbolSectionNdx(const ELFO *Obj, 2724 const Elf_Sym *Symbol, 2725 const Elf_Sym *FirstSym) { 2726 unsigned SectionIndex = Symbol->st_shndx; 2727 switch (SectionIndex) { 2728 case ELF::SHN_UNDEF: 2729 return "UND"; 2730 case ELF::SHN_ABS: 2731 return "ABS"; 2732 case ELF::SHN_COMMON: 2733 return "COM"; 2734 case ELF::SHN_XINDEX: 2735 SectionIndex = Obj->getExtendedSymbolTableIndex( 2736 Symbol, FirstSym, this->dumper()->getShndxTable()); 2737 default: 2738 // Find if: 2739 // Processor specific 2740 if (SectionIndex >= ELF::SHN_LOPROC && SectionIndex <= ELF::SHN_HIPROC) 2741 return std::string("PRC[0x") + 2742 to_string(format_hex_no_prefix(SectionIndex, 4)) + "]"; 2743 // OS specific 2744 if (SectionIndex >= ELF::SHN_LOOS && SectionIndex <= ELF::SHN_HIOS) 2745 return std::string("OS[0x") + 2746 to_string(format_hex_no_prefix(SectionIndex, 4)) + "]"; 2747 // Architecture reserved: 2748 if (SectionIndex >= ELF::SHN_LORESERVE && 2749 SectionIndex <= ELF::SHN_HIRESERVE) 2750 return std::string("RSV[0x") + 2751 to_string(format_hex_no_prefix(SectionIndex, 4)) + "]"; 2752 // A normal section with an index 2753 return to_string(format_decimal(SectionIndex, 3)); 2754 } 2755 } 2756 2757 template <class ELFT> 2758 void GNUStyle<ELFT>::printSymbol(const ELFO *Obj, const Elf_Sym *Symbol, 2759 const Elf_Sym *FirstSym, StringRef StrTable, 2760 bool IsDynamic) { 2761 static int Idx = 0; 2762 static bool Dynamic = true; 2763 size_t Width; 2764 2765 // If this function was called with a different value from IsDynamic 2766 // from last call, happens when we move from dynamic to static symbol 2767 // table, "Num" field should be reset. 2768 if (!Dynamic != !IsDynamic) { 2769 Idx = 0; 2770 Dynamic = false; 2771 } 2772 std::string Num, Name, Value, Size, Binding, Type, Visibility, Section; 2773 unsigned Bias = 0; 2774 if (ELFT::Is64Bits) { 2775 Bias = 8; 2776 Width = 16; 2777 } else { 2778 Bias = 0; 2779 Width = 8; 2780 } 2781 Field Fields[8] = {0, 8, 17 + Bias, 23 + Bias, 2782 31 + Bias, 38 + Bias, 47 + Bias, 51 + Bias}; 2783 Num = to_string(format_decimal(Idx++, 6)) + ":"; 2784 Value = to_string(format_hex_no_prefix(Symbol->st_value, Width)); 2785 Size = to_string(format_decimal(Symbol->st_size, 5)); 2786 unsigned char SymbolType = Symbol->getType(); 2787 if (Obj->getHeader()->e_machine == ELF::EM_AMDGPU && 2788 SymbolType >= ELF::STT_LOOS && SymbolType < ELF::STT_HIOS) 2789 Type = printEnum(SymbolType, makeArrayRef(AMDGPUSymbolTypes)); 2790 else 2791 Type = printEnum(SymbolType, makeArrayRef(ElfSymbolTypes)); 2792 unsigned Vis = Symbol->getVisibility(); 2793 Binding = printEnum(Symbol->getBinding(), makeArrayRef(ElfSymbolBindings)); 2794 Visibility = printEnum(Vis, makeArrayRef(ElfSymbolVisibilities)); 2795 Section = getSymbolSectionNdx(Obj, Symbol, FirstSym); 2796 Name = this->dumper()->getFullSymbolName(Symbol, StrTable, IsDynamic); 2797 Fields[0].Str = Num; 2798 Fields[1].Str = Value; 2799 Fields[2].Str = Size; 2800 Fields[3].Str = Type; 2801 Fields[4].Str = Binding; 2802 Fields[5].Str = Visibility; 2803 Fields[6].Str = Section; 2804 Fields[7].Str = Name; 2805 for (auto &Entry : Fields) 2806 printField(Entry); 2807 OS << "\n"; 2808 } 2809 2810 template <class ELFT> void GNUStyle<ELFT>::printSymbols(const ELFO *Obj) { 2811 this->dumper()->printSymbolsHelper(true); 2812 this->dumper()->printSymbolsHelper(false); 2813 } 2814 2815 template <class ELFT> 2816 void GNUStyle<ELFT>::printDynamicSymbols(const ELFO *Obj) { 2817 this->dumper()->printSymbolsHelper(true); 2818 } 2819 2820 static inline std::string printPhdrFlags(unsigned Flag) { 2821 std::string Str; 2822 Str = (Flag & PF_R) ? "R" : " "; 2823 Str += (Flag & PF_W) ? "W" : " "; 2824 Str += (Flag & PF_X) ? "E" : " "; 2825 return Str; 2826 } 2827 2828 // SHF_TLS sections are only in PT_TLS, PT_LOAD or PT_GNU_RELRO 2829 // PT_TLS must only have SHF_TLS sections 2830 template <class ELFT> 2831 bool GNUStyle<ELFT>::checkTLSSections(const Elf_Phdr &Phdr, 2832 const Elf_Shdr &Sec) { 2833 return (((Sec.sh_flags & ELF::SHF_TLS) && 2834 ((Phdr.p_type == ELF::PT_TLS) || (Phdr.p_type == ELF::PT_LOAD) || 2835 (Phdr.p_type == ELF::PT_GNU_RELRO))) || 2836 (!(Sec.sh_flags & ELF::SHF_TLS) && Phdr.p_type != ELF::PT_TLS)); 2837 } 2838 2839 // Non-SHT_NOBITS must have its offset inside the segment 2840 // Only non-zero section can be at end of segment 2841 template <class ELFT> 2842 bool GNUStyle<ELFT>::checkoffsets(const Elf_Phdr &Phdr, const Elf_Shdr &Sec) { 2843 if (Sec.sh_type == ELF::SHT_NOBITS) 2844 return true; 2845 bool IsSpecial = 2846 (Sec.sh_type == ELF::SHT_NOBITS) && ((Sec.sh_flags & ELF::SHF_TLS) != 0); 2847 // .tbss is special, it only has memory in PT_TLS and has NOBITS properties 2848 auto SectionSize = 2849 (IsSpecial && Phdr.p_type != ELF::PT_TLS) ? 0 : Sec.sh_size; 2850 if (Sec.sh_offset >= Phdr.p_offset) 2851 return ((Sec.sh_offset + SectionSize <= Phdr.p_filesz + Phdr.p_offset) 2852 /*only non-zero sized sections at end*/ && 2853 (Sec.sh_offset + 1 <= Phdr.p_offset + Phdr.p_filesz)); 2854 return false; 2855 } 2856 2857 // SHF_ALLOC must have VMA inside segment 2858 // Only non-zero section can be at end of segment 2859 template <class ELFT> 2860 bool GNUStyle<ELFT>::checkVMA(const Elf_Phdr &Phdr, const Elf_Shdr &Sec) { 2861 if (!(Sec.sh_flags & ELF::SHF_ALLOC)) 2862 return true; 2863 bool IsSpecial = 2864 (Sec.sh_type == ELF::SHT_NOBITS) && ((Sec.sh_flags & ELF::SHF_TLS) != 0); 2865 // .tbss is special, it only has memory in PT_TLS and has NOBITS properties 2866 auto SectionSize = 2867 (IsSpecial && Phdr.p_type != ELF::PT_TLS) ? 0 : Sec.sh_size; 2868 if (Sec.sh_addr >= Phdr.p_vaddr) 2869 return ((Sec.sh_addr + SectionSize <= Phdr.p_vaddr + Phdr.p_memsz) && 2870 (Sec.sh_addr + 1 <= Phdr.p_vaddr + Phdr.p_memsz)); 2871 return false; 2872 } 2873 2874 // No section with zero size must be at start or end of PT_DYNAMIC 2875 template <class ELFT> 2876 bool GNUStyle<ELFT>::checkPTDynamic(const Elf_Phdr &Phdr, const Elf_Shdr &Sec) { 2877 if (Phdr.p_type != ELF::PT_DYNAMIC || Sec.sh_size != 0 || Phdr.p_memsz == 0) 2878 return true; 2879 // Is section within the phdr both based on offset and VMA ? 2880 return ((Sec.sh_type == ELF::SHT_NOBITS) || 2881 (Sec.sh_offset > Phdr.p_offset && 2882 Sec.sh_offset < Phdr.p_offset + Phdr.p_filesz)) && 2883 (!(Sec.sh_flags & ELF::SHF_ALLOC) || 2884 (Sec.sh_addr > Phdr.p_vaddr && Sec.sh_addr < Phdr.p_memsz)); 2885 } 2886 2887 template <class ELFT> 2888 void GNUStyle<ELFT>::printProgramHeaders(const ELFO *Obj) { 2889 unsigned Bias = ELFT::Is64Bits ? 8 : 0; 2890 unsigned Width = ELFT::Is64Bits ? 18 : 10; 2891 unsigned SizeWidth = ELFT::Is64Bits ? 8 : 7; 2892 std::string Type, Offset, VMA, LMA, FileSz, MemSz, Flag, Align; 2893 2894 const Elf_Ehdr *Header = Obj->getHeader(); 2895 Field Fields[8] = {2, 17, 26, 37 + Bias, 2896 48 + Bias, 56 + Bias, 64 + Bias, 68 + Bias}; 2897 OS << "\nElf file type is " 2898 << printEnum(Header->e_type, makeArrayRef(ElfObjectFileType)) << "\n" 2899 << "Entry point " << format_hex(Header->e_entry, 3) << "\n" 2900 << "There are " << Header->e_phnum << " program headers," 2901 << " starting at offset " << Header->e_phoff << "\n\n" 2902 << "Program Headers:\n"; 2903 if (ELFT::Is64Bits) 2904 OS << " Type Offset VirtAddr PhysAddr " 2905 << " FileSiz MemSiz Flg Align\n"; 2906 else 2907 OS << " Type Offset VirtAddr PhysAddr FileSiz " 2908 << "MemSiz Flg Align\n"; 2909 for (const auto &Phdr : Obj->program_headers()) { 2910 Type = getElfPtType(Header->e_machine, Phdr.p_type); 2911 Offset = to_string(format_hex(Phdr.p_offset, 8)); 2912 VMA = to_string(format_hex(Phdr.p_vaddr, Width)); 2913 LMA = to_string(format_hex(Phdr.p_paddr, Width)); 2914 FileSz = to_string(format_hex(Phdr.p_filesz, SizeWidth)); 2915 MemSz = to_string(format_hex(Phdr.p_memsz, SizeWidth)); 2916 Flag = printPhdrFlags(Phdr.p_flags); 2917 Align = to_string(format_hex(Phdr.p_align, 1)); 2918 Fields[0].Str = Type; 2919 Fields[1].Str = Offset; 2920 Fields[2].Str = VMA; 2921 Fields[3].Str = LMA; 2922 Fields[4].Str = FileSz; 2923 Fields[5].Str = MemSz; 2924 Fields[6].Str = Flag; 2925 Fields[7].Str = Align; 2926 for (auto Field : Fields) 2927 printField(Field); 2928 if (Phdr.p_type == ELF::PT_INTERP) { 2929 OS << "\n [Requesting program interpreter: "; 2930 OS << reinterpret_cast<const char *>(Obj->base()) + Phdr.p_offset << "]"; 2931 } 2932 OS << "\n"; 2933 } 2934 OS << "\n Section to Segment mapping:\n Segment Sections...\n"; 2935 int Phnum = 0; 2936 for (const Elf_Phdr &Phdr : Obj->program_headers()) { 2937 std::string Sections; 2938 OS << format(" %2.2d ", Phnum++); 2939 for (const Elf_Shdr &Sec : Obj->sections()) { 2940 // Check if each section is in a segment and then print mapping. 2941 // readelf additionally makes sure it does not print zero sized sections 2942 // at end of segments and for PT_DYNAMIC both start and end of section 2943 // .tbss must only be shown in PT_TLS section. 2944 bool TbssInNonTLS = (Sec.sh_type == ELF::SHT_NOBITS) && 2945 ((Sec.sh_flags & ELF::SHF_TLS) != 0) && 2946 Phdr.p_type != ELF::PT_TLS; 2947 if (!TbssInNonTLS && checkTLSSections(Phdr, Sec) && 2948 checkoffsets(Phdr, Sec) && checkVMA(Phdr, Sec) && 2949 checkPTDynamic(Phdr, Sec) && (Sec.sh_type != ELF::SHT_NULL)) 2950 Sections += unwrapOrError(Obj->getSectionName(&Sec)).str() + " "; 2951 } 2952 OS << Sections << "\n"; 2953 OS.flush(); 2954 } 2955 } 2956 2957 template <class ELFT> 2958 void GNUStyle<ELFT>::printDynamicRelocation(const ELFO *Obj, Elf_Rela R, 2959 bool IsRela) { 2960 SmallString<32> RelocName; 2961 StringRef SymbolName; 2962 unsigned Width = ELFT::Is64Bits ? 16 : 8; 2963 unsigned Bias = ELFT::Is64Bits ? 8 : 0; 2964 // First two fields are bit width dependent. The rest of them are after are 2965 // fixed width. 2966 Field Fields[5] = {0, 10 + Bias, 19 + 2 * Bias, 42 + 2 * Bias, 53 + 2 * Bias}; 2967 2968 uint32_t SymIndex = R.getSymbol(Obj->isMips64EL()); 2969 const Elf_Sym *Sym = this->dumper()->dynamic_symbols().begin() + SymIndex; 2970 Obj->getRelocationTypeName(R.getType(Obj->isMips64EL()), RelocName); 2971 SymbolName = 2972 unwrapOrError(Sym->getName(this->dumper()->getDynamicStringTable())); 2973 std::string Addend = "", Info, Offset, Value; 2974 Offset = to_string(format_hex_no_prefix(R.r_offset, Width)); 2975 Info = to_string(format_hex_no_prefix(R.r_info, Width)); 2976 Value = to_string(format_hex_no_prefix(Sym->getValue(), Width)); 2977 int64_t RelAddend = R.r_addend; 2978 if (SymbolName.size() && IsRela) { 2979 if (R.r_addend < 0) 2980 Addend = " - "; 2981 else 2982 Addend = " + "; 2983 } 2984 2985 if (!SymbolName.size() && Sym->getValue() == 0) 2986 Value = ""; 2987 2988 if (IsRela) 2989 Addend += to_string(format_hex_no_prefix(std::abs(RelAddend), 1)); 2990 2991 2992 Fields[0].Str = Offset; 2993 Fields[1].Str = Info; 2994 Fields[2].Str = RelocName.c_str(); 2995 Fields[3].Str = Value; 2996 Fields[4].Str = SymbolName; 2997 for (auto &Field : Fields) 2998 printField(Field); 2999 OS << Addend; 3000 OS << "\n"; 3001 } 3002 3003 template <class ELFT> 3004 void GNUStyle<ELFT>::printDynamicRelocations(const ELFO *Obj) { 3005 const DynRegionInfo &DynRelRegion = this->dumper()->getDynRelRegion(); 3006 const DynRegionInfo &DynRelaRegion = this->dumper()->getDynRelaRegion(); 3007 const DynRegionInfo &DynPLTRelRegion = this->dumper()->getDynPLTRelRegion(); 3008 if (DynRelaRegion.Size > 0) { 3009 OS << "\n'RELA' relocation section at offset " 3010 << format_hex(reinterpret_cast<const uint8_t *>(DynRelaRegion.Addr) - 3011 Obj->base(), 3012 1) << " contains " << DynRelaRegion.Size << " bytes:\n"; 3013 printRelocHeader(OS, ELFT::Is64Bits, true); 3014 for (const Elf_Rela &Rela : this->dumper()->dyn_relas()) 3015 printDynamicRelocation(Obj, Rela, true); 3016 } 3017 if (DynRelRegion.Size > 0) { 3018 OS << "\n'REL' relocation section at offset " 3019 << format_hex(reinterpret_cast<const uint8_t *>(DynRelRegion.Addr) - 3020 Obj->base(), 3021 1) << " contains " << DynRelRegion.Size << " bytes:\n"; 3022 printRelocHeader(OS, ELFT::Is64Bits, false); 3023 for (const Elf_Rel &Rel : this->dumper()->dyn_rels()) { 3024 Elf_Rela Rela; 3025 Rela.r_offset = Rel.r_offset; 3026 Rela.r_info = Rel.r_info; 3027 Rela.r_addend = 0; 3028 printDynamicRelocation(Obj, Rela, false); 3029 } 3030 } 3031 if (DynPLTRelRegion.Size) { 3032 OS << "\n'PLT' relocation section at offset " 3033 << format_hex(reinterpret_cast<const uint8_t *>(DynPLTRelRegion.Addr) - 3034 Obj->base(), 3035 1) << " contains " << DynPLTRelRegion.Size << " bytes:\n"; 3036 } 3037 if (DynPLTRelRegion.EntSize == sizeof(Elf_Rela)) { 3038 printRelocHeader(OS, ELFT::Is64Bits, true); 3039 for (const Elf_Rela &Rela : DynPLTRelRegion.getAsArrayRef<Elf_Rela>()) 3040 printDynamicRelocation(Obj, Rela, true); 3041 } else { 3042 printRelocHeader(OS, ELFT::Is64Bits, false); 3043 for (const Elf_Rel &Rel : DynPLTRelRegion.getAsArrayRef<Elf_Rel>()) { 3044 Elf_Rela Rela; 3045 Rela.r_offset = Rel.r_offset; 3046 Rela.r_info = Rel.r_info; 3047 Rela.r_addend = 0; 3048 printDynamicRelocation(Obj, Rela, false); 3049 } 3050 } 3051 } 3052 3053 // Hash histogram shows statistics of how efficient the hash was for the 3054 // dynamic symbol table. The table shows number of hash buckets for different 3055 // lengths of chains as absolute number and percentage of the total buckets. 3056 // Additionally cumulative coverage of symbols for each set of buckets. 3057 template <class ELFT> 3058 void GNUStyle<ELFT>::printHashHistogram(const ELFFile<ELFT> *Obj) { 3059 3060 const Elf_Hash *HashTable = this->dumper()->getHashTable(); 3061 const Elf_GnuHash *GnuHashTable = this->dumper()->getGnuHashTable(); 3062 3063 // Print histogram for .hash section 3064 if (HashTable) { 3065 size_t NBucket = HashTable->nbucket; 3066 size_t NChain = HashTable->nchain; 3067 ArrayRef<Elf_Word> Buckets = HashTable->buckets(); 3068 ArrayRef<Elf_Word> Chains = HashTable->chains(); 3069 size_t TotalSyms = 0; 3070 // If hash table is correct, we have at least chains with 0 length 3071 size_t MaxChain = 1; 3072 size_t CumulativeNonZero = 0; 3073 3074 if (NChain == 0 || NBucket == 0) 3075 return; 3076 3077 std::vector<size_t> ChainLen(NBucket, 0); 3078 // Go over all buckets and and note chain lengths of each bucket (total 3079 // unique chain lengths). 3080 for (size_t B = 0; B < NBucket; B++) { 3081 for (size_t C = Buckets[B]; C > 0 && C < NChain; C = Chains[C]) 3082 if (MaxChain <= ++ChainLen[B]) 3083 MaxChain++; 3084 TotalSyms += ChainLen[B]; 3085 } 3086 3087 if (!TotalSyms) 3088 return; 3089 3090 std::vector<size_t> Count(MaxChain, 0) ; 3091 // Count how long is the chain for each bucket 3092 for (size_t B = 0; B < NBucket; B++) 3093 ++Count[ChainLen[B]]; 3094 // Print Number of buckets with each chain lengths and their cumulative 3095 // coverage of the symbols 3096 OS << "Histogram for bucket list length (total of " << NBucket 3097 << " buckets)\n" 3098 << " Length Number % of total Coverage\n"; 3099 for (size_t I = 0; I < MaxChain; I++) { 3100 CumulativeNonZero += Count[I] * I; 3101 OS << format("%7lu %-10lu (%5.1f%%) %5.1f%%\n", I, Count[I], 3102 (Count[I] * 100.0) / NBucket, 3103 (CumulativeNonZero * 100.0) / TotalSyms); 3104 } 3105 } 3106 3107 // Print histogram for .gnu.hash section 3108 if (GnuHashTable) { 3109 size_t NBucket = GnuHashTable->nbuckets; 3110 ArrayRef<Elf_Word> Buckets = GnuHashTable->buckets(); 3111 unsigned NumSyms = this->dumper()->dynamic_symbols().size(); 3112 if (!NumSyms) 3113 return; 3114 ArrayRef<Elf_Word> Chains = GnuHashTable->values(NumSyms); 3115 size_t Symndx = GnuHashTable->symndx; 3116 size_t TotalSyms = 0; 3117 size_t MaxChain = 1; 3118 size_t CumulativeNonZero = 0; 3119 3120 if (Chains.size() == 0 || NBucket == 0) 3121 return; 3122 3123 std::vector<size_t> ChainLen(NBucket, 0); 3124 3125 for (size_t B = 0; B < NBucket; B++) { 3126 if (!Buckets[B]) 3127 continue; 3128 size_t Len = 1; 3129 for (size_t C = Buckets[B] - Symndx; 3130 C < Chains.size() && (Chains[C] & 1) == 0; C++) 3131 if (MaxChain < ++Len) 3132 MaxChain++; 3133 ChainLen[B] = Len; 3134 TotalSyms += Len; 3135 } 3136 MaxChain++; 3137 3138 if (!TotalSyms) 3139 return; 3140 3141 std::vector<size_t> Count(MaxChain, 0) ; 3142 for (size_t B = 0; B < NBucket; B++) 3143 ++Count[ChainLen[B]]; 3144 // Print Number of buckets with each chain lengths and their cumulative 3145 // coverage of the symbols 3146 OS << "Histogram for `.gnu.hash' bucket list length (total of " << NBucket 3147 << " buckets)\n" 3148 << " Length Number % of total Coverage\n"; 3149 for (size_t I = 0; I <MaxChain; I++) { 3150 CumulativeNonZero += Count[I] * I; 3151 OS << format("%7lu %-10lu (%5.1f%%) %5.1f%%\n", I, Count[I], 3152 (Count[I] * 100.0) / NBucket, 3153 (CumulativeNonZero * 100.0) / TotalSyms); 3154 } 3155 } 3156 } 3157 3158 static std::string getGNUNoteTypeName(const uint32_t NT) { 3159 static const struct { 3160 uint32_t ID; 3161 const char *Name; 3162 } Notes[] = { 3163 {ELF::NT_GNU_ABI_TAG, "NT_GNU_ABI_TAG (ABI version tag)"}, 3164 {ELF::NT_GNU_HWCAP, "NT_GNU_HWCAP (DSO-supplied software HWCAP info)"}, 3165 {ELF::NT_GNU_BUILD_ID, "NT_GNU_BUILD_ID (unique build ID bitstring)"}, 3166 {ELF::NT_GNU_GOLD_VERSION, "NT_GNU_GOLD_VERSION (gold version)"}, 3167 }; 3168 3169 for (const auto &Note : Notes) 3170 if (Note.ID == NT) 3171 return std::string(Note.Name); 3172 3173 std::string string; 3174 raw_string_ostream OS(string); 3175 OS << format("Unknown note type (0x%08x)", NT); 3176 return string; 3177 } 3178 3179 template <typename ELFT> 3180 static void printGNUNote(raw_ostream &OS, uint32_t NoteType, 3181 ArrayRef<typename ELFFile<ELFT>::Elf_Word> Words) { 3182 switch (NoteType) { 3183 default: 3184 return; 3185 case ELF::NT_GNU_ABI_TAG: { 3186 static const char *OSNames[] = { 3187 "Linux", "Hurd", "Solaris", "FreeBSD", "NetBSD", "Syllable", "NaCl", 3188 }; 3189 3190 StringRef OSName = "Unknown"; 3191 if (Words[0] < array_lengthof(OSNames)) 3192 OSName = OSNames[Words[0]]; 3193 uint32_t Major = Words[1], Minor = Words[2], Patch = Words[3]; 3194 3195 if (Words.size() < 4) 3196 OS << " <corrupt GNU_ABI_TAG>"; 3197 else 3198 OS << " OS: " << OSName << ", ABI: " << Major << "." << Minor << "." 3199 << Patch; 3200 break; 3201 } 3202 case ELF::NT_GNU_BUILD_ID: { 3203 OS << " Build ID: "; 3204 ArrayRef<uint8_t> ID(reinterpret_cast<const uint8_t *>(Words.data()), 3205 Words.size() * 4); 3206 for (const auto &B : ID) 3207 OS << format_hex_no_prefix(B, 2); 3208 break; 3209 } 3210 case ELF::NT_GNU_GOLD_VERSION: 3211 OS << " Version: " 3212 << StringRef(reinterpret_cast<const char *>(Words.data()), 3213 Words.size() * 4); 3214 break; 3215 } 3216 3217 OS << '\n'; 3218 } 3219 3220 template <class ELFT> 3221 void GNUStyle<ELFT>::printNotes(const ELFFile<ELFT> *Obj) { 3222 const Elf_Ehdr *e = Obj->getHeader(); 3223 bool IsCore = e->e_type == ELF::ET_CORE; 3224 3225 auto process = [&](const typename ELFFile<ELFT>::Elf_Off Offset, 3226 const typename ELFFile<ELFT>::Elf_Addr Size) { 3227 if (Size <= 0) 3228 return; 3229 3230 const auto *P = static_cast<const uint8_t *>(Obj->base() + Offset); 3231 const auto *E = P + Size; 3232 3233 OS << "Displaying notes found at file offset " << format_hex(Offset, 10) 3234 << " with length " << format_hex(Size, 10) << ":\n" 3235 << " Owner Data size\tDescription\n"; 3236 3237 while (P < E) { 3238 const Elf_Word *Words = reinterpret_cast<const Elf_Word *>(&P[0]); 3239 3240 uint32_t NameSize = Words[0]; 3241 uint32_t DescriptorSize = Words[1]; 3242 uint32_t Type = Words[2]; 3243 3244 ArrayRef<Elf_Word> Descriptor(&Words[3 + (alignTo<4>(NameSize) / 4)], 3245 alignTo<4>(DescriptorSize) / 4); 3246 3247 StringRef Name; 3248 if (NameSize) 3249 Name = 3250 StringRef(reinterpret_cast<const char *>(&Words[3]), NameSize - 1); 3251 3252 OS << " " << Name << std::string(22 - NameSize, ' ') 3253 << format_hex(DescriptorSize, 10) << '\t'; 3254 3255 if (Name == "GNU") { 3256 OS << getGNUNoteTypeName(Type) << '\n'; 3257 printGNUNote<ELFT>(OS, Type, Descriptor); 3258 } 3259 OS << '\n'; 3260 3261 P = P + 3 * sizeof(Elf_Word) * alignTo<4>(NameSize) + 3262 alignTo<4>(DescriptorSize); 3263 } 3264 }; 3265 3266 if (IsCore) { 3267 for (const auto &P : Obj->program_headers()) 3268 if (P.p_type == PT_NOTE) 3269 process(P.p_offset, P.p_filesz); 3270 } else { 3271 for (const auto &S : Obj->sections()) 3272 if (S.sh_type == SHT_NOTE) 3273 process(S.sh_offset, S.sh_size); 3274 } 3275 } 3276 3277 template <class ELFT> void LLVMStyle<ELFT>::printFileHeaders(const ELFO *Obj) { 3278 const Elf_Ehdr *e = Obj->getHeader(); 3279 { 3280 DictScope D(W, "ElfHeader"); 3281 { 3282 DictScope D(W, "Ident"); 3283 W.printBinary("Magic", makeArrayRef(e->e_ident).slice(ELF::EI_MAG0, 4)); 3284 W.printEnum("Class", e->e_ident[ELF::EI_CLASS], makeArrayRef(ElfClass)); 3285 W.printEnum("DataEncoding", e->e_ident[ELF::EI_DATA], 3286 makeArrayRef(ElfDataEncoding)); 3287 W.printNumber("FileVersion", e->e_ident[ELF::EI_VERSION]); 3288 3289 // Handle architecture specific OS/ABI values. 3290 if (e->e_machine == ELF::EM_AMDGPU && 3291 e->e_ident[ELF::EI_OSABI] == ELF::ELFOSABI_AMDGPU_HSA) 3292 W.printHex("OS/ABI", "AMDGPU_HSA", ELF::ELFOSABI_AMDGPU_HSA); 3293 else 3294 W.printEnum("OS/ABI", e->e_ident[ELF::EI_OSABI], 3295 makeArrayRef(ElfOSABI)); 3296 W.printNumber("ABIVersion", e->e_ident[ELF::EI_ABIVERSION]); 3297 W.printBinary("Unused", makeArrayRef(e->e_ident).slice(ELF::EI_PAD)); 3298 } 3299 3300 W.printEnum("Type", e->e_type, makeArrayRef(ElfObjectFileType)); 3301 W.printEnum("Machine", e->e_machine, makeArrayRef(ElfMachineType)); 3302 W.printNumber("Version", e->e_version); 3303 W.printHex("Entry", e->e_entry); 3304 W.printHex("ProgramHeaderOffset", e->e_phoff); 3305 W.printHex("SectionHeaderOffset", e->e_shoff); 3306 if (e->e_machine == EM_MIPS) 3307 W.printFlags("Flags", e->e_flags, makeArrayRef(ElfHeaderMipsFlags), 3308 unsigned(ELF::EF_MIPS_ARCH), unsigned(ELF::EF_MIPS_ABI), 3309 unsigned(ELF::EF_MIPS_MACH)); 3310 else 3311 W.printFlags("Flags", e->e_flags); 3312 W.printNumber("HeaderSize", e->e_ehsize); 3313 W.printNumber("ProgramHeaderEntrySize", e->e_phentsize); 3314 W.printNumber("ProgramHeaderCount", e->e_phnum); 3315 W.printNumber("SectionHeaderEntrySize", e->e_shentsize); 3316 W.printNumber("SectionHeaderCount", e->e_shnum); 3317 W.printNumber("StringTableSectionIndex", e->e_shstrndx); 3318 } 3319 } 3320 3321 template <class ELFT> 3322 void LLVMStyle<ELFT>::printGroupSections(const ELFO *Obj) { 3323 DictScope Lists(W, "Groups"); 3324 uint32_t SectionIndex = 0; 3325 bool HasGroups = false; 3326 for (const Elf_Shdr &Sec : Obj->sections()) { 3327 if (Sec.sh_type == ELF::SHT_GROUP) { 3328 HasGroups = true; 3329 const Elf_Shdr *Symtab = unwrapOrError(Obj->getSection(Sec.sh_link)); 3330 StringRef StrTable = unwrapOrError(Obj->getStringTableForSymtab(*Symtab)); 3331 const Elf_Sym *Sym = Obj->template getEntry<Elf_Sym>(Symtab, Sec.sh_info); 3332 auto Data = unwrapOrError( 3333 Obj->template getSectionContentsAsArray<Elf_Word>(&Sec)); 3334 DictScope D(W, "Group"); 3335 StringRef Name = unwrapOrError(Obj->getSectionName(&Sec)); 3336 W.printNumber("Name", Name, Sec.sh_name); 3337 W.printNumber("Index", SectionIndex); 3338 W.printHex("Type", getGroupType(Data[0]), Data[0]); 3339 W.startLine() << "Signature: " << StrTable.data() + Sym->st_name << "\n"; 3340 { 3341 ListScope L(W, "Section(s) in group"); 3342 size_t Member = 1; 3343 while (Member < Data.size()) { 3344 auto Sec = unwrapOrError(Obj->getSection(Data[Member])); 3345 const StringRef Name = unwrapOrError(Obj->getSectionName(Sec)); 3346 W.startLine() << Name << " (" << Data[Member++] << ")\n"; 3347 } 3348 } 3349 } 3350 ++SectionIndex; 3351 } 3352 if (!HasGroups) 3353 W.startLine() << "There are no group sections in the file.\n"; 3354 } 3355 3356 template <class ELFT> void LLVMStyle<ELFT>::printRelocations(const ELFO *Obj) { 3357 ListScope D(W, "Relocations"); 3358 3359 int SectionNumber = -1; 3360 for (const Elf_Shdr &Sec : Obj->sections()) { 3361 ++SectionNumber; 3362 3363 if (Sec.sh_type != ELF::SHT_REL && Sec.sh_type != ELF::SHT_RELA) 3364 continue; 3365 3366 StringRef Name = unwrapOrError(Obj->getSectionName(&Sec)); 3367 3368 W.startLine() << "Section (" << SectionNumber << ") " << Name << " {\n"; 3369 W.indent(); 3370 3371 printRelocations(&Sec, Obj); 3372 3373 W.unindent(); 3374 W.startLine() << "}\n"; 3375 } 3376 } 3377 3378 template <class ELFT> 3379 void LLVMStyle<ELFT>::printRelocations(const Elf_Shdr *Sec, const ELFO *Obj) { 3380 const Elf_Shdr *SymTab = unwrapOrError(Obj->getSection(Sec->sh_link)); 3381 3382 switch (Sec->sh_type) { 3383 case ELF::SHT_REL: 3384 for (const Elf_Rel &R : Obj->rels(Sec)) { 3385 Elf_Rela Rela; 3386 Rela.r_offset = R.r_offset; 3387 Rela.r_info = R.r_info; 3388 Rela.r_addend = 0; 3389 printRelocation(Obj, Rela, SymTab); 3390 } 3391 break; 3392 case ELF::SHT_RELA: 3393 for (const Elf_Rela &R : Obj->relas(Sec)) 3394 printRelocation(Obj, R, SymTab); 3395 break; 3396 } 3397 } 3398 3399 template <class ELFT> 3400 void LLVMStyle<ELFT>::printRelocation(const ELFO *Obj, Elf_Rela Rel, 3401 const Elf_Shdr *SymTab) { 3402 SmallString<32> RelocName; 3403 Obj->getRelocationTypeName(Rel.getType(Obj->isMips64EL()), RelocName); 3404 StringRef TargetName; 3405 const Elf_Sym *Sym = Obj->getRelocationSymbol(&Rel, SymTab); 3406 if (Sym && Sym->getType() == ELF::STT_SECTION) { 3407 const Elf_Shdr *Sec = unwrapOrError( 3408 Obj->getSection(Sym, SymTab, this->dumper()->getShndxTable())); 3409 TargetName = unwrapOrError(Obj->getSectionName(Sec)); 3410 } else if (Sym) { 3411 StringRef StrTable = unwrapOrError(Obj->getStringTableForSymtab(*SymTab)); 3412 TargetName = unwrapOrError(Sym->getName(StrTable)); 3413 } 3414 3415 if (opts::ExpandRelocs) { 3416 DictScope Group(W, "Relocation"); 3417 W.printHex("Offset", Rel.r_offset); 3418 W.printNumber("Type", RelocName, (int)Rel.getType(Obj->isMips64EL())); 3419 W.printNumber("Symbol", TargetName.size() > 0 ? TargetName : "-", 3420 Rel.getSymbol(Obj->isMips64EL())); 3421 W.printHex("Addend", Rel.r_addend); 3422 } else { 3423 raw_ostream &OS = W.startLine(); 3424 OS << W.hex(Rel.r_offset) << " " << RelocName << " " 3425 << (TargetName.size() > 0 ? TargetName : "-") << " " 3426 << W.hex(Rel.r_addend) << "\n"; 3427 } 3428 } 3429 3430 template <class ELFT> void LLVMStyle<ELFT>::printSections(const ELFO *Obj) { 3431 ListScope SectionsD(W, "Sections"); 3432 3433 int SectionIndex = -1; 3434 for (const Elf_Shdr &Sec : Obj->sections()) { 3435 ++SectionIndex; 3436 3437 StringRef Name = unwrapOrError(Obj->getSectionName(&Sec)); 3438 3439 DictScope SectionD(W, "Section"); 3440 W.printNumber("Index", SectionIndex); 3441 W.printNumber("Name", Name, Sec.sh_name); 3442 W.printHex("Type", 3443 getElfSectionType(Obj->getHeader()->e_machine, Sec.sh_type), 3444 Sec.sh_type); 3445 std::vector<EnumEntry<unsigned>> SectionFlags(std::begin(ElfSectionFlags), 3446 std::end(ElfSectionFlags)); 3447 switch (Obj->getHeader()->e_machine) { 3448 case EM_AMDGPU: 3449 SectionFlags.insert(SectionFlags.end(), std::begin(ElfAMDGPUSectionFlags), 3450 std::end(ElfAMDGPUSectionFlags)); 3451 break; 3452 case EM_HEXAGON: 3453 SectionFlags.insert(SectionFlags.end(), 3454 std::begin(ElfHexagonSectionFlags), 3455 std::end(ElfHexagonSectionFlags)); 3456 break; 3457 case EM_MIPS: 3458 SectionFlags.insert(SectionFlags.end(), std::begin(ElfMipsSectionFlags), 3459 std::end(ElfMipsSectionFlags)); 3460 break; 3461 case EM_X86_64: 3462 SectionFlags.insert(SectionFlags.end(), std::begin(ElfX86_64SectionFlags), 3463 std::end(ElfX86_64SectionFlags)); 3464 break; 3465 case EM_XCORE: 3466 SectionFlags.insert(SectionFlags.end(), std::begin(ElfXCoreSectionFlags), 3467 std::end(ElfXCoreSectionFlags)); 3468 break; 3469 default: 3470 // Nothing to do. 3471 break; 3472 } 3473 W.printFlags("Flags", Sec.sh_flags, makeArrayRef(SectionFlags)); 3474 W.printHex("Address", Sec.sh_addr); 3475 W.printHex("Offset", Sec.sh_offset); 3476 W.printNumber("Size", Sec.sh_size); 3477 W.printNumber("Link", Sec.sh_link); 3478 W.printNumber("Info", Sec.sh_info); 3479 W.printNumber("AddressAlignment", Sec.sh_addralign); 3480 W.printNumber("EntrySize", Sec.sh_entsize); 3481 3482 if (opts::SectionRelocations) { 3483 ListScope D(W, "Relocations"); 3484 printRelocations(&Sec, Obj); 3485 } 3486 3487 if (opts::SectionSymbols) { 3488 ListScope D(W, "Symbols"); 3489 const Elf_Shdr *Symtab = this->dumper()->getDotSymtabSec(); 3490 StringRef StrTable = unwrapOrError(Obj->getStringTableForSymtab(*Symtab)); 3491 3492 for (const Elf_Sym &Sym : Obj->symbols(Symtab)) { 3493 const Elf_Shdr *SymSec = unwrapOrError( 3494 Obj->getSection(&Sym, Symtab, this->dumper()->getShndxTable())); 3495 if (SymSec == &Sec) 3496 printSymbol(Obj, &Sym, Obj->symbols(Symtab).begin(), StrTable, false); 3497 } 3498 } 3499 3500 if (opts::SectionData && Sec.sh_type != ELF::SHT_NOBITS) { 3501 ArrayRef<uint8_t> Data = unwrapOrError(Obj->getSectionContents(&Sec)); 3502 W.printBinaryBlock("SectionData", 3503 StringRef((const char *)Data.data(), Data.size())); 3504 } 3505 } 3506 } 3507 3508 template <class ELFT> 3509 void LLVMStyle<ELFT>::printSymbol(const ELFO *Obj, const Elf_Sym *Symbol, 3510 const Elf_Sym *First, StringRef StrTable, 3511 bool IsDynamic) { 3512 unsigned SectionIndex = 0; 3513 StringRef SectionName; 3514 getSectionNameIndex(*Obj, Symbol, First, this->dumper()->getShndxTable(), 3515 SectionName, SectionIndex); 3516 std::string FullSymbolName = 3517 this->dumper()->getFullSymbolName(Symbol, StrTable, IsDynamic); 3518 unsigned char SymbolType = Symbol->getType(); 3519 3520 DictScope D(W, "Symbol"); 3521 W.printNumber("Name", FullSymbolName, Symbol->st_name); 3522 W.printHex("Value", Symbol->st_value); 3523 W.printNumber("Size", Symbol->st_size); 3524 W.printEnum("Binding", Symbol->getBinding(), makeArrayRef(ElfSymbolBindings)); 3525 if (Obj->getHeader()->e_machine == ELF::EM_AMDGPU && 3526 SymbolType >= ELF::STT_LOOS && SymbolType < ELF::STT_HIOS) 3527 W.printEnum("Type", SymbolType, makeArrayRef(AMDGPUSymbolTypes)); 3528 else 3529 W.printEnum("Type", SymbolType, makeArrayRef(ElfSymbolTypes)); 3530 if (Symbol->st_other == 0) 3531 // Usually st_other flag is zero. Do not pollute the output 3532 // by flags enumeration in that case. 3533 W.printNumber("Other", 0); 3534 else { 3535 std::vector<EnumEntry<unsigned>> SymOtherFlags(std::begin(ElfSymOtherFlags), 3536 std::end(ElfSymOtherFlags)); 3537 if (Obj->getHeader()->e_machine == EM_MIPS) { 3538 // Someones in their infinite wisdom decided to make STO_MIPS_MIPS16 3539 // flag overlapped with other ST_MIPS_xxx flags. So consider both 3540 // cases separately. 3541 if ((Symbol->st_other & STO_MIPS_MIPS16) == STO_MIPS_MIPS16) 3542 SymOtherFlags.insert(SymOtherFlags.end(), 3543 std::begin(ElfMips16SymOtherFlags), 3544 std::end(ElfMips16SymOtherFlags)); 3545 else 3546 SymOtherFlags.insert(SymOtherFlags.end(), 3547 std::begin(ElfMipsSymOtherFlags), 3548 std::end(ElfMipsSymOtherFlags)); 3549 } 3550 W.printFlags("Other", Symbol->st_other, makeArrayRef(SymOtherFlags), 0x3u); 3551 } 3552 W.printHex("Section", SectionName, SectionIndex); 3553 } 3554 3555 template <class ELFT> void LLVMStyle<ELFT>::printSymbols(const ELFO *Obj) { 3556 ListScope Group(W, "Symbols"); 3557 this->dumper()->printSymbolsHelper(false); 3558 } 3559 3560 template <class ELFT> 3561 void LLVMStyle<ELFT>::printDynamicSymbols(const ELFO *Obj) { 3562 ListScope Group(W, "DynamicSymbols"); 3563 this->dumper()->printSymbolsHelper(true); 3564 } 3565 3566 template <class ELFT> 3567 void LLVMStyle<ELFT>::printDynamicRelocations(const ELFO *Obj) { 3568 const DynRegionInfo &DynRelRegion = this->dumper()->getDynRelRegion(); 3569 const DynRegionInfo &DynRelaRegion = this->dumper()->getDynRelaRegion(); 3570 const DynRegionInfo &DynPLTRelRegion = this->dumper()->getDynPLTRelRegion(); 3571 if (DynRelRegion.Size && DynRelaRegion.Size) 3572 report_fatal_error("There are both REL and RELA dynamic relocations"); 3573 W.startLine() << "Dynamic Relocations {\n"; 3574 W.indent(); 3575 if (DynRelaRegion.Size > 0) 3576 for (const Elf_Rela &Rela : this->dumper()->dyn_relas()) 3577 printDynamicRelocation(Obj, Rela); 3578 else 3579 for (const Elf_Rel &Rel : this->dumper()->dyn_rels()) { 3580 Elf_Rela Rela; 3581 Rela.r_offset = Rel.r_offset; 3582 Rela.r_info = Rel.r_info; 3583 Rela.r_addend = 0; 3584 printDynamicRelocation(Obj, Rela); 3585 } 3586 if (DynPLTRelRegion.EntSize == sizeof(Elf_Rela)) 3587 for (const Elf_Rela &Rela : DynPLTRelRegion.getAsArrayRef<Elf_Rela>()) 3588 printDynamicRelocation(Obj, Rela); 3589 else 3590 for (const Elf_Rel &Rel : DynPLTRelRegion.getAsArrayRef<Elf_Rel>()) { 3591 Elf_Rela Rela; 3592 Rela.r_offset = Rel.r_offset; 3593 Rela.r_info = Rel.r_info; 3594 Rela.r_addend = 0; 3595 printDynamicRelocation(Obj, Rela); 3596 } 3597 W.unindent(); 3598 W.startLine() << "}\n"; 3599 } 3600 3601 template <class ELFT> 3602 void LLVMStyle<ELFT>::printDynamicRelocation(const ELFO *Obj, Elf_Rela Rel) { 3603 SmallString<32> RelocName; 3604 Obj->getRelocationTypeName(Rel.getType(Obj->isMips64EL()), RelocName); 3605 StringRef SymbolName; 3606 uint32_t SymIndex = Rel.getSymbol(Obj->isMips64EL()); 3607 const Elf_Sym *Sym = this->dumper()->dynamic_symbols().begin() + SymIndex; 3608 SymbolName = 3609 unwrapOrError(Sym->getName(this->dumper()->getDynamicStringTable())); 3610 if (opts::ExpandRelocs) { 3611 DictScope Group(W, "Relocation"); 3612 W.printHex("Offset", Rel.r_offset); 3613 W.printNumber("Type", RelocName, (int)Rel.getType(Obj->isMips64EL())); 3614 W.printString("Symbol", SymbolName.size() > 0 ? SymbolName : "-"); 3615 W.printHex("Addend", Rel.r_addend); 3616 } else { 3617 raw_ostream &OS = W.startLine(); 3618 OS << W.hex(Rel.r_offset) << " " << RelocName << " " 3619 << (SymbolName.size() > 0 ? SymbolName : "-") << " " 3620 << W.hex(Rel.r_addend) << "\n"; 3621 } 3622 } 3623 3624 template <class ELFT> 3625 void LLVMStyle<ELFT>::printProgramHeaders(const ELFO *Obj) { 3626 ListScope L(W, "ProgramHeaders"); 3627 3628 for (const Elf_Phdr &Phdr : Obj->program_headers()) { 3629 DictScope P(W, "ProgramHeader"); 3630 W.printHex("Type", 3631 getElfSegmentType(Obj->getHeader()->e_machine, Phdr.p_type), 3632 Phdr.p_type); 3633 W.printHex("Offset", Phdr.p_offset); 3634 W.printHex("VirtualAddress", Phdr.p_vaddr); 3635 W.printHex("PhysicalAddress", Phdr.p_paddr); 3636 W.printNumber("FileSize", Phdr.p_filesz); 3637 W.printNumber("MemSize", Phdr.p_memsz); 3638 W.printFlags("Flags", Phdr.p_flags, makeArrayRef(ElfSegmentFlags)); 3639 W.printNumber("Alignment", Phdr.p_align); 3640 } 3641 } 3642 3643 template <class ELFT> 3644 void LLVMStyle<ELFT>::printHashHistogram(const ELFFile<ELFT> *Obj) { 3645 W.startLine() << "Hash Histogram not implemented!\n"; 3646 } 3647 3648 template <class ELFT> 3649 void LLVMStyle<ELFT>::printNotes(const ELFFile<ELFT> *Obj) { 3650 W.startLine() << "printNotes not implemented!\n"; 3651 } 3652 3653