1 //===- DumpOutputStyle.cpp ------------------------------------ *- C++ --*-===// 2 // 3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4 // See https://llvm.org/LICENSE.txt for license information. 5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6 // 7 //===----------------------------------------------------------------------===// 8 9 #include "DumpOutputStyle.h" 10 11 #include "FormatUtil.h" 12 #include "InputFile.h" 13 #include "MinimalSymbolDumper.h" 14 #include "MinimalTypeDumper.h" 15 #include "StreamUtil.h" 16 #include "TypeReferenceTracker.h" 17 #include "llvm-pdbutil.h" 18 19 #include "llvm/ADT/STLExtras.h" 20 #include "llvm/DebugInfo/CodeView/CVSymbolVisitor.h" 21 #include "llvm/DebugInfo/CodeView/CVTypeVisitor.h" 22 #include "llvm/DebugInfo/CodeView/DebugChecksumsSubsection.h" 23 #include "llvm/DebugInfo/CodeView/DebugCrossExSubsection.h" 24 #include "llvm/DebugInfo/CodeView/DebugCrossImpSubsection.h" 25 #include "llvm/DebugInfo/CodeView/DebugFrameDataSubsection.h" 26 #include "llvm/DebugInfo/CodeView/DebugInlineeLinesSubsection.h" 27 #include "llvm/DebugInfo/CodeView/DebugLinesSubsection.h" 28 #include "llvm/DebugInfo/CodeView/DebugStringTableSubsection.h" 29 #include "llvm/DebugInfo/CodeView/DebugSymbolsSubsection.h" 30 #include "llvm/DebugInfo/CodeView/Formatters.h" 31 #include "llvm/DebugInfo/CodeView/LazyRandomTypeCollection.h" 32 #include "llvm/DebugInfo/CodeView/Line.h" 33 #include "llvm/DebugInfo/CodeView/SymbolDeserializer.h" 34 #include "llvm/DebugInfo/CodeView/SymbolVisitorCallbackPipeline.h" 35 #include "llvm/DebugInfo/CodeView/SymbolVisitorCallbacks.h" 36 #include "llvm/DebugInfo/CodeView/TypeHashing.h" 37 #include "llvm/DebugInfo/CodeView/TypeIndexDiscovery.h" 38 #include "llvm/DebugInfo/MSF/MappedBlockStream.h" 39 #include "llvm/DebugInfo/PDB/Native/DbiModuleDescriptor.h" 40 #include "llvm/DebugInfo/PDB/Native/DbiStream.h" 41 #include "llvm/DebugInfo/PDB/Native/GlobalsStream.h" 42 #include "llvm/DebugInfo/PDB/Native/ISectionContribVisitor.h" 43 #include "llvm/DebugInfo/PDB/Native/InfoStream.h" 44 #include "llvm/DebugInfo/PDB/Native/ModuleDebugStream.h" 45 #include "llvm/DebugInfo/PDB/Native/PDBFile.h" 46 #include "llvm/DebugInfo/PDB/Native/PublicsStream.h" 47 #include "llvm/DebugInfo/PDB/Native/RawError.h" 48 #include "llvm/DebugInfo/PDB/Native/SymbolStream.h" 49 #include "llvm/DebugInfo/PDB/Native/TpiHashing.h" 50 #include "llvm/DebugInfo/PDB/Native/TpiStream.h" 51 #include "llvm/Object/COFF.h" 52 #include "llvm/Support/BinaryStreamReader.h" 53 #include "llvm/Support/FormatAdapters.h" 54 #include "llvm/Support/FormatVariadic.h" 55 56 #include <cctype> 57 58 using namespace llvm; 59 using namespace llvm::codeview; 60 using namespace llvm::msf; 61 using namespace llvm::pdb; 62 63 DumpOutputStyle::DumpOutputStyle(InputFile &File) 64 : File(File), P(2, false, outs()) { 65 if (opts::dump::DumpTypeRefStats) 66 RefTracker.reset(new TypeReferenceTracker(File)); 67 } 68 69 DumpOutputStyle::~DumpOutputStyle() {} 70 71 PDBFile &DumpOutputStyle::getPdb() { return File.pdb(); } 72 object::COFFObjectFile &DumpOutputStyle::getObj() { return File.obj(); } 73 74 void DumpOutputStyle::printStreamNotValidForObj() { 75 AutoIndent Indent(P, 4); 76 P.formatLine("Dumping this stream is not valid for object files"); 77 } 78 79 void DumpOutputStyle::printStreamNotPresent(StringRef StreamName) { 80 AutoIndent Indent(P, 4); 81 P.formatLine("{0} stream not present", StreamName); 82 } 83 84 Error DumpOutputStyle::dump() { 85 // Walk symbols & globals if we are supposed to mark types referenced. 86 if (opts::dump::DumpTypeRefStats) 87 RefTracker->mark(); 88 89 if (opts::dump::DumpSummary) { 90 if (auto EC = dumpFileSummary()) 91 return EC; 92 P.NewLine(); 93 } 94 95 if (opts::dump::DumpStreams) { 96 if (auto EC = dumpStreamSummary()) 97 return EC; 98 P.NewLine(); 99 } 100 101 if (opts::dump::DumpSymbolStats) { 102 if (auto EC = dumpSymbolStats()) 103 return EC; 104 P.NewLine(); 105 } 106 107 if (opts::dump::DumpUdtStats) { 108 if (auto EC = dumpUdtStats()) 109 return EC; 110 P.NewLine(); 111 } 112 113 if (opts::dump::DumpTypeStats) { 114 if (auto EC = dumpTypeStats()) 115 return EC; 116 P.NewLine(); 117 } 118 119 if (opts::dump::DumpNamedStreams) { 120 if (auto EC = dumpNamedStreams()) 121 return EC; 122 P.NewLine(); 123 } 124 125 if (opts::dump::DumpStringTable || opts::dump::DumpStringTableDetails) { 126 if (auto EC = dumpStringTable()) 127 return EC; 128 P.NewLine(); 129 } 130 131 if (opts::dump::DumpModules) { 132 if (auto EC = dumpModules()) 133 return EC; 134 } 135 136 if (opts::dump::DumpModuleFiles) { 137 if (auto EC = dumpModuleFiles()) 138 return EC; 139 } 140 141 if (opts::dump::DumpLines) { 142 if (auto EC = dumpLines()) 143 return EC; 144 } 145 146 if (opts::dump::DumpInlineeLines) { 147 if (auto EC = dumpInlineeLines()) 148 return EC; 149 } 150 151 if (opts::dump::DumpXmi) { 152 if (auto EC = dumpXmi()) 153 return EC; 154 } 155 156 if (opts::dump::DumpXme) { 157 if (auto EC = dumpXme()) 158 return EC; 159 } 160 161 if (opts::dump::DumpFpo) { 162 if (auto EC = dumpFpo()) 163 return EC; 164 } 165 166 if (File.isObj()) { 167 if (opts::dump::DumpTypes || !opts::dump::DumpTypeIndex.empty() || 168 opts::dump::DumpTypeExtras) 169 if (auto EC = dumpTypesFromObjectFile()) 170 return EC; 171 } else { 172 if (opts::dump::DumpTypes || !opts::dump::DumpTypeIndex.empty() || 173 opts::dump::DumpTypeExtras) { 174 if (auto EC = dumpTpiStream(StreamTPI)) 175 return EC; 176 } 177 178 if (opts::dump::DumpIds || !opts::dump::DumpIdIndex.empty() || 179 opts::dump::DumpIdExtras) { 180 if (auto EC = dumpTpiStream(StreamIPI)) 181 return EC; 182 } 183 } 184 185 if (opts::dump::DumpGSIRecords) { 186 if (auto EC = dumpGSIRecords()) 187 return EC; 188 } 189 190 if (opts::dump::DumpGlobals) { 191 if (auto EC = dumpGlobals()) 192 return EC; 193 } 194 195 if (opts::dump::DumpPublics) { 196 if (auto EC = dumpPublics()) 197 return EC; 198 } 199 200 if (opts::dump::DumpSymbols) { 201 auto EC = File.isPdb() ? dumpModuleSymsForPdb() : dumpModuleSymsForObj(); 202 if (EC) 203 return EC; 204 } 205 206 if (opts::dump::DumpTypeRefStats) { 207 if (auto EC = dumpTypeRefStats()) 208 return EC; 209 } 210 211 if (opts::dump::DumpSectionHeaders) { 212 if (auto EC = dumpSectionHeaders()) 213 return EC; 214 } 215 216 if (opts::dump::DumpSectionContribs) { 217 if (auto EC = dumpSectionContribs()) 218 return EC; 219 } 220 221 if (opts::dump::DumpSectionMap) { 222 if (auto EC = dumpSectionMap()) 223 return EC; 224 } 225 226 P.NewLine(); 227 228 return Error::success(); 229 } 230 231 static void printHeader(LinePrinter &P, const Twine &S) { 232 P.NewLine(); 233 P.formatLine("{0,=60}", S); 234 P.formatLine("{0}", fmt_repeat('=', 60)); 235 } 236 237 Error DumpOutputStyle::dumpFileSummary() { 238 printHeader(P, "Summary"); 239 240 if (File.isObj()) { 241 printStreamNotValidForObj(); 242 return Error::success(); 243 } 244 245 AutoIndent Indent(P); 246 ExitOnError Err("Invalid PDB Format: "); 247 248 P.formatLine("Block Size: {0}", getPdb().getBlockSize()); 249 P.formatLine("Number of blocks: {0}", getPdb().getBlockCount()); 250 P.formatLine("Number of streams: {0}", getPdb().getNumStreams()); 251 252 auto &PS = Err(getPdb().getPDBInfoStream()); 253 P.formatLine("Signature: {0}", PS.getSignature()); 254 P.formatLine("Age: {0}", PS.getAge()); 255 P.formatLine("GUID: {0}", fmt_guid(PS.getGuid().Guid)); 256 P.formatLine("Features: {0:x+}", static_cast<uint32_t>(PS.getFeatures())); 257 P.formatLine("Has Debug Info: {0}", getPdb().hasPDBDbiStream()); 258 P.formatLine("Has Types: {0}", getPdb().hasPDBTpiStream()); 259 P.formatLine("Has IDs: {0}", getPdb().hasPDBIpiStream()); 260 P.formatLine("Has Globals: {0}", getPdb().hasPDBGlobalsStream()); 261 P.formatLine("Has Publics: {0}", getPdb().hasPDBPublicsStream()); 262 if (getPdb().hasPDBDbiStream()) { 263 auto &DBI = Err(getPdb().getPDBDbiStream()); 264 P.formatLine("Is incrementally linked: {0}", DBI.isIncrementallyLinked()); 265 P.formatLine("Has conflicting types: {0}", DBI.hasCTypes()); 266 P.formatLine("Is stripped: {0}", DBI.isStripped()); 267 } 268 269 return Error::success(); 270 } 271 272 static StatCollection getSymbolStats(const SymbolGroup &SG, 273 StatCollection &CumulativeStats) { 274 StatCollection Stats; 275 if (SG.getFile().isPdb() && SG.hasDebugStream()) { 276 // For PDB files, all symbols are packed into one stream. 277 for (const auto &S : SG.getPdbModuleStream().symbols(nullptr)) { 278 Stats.update(S.kind(), S.length()); 279 CumulativeStats.update(S.kind(), S.length()); 280 } 281 return Stats; 282 } 283 284 for (const auto &SS : SG.getDebugSubsections()) { 285 // For object files, all symbols are spread across multiple Symbol 286 // subsections of a given .debug$S section. 287 if (SS.kind() != DebugSubsectionKind::Symbols) 288 continue; 289 DebugSymbolsSubsectionRef Symbols; 290 BinaryStreamReader Reader(SS.getRecordData()); 291 cantFail(Symbols.initialize(Reader)); 292 for (const auto &S : Symbols) { 293 Stats.update(S.kind(), S.length()); 294 CumulativeStats.update(S.kind(), S.length()); 295 } 296 } 297 return Stats; 298 } 299 300 static StatCollection getChunkStats(const SymbolGroup &SG, 301 StatCollection &CumulativeStats) { 302 StatCollection Stats; 303 for (const auto &Chunk : SG.getDebugSubsections()) { 304 Stats.update(uint32_t(Chunk.kind()), Chunk.getRecordLength()); 305 CumulativeStats.update(uint32_t(Chunk.kind()), Chunk.getRecordLength()); 306 } 307 return Stats; 308 } 309 310 static inline std::string formatModuleDetailKind(DebugSubsectionKind K) { 311 return formatChunkKind(K, false); 312 } 313 314 static inline std::string formatModuleDetailKind(SymbolKind K) { 315 return formatSymbolKind(K); 316 } 317 318 // Get the stats sorted by size, descending. 319 std::vector<StatCollection::KindAndStat> 320 StatCollection::getStatsSortedBySize() const { 321 std::vector<KindAndStat> SortedStats(Individual.begin(), Individual.end()); 322 llvm::stable_sort(SortedStats, 323 [](const KindAndStat &LHS, const KindAndStat &RHS) { 324 return LHS.second.Size > RHS.second.Size; 325 }); 326 return SortedStats; 327 } 328 329 template <typename Kind> 330 static void printModuleDetailStats(LinePrinter &P, StringRef Label, 331 const StatCollection &Stats) { 332 P.NewLine(); 333 P.formatLine(" {0}", Label); 334 AutoIndent Indent(P); 335 P.formatLine("{0,40}: {1,7} entries ({2,12:N} bytes)", "Total", 336 Stats.Totals.Count, Stats.Totals.Size); 337 P.formatLine("{0}", fmt_repeat('-', 74)); 338 339 for (const auto &K : Stats.getStatsSortedBySize()) { 340 std::string KindName = formatModuleDetailKind(Kind(K.first)); 341 P.formatLine("{0,40}: {1,7} entries ({2,12:N} bytes)", KindName, 342 K.second.Count, K.second.Size); 343 } 344 } 345 346 static bool isMyCode(const SymbolGroup &Group) { 347 if (Group.getFile().isObj()) 348 return true; 349 350 StringRef Name = Group.name(); 351 if (Name.startswith("Import:")) 352 return false; 353 if (Name.endswith_lower(".dll")) 354 return false; 355 if (Name.equals_lower("* linker *")) 356 return false; 357 if (Name.startswith_lower("f:\\binaries\\Intermediate\\vctools")) 358 return false; 359 if (Name.startswith_lower("f:\\dd\\vctools\\crt")) 360 return false; 361 return true; 362 } 363 364 static bool shouldDumpSymbolGroup(uint32_t Idx, const SymbolGroup &Group) { 365 if (opts::dump::JustMyCode && !isMyCode(Group)) 366 return false; 367 368 // If the arg was not specified on the command line, always dump all modules. 369 if (opts::dump::DumpModi.getNumOccurrences() == 0) 370 return true; 371 372 // Otherwise, only dump if this is the same module specified. 373 return (opts::dump::DumpModi == Idx); 374 } 375 376 Error DumpOutputStyle::dumpStreamSummary() { 377 printHeader(P, "Streams"); 378 379 if (File.isObj()) { 380 printStreamNotValidForObj(); 381 return Error::success(); 382 } 383 384 AutoIndent Indent(P); 385 386 if (StreamPurposes.empty()) 387 discoverStreamPurposes(getPdb(), StreamPurposes); 388 389 uint32_t StreamCount = getPdb().getNumStreams(); 390 uint32_t MaxStreamSize = getPdb().getMaxStreamSize(); 391 392 for (uint16_t StreamIdx = 0; StreamIdx < StreamCount; ++StreamIdx) { 393 P.formatLine( 394 "Stream {0} ({1} bytes): [{2}]", 395 fmt_align(StreamIdx, AlignStyle::Right, NumDigits(StreamCount)), 396 fmt_align(getPdb().getStreamByteSize(StreamIdx), AlignStyle::Right, 397 NumDigits(MaxStreamSize)), 398 StreamPurposes[StreamIdx].getLongName()); 399 400 if (opts::dump::DumpStreamBlocks) { 401 auto Blocks = getPdb().getStreamBlockList(StreamIdx); 402 std::vector<uint32_t> BV(Blocks.begin(), Blocks.end()); 403 P.formatLine(" {0} Blocks: [{1}]", 404 fmt_repeat(' ', NumDigits(StreamCount)), 405 make_range(BV.begin(), BV.end())); 406 } 407 } 408 409 return Error::success(); 410 } 411 412 static Expected<ModuleDebugStreamRef> getModuleDebugStream(PDBFile &File, 413 uint32_t Index) { 414 ExitOnError Err("Unexpected error: "); 415 416 auto &Dbi = Err(File.getPDBDbiStream()); 417 const auto &Modules = Dbi.modules(); 418 auto Modi = Modules.getModuleDescriptor(Index); 419 420 uint16_t ModiStream = Modi.getModuleStreamIndex(); 421 if (ModiStream == kInvalidStreamIndex) 422 return make_error<RawError>(raw_error_code::no_stream, 423 "Module stream not present"); 424 425 auto ModStreamData = File.createIndexedStream(ModiStream); 426 427 ModuleDebugStreamRef ModS(Modi, std::move(ModStreamData)); 428 if (auto EC = ModS.reload()) 429 return make_error<RawError>(raw_error_code::corrupt_file, 430 "Invalid module stream"); 431 432 return std::move(ModS); 433 } 434 435 template <typename CallbackT> 436 static void 437 iterateOneModule(InputFile &File, const Optional<PrintScope> &HeaderScope, 438 const SymbolGroup &SG, uint32_t Modi, CallbackT Callback) { 439 if (HeaderScope) { 440 HeaderScope->P.formatLine( 441 "Mod {0:4} | `{1}`: ", 442 fmt_align(Modi, AlignStyle::Right, HeaderScope->LabelWidth), SG.name()); 443 } 444 445 AutoIndent Indent(HeaderScope); 446 Callback(Modi, SG); 447 } 448 449 template <typename CallbackT> 450 static void iterateSymbolGroups(InputFile &Input, 451 const Optional<PrintScope> &HeaderScope, 452 CallbackT Callback) { 453 AutoIndent Indent(HeaderScope); 454 455 ExitOnError Err("Unexpected error processing modules: "); 456 457 if (opts::dump::DumpModi.getNumOccurrences() > 0) { 458 assert(opts::dump::DumpModi.getNumOccurrences() == 1); 459 uint32_t Modi = opts::dump::DumpModi; 460 SymbolGroup SG(&Input, Modi); 461 iterateOneModule(Input, withLabelWidth(HeaderScope, NumDigits(Modi)), SG, 462 Modi, Callback); 463 return; 464 } 465 466 uint32_t I = 0; 467 468 for (const auto &SG : Input.symbol_groups()) { 469 if (shouldDumpSymbolGroup(I, SG)) 470 iterateOneModule(Input, withLabelWidth(HeaderScope, NumDigits(I)), SG, I, 471 Callback); 472 473 ++I; 474 } 475 } 476 477 template <typename SubsectionT> 478 static void iterateModuleSubsections( 479 InputFile &File, const Optional<PrintScope> &HeaderScope, 480 llvm::function_ref<void(uint32_t, const SymbolGroup &, SubsectionT &)> 481 Callback) { 482 483 iterateSymbolGroups(File, HeaderScope, 484 [&](uint32_t Modi, const SymbolGroup &SG) { 485 for (const auto &SS : SG.getDebugSubsections()) { 486 SubsectionT Subsection; 487 488 if (SS.kind() != Subsection.kind()) 489 continue; 490 491 BinaryStreamReader Reader(SS.getRecordData()); 492 if (auto EC = Subsection.initialize(Reader)) 493 continue; 494 Callback(Modi, SG, Subsection); 495 } 496 }); 497 } 498 499 static Expected<std::pair<std::unique_ptr<MappedBlockStream>, 500 ArrayRef<llvm::object::coff_section>>> 501 loadSectionHeaders(PDBFile &File, DbgHeaderType Type) { 502 if (!File.hasPDBDbiStream()) 503 return make_error<StringError>( 504 "Section headers require a DBI Stream, which could not be loaded", 505 inconvertibleErrorCode()); 506 507 auto &Dbi = cantFail(File.getPDBDbiStream()); 508 uint32_t SI = Dbi.getDebugStreamIndex(Type); 509 510 if (SI == kInvalidStreamIndex) 511 return make_error<StringError>( 512 "PDB does not contain the requested image section header type", 513 inconvertibleErrorCode()); 514 515 auto Stream = File.createIndexedStream(SI); 516 if (!Stream) 517 return make_error<StringError>("Could not load the required stream data", 518 inconvertibleErrorCode()); 519 520 ArrayRef<object::coff_section> Headers; 521 if (Stream->getLength() % sizeof(object::coff_section) != 0) 522 return make_error<StringError>( 523 "Section header array size is not a multiple of section header size", 524 inconvertibleErrorCode()); 525 526 uint32_t NumHeaders = Stream->getLength() / sizeof(object::coff_section); 527 BinaryStreamReader Reader(*Stream); 528 cantFail(Reader.readArray(Headers, NumHeaders)); 529 return std::make_pair(std::move(Stream), Headers); 530 } 531 532 static std::vector<std::string> getSectionNames(PDBFile &File) { 533 auto ExpectedHeaders = loadSectionHeaders(File, DbgHeaderType::SectionHdr); 534 if (!ExpectedHeaders) 535 return {}; 536 537 std::unique_ptr<MappedBlockStream> Stream; 538 ArrayRef<object::coff_section> Headers; 539 std::tie(Stream, Headers) = std::move(*ExpectedHeaders); 540 std::vector<std::string> Names; 541 for (const auto &H : Headers) 542 Names.push_back(H.Name); 543 return Names; 544 } 545 546 static void dumpSectionContrib(LinePrinter &P, const SectionContrib &SC, 547 ArrayRef<std::string> SectionNames, 548 uint32_t FieldWidth) { 549 std::string NameInsert; 550 if (SC.ISect > 0 && SC.ISect <= SectionNames.size()) { 551 StringRef SectionName = SectionNames[SC.ISect - 1]; 552 NameInsert = formatv("[{0}]", SectionName).str(); 553 } else 554 NameInsert = "[???]"; 555 P.formatLine("SC{5} | mod = {2}, {0}, size = {1}, data crc = {3}, reloc " 556 "crc = {4}", 557 formatSegmentOffset(SC.ISect, SC.Off), fmtle(SC.Size), 558 fmtle(SC.Imod), fmtle(SC.DataCrc), fmtle(SC.RelocCrc), 559 fmt_align(NameInsert, AlignStyle::Left, FieldWidth + 2)); 560 AutoIndent Indent(P, FieldWidth + 2); 561 P.formatLine(" {0}", 562 formatSectionCharacteristics(P.getIndentLevel() + 6, 563 SC.Characteristics, 3, " | ")); 564 } 565 566 static void dumpSectionContrib(LinePrinter &P, const SectionContrib2 &SC, 567 ArrayRef<std::string> SectionNames, 568 uint32_t FieldWidth) { 569 P.formatLine("SC2[{6}] | mod = {2}, {0}, size = {1}, data crc = {3}, reloc " 570 "crc = {4}, coff section = {5}", 571 formatSegmentOffset(SC.Base.ISect, SC.Base.Off), 572 fmtle(SC.Base.Size), fmtle(SC.Base.Imod), fmtle(SC.Base.DataCrc), 573 fmtle(SC.Base.RelocCrc), fmtle(SC.ISectCoff)); 574 P.formatLine(" {0}", 575 formatSectionCharacteristics(P.getIndentLevel() + 6, 576 SC.Base.Characteristics, 3, " | ")); 577 } 578 579 Error DumpOutputStyle::dumpModules() { 580 printHeader(P, "Modules"); 581 582 if (File.isObj()) { 583 printStreamNotValidForObj(); 584 return Error::success(); 585 } 586 587 if (!getPdb().hasPDBDbiStream()) { 588 printStreamNotPresent("DBI"); 589 return Error::success(); 590 } 591 592 AutoIndent Indent(P); 593 ExitOnError Err("Unexpected error processing modules: "); 594 595 auto &Stream = Err(getPdb().getPDBDbiStream()); 596 597 const DbiModuleList &Modules = Stream.modules(); 598 iterateSymbolGroups( 599 File, PrintScope{P, 11}, [&](uint32_t Modi, const SymbolGroup &Strings) { 600 auto Desc = Modules.getModuleDescriptor(Modi); 601 if (opts::dump::DumpSectionContribs) { 602 std::vector<std::string> Sections = getSectionNames(getPdb()); 603 dumpSectionContrib(P, Desc.getSectionContrib(), Sections, 0); 604 } 605 P.formatLine("Obj: `{0}`: ", Desc.getObjFileName()); 606 P.formatLine("debug stream: {0}, # files: {1}, has ec info: {2}", 607 Desc.getModuleStreamIndex(), Desc.getNumberOfFiles(), 608 Desc.hasECInfo()); 609 StringRef PdbFilePath = 610 Err(Stream.getECName(Desc.getPdbFilePathNameIndex())); 611 StringRef SrcFilePath = 612 Err(Stream.getECName(Desc.getSourceFileNameIndex())); 613 P.formatLine("pdb file ni: {0} `{1}`, src file ni: {2} `{3}`", 614 Desc.getPdbFilePathNameIndex(), PdbFilePath, 615 Desc.getSourceFileNameIndex(), SrcFilePath); 616 }); 617 return Error::success(); 618 } 619 620 Error DumpOutputStyle::dumpModuleFiles() { 621 printHeader(P, "Files"); 622 623 if (File.isObj()) { 624 printStreamNotValidForObj(); 625 return Error::success(); 626 } 627 628 if (!getPdb().hasPDBDbiStream()) { 629 printStreamNotPresent("DBI"); 630 return Error::success(); 631 } 632 633 ExitOnError Err("Unexpected error processing modules: "); 634 635 iterateSymbolGroups(File, PrintScope{P, 11}, 636 [this, &Err](uint32_t Modi, const SymbolGroup &Strings) { 637 auto &Stream = Err(getPdb().getPDBDbiStream()); 638 639 const DbiModuleList &Modules = Stream.modules(); 640 for (const auto &F : Modules.source_files(Modi)) { 641 Strings.formatFromFileName(P, F); 642 } 643 }); 644 return Error::success(); 645 } 646 647 Error DumpOutputStyle::dumpSymbolStats() { 648 printHeader(P, "Module Stats"); 649 650 if (File.isPdb() && !getPdb().hasPDBDbiStream()) { 651 printStreamNotPresent("DBI"); 652 return Error::success(); 653 } 654 655 ExitOnError Err("Unexpected error processing modules: "); 656 657 StatCollection SymStats; 658 StatCollection ChunkStats; 659 660 Optional<PrintScope> Scope; 661 if (File.isPdb()) 662 Scope.emplace(P, 2); 663 664 iterateSymbolGroups(File, Scope, [&](uint32_t Modi, const SymbolGroup &SG) { 665 StatCollection SS = getSymbolStats(SG, SymStats); 666 StatCollection CS = getChunkStats(SG, ChunkStats); 667 668 if (SG.getFile().isPdb()) { 669 AutoIndent Indent(P); 670 auto Modules = cantFail(File.pdb().getPDBDbiStream()).modules(); 671 uint32_t ModCount = Modules.getModuleCount(); 672 DbiModuleDescriptor Desc = Modules.getModuleDescriptor(Modi); 673 uint32_t StreamIdx = Desc.getModuleStreamIndex(); 674 675 if (StreamIdx == kInvalidStreamIndex) { 676 P.formatLine("Mod {0} (debug info not present): [{1}]", 677 fmt_align(Modi, AlignStyle::Right, NumDigits(ModCount)), 678 Desc.getModuleName()); 679 return; 680 } 681 P.formatLine("Stream {0}, {1} bytes", StreamIdx, 682 getPdb().getStreamByteSize(StreamIdx)); 683 684 printModuleDetailStats<SymbolKind>(P, "Symbols", SS); 685 printModuleDetailStats<DebugSubsectionKind>(P, "Chunks", CS); 686 } 687 }); 688 689 if (SymStats.Totals.Count > 0) { 690 P.printLine(" Summary |"); 691 AutoIndent Indent(P, 4); 692 printModuleDetailStats<SymbolKind>(P, "Symbols", SymStats); 693 printModuleDetailStats<DebugSubsectionKind>(P, "Chunks", ChunkStats); 694 } 695 696 return Error::success(); 697 } 698 699 Error DumpOutputStyle::dumpTypeStats() { 700 printHeader(P, "Type Record Stats"); 701 702 // Iterate the types, categorize by kind, accumulate size stats. 703 StatCollection TypeStats; 704 LazyRandomTypeCollection &Types = File.types(); 705 for (Optional<TypeIndex> TI = Types.getFirst(); TI; TI = Types.getNext(*TI)) { 706 CVType Type = Types.getType(*TI); 707 TypeStats.update(uint32_t(Type.kind()), Type.length()); 708 } 709 710 P.NewLine(); 711 P.formatLine(" Types"); 712 AutoIndent Indent(P); 713 P.formatLine("{0,14}: {1,7} entries ({2,12:N} bytes, {3,7} avg)", "Total", 714 TypeStats.Totals.Count, TypeStats.Totals.Size, 715 (double)TypeStats.Totals.Size / TypeStats.Totals.Count); 716 P.formatLine("{0}", fmt_repeat('-', 74)); 717 718 for (const auto &K : TypeStats.getStatsSortedBySize()) { 719 P.formatLine("{0,14}: {1,7} entries ({2,12:N} bytes, {3,7} avg)", 720 formatTypeLeafKind(TypeLeafKind(K.first)), K.second.Count, 721 K.second.Size, (double)K.second.Size / K.second.Count); 722 } 723 724 725 return Error::success(); 726 } 727 728 static bool isValidNamespaceIdentifier(StringRef S) { 729 if (S.empty()) 730 return false; 731 732 if (std::isdigit(S[0])) 733 return false; 734 735 return llvm::all_of(S, [](char C) { return std::isalnum(C); }); 736 } 737 738 namespace { 739 constexpr uint32_t kNoneUdtKind = 0; 740 constexpr uint32_t kSimpleUdtKind = 1; 741 constexpr uint32_t kUnknownUdtKind = 2; 742 const StringRef NoneLabel("<none type>"); 743 const StringRef SimpleLabel("<simple type>"); 744 const StringRef UnknownLabel("<unknown type>"); 745 746 } // namespace 747 748 static StringRef getUdtStatLabel(uint32_t Kind) { 749 if (Kind == kNoneUdtKind) 750 return NoneLabel; 751 752 if (Kind == kSimpleUdtKind) 753 return SimpleLabel; 754 755 if (Kind == kUnknownUdtKind) 756 return UnknownLabel; 757 758 return formatTypeLeafKind(static_cast<TypeLeafKind>(Kind)); 759 } 760 761 static uint32_t getLongestTypeLeafName(const StatCollection &Stats) { 762 size_t L = 0; 763 for (const auto &Stat : Stats.Individual) { 764 StringRef Label = getUdtStatLabel(Stat.first); 765 L = std::max(L, Label.size()); 766 } 767 return static_cast<uint32_t>(L); 768 } 769 770 Error DumpOutputStyle::dumpUdtStats() { 771 printHeader(P, "S_UDT Record Stats"); 772 773 if (File.isPdb() && !getPdb().hasPDBGlobalsStream()) { 774 printStreamNotPresent("Globals"); 775 return Error::success(); 776 } 777 778 StatCollection UdtStats; 779 StatCollection UdtTargetStats; 780 AutoIndent Indent(P, 4); 781 782 auto &TpiTypes = File.types(); 783 784 StringMap<StatCollection::Stat> NamespacedStats; 785 786 size_t LongestNamespace = 0; 787 auto HandleOneSymbol = [&](const CVSymbol &Sym) { 788 if (Sym.kind() != SymbolKind::S_UDT) 789 return; 790 UdtStats.update(SymbolKind::S_UDT, Sym.length()); 791 792 UDTSym UDT = cantFail(SymbolDeserializer::deserializeAs<UDTSym>(Sym)); 793 794 uint32_t Kind = 0; 795 uint32_t RecordSize = 0; 796 797 if (UDT.Type.isNoneType()) 798 Kind = kNoneUdtKind; 799 else if (UDT.Type.isSimple()) 800 Kind = kSimpleUdtKind; 801 else if (Optional<CVType> T = TpiTypes.tryGetType(UDT.Type)) { 802 Kind = T->kind(); 803 RecordSize = T->length(); 804 } else 805 Kind = kUnknownUdtKind; 806 807 UdtTargetStats.update(Kind, RecordSize); 808 809 size_t Pos = UDT.Name.find("::"); 810 if (Pos == StringRef::npos) 811 return; 812 813 StringRef Scope = UDT.Name.take_front(Pos); 814 if (Scope.empty() || !isValidNamespaceIdentifier(Scope)) 815 return; 816 817 LongestNamespace = std::max(LongestNamespace, Scope.size()); 818 NamespacedStats[Scope].update(RecordSize); 819 }; 820 821 P.NewLine(); 822 823 if (File.isPdb()) { 824 auto &SymbolRecords = cantFail(getPdb().getPDBSymbolStream()); 825 auto ExpGlobals = getPdb().getPDBGlobalsStream(); 826 if (!ExpGlobals) 827 return ExpGlobals.takeError(); 828 829 for (uint32_t PubSymOff : ExpGlobals->getGlobalsTable()) { 830 CVSymbol Sym = SymbolRecords.readRecord(PubSymOff); 831 HandleOneSymbol(Sym); 832 } 833 } else { 834 for (const auto &Sec : File.symbol_groups()) { 835 for (const auto &SS : Sec.getDebugSubsections()) { 836 if (SS.kind() != DebugSubsectionKind::Symbols) 837 continue; 838 839 DebugSymbolsSubsectionRef Symbols; 840 BinaryStreamReader Reader(SS.getRecordData()); 841 cantFail(Symbols.initialize(Reader)); 842 for (const auto &S : Symbols) 843 HandleOneSymbol(S); 844 } 845 } 846 } 847 848 LongestNamespace += StringRef(" namespace ''").size(); 849 size_t LongestTypeLeafKind = getLongestTypeLeafName(UdtTargetStats); 850 size_t FieldWidth = std::max(LongestNamespace, LongestTypeLeafKind); 851 852 // Compute the max number of digits for count and size fields, including comma 853 // separators. 854 StringRef CountHeader("Count"); 855 StringRef SizeHeader("Size"); 856 size_t CD = NumDigits(UdtStats.Totals.Count); 857 CD += (CD - 1) / 3; 858 CD = std::max(CD, CountHeader.size()); 859 860 size_t SD = NumDigits(UdtStats.Totals.Size); 861 SD += (SD - 1) / 3; 862 SD = std::max(SD, SizeHeader.size()); 863 864 uint32_t TableWidth = FieldWidth + 3 + CD + 2 + SD + 1; 865 866 P.formatLine("{0} | {1} {2}", 867 fmt_align("Record Kind", AlignStyle::Right, FieldWidth), 868 fmt_align(CountHeader, AlignStyle::Right, CD), 869 fmt_align(SizeHeader, AlignStyle::Right, SD)); 870 871 P.formatLine("{0}", fmt_repeat('-', TableWidth)); 872 for (const auto &Stat : UdtTargetStats.getStatsSortedBySize()) { 873 StringRef Label = getUdtStatLabel(Stat.first); 874 P.formatLine("{0} | {1:N} {2:N}", 875 fmt_align(Label, AlignStyle::Right, FieldWidth), 876 fmt_align(Stat.second.Count, AlignStyle::Right, CD), 877 fmt_align(Stat.second.Size, AlignStyle::Right, SD)); 878 } 879 P.formatLine("{0}", fmt_repeat('-', TableWidth)); 880 P.formatLine("{0} | {1:N} {2:N}", 881 fmt_align("Total (S_UDT)", AlignStyle::Right, FieldWidth), 882 fmt_align(UdtStats.Totals.Count, AlignStyle::Right, CD), 883 fmt_align(UdtStats.Totals.Size, AlignStyle::Right, SD)); 884 P.formatLine("{0}", fmt_repeat('-', TableWidth)); 885 struct StrAndStat { 886 StringRef Key; 887 StatCollection::Stat Stat; 888 }; 889 890 // Print namespace stats in descending order of size. 891 std::vector<StrAndStat> NamespacedStatsSorted; 892 for (const auto &Stat : NamespacedStats) 893 NamespacedStatsSorted.push_back({Stat.getKey(), Stat.second}); 894 llvm::stable_sort(NamespacedStatsSorted, 895 [](const StrAndStat &L, const StrAndStat &R) { 896 return L.Stat.Size > R.Stat.Size; 897 }); 898 for (const auto &Stat : NamespacedStatsSorted) { 899 std::string Label = formatv("namespace '{0}'", Stat.Key); 900 P.formatLine("{0} | {1:N} {2:N}", 901 fmt_align(Label, AlignStyle::Right, FieldWidth), 902 fmt_align(Stat.Stat.Count, AlignStyle::Right, CD), 903 fmt_align(Stat.Stat.Size, AlignStyle::Right, SD)); 904 } 905 return Error::success(); 906 } 907 908 static void typesetLinesAndColumns(LinePrinter &P, uint32_t Start, 909 const LineColumnEntry &E) { 910 const uint32_t kMaxCharsPerLineNumber = 4; // 4 digit line number 911 uint32_t MinColumnWidth = kMaxCharsPerLineNumber + 5; 912 913 // Let's try to keep it under 100 characters 914 constexpr uint32_t kMaxRowLength = 100; 915 // At least 3 spaces between columns. 916 uint32_t ColumnsPerRow = kMaxRowLength / (MinColumnWidth + 3); 917 uint32_t ItemsLeft = E.LineNumbers.size(); 918 auto LineIter = E.LineNumbers.begin(); 919 while (ItemsLeft != 0) { 920 uint32_t RowColumns = std::min(ItemsLeft, ColumnsPerRow); 921 for (uint32_t I = 0; I < RowColumns; ++I) { 922 LineInfo Line(LineIter->Flags); 923 std::string LineStr; 924 if (Line.isAlwaysStepInto()) 925 LineStr = "ASI"; 926 else if (Line.isNeverStepInto()) 927 LineStr = "NSI"; 928 else 929 LineStr = utostr(Line.getStartLine()); 930 char Statement = Line.isStatement() ? ' ' : '!'; 931 P.format("{0} {1:X-} {2} ", 932 fmt_align(LineStr, AlignStyle::Right, kMaxCharsPerLineNumber), 933 fmt_align(Start + LineIter->Offset, AlignStyle::Right, 8, '0'), 934 Statement); 935 ++LineIter; 936 --ItemsLeft; 937 } 938 P.NewLine(); 939 } 940 } 941 942 Error DumpOutputStyle::dumpLines() { 943 printHeader(P, "Lines"); 944 945 if (File.isPdb() && !getPdb().hasPDBDbiStream()) { 946 printStreamNotPresent("DBI"); 947 return Error::success(); 948 } 949 950 uint32_t LastModi = UINT32_MAX; 951 uint32_t LastNameIndex = UINT32_MAX; 952 iterateModuleSubsections<DebugLinesSubsectionRef>( 953 File, PrintScope{P, 4}, 954 [this, &LastModi, &LastNameIndex](uint32_t Modi, 955 const SymbolGroup &Strings, 956 DebugLinesSubsectionRef &Lines) { 957 uint16_t Segment = Lines.header()->RelocSegment; 958 uint32_t Begin = Lines.header()->RelocOffset; 959 uint32_t End = Begin + Lines.header()->CodeSize; 960 for (const auto &Block : Lines) { 961 if (LastModi != Modi || LastNameIndex != Block.NameIndex) { 962 LastModi = Modi; 963 LastNameIndex = Block.NameIndex; 964 Strings.formatFromChecksumsOffset(P, Block.NameIndex); 965 } 966 967 AutoIndent Indent(P, 2); 968 P.formatLine("{0:X-4}:{1:X-8}-{2:X-8}, ", Segment, Begin, End); 969 uint32_t Count = Block.LineNumbers.size(); 970 if (Lines.hasColumnInfo()) 971 P.format("line/column/addr entries = {0}", Count); 972 else 973 P.format("line/addr entries = {0}", Count); 974 975 P.NewLine(); 976 typesetLinesAndColumns(P, Begin, Block); 977 } 978 }); 979 980 return Error::success(); 981 } 982 983 Error DumpOutputStyle::dumpInlineeLines() { 984 printHeader(P, "Inlinee Lines"); 985 986 if (File.isPdb() && !getPdb().hasPDBDbiStream()) { 987 printStreamNotPresent("DBI"); 988 return Error::success(); 989 } 990 991 iterateModuleSubsections<DebugInlineeLinesSubsectionRef>( 992 File, PrintScope{P, 2}, 993 [this](uint32_t Modi, const SymbolGroup &Strings, 994 DebugInlineeLinesSubsectionRef &Lines) { 995 P.formatLine("{0,+8} | {1,+5} | {2}", "Inlinee", "Line", "Source File"); 996 for (const auto &Entry : Lines) { 997 P.formatLine("{0,+8} | {1,+5} | ", Entry.Header->Inlinee, 998 fmtle(Entry.Header->SourceLineNum)); 999 Strings.formatFromChecksumsOffset(P, Entry.Header->FileID, true); 1000 for (const auto &ExtraFileID : Entry.ExtraFiles) { 1001 P.formatLine(" "); 1002 Strings.formatFromChecksumsOffset(P, ExtraFileID, true); 1003 } 1004 } 1005 P.NewLine(); 1006 }); 1007 1008 return Error::success(); 1009 } 1010 1011 Error DumpOutputStyle::dumpXmi() { 1012 printHeader(P, "Cross Module Imports"); 1013 1014 if (File.isPdb() && !getPdb().hasPDBDbiStream()) { 1015 printStreamNotPresent("DBI"); 1016 return Error::success(); 1017 } 1018 1019 iterateModuleSubsections<DebugCrossModuleImportsSubsectionRef>( 1020 File, PrintScope{P, 2}, 1021 [this](uint32_t Modi, const SymbolGroup &Strings, 1022 DebugCrossModuleImportsSubsectionRef &Imports) { 1023 P.formatLine("{0,=32} | {1}", "Imported Module", "Type IDs"); 1024 1025 for (const auto &Xmi : Imports) { 1026 auto ExpectedModule = 1027 Strings.getNameFromStringTable(Xmi.Header->ModuleNameOffset); 1028 StringRef Module; 1029 SmallString<32> ModuleStorage; 1030 if (!ExpectedModule) { 1031 Module = "(unknown module)"; 1032 consumeError(ExpectedModule.takeError()); 1033 } else 1034 Module = *ExpectedModule; 1035 if (Module.size() > 32) { 1036 ModuleStorage = "..."; 1037 ModuleStorage += Module.take_back(32 - 3); 1038 Module = ModuleStorage; 1039 } 1040 std::vector<std::string> TIs; 1041 for (const auto I : Xmi.Imports) 1042 TIs.push_back(formatv("{0,+10:X+}", fmtle(I))); 1043 std::string Result = 1044 typesetItemList(TIs, P.getIndentLevel() + 35, 12, " "); 1045 P.formatLine("{0,+32} | {1}", Module, Result); 1046 } 1047 }); 1048 1049 return Error::success(); 1050 } 1051 1052 Error DumpOutputStyle::dumpXme() { 1053 printHeader(P, "Cross Module Exports"); 1054 1055 if (File.isPdb() && !getPdb().hasPDBDbiStream()) { 1056 printStreamNotPresent("DBI"); 1057 return Error::success(); 1058 } 1059 1060 iterateModuleSubsections<DebugCrossModuleExportsSubsectionRef>( 1061 File, PrintScope{P, 2}, 1062 [this](uint32_t Modi, const SymbolGroup &Strings, 1063 DebugCrossModuleExportsSubsectionRef &Exports) { 1064 P.formatLine("{0,-10} | {1}", "Local ID", "Global ID"); 1065 for (const auto &Export : Exports) { 1066 P.formatLine("{0,+10:X+} | {1}", TypeIndex(Export.Local), 1067 TypeIndex(Export.Global)); 1068 } 1069 }); 1070 1071 return Error::success(); 1072 } 1073 1074 std::string formatFrameType(object::frame_type FT) { 1075 switch (FT) { 1076 case object::frame_type::Fpo: 1077 return "FPO"; 1078 case object::frame_type::NonFpo: 1079 return "Non-FPO"; 1080 case object::frame_type::Trap: 1081 return "Trap"; 1082 case object::frame_type::Tss: 1083 return "TSS"; 1084 } 1085 return "<unknown>"; 1086 } 1087 1088 Error DumpOutputStyle::dumpOldFpo(PDBFile &File) { 1089 printHeader(P, "Old FPO Data"); 1090 1091 ExitOnError Err("Error dumping old fpo data:"); 1092 auto &Dbi = Err(File.getPDBDbiStream()); 1093 1094 if (!Dbi.hasOldFpoRecords()) { 1095 printStreamNotPresent("FPO"); 1096 return Error::success(); 1097 } 1098 1099 const FixedStreamArray<object::FpoData>& Records = Dbi.getOldFpoRecords(); 1100 1101 P.printLine(" RVA | Code | Locals | Params | Prolog | Saved Regs | Use " 1102 "BP | Has SEH | Frame Type"); 1103 1104 for (const object::FpoData &FD : Records) { 1105 P.formatLine("{0:X-8} | {1,4} | {2,6} | {3,6} | {4,6} | {5,10} | {6,6} | " 1106 "{7,7} | {8,9}", 1107 uint32_t(FD.Offset), uint32_t(FD.Size), uint32_t(FD.NumLocals), 1108 uint32_t(FD.NumParams), FD.getPrologSize(), 1109 FD.getNumSavedRegs(), FD.useBP(), FD.hasSEH(), 1110 formatFrameType(FD.getFP())); 1111 } 1112 return Error::success(); 1113 } 1114 1115 Error DumpOutputStyle::dumpNewFpo(PDBFile &File) { 1116 printHeader(P, "New FPO Data"); 1117 1118 ExitOnError Err("Error dumping new fpo data:"); 1119 auto &Dbi = Err(File.getPDBDbiStream()); 1120 1121 if (!Dbi.hasNewFpoRecords()) { 1122 printStreamNotPresent("New FPO"); 1123 return Error::success(); 1124 } 1125 1126 const DebugFrameDataSubsectionRef& FDS = Dbi.getNewFpoRecords(); 1127 1128 P.printLine(" RVA | Code | Locals | Params | Stack | Prolog | Saved Regs " 1129 "| Has SEH | Has C++EH | Start | Program"); 1130 for (const FrameData &FD : FDS) { 1131 bool IsFuncStart = FD.Flags & FrameData::IsFunctionStart; 1132 bool HasEH = FD.Flags & FrameData::HasEH; 1133 bool HasSEH = FD.Flags & FrameData::HasSEH; 1134 1135 auto &StringTable = Err(File.getStringTable()); 1136 1137 auto Program = Err(StringTable.getStringForID(FD.FrameFunc)); 1138 P.formatLine("{0:X-8} | {1,4} | {2,6} | {3,6} | {4,5} | {5,6} | {6,10} | " 1139 "{7,7} | {8,9} | {9,5} | {10}", 1140 uint32_t(FD.RvaStart), uint32_t(FD.CodeSize), 1141 uint32_t(FD.LocalSize), uint32_t(FD.ParamsSize), 1142 uint32_t(FD.MaxStackSize), uint16_t(FD.PrologSize), 1143 uint16_t(FD.SavedRegsSize), HasSEH, HasEH, IsFuncStart, 1144 Program); 1145 } 1146 return Error::success(); 1147 } 1148 1149 Error DumpOutputStyle::dumpFpo() { 1150 if (!File.isPdb()) { 1151 printStreamNotValidForObj(); 1152 return Error::success(); 1153 } 1154 1155 PDBFile &File = getPdb(); 1156 if (!File.hasPDBDbiStream()) { 1157 printStreamNotPresent("DBI"); 1158 return Error::success(); 1159 } 1160 1161 if (auto EC = dumpOldFpo(File)) 1162 return EC; 1163 if (auto EC = dumpNewFpo(File)) 1164 return EC; 1165 return Error::success(); 1166 } 1167 1168 Error DumpOutputStyle::dumpStringTableFromPdb() { 1169 AutoIndent Indent(P); 1170 auto IS = getPdb().getStringTable(); 1171 if (!IS) { 1172 P.formatLine("Not present in file"); 1173 consumeError(IS.takeError()); 1174 return Error::success(); 1175 } 1176 1177 if (opts::dump::DumpStringTable) { 1178 if (IS->name_ids().empty()) 1179 P.formatLine("Empty"); 1180 else { 1181 auto MaxID = 1182 std::max_element(IS->name_ids().begin(), IS->name_ids().end()); 1183 uint32_t Digits = NumDigits(*MaxID); 1184 1185 P.formatLine("{0} | {1}", fmt_align("ID", AlignStyle::Right, Digits), 1186 "String"); 1187 1188 std::vector<uint32_t> SortedIDs(IS->name_ids().begin(), 1189 IS->name_ids().end()); 1190 llvm::sort(SortedIDs); 1191 for (uint32_t I : SortedIDs) { 1192 auto ES = IS->getStringForID(I); 1193 llvm::SmallString<32> Str; 1194 if (!ES) { 1195 consumeError(ES.takeError()); 1196 Str = "Error reading string"; 1197 } else if (!ES->empty()) { 1198 Str.append("'"); 1199 Str.append(*ES); 1200 Str.append("'"); 1201 } 1202 1203 if (!Str.empty()) 1204 P.formatLine("{0} | {1}", fmt_align(I, AlignStyle::Right, Digits), 1205 Str); 1206 } 1207 } 1208 } 1209 1210 if (opts::dump::DumpStringTableDetails) { 1211 P.NewLine(); 1212 { 1213 P.printLine("String Table Header:"); 1214 AutoIndent Indent(P); 1215 P.formatLine("Signature: {0}", IS->getSignature()); 1216 P.formatLine("Hash Version: {0}", IS->getHashVersion()); 1217 P.formatLine("Name Buffer Size: {0}", IS->getByteSize()); 1218 P.NewLine(); 1219 } 1220 1221 BinaryStreamRef NameBuffer = IS->getStringTable().getBuffer(); 1222 ArrayRef<uint8_t> Contents; 1223 cantFail(NameBuffer.readBytes(0, NameBuffer.getLength(), Contents)); 1224 P.formatBinary("Name Buffer", Contents, 0); 1225 P.NewLine(); 1226 { 1227 P.printLine("Hash Table:"); 1228 AutoIndent Indent(P); 1229 P.formatLine("Bucket Count: {0}", IS->name_ids().size()); 1230 for (const auto &Entry : enumerate(IS->name_ids())) 1231 P.formatLine("Bucket[{0}] : {1}", Entry.index(), 1232 uint32_t(Entry.value())); 1233 P.formatLine("Name Count: {0}", IS->getNameCount()); 1234 } 1235 } 1236 return Error::success(); 1237 } 1238 1239 Error DumpOutputStyle::dumpStringTableFromObj() { 1240 iterateModuleSubsections<DebugStringTableSubsectionRef>( 1241 File, PrintScope{P, 4}, 1242 [&](uint32_t Modi, const SymbolGroup &Strings, 1243 DebugStringTableSubsectionRef &Strings2) { 1244 BinaryStreamRef StringTableBuffer = Strings2.getBuffer(); 1245 BinaryStreamReader Reader(StringTableBuffer); 1246 while (Reader.bytesRemaining() > 0) { 1247 StringRef Str; 1248 uint32_t Offset = Reader.getOffset(); 1249 cantFail(Reader.readCString(Str)); 1250 if (Str.empty()) 1251 continue; 1252 1253 P.formatLine("{0} | {1}", fmt_align(Offset, AlignStyle::Right, 4), 1254 Str); 1255 } 1256 }); 1257 return Error::success(); 1258 } 1259 1260 Error DumpOutputStyle::dumpNamedStreams() { 1261 printHeader(P, "Named Streams"); 1262 1263 if (File.isObj()) { 1264 printStreamNotValidForObj(); 1265 return Error::success(); 1266 } 1267 1268 AutoIndent Indent(P); 1269 ExitOnError Err("Invalid PDB File: "); 1270 1271 auto &IS = Err(File.pdb().getPDBInfoStream()); 1272 const NamedStreamMap &NS = IS.getNamedStreams(); 1273 for (const auto &Entry : NS.entries()) { 1274 P.printLine(Entry.getKey()); 1275 AutoIndent Indent2(P, 2); 1276 P.formatLine("Index: {0}", Entry.getValue()); 1277 P.formatLine("Size in bytes: {0}", 1278 File.pdb().getStreamByteSize(Entry.getValue())); 1279 } 1280 1281 return Error::success(); 1282 } 1283 1284 Error DumpOutputStyle::dumpStringTable() { 1285 printHeader(P, "String Table"); 1286 1287 if (File.isPdb()) 1288 return dumpStringTableFromPdb(); 1289 1290 return dumpStringTableFromObj(); 1291 } 1292 1293 static void buildDepSet(LazyRandomTypeCollection &Types, 1294 ArrayRef<TypeIndex> Indices, 1295 std::map<TypeIndex, CVType> &DepSet) { 1296 SmallVector<TypeIndex, 4> DepList; 1297 for (const auto &I : Indices) { 1298 TypeIndex TI(I); 1299 if (DepSet.find(TI) != DepSet.end() || TI.isSimple() || TI.isNoneType()) 1300 continue; 1301 1302 CVType Type = Types.getType(TI); 1303 DepSet[TI] = Type; 1304 codeview::discoverTypeIndices(Type, DepList); 1305 buildDepSet(Types, DepList, DepSet); 1306 } 1307 } 1308 1309 static void 1310 dumpFullTypeStream(LinePrinter &Printer, LazyRandomTypeCollection &Types, 1311 TypeReferenceTracker *RefTracker, uint32_t NumTypeRecords, 1312 uint32_t NumHashBuckets, 1313 FixedStreamArray<support::ulittle32_t> HashValues, 1314 TpiStream *Stream, bool Bytes, bool Extras) { 1315 1316 Printer.formatLine("Showing {0:N} records", NumTypeRecords); 1317 uint32_t Width = NumDigits(TypeIndex::FirstNonSimpleIndex + NumTypeRecords); 1318 1319 MinimalTypeDumpVisitor V(Printer, Width + 2, Bytes, Extras, Types, RefTracker, 1320 NumHashBuckets, HashValues, Stream); 1321 1322 if (auto EC = codeview::visitTypeStream(Types, V)) { 1323 Printer.formatLine("An error occurred dumping type records: {0}", 1324 toString(std::move(EC))); 1325 } 1326 } 1327 1328 static void dumpPartialTypeStream(LinePrinter &Printer, 1329 LazyRandomTypeCollection &Types, 1330 TypeReferenceTracker *RefTracker, 1331 TpiStream &Stream, ArrayRef<TypeIndex> TiList, 1332 bool Bytes, bool Extras, bool Deps) { 1333 uint32_t Width = 1334 NumDigits(TypeIndex::FirstNonSimpleIndex + Stream.getNumTypeRecords()); 1335 1336 MinimalTypeDumpVisitor V(Printer, Width + 2, Bytes, Extras, Types, RefTracker, 1337 Stream.getNumHashBuckets(), Stream.getHashValues(), 1338 &Stream); 1339 1340 if (opts::dump::DumpTypeDependents) { 1341 // If we need to dump all dependents, then iterate each index and find 1342 // all dependents, adding them to a map ordered by TypeIndex. 1343 std::map<TypeIndex, CVType> DepSet; 1344 buildDepSet(Types, TiList, DepSet); 1345 1346 Printer.formatLine( 1347 "Showing {0:N} records and their dependents ({1:N} records total)", 1348 TiList.size(), DepSet.size()); 1349 1350 for (auto &Dep : DepSet) { 1351 if (auto EC = codeview::visitTypeRecord(Dep.second, Dep.first, V)) 1352 Printer.formatLine("An error occurred dumping type record {0}: {1}", 1353 Dep.first, toString(std::move(EC))); 1354 } 1355 } else { 1356 Printer.formatLine("Showing {0:N} records.", TiList.size()); 1357 1358 for (const auto &I : TiList) { 1359 TypeIndex TI(I); 1360 CVType Type = Types.getType(TI); 1361 if (auto EC = codeview::visitTypeRecord(Type, TI, V)) 1362 Printer.formatLine("An error occurred dumping type record {0}: {1}", TI, 1363 toString(std::move(EC))); 1364 } 1365 } 1366 } 1367 1368 Error DumpOutputStyle::dumpTypesFromObjectFile() { 1369 LazyRandomTypeCollection Types(100); 1370 1371 for (const auto &S : getObj().sections()) { 1372 StringRef SectionName; 1373 if (auto EC = S.getName(SectionName)) 1374 return errorCodeToError(EC); 1375 1376 // .debug$T is a standard CodeView type section, while .debug$P is the same 1377 // format but used for MSVC precompiled header object files. 1378 if (SectionName == ".debug$T") 1379 printHeader(P, "Types (.debug$T)"); 1380 else if (SectionName == ".debug$P") 1381 printHeader(P, "Precompiled Types (.debug$P)"); 1382 else 1383 continue; 1384 1385 Expected<StringRef> ContentsOrErr = S.getContents(); 1386 if (!ContentsOrErr) 1387 return ContentsOrErr.takeError(); 1388 1389 uint32_t Magic; 1390 BinaryStreamReader Reader(*ContentsOrErr, llvm::support::little); 1391 if (auto EC = Reader.readInteger(Magic)) 1392 return EC; 1393 if (Magic != COFF::DEBUG_SECTION_MAGIC) 1394 return make_error<StringError>("Invalid CodeView debug section.", 1395 inconvertibleErrorCode()); 1396 1397 Types.reset(Reader, 100); 1398 1399 if (opts::dump::DumpTypes) { 1400 dumpFullTypeStream(P, Types, RefTracker.get(), 0, 0, {}, nullptr, 1401 opts::dump::DumpTypeData, false); 1402 } else if (opts::dump::DumpTypeExtras) { 1403 auto LocalHashes = LocallyHashedType::hashTypeCollection(Types); 1404 auto GlobalHashes = GloballyHashedType::hashTypeCollection(Types); 1405 assert(LocalHashes.size() == GlobalHashes.size()); 1406 1407 P.formatLine("Local / Global hashes:"); 1408 TypeIndex TI(TypeIndex::FirstNonSimpleIndex); 1409 for (const auto &H : zip(LocalHashes, GlobalHashes)) { 1410 AutoIndent Indent2(P); 1411 LocallyHashedType &L = std::get<0>(H); 1412 GloballyHashedType &G = std::get<1>(H); 1413 1414 P.formatLine("TI: {0}, LocalHash: {1:X}, GlobalHash: {2}", TI, L, G); 1415 1416 ++TI; 1417 } 1418 P.NewLine(); 1419 } 1420 } 1421 1422 return Error::success(); 1423 } 1424 1425 Error DumpOutputStyle::dumpTpiStream(uint32_t StreamIdx) { 1426 assert(StreamIdx == StreamTPI || StreamIdx == StreamIPI); 1427 1428 if (StreamIdx == StreamTPI) { 1429 printHeader(P, "Types (TPI Stream)"); 1430 } else if (StreamIdx == StreamIPI) { 1431 printHeader(P, "Types (IPI Stream)"); 1432 } 1433 1434 assert(!File.isObj()); 1435 1436 bool Present = false; 1437 bool DumpTypes = false; 1438 bool DumpBytes = false; 1439 bool DumpExtras = false; 1440 std::vector<uint32_t> Indices; 1441 if (StreamIdx == StreamTPI) { 1442 Present = getPdb().hasPDBTpiStream(); 1443 DumpTypes = opts::dump::DumpTypes; 1444 DumpBytes = opts::dump::DumpTypeData; 1445 DumpExtras = opts::dump::DumpTypeExtras; 1446 Indices.assign(opts::dump::DumpTypeIndex.begin(), 1447 opts::dump::DumpTypeIndex.end()); 1448 } else if (StreamIdx == StreamIPI) { 1449 Present = getPdb().hasPDBIpiStream(); 1450 DumpTypes = opts::dump::DumpIds; 1451 DumpBytes = opts::dump::DumpIdData; 1452 DumpExtras = opts::dump::DumpIdExtras; 1453 Indices.assign(opts::dump::DumpIdIndex.begin(), 1454 opts::dump::DumpIdIndex.end()); 1455 } 1456 1457 if (!Present) { 1458 printStreamNotPresent(StreamIdx == StreamTPI ? "TPI" : "IPI"); 1459 return Error::success(); 1460 } 1461 1462 AutoIndent Indent(P); 1463 ExitOnError Err("Unexpected error processing types: "); 1464 1465 auto &Stream = Err((StreamIdx == StreamTPI) ? getPdb().getPDBTpiStream() 1466 : getPdb().getPDBIpiStream()); 1467 1468 auto &Types = (StreamIdx == StreamTPI) ? File.types() : File.ids(); 1469 1470 // Only emit notes about referenced/unreferenced for types. 1471 TypeReferenceTracker *MaybeTracker = 1472 (StreamIdx == StreamTPI) ? RefTracker.get() : nullptr; 1473 1474 // Enable resolving forward decls. 1475 Stream.buildHashMap(); 1476 1477 if (DumpTypes || !Indices.empty()) { 1478 if (Indices.empty()) 1479 dumpFullTypeStream(P, Types, MaybeTracker, Stream.getNumTypeRecords(), 1480 Stream.getNumHashBuckets(), Stream.getHashValues(), 1481 &Stream, DumpBytes, DumpExtras); 1482 else { 1483 std::vector<TypeIndex> TiList(Indices.begin(), Indices.end()); 1484 dumpPartialTypeStream(P, Types, MaybeTracker, Stream, TiList, DumpBytes, 1485 DumpExtras, opts::dump::DumpTypeDependents); 1486 } 1487 } 1488 1489 if (DumpExtras) { 1490 P.NewLine(); 1491 1492 P.formatLine("Header Version: {0}", 1493 static_cast<uint32_t>(Stream.getTpiVersion())); 1494 P.formatLine("Hash Stream Index: {0}", Stream.getTypeHashStreamIndex()); 1495 P.formatLine("Aux Hash Stream Index: {0}", 1496 Stream.getTypeHashStreamAuxIndex()); 1497 P.formatLine("Hash Key Size: {0}", Stream.getHashKeySize()); 1498 P.formatLine("Num Hash Buckets: {0}", Stream.getNumHashBuckets()); 1499 1500 auto IndexOffsets = Stream.getTypeIndexOffsets(); 1501 P.formatLine("Type Index Offsets:"); 1502 for (const auto &IO : IndexOffsets) { 1503 AutoIndent Indent2(P); 1504 P.formatLine("TI: {0}, Offset: {1}", IO.Type, fmtle(IO.Offset)); 1505 } 1506 1507 if (getPdb().hasPDBStringTable()) { 1508 P.NewLine(); 1509 P.formatLine("Hash Adjusters:"); 1510 auto &Adjusters = Stream.getHashAdjusters(); 1511 auto &Strings = Err(getPdb().getStringTable()); 1512 for (const auto &A : Adjusters) { 1513 AutoIndent Indent2(P); 1514 auto ExpectedStr = Strings.getStringForID(A.first); 1515 TypeIndex TI(A.second); 1516 if (ExpectedStr) 1517 P.formatLine("`{0}` -> {1}", *ExpectedStr, TI); 1518 else { 1519 P.formatLine("unknown str id ({0}) -> {1}", A.first, TI); 1520 consumeError(ExpectedStr.takeError()); 1521 } 1522 } 1523 } 1524 } 1525 return Error::success(); 1526 } 1527 1528 Error DumpOutputStyle::dumpModuleSymsForObj() { 1529 printHeader(P, "Symbols"); 1530 1531 AutoIndent Indent(P); 1532 1533 ExitOnError Err("Unexpected error processing symbols: "); 1534 1535 auto &Types = File.types(); 1536 1537 SymbolVisitorCallbackPipeline Pipeline; 1538 SymbolDeserializer Deserializer(nullptr, CodeViewContainer::ObjectFile); 1539 MinimalSymbolDumper Dumper(P, opts::dump::DumpSymRecordBytes, Types, Types); 1540 1541 Pipeline.addCallbackToPipeline(Deserializer); 1542 Pipeline.addCallbackToPipeline(Dumper); 1543 CVSymbolVisitor Visitor(Pipeline); 1544 1545 std::unique_ptr<llvm::Error> SymbolError; 1546 1547 iterateModuleSubsections<DebugSymbolsSubsectionRef>( 1548 File, PrintScope{P, 2}, 1549 [&](uint32_t Modi, const SymbolGroup &Strings, 1550 DebugSymbolsSubsectionRef &Symbols) { 1551 Dumper.setSymbolGroup(&Strings); 1552 for (auto Symbol : Symbols) { 1553 if (auto EC = Visitor.visitSymbolRecord(Symbol)) { 1554 SymbolError = llvm::make_unique<Error>(std::move(EC)); 1555 return; 1556 } 1557 } 1558 }); 1559 1560 if (SymbolError) 1561 return std::move(*SymbolError); 1562 1563 return Error::success(); 1564 } 1565 1566 Error DumpOutputStyle::dumpModuleSymsForPdb() { 1567 printHeader(P, "Symbols"); 1568 1569 if (File.isPdb() && !getPdb().hasPDBDbiStream()) { 1570 printStreamNotPresent("DBI"); 1571 return Error::success(); 1572 } 1573 1574 AutoIndent Indent(P); 1575 ExitOnError Err("Unexpected error processing symbols: "); 1576 1577 auto &Ids = File.ids(); 1578 auto &Types = File.types(); 1579 1580 iterateSymbolGroups( 1581 File, PrintScope{P, 2}, [&](uint32_t I, const SymbolGroup &Strings) { 1582 auto ExpectedModS = getModuleDebugStream(File.pdb(), I); 1583 if (!ExpectedModS) { 1584 P.formatLine("Error loading module stream {0}. {1}", I, 1585 toString(ExpectedModS.takeError())); 1586 return; 1587 } 1588 1589 ModuleDebugStreamRef &ModS = *ExpectedModS; 1590 1591 SymbolVisitorCallbackPipeline Pipeline; 1592 SymbolDeserializer Deserializer(nullptr, CodeViewContainer::Pdb); 1593 MinimalSymbolDumper Dumper(P, opts::dump::DumpSymRecordBytes, Strings, 1594 Ids, Types); 1595 1596 Pipeline.addCallbackToPipeline(Deserializer); 1597 Pipeline.addCallbackToPipeline(Dumper); 1598 CVSymbolVisitor Visitor(Pipeline); 1599 auto SS = ModS.getSymbolsSubstream(); 1600 if (auto EC = 1601 Visitor.visitSymbolStream(ModS.getSymbolArray(), SS.Offset)) { 1602 P.formatLine("Error while processing symbol records. {0}", 1603 toString(std::move(EC))); 1604 return; 1605 } 1606 }); 1607 return Error::success(); 1608 } 1609 1610 Error DumpOutputStyle::dumpTypeRefStats() { 1611 printHeader(P, "Type Reference Statistics"); 1612 AutoIndent Indent(P); 1613 1614 // Sum the byte size of all type records, and the size and count of all 1615 // referenced records. 1616 size_t TotalRecs = File.types().size(); 1617 size_t RefRecs = 0; 1618 size_t TotalBytes = 0; 1619 size_t RefBytes = 0; 1620 auto &Types = File.types(); 1621 for (Optional<TypeIndex> TI = Types.getFirst(); TI; TI = Types.getNext(*TI)) { 1622 CVType Type = File.types().getType(*TI); 1623 TotalBytes += Type.length(); 1624 if (RefTracker->isTypeReferenced(*TI)) { 1625 ++RefRecs; 1626 RefBytes += Type.length(); 1627 } 1628 } 1629 1630 P.formatLine("Records referenced: {0:N} / {1:N} {2:P}", RefRecs, TotalRecs, 1631 (double)RefRecs / TotalRecs); 1632 P.formatLine("Bytes referenced: {0:N} / {1:N} {2:P}", RefBytes, TotalBytes, 1633 (double)RefBytes / TotalBytes); 1634 1635 return Error::success(); 1636 } 1637 1638 Error DumpOutputStyle::dumpGSIRecords() { 1639 printHeader(P, "GSI Records"); 1640 1641 if (File.isObj()) { 1642 printStreamNotValidForObj(); 1643 return Error::success(); 1644 } 1645 1646 if (!getPdb().hasPDBSymbolStream()) { 1647 printStreamNotPresent("GSI Common Symbol"); 1648 return Error::success(); 1649 } 1650 1651 AutoIndent Indent(P); 1652 1653 auto &Records = cantFail(getPdb().getPDBSymbolStream()); 1654 auto &Types = File.types(); 1655 auto &Ids = File.ids(); 1656 1657 P.printLine("Records"); 1658 SymbolVisitorCallbackPipeline Pipeline; 1659 SymbolDeserializer Deserializer(nullptr, CodeViewContainer::Pdb); 1660 MinimalSymbolDumper Dumper(P, opts::dump::DumpSymRecordBytes, Ids, Types); 1661 1662 Pipeline.addCallbackToPipeline(Deserializer); 1663 Pipeline.addCallbackToPipeline(Dumper); 1664 CVSymbolVisitor Visitor(Pipeline); 1665 1666 BinaryStreamRef SymStream = Records.getSymbolArray().getUnderlyingStream(); 1667 if (auto E = Visitor.visitSymbolStream(Records.getSymbolArray(), 0)) 1668 return E; 1669 return Error::success(); 1670 } 1671 1672 Error DumpOutputStyle::dumpGlobals() { 1673 printHeader(P, "Global Symbols"); 1674 1675 if (File.isObj()) { 1676 printStreamNotValidForObj(); 1677 return Error::success(); 1678 } 1679 1680 if (!getPdb().hasPDBGlobalsStream()) { 1681 printStreamNotPresent("Globals"); 1682 return Error::success(); 1683 } 1684 1685 AutoIndent Indent(P); 1686 ExitOnError Err("Error dumping globals stream: "); 1687 auto &Globals = Err(getPdb().getPDBGlobalsStream()); 1688 1689 if (opts::dump::DumpGlobalNames.empty()) { 1690 const GSIHashTable &Table = Globals.getGlobalsTable(); 1691 Err(dumpSymbolsFromGSI(Table, opts::dump::DumpGlobalExtras)); 1692 } else { 1693 SymbolStream &SymRecords = cantFail(getPdb().getPDBSymbolStream()); 1694 auto &Types = File.types(); 1695 auto &Ids = File.ids(); 1696 1697 SymbolVisitorCallbackPipeline Pipeline; 1698 SymbolDeserializer Deserializer(nullptr, CodeViewContainer::Pdb); 1699 MinimalSymbolDumper Dumper(P, opts::dump::DumpSymRecordBytes, Ids, Types); 1700 1701 Pipeline.addCallbackToPipeline(Deserializer); 1702 Pipeline.addCallbackToPipeline(Dumper); 1703 CVSymbolVisitor Visitor(Pipeline); 1704 1705 using ResultEntryType = std::pair<uint32_t, CVSymbol>; 1706 for (StringRef Name : opts::dump::DumpGlobalNames) { 1707 AutoIndent Indent(P); 1708 P.formatLine("Global Name `{0}`", Name); 1709 std::vector<ResultEntryType> Results = 1710 Globals.findRecordsByName(Name, SymRecords); 1711 if (Results.empty()) { 1712 AutoIndent Indent(P); 1713 P.printLine("(no matching records found)"); 1714 continue; 1715 } 1716 1717 for (ResultEntryType Result : Results) { 1718 if (auto E = Visitor.visitSymbolRecord(Result.second, Result.first)) 1719 return E; 1720 } 1721 } 1722 } 1723 return Error::success(); 1724 } 1725 1726 Error DumpOutputStyle::dumpPublics() { 1727 printHeader(P, "Public Symbols"); 1728 1729 if (File.isObj()) { 1730 printStreamNotValidForObj(); 1731 return Error::success(); 1732 } 1733 1734 if (!getPdb().hasPDBPublicsStream()) { 1735 printStreamNotPresent("Publics"); 1736 return Error::success(); 1737 } 1738 1739 AutoIndent Indent(P); 1740 ExitOnError Err("Error dumping publics stream: "); 1741 auto &Publics = Err(getPdb().getPDBPublicsStream()); 1742 1743 const GSIHashTable &PublicsTable = Publics.getPublicsTable(); 1744 if (opts::dump::DumpPublicExtras) { 1745 P.printLine("Publics Header"); 1746 AutoIndent Indent(P); 1747 P.formatLine("sym hash = {0}, thunk table addr = {1}", Publics.getSymHash(), 1748 formatSegmentOffset(Publics.getThunkTableSection(), 1749 Publics.getThunkTableOffset())); 1750 } 1751 Err(dumpSymbolsFromGSI(PublicsTable, opts::dump::DumpPublicExtras)); 1752 1753 // Skip the rest if we aren't dumping extras. 1754 if (!opts::dump::DumpPublicExtras) 1755 return Error::success(); 1756 1757 P.formatLine("Address Map"); 1758 { 1759 // These are offsets into the publics stream sorted by secidx:secrel. 1760 AutoIndent Indent2(P); 1761 for (uint32_t Addr : Publics.getAddressMap()) 1762 P.formatLine("off = {0}", Addr); 1763 } 1764 1765 // The thunk map is optional debug info used for ILT thunks. 1766 if (!Publics.getThunkMap().empty()) { 1767 P.formatLine("Thunk Map"); 1768 AutoIndent Indent2(P); 1769 for (uint32_t Addr : Publics.getThunkMap()) 1770 P.formatLine("{0:x8}", Addr); 1771 } 1772 1773 // The section offsets table appears to be empty when incremental linking 1774 // isn't in use. 1775 if (!Publics.getSectionOffsets().empty()) { 1776 P.formatLine("Section Offsets"); 1777 AutoIndent Indent2(P); 1778 for (const SectionOffset &SO : Publics.getSectionOffsets()) 1779 P.formatLine("{0:x4}:{1:x8}", uint16_t(SO.Isect), uint32_t(SO.Off)); 1780 } 1781 1782 return Error::success(); 1783 } 1784 1785 Error DumpOutputStyle::dumpSymbolsFromGSI(const GSIHashTable &Table, 1786 bool HashExtras) { 1787 auto ExpectedSyms = getPdb().getPDBSymbolStream(); 1788 if (!ExpectedSyms) 1789 return ExpectedSyms.takeError(); 1790 auto &Types = File.types(); 1791 auto &Ids = File.ids(); 1792 1793 if (HashExtras) { 1794 P.printLine("GSI Header"); 1795 AutoIndent Indent(P); 1796 P.formatLine("sig = {0:X}, hdr = {1:X}, hr size = {2}, num buckets = {3}", 1797 Table.getVerSignature(), Table.getVerHeader(), 1798 Table.getHashRecordSize(), Table.getNumBuckets()); 1799 } 1800 1801 { 1802 P.printLine("Records"); 1803 SymbolVisitorCallbackPipeline Pipeline; 1804 SymbolDeserializer Deserializer(nullptr, CodeViewContainer::Pdb); 1805 MinimalSymbolDumper Dumper(P, opts::dump::DumpSymRecordBytes, Ids, Types); 1806 1807 Pipeline.addCallbackToPipeline(Deserializer); 1808 Pipeline.addCallbackToPipeline(Dumper); 1809 CVSymbolVisitor Visitor(Pipeline); 1810 1811 1812 BinaryStreamRef SymStream = 1813 ExpectedSyms->getSymbolArray().getUnderlyingStream(); 1814 for (uint32_t PubSymOff : Table) { 1815 Expected<CVSymbol> Sym = readSymbolFromStream(SymStream, PubSymOff); 1816 if (!Sym) 1817 return Sym.takeError(); 1818 if (auto E = Visitor.visitSymbolRecord(*Sym, PubSymOff)) 1819 return E; 1820 } 1821 } 1822 1823 // Return early if we aren't dumping public hash table and address map info. 1824 if (HashExtras) { 1825 P.formatLine("Hash Entries"); 1826 { 1827 AutoIndent Indent2(P); 1828 for (const PSHashRecord &HR : Table.HashRecords) 1829 P.formatLine("off = {0}, refcnt = {1}", uint32_t(HR.Off), 1830 uint32_t(HR.CRef)); 1831 } 1832 1833 P.formatLine("Hash Buckets"); 1834 { 1835 AutoIndent Indent2(P); 1836 for (uint32_t Hash : Table.HashBuckets) 1837 P.formatLine("{0:x8}", Hash); 1838 } 1839 } 1840 1841 return Error::success(); 1842 } 1843 1844 static std::string formatSegMapDescriptorFlag(uint32_t IndentLevel, 1845 OMFSegDescFlags Flags) { 1846 std::vector<std::string> Opts; 1847 if (Flags == OMFSegDescFlags::None) 1848 return "none"; 1849 1850 PUSH_FLAG(OMFSegDescFlags, Read, Flags, "read"); 1851 PUSH_FLAG(OMFSegDescFlags, Write, Flags, "write"); 1852 PUSH_FLAG(OMFSegDescFlags, Execute, Flags, "execute"); 1853 PUSH_FLAG(OMFSegDescFlags, AddressIs32Bit, Flags, "32 bit addr"); 1854 PUSH_FLAG(OMFSegDescFlags, IsSelector, Flags, "selector"); 1855 PUSH_FLAG(OMFSegDescFlags, IsAbsoluteAddress, Flags, "absolute addr"); 1856 PUSH_FLAG(OMFSegDescFlags, IsGroup, Flags, "group"); 1857 return typesetItemList(Opts, IndentLevel, 4, " | "); 1858 } 1859 1860 Error DumpOutputStyle::dumpSectionHeaders() { 1861 dumpSectionHeaders("Section Headers", DbgHeaderType::SectionHdr); 1862 dumpSectionHeaders("Original Section Headers", DbgHeaderType::SectionHdrOrig); 1863 return Error::success(); 1864 } 1865 1866 void DumpOutputStyle::dumpSectionHeaders(StringRef Label, DbgHeaderType Type) { 1867 printHeader(P, Label); 1868 1869 if (File.isObj()) { 1870 printStreamNotValidForObj(); 1871 return; 1872 } 1873 1874 if (!getPdb().hasPDBDbiStream()) { 1875 printStreamNotPresent("DBI"); 1876 return; 1877 } 1878 1879 AutoIndent Indent(P); 1880 ExitOnError Err("Error dumping section headers: "); 1881 std::unique_ptr<MappedBlockStream> Stream; 1882 ArrayRef<object::coff_section> Headers; 1883 auto ExpectedHeaders = loadSectionHeaders(getPdb(), Type); 1884 if (!ExpectedHeaders) { 1885 P.printLine(toString(ExpectedHeaders.takeError())); 1886 return; 1887 } 1888 std::tie(Stream, Headers) = std::move(*ExpectedHeaders); 1889 1890 uint32_t I = 1; 1891 for (const auto &Header : Headers) { 1892 P.NewLine(); 1893 P.formatLine("SECTION HEADER #{0}", I); 1894 P.formatLine("{0,8} name", Header.Name); 1895 P.formatLine("{0,8:X-} virtual size", uint32_t(Header.VirtualSize)); 1896 P.formatLine("{0,8:X-} virtual address", uint32_t(Header.VirtualAddress)); 1897 P.formatLine("{0,8:X-} size of raw data", uint32_t(Header.SizeOfRawData)); 1898 P.formatLine("{0,8:X-} file pointer to raw data", 1899 uint32_t(Header.PointerToRawData)); 1900 P.formatLine("{0,8:X-} file pointer to relocation table", 1901 uint32_t(Header.PointerToRelocations)); 1902 P.formatLine("{0,8:X-} file pointer to line numbers", 1903 uint32_t(Header.PointerToLinenumbers)); 1904 P.formatLine("{0,8:X-} number of relocations", 1905 uint32_t(Header.NumberOfRelocations)); 1906 P.formatLine("{0,8:X-} number of line numbers", 1907 uint32_t(Header.NumberOfLinenumbers)); 1908 P.formatLine("{0,8:X-} flags", uint32_t(Header.Characteristics)); 1909 AutoIndent IndentMore(P, 9); 1910 P.formatLine("{0}", formatSectionCharacteristics( 1911 P.getIndentLevel(), Header.Characteristics, 1, "")); 1912 ++I; 1913 } 1914 return; 1915 } 1916 1917 Error DumpOutputStyle::dumpSectionContribs() { 1918 printHeader(P, "Section Contributions"); 1919 1920 if (File.isObj()) { 1921 printStreamNotValidForObj(); 1922 return Error::success(); 1923 } 1924 1925 if (!getPdb().hasPDBDbiStream()) { 1926 printStreamNotPresent("DBI"); 1927 return Error::success(); 1928 } 1929 1930 AutoIndent Indent(P); 1931 ExitOnError Err("Error dumping section contributions: "); 1932 1933 auto &Dbi = Err(getPdb().getPDBDbiStream()); 1934 1935 class Visitor : public ISectionContribVisitor { 1936 public: 1937 Visitor(LinePrinter &P, ArrayRef<std::string> Names) : P(P), Names(Names) { 1938 auto Max = std::max_element( 1939 Names.begin(), Names.end(), 1940 [](StringRef S1, StringRef S2) { return S1.size() < S2.size(); }); 1941 MaxNameLen = (Max == Names.end() ? 0 : Max->size()); 1942 } 1943 void visit(const SectionContrib &SC) override { 1944 dumpSectionContrib(P, SC, Names, MaxNameLen); 1945 } 1946 void visit(const SectionContrib2 &SC) override { 1947 dumpSectionContrib(P, SC, Names, MaxNameLen); 1948 } 1949 1950 private: 1951 LinePrinter &P; 1952 uint32_t MaxNameLen; 1953 ArrayRef<std::string> Names; 1954 }; 1955 1956 std::vector<std::string> Names = getSectionNames(getPdb()); 1957 Visitor V(P, makeArrayRef(Names)); 1958 Dbi.visitSectionContributions(V); 1959 return Error::success(); 1960 } 1961 1962 Error DumpOutputStyle::dumpSectionMap() { 1963 printHeader(P, "Section Map"); 1964 1965 if (File.isObj()) { 1966 printStreamNotValidForObj(); 1967 return Error::success(); 1968 } 1969 1970 if (!getPdb().hasPDBDbiStream()) { 1971 printStreamNotPresent("DBI"); 1972 return Error::success(); 1973 } 1974 1975 AutoIndent Indent(P); 1976 ExitOnError Err("Error dumping section map: "); 1977 1978 auto &Dbi = Err(getPdb().getPDBDbiStream()); 1979 1980 uint32_t I = 0; 1981 for (auto &M : Dbi.getSectionMap()) { 1982 P.formatLine( 1983 "Section {0:4} | ovl = {1}, group = {2}, frame = {3}, name = {4}", I, 1984 fmtle(M.Ovl), fmtle(M.Group), fmtle(M.Frame), fmtle(M.SecName)); 1985 P.formatLine(" class = {0}, offset = {1}, size = {2}", 1986 fmtle(M.ClassName), fmtle(M.Offset), fmtle(M.SecByteLength)); 1987 P.formatLine(" flags = {0}", 1988 formatSegMapDescriptorFlag( 1989 P.getIndentLevel() + 13, 1990 static_cast<OMFSegDescFlags>(uint16_t(M.Flags)))); 1991 ++I; 1992 } 1993 return Error::success(); 1994 } 1995