| 1 | //===- llvm-pdbutil.cpp - Dump debug info from a PDB file -------*- 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 | // Dumps debug information present in PDB files. |
| 10 | // |
| 11 | //===----------------------------------------------------------------------===// |
| 12 | |
| 13 | #include "llvm-pdbutil.h" |
| 14 | |
| 15 | #include "BytesOutputStyle.h" |
| 16 | #include "DumpOutputStyle.h" |
| 17 | #include "ExplainOutputStyle.h" |
| 18 | #include "OutputStyle.h" |
| 19 | #include "PrettyClassDefinitionDumper.h" |
| 20 | #include "PrettyCompilandDumper.h" |
| 21 | #include "PrettyEnumDumper.h" |
| 22 | #include "PrettyExternalSymbolDumper.h" |
| 23 | #include "PrettyFunctionDumper.h" |
| 24 | #include "PrettyTypeDumper.h" |
| 25 | #include "PrettyTypedefDumper.h" |
| 26 | #include "PrettyVariableDumper.h" |
| 27 | #include "YAMLOutputStyle.h" |
| 28 | |
| 29 | #include "llvm/ADT/ArrayRef.h" |
| 30 | #include "llvm/ADT/STLExtras.h" |
| 31 | #include "llvm/ADT/StringExtras.h" |
| 32 | #include "llvm/BinaryFormat/Magic.h" |
| 33 | #include "llvm/Config/config.h" |
| 34 | #include "llvm/DebugInfo/CodeView/AppendingTypeTableBuilder.h" |
| 35 | #include "llvm/DebugInfo/CodeView/DebugChecksumsSubsection.h" |
| 36 | #include "llvm/DebugInfo/CodeView/DebugInlineeLinesSubsection.h" |
| 37 | #include "llvm/DebugInfo/CodeView/DebugLinesSubsection.h" |
| 38 | #include "llvm/DebugInfo/CodeView/LazyRandomTypeCollection.h" |
| 39 | #include "llvm/DebugInfo/CodeView/MergingTypeTableBuilder.h" |
| 40 | #include "llvm/DebugInfo/CodeView/StringsAndChecksums.h" |
| 41 | #include "llvm/DebugInfo/CodeView/TypeStreamMerger.h" |
| 42 | #include "llvm/DebugInfo/MSF/MSFBuilder.h" |
| 43 | #include "llvm/DebugInfo/MSF/MappedBlockStream.h" |
| 44 | #include "llvm/DebugInfo/PDB/ConcreteSymbolEnumerator.h" |
| 45 | #include "llvm/DebugInfo/PDB/IPDBEnumChildren.h" |
| 46 | #include "llvm/DebugInfo/PDB/IPDBInjectedSource.h" |
| 47 | #include "llvm/DebugInfo/PDB/IPDBLineNumber.h" |
| 48 | #include "llvm/DebugInfo/PDB/IPDBRawSymbol.h" |
| 49 | #include "llvm/DebugInfo/PDB/IPDBSession.h" |
| 50 | #include "llvm/DebugInfo/PDB/Native/DbiModuleDescriptorBuilder.h" |
| 51 | #include "llvm/DebugInfo/PDB/Native/DbiStreamBuilder.h" |
| 52 | #include "llvm/DebugInfo/PDB/Native/InfoStream.h" |
| 53 | #include "llvm/DebugInfo/PDB/Native/InfoStreamBuilder.h" |
| 54 | #include "llvm/DebugInfo/PDB/Native/InputFile.h" |
| 55 | #include "llvm/DebugInfo/PDB/Native/NativeSession.h" |
| 56 | #include "llvm/DebugInfo/PDB/Native/PDBFile.h" |
| 57 | #include "llvm/DebugInfo/PDB/Native/PDBFileBuilder.h" |
| 58 | #include "llvm/DebugInfo/PDB/Native/PDBStringTableBuilder.h" |
| 59 | #include "llvm/DebugInfo/PDB/Native/RawConstants.h" |
| 60 | #include "llvm/DebugInfo/PDB/Native/RawError.h" |
| 61 | #include "llvm/DebugInfo/PDB/Native/TpiStream.h" |
| 62 | #include "llvm/DebugInfo/PDB/Native/TpiStreamBuilder.h" |
| 63 | #include "llvm/DebugInfo/PDB/PDB.h" |
| 64 | #include "llvm/DebugInfo/PDB/PDBSymbolCompiland.h" |
| 65 | #include "llvm/DebugInfo/PDB/PDBSymbolData.h" |
| 66 | #include "llvm/DebugInfo/PDB/PDBSymbolExe.h" |
| 67 | #include "llvm/DebugInfo/PDB/PDBSymbolFunc.h" |
| 68 | #include "llvm/DebugInfo/PDB/PDBSymbolPublicSymbol.h" |
| 69 | #include "llvm/DebugInfo/PDB/PDBSymbolThunk.h" |
| 70 | #include "llvm/DebugInfo/PDB/PDBSymbolTypeBuiltin.h" |
| 71 | #include "llvm/DebugInfo/PDB/PDBSymbolTypeEnum.h" |
| 72 | #include "llvm/DebugInfo/PDB/PDBSymbolTypeFunctionArg.h" |
| 73 | #include "llvm/DebugInfo/PDB/PDBSymbolTypeFunctionSig.h" |
| 74 | #include "llvm/DebugInfo/PDB/PDBSymbolTypeTypedef.h" |
| 75 | #include "llvm/DebugInfo/PDB/PDBSymbolTypeUDT.h" |
| 76 | #include "llvm/Support/BinaryByteStream.h" |
| 77 | #include "llvm/Support/COM.h" |
| 78 | #include "llvm/Support/CommandLine.h" |
| 79 | #include "llvm/Support/ConvertUTF.h" |
| 80 | #include "llvm/Support/FileOutputBuffer.h" |
| 81 | #include "llvm/Support/FileSystem.h" |
| 82 | #include "llvm/Support/Format.h" |
| 83 | #include "llvm/Support/InitLLVM.h" |
| 84 | #include "llvm/Support/LineIterator.h" |
| 85 | #include "llvm/Support/MemoryBuffer.h" |
| 86 | #include "llvm/Support/Path.h" |
| 87 | #include "llvm/Support/PrettyStackTrace.h" |
| 88 | #include "llvm/Support/Process.h" |
| 89 | #include "llvm/Support/Regex.h" |
| 90 | #include "llvm/Support/ScopedPrinter.h" |
| 91 | #include "llvm/Support/Signals.h" |
| 92 | #include "llvm/Support/raw_ostream.h" |
| 93 | |
| 94 | using namespace llvm; |
| 95 | using namespace llvm::codeview; |
| 96 | using namespace llvm::msf; |
| 97 | using namespace llvm::pdb; |
| 98 | |
| 99 | namespace opts { |
| 100 | |
| 101 | cl::SubCommand DumpSubcommand("dump" , "Dump MSF and CodeView debug info" ); |
| 102 | cl::SubCommand BytesSubcommand("bytes" , "Dump raw bytes from the PDB file" ); |
| 103 | |
| 104 | cl::SubCommand DiaDumpSubcommand("diadump" , |
| 105 | "Dump debug information using a DIA-like API" ); |
| 106 | |
| 107 | cl::SubCommand |
| 108 | PrettySubcommand("pretty" , |
| 109 | "Dump semantic information about types and symbols" ); |
| 110 | |
| 111 | cl::SubCommand |
| 112 | YamlToPdbSubcommand("yaml2pdb" , |
| 113 | "Generate a PDB file from a YAML description" ); |
| 114 | cl::SubCommand |
| 115 | PdbToYamlSubcommand("pdb2yaml" , |
| 116 | "Generate a detailed YAML description of a PDB File" ); |
| 117 | |
| 118 | cl::SubCommand MergeSubcommand("merge" , |
| 119 | "Merge multiple PDBs into a single PDB" ); |
| 120 | |
| 121 | cl::SubCommand ExplainSubcommand("explain" , |
| 122 | "Explain the meaning of a file offset" ); |
| 123 | |
| 124 | cl::SubCommand ExportSubcommand("export" , |
| 125 | "Write binary data from a stream to a file" ); |
| 126 | |
| 127 | static cl::OptionCategory TypeCategory("Symbol Type Options" ); |
| 128 | static cl::OptionCategory FilterCategory("Filtering and Sorting Options" ); |
| 129 | static cl::OptionCategory OtherOptions("Other Options" ); |
| 130 | |
| 131 | cl::ValuesClass ChunkValues = cl::values( |
| 132 | clEnumValN(ModuleSubsection::CrossScopeExports, "cme" , |
| 133 | "Cross module exports (DEBUG_S_CROSSSCOPEEXPORTS subsection)" ), |
| 134 | clEnumValN(ModuleSubsection::CrossScopeImports, "cmi" , |
| 135 | "Cross module imports (DEBUG_S_CROSSSCOPEIMPORTS subsection)" ), |
| 136 | clEnumValN(ModuleSubsection::FileChecksums, "fc" , |
| 137 | "File checksums (DEBUG_S_CHECKSUMS subsection)" ), |
| 138 | clEnumValN(ModuleSubsection::InlineeLines, "ilines" , |
| 139 | "Inlinee lines (DEBUG_S_INLINEELINES subsection)" ), |
| 140 | clEnumValN(ModuleSubsection::Lines, "lines" , |
| 141 | "Lines (DEBUG_S_LINES subsection)" ), |
| 142 | clEnumValN(ModuleSubsection::StringTable, "strings" , |
| 143 | "String Table (DEBUG_S_STRINGTABLE subsection) (not " |
| 144 | "typically present in PDB file)" ), |
| 145 | clEnumValN(ModuleSubsection::FrameData, "frames" , |
| 146 | "Frame Data (DEBUG_S_FRAMEDATA subsection)" ), |
| 147 | clEnumValN(ModuleSubsection::Symbols, "symbols" , |
| 148 | "Symbols (DEBUG_S_SYMBOLS subsection) (not typically " |
| 149 | "present in PDB file)" ), |
| 150 | clEnumValN(ModuleSubsection::CoffSymbolRVAs, "rvas" , |
| 151 | "COFF Symbol RVAs (DEBUG_S_COFF_SYMBOL_RVA subsection)" ), |
| 152 | clEnumValN(ModuleSubsection::Unknown, "unknown" , |
| 153 | "Any subsection not covered by another option" ), |
| 154 | clEnumValN(ModuleSubsection::All, "all" , "All known subsections" )); |
| 155 | |
| 156 | namespace diadump { |
| 157 | static cl::list<std::string> InputFilenames(cl::Positional, |
| 158 | cl::desc("<input PDB files>" ), |
| 159 | cl::OneOrMore, |
| 160 | cl::sub(DiaDumpSubcommand)); |
| 161 | |
| 162 | cl::opt<bool> Native("native" , cl::desc("Use native PDB reader instead of DIA" ), |
| 163 | cl::sub(DiaDumpSubcommand)); |
| 164 | |
| 165 | static cl::opt<bool> |
| 166 | ShowClassHierarchy("hierarchy" , cl::desc("Show lexical and class parents" ), |
| 167 | cl::sub(DiaDumpSubcommand)); |
| 168 | static cl::opt<bool> NoSymIndexIds( |
| 169 | "no-ids" , |
| 170 | cl::desc("Don't show any SymIndexId fields (overrides -hierarchy)" ), |
| 171 | cl::sub(DiaDumpSubcommand)); |
| 172 | |
| 173 | static cl::opt<bool> |
| 174 | Recurse("recurse" , |
| 175 | cl::desc("When dumping a SymIndexId, dump the full details of the " |
| 176 | "corresponding record" ), |
| 177 | cl::sub(DiaDumpSubcommand)); |
| 178 | |
| 179 | static cl::opt<bool> Enums("enums" , cl::desc("Dump enum types" ), |
| 180 | cl::sub(DiaDumpSubcommand)); |
| 181 | static cl::opt<bool> Pointers("pointers" , cl::desc("Dump enum types" ), |
| 182 | cl::sub(DiaDumpSubcommand)); |
| 183 | static cl::opt<bool> UDTs("udts" , cl::desc("Dump udt types" ), |
| 184 | cl::sub(DiaDumpSubcommand)); |
| 185 | static cl::opt<bool> Compilands("compilands" , |
| 186 | cl::desc("Dump compiland information" ), |
| 187 | cl::sub(DiaDumpSubcommand)); |
| 188 | static cl::opt<bool> Funcsigs("funcsigs" , |
| 189 | cl::desc("Dump function signature information" ), |
| 190 | cl::sub(DiaDumpSubcommand)); |
| 191 | static cl::opt<bool> Arrays("arrays" , cl::desc("Dump array types" ), |
| 192 | cl::sub(DiaDumpSubcommand)); |
| 193 | static cl::opt<bool> VTShapes("vtshapes" , cl::desc("Dump virtual table shapes" ), |
| 194 | cl::sub(DiaDumpSubcommand)); |
| 195 | static cl::opt<bool> Typedefs("typedefs" , cl::desc("Dump typedefs" ), |
| 196 | cl::sub(DiaDumpSubcommand)); |
| 197 | } // namespace diadump |
| 198 | |
| 199 | FilterOptions Filters; |
| 200 | |
| 201 | namespace pretty { |
| 202 | static cl::list<std::string> InputFilenames(cl::Positional, |
| 203 | cl::desc("<input PDB files>" ), |
| 204 | cl::OneOrMore, |
| 205 | cl::sub(PrettySubcommand)); |
| 206 | |
| 207 | cl::opt<bool> InjectedSources("injected-sources" , |
| 208 | cl::desc("Display injected sources" ), |
| 209 | cl::cat(OtherOptions), cl::sub(PrettySubcommand)); |
| 210 | cl::opt<bool> ShowInjectedSourceContent( |
| 211 | "injected-source-content" , |
| 212 | cl::desc("When displaying an injected source, display the file content" ), |
| 213 | cl::cat(OtherOptions), cl::sub(PrettySubcommand)); |
| 214 | |
| 215 | cl::list<std::string> WithName( |
| 216 | "with-name" , |
| 217 | cl::desc("Display any symbol or type with the specified exact name" ), |
| 218 | cl::cat(TypeCategory), cl::sub(PrettySubcommand)); |
| 219 | |
| 220 | cl::opt<bool> Compilands("compilands" , cl::desc("Display compilands" ), |
| 221 | cl::cat(TypeCategory), cl::sub(PrettySubcommand)); |
| 222 | cl::opt<bool> Symbols("module-syms" , |
| 223 | cl::desc("Display symbols for each compiland" ), |
| 224 | cl::cat(TypeCategory), cl::sub(PrettySubcommand)); |
| 225 | cl::opt<bool> Globals("globals" , cl::desc("Dump global symbols" ), |
| 226 | cl::cat(TypeCategory), cl::sub(PrettySubcommand)); |
| 227 | cl::opt<bool> Externals("externals" , cl::desc("Dump external symbols" ), |
| 228 | cl::cat(TypeCategory), cl::sub(PrettySubcommand)); |
| 229 | static cl::list<SymLevel> SymTypes( |
| 230 | "sym-types" , cl::desc("Type of symbols to dump (default all)" ), |
| 231 | cl::cat(TypeCategory), cl::sub(PrettySubcommand), |
| 232 | cl::values( |
| 233 | clEnumValN(SymLevel::Thunks, "thunks" , "Display thunk symbols" ), |
| 234 | clEnumValN(SymLevel::Data, "data" , "Display data symbols" ), |
| 235 | clEnumValN(SymLevel::Functions, "funcs" , "Display function symbols" ), |
| 236 | clEnumValN(SymLevel::All, "all" , "Display all symbols (default)" ))); |
| 237 | |
| 238 | cl::opt<bool> |
| 239 | Types("types" , |
| 240 | cl::desc("Display all types (implies -classes, -enums, -typedefs)" ), |
| 241 | cl::cat(TypeCategory), cl::sub(PrettySubcommand)); |
| 242 | cl::opt<bool> Classes("classes" , cl::desc("Display class types" ), |
| 243 | cl::cat(TypeCategory), cl::sub(PrettySubcommand)); |
| 244 | cl::opt<bool> Enums("enums" , cl::desc("Display enum types" ), |
| 245 | cl::cat(TypeCategory), cl::sub(PrettySubcommand)); |
| 246 | cl::opt<bool> Typedefs("typedefs" , cl::desc("Display typedef types" ), |
| 247 | cl::cat(TypeCategory), cl::sub(PrettySubcommand)); |
| 248 | cl::opt<bool> Funcsigs("funcsigs" , cl::desc("Display function signatures" ), |
| 249 | cl::cat(TypeCategory), cl::sub(PrettySubcommand)); |
| 250 | cl::opt<bool> Pointers("pointers" , cl::desc("Display pointer types" ), |
| 251 | cl::cat(TypeCategory), cl::sub(PrettySubcommand)); |
| 252 | cl::opt<bool> Arrays("arrays" , cl::desc("Display arrays" ), |
| 253 | cl::cat(TypeCategory), cl::sub(PrettySubcommand)); |
| 254 | cl::opt<bool> VTShapes("vtshapes" , cl::desc("Display vftable shapes" ), |
| 255 | cl::cat(TypeCategory), cl::sub(PrettySubcommand)); |
| 256 | |
| 257 | cl::opt<SymbolSortMode> SymbolOrder( |
| 258 | "symbol-order" , cl::desc("symbol sort order" ), |
| 259 | cl::init(Val: SymbolSortMode::None), |
| 260 | cl::values(clEnumValN(SymbolSortMode::None, "none" , |
| 261 | "Undefined / no particular sort order" ), |
| 262 | clEnumValN(SymbolSortMode::Name, "name" , "Sort symbols by name" ), |
| 263 | clEnumValN(SymbolSortMode::Size, "size" , |
| 264 | "Sort symbols by size" )), |
| 265 | cl::cat(TypeCategory), cl::sub(PrettySubcommand)); |
| 266 | |
| 267 | cl::opt<ClassSortMode> ClassOrder( |
| 268 | "class-order" , cl::desc("Class sort order" ), cl::init(Val: ClassSortMode::None), |
| 269 | cl::values( |
| 270 | clEnumValN(ClassSortMode::None, "none" , |
| 271 | "Undefined / no particular sort order" ), |
| 272 | clEnumValN(ClassSortMode::Name, "name" , "Sort classes by name" ), |
| 273 | clEnumValN(ClassSortMode::Size, "size" , "Sort classes by size" ), |
| 274 | clEnumValN(ClassSortMode::Padding, "padding" , |
| 275 | "Sort classes by amount of padding" ), |
| 276 | clEnumValN(ClassSortMode::PaddingPct, "padding-pct" , |
| 277 | "Sort classes by percentage of space consumed by padding" ), |
| 278 | clEnumValN(ClassSortMode::PaddingImmediate, "padding-imm" , |
| 279 | "Sort classes by amount of immediate padding" ), |
| 280 | clEnumValN(ClassSortMode::PaddingPctImmediate, "padding-pct-imm" , |
| 281 | "Sort classes by percentage of space consumed by immediate " |
| 282 | "padding" )), |
| 283 | cl::cat(TypeCategory), cl::sub(PrettySubcommand)); |
| 284 | |
| 285 | cl::opt<ClassDefinitionFormat> ClassFormat( |
| 286 | "class-definitions" , cl::desc("Class definition format" ), |
| 287 | cl::init(Val: ClassDefinitionFormat::All), |
| 288 | cl::values( |
| 289 | clEnumValN(ClassDefinitionFormat::All, "all" , |
| 290 | "Display all class members including data, constants, " |
| 291 | "typedefs, functions, etc" ), |
| 292 | clEnumValN(ClassDefinitionFormat::Layout, "layout" , |
| 293 | "Only display members that contribute to class size." ), |
| 294 | clEnumValN(ClassDefinitionFormat::None, "none" , |
| 295 | "Don't display class definitions" )), |
| 296 | cl::cat(TypeCategory), cl::sub(PrettySubcommand)); |
| 297 | cl::opt<uint32_t> ClassRecursionDepth( |
| 298 | "class-recurse-depth" , cl::desc("Class recursion depth (0=no limit)" ), |
| 299 | cl::init(Val: 0), cl::cat(TypeCategory), cl::sub(PrettySubcommand)); |
| 300 | |
| 301 | cl::opt<bool> Lines("lines" , cl::desc("Line tables" ), cl::cat(TypeCategory), |
| 302 | cl::sub(PrettySubcommand)); |
| 303 | cl::opt<bool> |
| 304 | All("all" , cl::desc("Implies all other options in 'Symbol Types' category" ), |
| 305 | cl::cat(TypeCategory), cl::sub(PrettySubcommand)); |
| 306 | |
| 307 | cl::opt<uint64_t> LoadAddress( |
| 308 | "load-address" , |
| 309 | cl::desc("Assume the module is loaded at the specified address" ), |
| 310 | cl::cat(OtherOptions), cl::sub(PrettySubcommand)); |
| 311 | cl::opt<bool> Native("native" , cl::desc("Use native PDB reader instead of DIA" ), |
| 312 | cl::cat(OtherOptions), cl::sub(PrettySubcommand)); |
| 313 | cl::opt<cl::boolOrDefault> |
| 314 | ColorOutput("color-output" , |
| 315 | cl::desc("Override use of color (default = isatty)" ), |
| 316 | cl::cat(OtherOptions), cl::sub(PrettySubcommand)); |
| 317 | cl::list<std::string> |
| 318 | ExcludeTypes("exclude-types" , |
| 319 | cl::desc("Exclude types by regular expression" ), |
| 320 | cl::cat(FilterCategory), cl::sub(PrettySubcommand)); |
| 321 | cl::list<std::string> |
| 322 | ExcludeSymbols("exclude-symbols" , |
| 323 | cl::desc("Exclude symbols by regular expression" ), |
| 324 | cl::cat(FilterCategory), cl::sub(PrettySubcommand)); |
| 325 | cl::list<std::string> |
| 326 | ExcludeCompilands("exclude-compilands" , |
| 327 | cl::desc("Exclude compilands by regular expression" ), |
| 328 | cl::cat(FilterCategory), cl::sub(PrettySubcommand)); |
| 329 | |
| 330 | cl::list<std::string> IncludeTypes( |
| 331 | "include-types" , |
| 332 | cl::desc("Include only types which match a regular expression" ), |
| 333 | cl::cat(FilterCategory), cl::sub(PrettySubcommand)); |
| 334 | cl::list<std::string> IncludeSymbols( |
| 335 | "include-symbols" , |
| 336 | cl::desc("Include only symbols which match a regular expression" ), |
| 337 | cl::cat(FilterCategory), cl::sub(PrettySubcommand)); |
| 338 | cl::list<std::string> IncludeCompilands( |
| 339 | "include-compilands" , |
| 340 | cl::desc("Include only compilands those which match a regular expression" ), |
| 341 | cl::cat(FilterCategory), cl::sub(PrettySubcommand)); |
| 342 | cl::opt<uint32_t> SizeThreshold( |
| 343 | "min-type-size" , cl::desc("Displays only those types which are greater " |
| 344 | "than or equal to the specified size." ), |
| 345 | cl::init(Val: 0), cl::cat(FilterCategory), cl::sub(PrettySubcommand)); |
| 346 | cl::opt<uint32_t> PaddingThreshold( |
| 347 | "min-class-padding" , cl::desc("Displays only those classes which have at " |
| 348 | "least the specified amount of padding." ), |
| 349 | cl::init(Val: 0), cl::cat(FilterCategory), cl::sub(PrettySubcommand)); |
| 350 | cl::opt<uint32_t> ImmediatePaddingThreshold( |
| 351 | "min-class-padding-imm" , |
| 352 | cl::desc("Displays only those classes which have at least the specified " |
| 353 | "amount of immediate padding, ignoring padding internal to bases " |
| 354 | "and aggregates." ), |
| 355 | cl::init(Val: 0), cl::cat(FilterCategory), cl::sub(PrettySubcommand)); |
| 356 | |
| 357 | cl::opt<bool> ExcludeCompilerGenerated( |
| 358 | "no-compiler-generated" , |
| 359 | cl::desc("Don't show compiler generated types and symbols" ), |
| 360 | cl::cat(FilterCategory), cl::sub(PrettySubcommand)); |
| 361 | cl::opt<bool> |
| 362 | ExcludeSystemLibraries("no-system-libs" , |
| 363 | cl::desc("Don't show symbols from system libraries" ), |
| 364 | cl::cat(FilterCategory), cl::sub(PrettySubcommand)); |
| 365 | |
| 366 | cl::opt<bool> NoEnumDefs("no-enum-definitions" , |
| 367 | cl::desc("Don't display full enum definitions" ), |
| 368 | cl::cat(FilterCategory), cl::sub(PrettySubcommand)); |
| 369 | } |
| 370 | |
| 371 | static cl::OptionCategory FileOptions("Module & File Options" ); |
| 372 | |
| 373 | namespace bytes { |
| 374 | static cl::OptionCategory MsfBytes("MSF File Options" ); |
| 375 | static cl::OptionCategory DbiBytes("Dbi Stream Options" ); |
| 376 | static cl::OptionCategory PdbBytes("PDB Stream Options" ); |
| 377 | static cl::OptionCategory Types("Type Options" ); |
| 378 | static cl::OptionCategory ModuleCategory("Module Options" ); |
| 379 | |
| 380 | std::optional<NumberRange> DumpBlockRange; |
| 381 | std::optional<NumberRange> DumpByteRange; |
| 382 | |
| 383 | cl::opt<std::string> DumpBlockRangeOpt( |
| 384 | "block-range" , cl::value_desc("start[-end]" ), |
| 385 | cl::desc("Dump binary data from specified range of blocks." ), |
| 386 | cl::sub(BytesSubcommand), cl::cat(MsfBytes)); |
| 387 | |
| 388 | cl::opt<std::string> |
| 389 | DumpByteRangeOpt("byte-range" , cl::value_desc("start[-end]" ), |
| 390 | cl::desc("Dump binary data from specified range of bytes" ), |
| 391 | cl::sub(BytesSubcommand), cl::cat(MsfBytes)); |
| 392 | |
| 393 | cl::list<std::string> |
| 394 | DumpStreamData("stream-data" , cl::CommaSeparated, |
| 395 | cl::desc("Dump binary data from specified streams. Format " |
| 396 | "is SN[:Start][@Size]" ), |
| 397 | cl::sub(BytesSubcommand), cl::cat(MsfBytes)); |
| 398 | |
| 399 | cl::opt<bool> NameMap("name-map" , cl::desc("Dump bytes of PDB Name Map" ), |
| 400 | cl::sub(BytesSubcommand), cl::cat(PdbBytes)); |
| 401 | cl::opt<bool> Fpm("fpm" , cl::desc("Dump free page map" ), |
| 402 | cl::sub(BytesSubcommand), cl::cat(MsfBytes)); |
| 403 | |
| 404 | cl::opt<bool> SectionContributions("sc" , cl::desc("Dump section contributions" ), |
| 405 | cl::sub(BytesSubcommand), cl::cat(DbiBytes)); |
| 406 | cl::opt<bool> SectionMap("sm" , cl::desc("Dump section map" ), |
| 407 | cl::sub(BytesSubcommand), cl::cat(DbiBytes)); |
| 408 | cl::opt<bool> ModuleInfos("modi" , cl::desc("Dump module info" ), |
| 409 | cl::sub(BytesSubcommand), cl::cat(DbiBytes)); |
| 410 | cl::opt<bool> FileInfo("files" , cl::desc("Dump source file info" ), |
| 411 | cl::sub(BytesSubcommand), cl::cat(DbiBytes)); |
| 412 | cl::opt<bool> TypeServerMap("type-server" , cl::desc("Dump type server map" ), |
| 413 | cl::sub(BytesSubcommand), cl::cat(DbiBytes)); |
| 414 | cl::opt<bool> ECData("ec" , cl::desc("Dump edit and continue map" ), |
| 415 | cl::sub(BytesSubcommand), cl::cat(DbiBytes)); |
| 416 | |
| 417 | cl::list<uint32_t> TypeIndex( |
| 418 | "type" , cl::desc("Dump the type record with the given type index" ), |
| 419 | cl::CommaSeparated, cl::sub(BytesSubcommand), cl::cat(TypeCategory)); |
| 420 | cl::list<uint32_t> |
| 421 | IdIndex("id" , cl::desc("Dump the id record with the given type index" ), |
| 422 | cl::CommaSeparated, cl::sub(BytesSubcommand), |
| 423 | cl::cat(TypeCategory)); |
| 424 | |
| 425 | cl::opt<uint32_t> ModuleIndex( |
| 426 | "mod" , |
| 427 | cl::desc( |
| 428 | "Limit options in the Modules category to the specified module index" ), |
| 429 | cl::Optional, cl::sub(BytesSubcommand), cl::cat(ModuleCategory)); |
| 430 | cl::opt<bool> ModuleSyms("syms" , cl::desc("Dump symbol record substream" ), |
| 431 | cl::sub(BytesSubcommand), cl::cat(ModuleCategory)); |
| 432 | cl::opt<bool> ModuleC11("c11-chunks" , cl::Hidden, |
| 433 | cl::desc("Dump C11 CodeView debug chunks" ), |
| 434 | cl::sub(BytesSubcommand), cl::cat(ModuleCategory)); |
| 435 | cl::opt<bool> ModuleC13("chunks" , |
| 436 | cl::desc("Dump C13 CodeView debug chunk subsection" ), |
| 437 | cl::sub(BytesSubcommand), cl::cat(ModuleCategory)); |
| 438 | cl::opt<bool> SplitChunks( |
| 439 | "split-chunks" , |
| 440 | cl::desc( |
| 441 | "When dumping debug chunks, show a different section for each chunk" ), |
| 442 | cl::sub(BytesSubcommand), cl::cat(ModuleCategory)); |
| 443 | static cl::list<std::string> InputFilenames(cl::Positional, |
| 444 | cl::desc("<input PDB files>" ), |
| 445 | cl::OneOrMore, |
| 446 | cl::sub(BytesSubcommand)); |
| 447 | |
| 448 | } // namespace bytes |
| 449 | |
| 450 | namespace dump { |
| 451 | |
| 452 | static cl::OptionCategory MsfOptions("MSF Container Options" ); |
| 453 | static cl::OptionCategory TypeOptions("Type Record Options" ); |
| 454 | static cl::OptionCategory SymbolOptions("Symbol Options" ); |
| 455 | static cl::OptionCategory MiscOptions("Miscellaneous Options" ); |
| 456 | |
| 457 | // MSF OPTIONS |
| 458 | cl::opt<bool> DumpSummary("summary" , cl::desc("dump file summary" ), |
| 459 | cl::cat(MsfOptions), cl::sub(DumpSubcommand)); |
| 460 | cl::opt<bool> DumpStreams("streams" , |
| 461 | cl::desc("dump summary of the PDB streams" ), |
| 462 | cl::cat(MsfOptions), cl::sub(DumpSubcommand)); |
| 463 | cl::opt<bool> DumpStreamBlocks( |
| 464 | "stream-blocks" , |
| 465 | cl::desc("Add block information to the output of -streams" ), |
| 466 | cl::cat(MsfOptions), cl::sub(DumpSubcommand)); |
| 467 | cl::opt<bool> DumpSymbolStats( |
| 468 | "sym-stats" , |
| 469 | cl::desc("Dump a detailed breakdown of symbol usage/size for each module" ), |
| 470 | cl::cat(MsfOptions), cl::sub(DumpSubcommand)); |
| 471 | cl::opt<bool> DumpTypeStats( |
| 472 | "type-stats" , |
| 473 | cl::desc("Dump a detailed breakdown of type usage/size" ), |
| 474 | cl::cat(MsfOptions), cl::sub(DumpSubcommand)); |
| 475 | cl::opt<bool> DumpIDStats( |
| 476 | "id-stats" , |
| 477 | cl::desc("Dump a detailed breakdown of IPI types usage/size" ), |
| 478 | cl::cat(MsfOptions), cl::sub(DumpSubcommand)); |
| 479 | cl::opt<bool> DumpUdtStats( |
| 480 | "udt-stats" , |
| 481 | cl::desc("Dump a detailed breakdown of S_UDT record usage / stats" ), |
| 482 | cl::cat(MsfOptions), cl::sub(DumpSubcommand)); |
| 483 | |
| 484 | // TYPE OPTIONS |
| 485 | cl::opt<bool> DumpTypes("types" , |
| 486 | cl::desc("dump CodeView type records from TPI stream" ), |
| 487 | cl::cat(TypeOptions), cl::sub(DumpSubcommand)); |
| 488 | cl::opt<bool> DumpTypeData( |
| 489 | "type-data" , |
| 490 | cl::desc("dump CodeView type record raw bytes from TPI stream" ), |
| 491 | cl::cat(TypeOptions), cl::sub(DumpSubcommand)); |
| 492 | cl::opt<bool> |
| 493 | DumpTypeRefStats("type-ref-stats" , |
| 494 | cl::desc("dump statistics on the number and size of types " |
| 495 | "transitively referenced by symbol records" ), |
| 496 | cl::cat(TypeOptions), cl::sub(DumpSubcommand)); |
| 497 | |
| 498 | cl::opt<bool> ("type-extras" , |
| 499 | cl::desc("dump type hashes and index offsets" ), |
| 500 | cl::cat(TypeOptions), cl::sub(DumpSubcommand)); |
| 501 | |
| 502 | cl::opt<bool> DontResolveForwardRefs( |
| 503 | "dont-resolve-forward-refs" , |
| 504 | cl::desc("When dumping type records for classes, unions, enums, and " |
| 505 | "structs, don't try to resolve forward references" ), |
| 506 | cl::cat(TypeOptions), cl::sub(DumpSubcommand)); |
| 507 | |
| 508 | cl::list<uint32_t> DumpTypeIndex( |
| 509 | "type-index" , cl::CommaSeparated, |
| 510 | cl::desc("only dump types with the specified hexadecimal type index" ), |
| 511 | cl::cat(TypeOptions), cl::sub(DumpSubcommand)); |
| 512 | |
| 513 | cl::opt<bool> DumpIds("ids" , |
| 514 | cl::desc("dump CodeView type records from IPI stream" ), |
| 515 | cl::cat(TypeOptions), cl::sub(DumpSubcommand)); |
| 516 | cl::opt<bool> |
| 517 | DumpIdData("id-data" , |
| 518 | cl::desc("dump CodeView type record raw bytes from IPI stream" ), |
| 519 | cl::cat(TypeOptions), cl::sub(DumpSubcommand)); |
| 520 | |
| 521 | cl::opt<bool> ("id-extras" , |
| 522 | cl::desc("dump id hashes and index offsets" ), |
| 523 | cl::cat(TypeOptions), cl::sub(DumpSubcommand)); |
| 524 | cl::list<uint32_t> DumpIdIndex( |
| 525 | "id-index" , cl::CommaSeparated, |
| 526 | cl::desc("only dump ids with the specified hexadecimal type index" ), |
| 527 | cl::cat(TypeOptions), cl::sub(DumpSubcommand)); |
| 528 | |
| 529 | cl::opt<bool> DumpTypeDependents( |
| 530 | "dependents" , |
| 531 | cl::desc("In conjunection with -type-index and -id-index, dumps the entire " |
| 532 | "dependency graph for the specified index instead of " |
| 533 | "just the single record with the specified index" ), |
| 534 | cl::cat(TypeOptions), cl::sub(DumpSubcommand)); |
| 535 | |
| 536 | // SYMBOL OPTIONS |
| 537 | cl::opt<bool> DumpGlobals("globals" , cl::desc("dump Globals symbol records" ), |
| 538 | cl::cat(SymbolOptions), cl::sub(DumpSubcommand)); |
| 539 | cl::opt<bool> ("global-extras" , cl::desc("dump Globals hashes" ), |
| 540 | cl::cat(SymbolOptions), cl::sub(DumpSubcommand)); |
| 541 | cl::list<std::string> DumpGlobalNames( |
| 542 | "global-name" , |
| 543 | cl::desc( |
| 544 | "With -globals, only dump globals whose name matches the given value" ), |
| 545 | cl::cat(SymbolOptions), cl::sub(DumpSubcommand)); |
| 546 | cl::opt<bool> DumpPublics("publics" , cl::desc("dump Publics stream data" ), |
| 547 | cl::cat(SymbolOptions), cl::sub(DumpSubcommand)); |
| 548 | cl::opt<bool> ("public-extras" , |
| 549 | cl::desc("dump Publics hashes and address maps" ), |
| 550 | cl::cat(SymbolOptions), cl::sub(DumpSubcommand)); |
| 551 | cl::opt<bool> |
| 552 | DumpGSIRecords("gsi-records" , |
| 553 | cl::desc("dump public / global common record stream" ), |
| 554 | cl::cat(SymbolOptions), cl::sub(DumpSubcommand)); |
| 555 | cl::opt<bool> DumpSymbols("symbols" , cl::desc("dump module symbols" ), |
| 556 | cl::cat(SymbolOptions), cl::sub(DumpSubcommand)); |
| 557 | |
| 558 | cl::opt<bool> |
| 559 | DumpSymRecordBytes("sym-data" , |
| 560 | cl::desc("dump CodeView symbol record raw bytes" ), |
| 561 | cl::cat(SymbolOptions), cl::sub(DumpSubcommand)); |
| 562 | |
| 563 | cl::opt<bool> DumpFpo("fpo" , cl::desc("dump FPO records" ), |
| 564 | cl::cat(SymbolOptions), cl::sub(DumpSubcommand)); |
| 565 | |
| 566 | cl::opt<uint32_t> DumpSymbolOffset( |
| 567 | "symbol-offset" , cl::Optional, |
| 568 | cl::desc("only dump symbol record with the specified symbol offset" ), |
| 569 | cl::cat(SymbolOptions), cl::sub(DumpSubcommand)); |
| 570 | cl::opt<bool> DumpParents("show-parents" , |
| 571 | cl::desc("dump the symbols record's all parents." ), |
| 572 | cl::cat(SymbolOptions), cl::sub(DumpSubcommand)); |
| 573 | cl::opt<uint32_t> |
| 574 | DumpParentDepth("parent-recurse-depth" , cl::Optional, cl::init(Val: -1U), |
| 575 | cl::desc("only recurse to a depth of N when displaying " |
| 576 | "parents of a symbol record." ), |
| 577 | cl::cat(SymbolOptions), cl::sub(DumpSubcommand)); |
| 578 | cl::opt<bool> DumpChildren("show-children" , |
| 579 | cl::desc("dump the symbols record's all children." ), |
| 580 | cl::cat(SymbolOptions), cl::sub(DumpSubcommand)); |
| 581 | cl::opt<uint32_t> |
| 582 | DumpChildrenDepth("children-recurse-depth" , cl::Optional, cl::init(Val: -1U), |
| 583 | cl::desc("only recurse to a depth of N when displaying " |
| 584 | "children of a symbol record." ), |
| 585 | cl::cat(SymbolOptions), cl::sub(DumpSubcommand)); |
| 586 | |
| 587 | // MODULE & FILE OPTIONS |
| 588 | cl::opt<bool> DumpModules("modules" , cl::desc("dump compiland information" ), |
| 589 | cl::cat(FileOptions), cl::sub(DumpSubcommand)); |
| 590 | cl::opt<bool> DumpModuleFiles( |
| 591 | "files" , |
| 592 | cl::desc("Dump the source files that contribute to each module's." ), |
| 593 | cl::cat(FileOptions), cl::sub(DumpSubcommand)); |
| 594 | cl::opt<bool> DumpLines( |
| 595 | "l" , |
| 596 | cl::desc("dump source file/line information (DEBUG_S_LINES subsection)" ), |
| 597 | cl::cat(FileOptions), cl::sub(DumpSubcommand)); |
| 598 | cl::opt<bool> DumpInlineeLines( |
| 599 | "il" , |
| 600 | cl::desc("dump inlinee line information (DEBUG_S_INLINEELINES subsection)" ), |
| 601 | cl::cat(FileOptions), cl::sub(DumpSubcommand)); |
| 602 | cl::opt<bool> DumpXmi( |
| 603 | "xmi" , |
| 604 | cl::desc( |
| 605 | "dump cross module imports (DEBUG_S_CROSSSCOPEIMPORTS subsection)" ), |
| 606 | cl::cat(FileOptions), cl::sub(DumpSubcommand)); |
| 607 | cl::opt<bool> DumpXme( |
| 608 | "xme" , |
| 609 | cl::desc( |
| 610 | "dump cross module exports (DEBUG_S_CROSSSCOPEEXPORTS subsection)" ), |
| 611 | cl::cat(FileOptions), cl::sub(DumpSubcommand)); |
| 612 | cl::opt<uint32_t> DumpModi("modi" , cl::Optional, |
| 613 | cl::desc("For all options that iterate over " |
| 614 | "modules, limit to the specified module" ), |
| 615 | cl::cat(FileOptions), cl::sub(DumpSubcommand)); |
| 616 | cl::opt<bool> JustMyCode("jmc" , cl::Optional, |
| 617 | cl::desc("For all options that iterate over modules, " |
| 618 | "ignore modules from system libraries" ), |
| 619 | cl::cat(FileOptions), cl::sub(DumpSubcommand)); |
| 620 | |
| 621 | // MISCELLANEOUS OPTIONS |
| 622 | cl::opt<bool> DumpNamedStreams("named-streams" , |
| 623 | cl::desc("dump PDB named stream table" ), |
| 624 | cl::cat(MiscOptions), cl::sub(DumpSubcommand)); |
| 625 | |
| 626 | cl::opt<bool> DumpStringTable("string-table" , cl::desc("dump PDB String Table" ), |
| 627 | cl::cat(MiscOptions), cl::sub(DumpSubcommand)); |
| 628 | cl::opt<bool> DumpStringTableDetails("string-table-details" , |
| 629 | cl::desc("dump PDB String Table Details" ), |
| 630 | cl::cat(MiscOptions), |
| 631 | cl::sub(DumpSubcommand)); |
| 632 | |
| 633 | cl::opt<bool> DumpSectionContribs("section-contribs" , |
| 634 | cl::desc("dump section contributions" ), |
| 635 | cl::cat(MiscOptions), |
| 636 | cl::sub(DumpSubcommand)); |
| 637 | cl::opt<bool> DumpSectionMap("section-map" , cl::desc("dump section map" ), |
| 638 | cl::cat(MiscOptions), cl::sub(DumpSubcommand)); |
| 639 | cl::opt<bool> ("section-headers" , |
| 640 | cl::desc("Dump image section headers" ), |
| 641 | cl::cat(MiscOptions), cl::sub(DumpSubcommand)); |
| 642 | |
| 643 | cl::opt<bool> RawAll("all" , cl::desc("Implies most other options." ), |
| 644 | cl::cat(MiscOptions), cl::sub(DumpSubcommand)); |
| 645 | |
| 646 | static cl::list<std::string> InputFilenames(cl::Positional, |
| 647 | cl::desc("<input PDB files>" ), |
| 648 | cl::OneOrMore, |
| 649 | cl::sub(DumpSubcommand)); |
| 650 | } |
| 651 | |
| 652 | namespace yaml2pdb { |
| 653 | cl::opt<std::string> |
| 654 | YamlPdbOutputFile("pdb" , cl::desc("the name of the PDB file to write" ), |
| 655 | cl::sub(YamlToPdbSubcommand)); |
| 656 | |
| 657 | cl::opt<std::string> InputFilename(cl::Positional, |
| 658 | cl::desc("<input YAML file>" ), cl::Required, |
| 659 | cl::sub(YamlToPdbSubcommand)); |
| 660 | } |
| 661 | |
| 662 | namespace pdb2yaml { |
| 663 | cl::opt<bool> All("all" , |
| 664 | cl::desc("Dump everything we know how to dump." ), |
| 665 | cl::sub(PdbToYamlSubcommand), cl::init(Val: false)); |
| 666 | cl::opt<bool> ("no-file-headers" , |
| 667 | cl::desc("Do not dump MSF file headers" ), |
| 668 | cl::sub(PdbToYamlSubcommand), cl::init(Val: false)); |
| 669 | cl::opt<bool> Minimal("minimal" , |
| 670 | cl::desc("Don't write fields with default values" ), |
| 671 | cl::sub(PdbToYamlSubcommand), cl::init(Val: false)); |
| 672 | |
| 673 | cl::opt<bool> StreamMetadata( |
| 674 | "stream-metadata" , |
| 675 | cl::desc("Dump the number of streams and each stream's size" ), |
| 676 | cl::sub(PdbToYamlSubcommand), cl::init(Val: false)); |
| 677 | cl::opt<bool> StreamDirectory( |
| 678 | "stream-directory" , |
| 679 | cl::desc("Dump each stream's block map (implies -stream-metadata)" ), |
| 680 | cl::sub(PdbToYamlSubcommand), cl::init(Val: false)); |
| 681 | cl::opt<bool> PdbStream("pdb-stream" , |
| 682 | cl::desc("Dump the PDB Stream (Stream 1)" ), |
| 683 | cl::sub(PdbToYamlSubcommand), cl::init(Val: false)); |
| 684 | |
| 685 | cl::opt<bool> StringTable("string-table" , cl::desc("Dump the PDB String Table" ), |
| 686 | cl::sub(PdbToYamlSubcommand), cl::init(Val: false)); |
| 687 | |
| 688 | cl::opt<bool> DbiStream("dbi-stream" , |
| 689 | cl::desc("Dump the DBI Stream Headers (Stream 2)" ), |
| 690 | cl::sub(PdbToYamlSubcommand), cl::init(Val: false)); |
| 691 | |
| 692 | cl::opt<bool> TpiStream("tpi-stream" , |
| 693 | cl::desc("Dump the TPI Stream (Stream 3)" ), |
| 694 | cl::sub(PdbToYamlSubcommand), cl::init(Val: false)); |
| 695 | |
| 696 | cl::opt<bool> IpiStream("ipi-stream" , |
| 697 | cl::desc("Dump the IPI Stream (Stream 5)" ), |
| 698 | cl::sub(PdbToYamlSubcommand), cl::init(Val: false)); |
| 699 | |
| 700 | cl::opt<bool> PublicsStream("publics-stream" , |
| 701 | cl::desc("Dump the Publics Stream" ), |
| 702 | cl::sub(PdbToYamlSubcommand), cl::init(Val: false)); |
| 703 | |
| 704 | // MODULE & FILE OPTIONS |
| 705 | cl::opt<bool> DumpModules("modules" , cl::desc("dump compiland information" ), |
| 706 | cl::cat(FileOptions), cl::sub(PdbToYamlSubcommand)); |
| 707 | cl::opt<bool> DumpModuleFiles("module-files" , cl::desc("dump file information" ), |
| 708 | cl::cat(FileOptions), |
| 709 | cl::sub(PdbToYamlSubcommand)); |
| 710 | cl::list<ModuleSubsection> DumpModuleSubsections( |
| 711 | "subsections" , cl::CommaSeparated, |
| 712 | cl::desc("dump subsections from each module's debug stream" ), ChunkValues, |
| 713 | cl::cat(FileOptions), cl::sub(PdbToYamlSubcommand)); |
| 714 | cl::opt<bool> DumpModuleSyms("module-syms" , cl::desc("dump module symbols" ), |
| 715 | cl::cat(FileOptions), |
| 716 | cl::sub(PdbToYamlSubcommand)); |
| 717 | |
| 718 | cl::list<std::string> InputFilename(cl::Positional, |
| 719 | cl::desc("<input PDB file>" ), cl::Required, |
| 720 | cl::sub(PdbToYamlSubcommand)); |
| 721 | } // namespace pdb2yaml |
| 722 | |
| 723 | namespace merge { |
| 724 | static cl::list<std::string> InputFilenames(cl::Positional, |
| 725 | cl::desc("<input PDB files>" ), |
| 726 | cl::OneOrMore, |
| 727 | cl::sub(MergeSubcommand)); |
| 728 | cl::opt<std::string> |
| 729 | PdbOutputFile("pdb" , cl::desc("the name of the PDB file to write" ), |
| 730 | cl::sub(MergeSubcommand)); |
| 731 | } |
| 732 | |
| 733 | namespace explain { |
| 734 | cl::list<std::string> InputFilename(cl::Positional, |
| 735 | cl::desc("<input PDB file>" ), cl::Required, |
| 736 | cl::sub(ExplainSubcommand)); |
| 737 | |
| 738 | cl::list<uint64_t> Offsets("offset" , cl::desc("The file offset to explain" ), |
| 739 | cl::sub(ExplainSubcommand), cl::OneOrMore); |
| 740 | |
| 741 | cl::opt<InputFileType> InputType( |
| 742 | "input-type" , cl::desc("Specify how to interpret the input file" ), |
| 743 | cl::init(Val: InputFileType::PDBFile), cl::Optional, cl::sub(ExplainSubcommand), |
| 744 | cl::values(clEnumValN(InputFileType::PDBFile, "pdb-file" , |
| 745 | "Treat input as a PDB file (default)" ), |
| 746 | clEnumValN(InputFileType::PDBStream, "pdb-stream" , |
| 747 | "Treat input as raw contents of PDB stream" ), |
| 748 | clEnumValN(InputFileType::DBIStream, "dbi-stream" , |
| 749 | "Treat input as raw contents of DBI stream" ), |
| 750 | clEnumValN(InputFileType::Names, "names-stream" , |
| 751 | "Treat input as raw contents of /names named stream" ), |
| 752 | clEnumValN(InputFileType::ModuleStream, "mod-stream" , |
| 753 | "Treat input as raw contents of a module stream" ))); |
| 754 | } // namespace explain |
| 755 | |
| 756 | namespace exportstream { |
| 757 | static cl::list<std::string> InputFilename(cl::Positional, |
| 758 | cl::desc("<input PDB file>" ), |
| 759 | cl::Required, |
| 760 | cl::sub(ExportSubcommand)); |
| 761 | cl::opt<std::string> OutputFile("out" , |
| 762 | cl::desc("The file to write the stream to" ), |
| 763 | cl::Required, cl::sub(ExportSubcommand)); |
| 764 | cl::opt<std::string> |
| 765 | Stream("stream" , cl::Required, |
| 766 | cl::desc("The index or name of the stream whose contents to export" ), |
| 767 | cl::sub(ExportSubcommand)); |
| 768 | cl::opt<bool> ForceName("name" , |
| 769 | cl::desc("Force the interpretation of -stream as a " |
| 770 | "string, even if it is a valid integer" ), |
| 771 | cl::sub(ExportSubcommand), cl::Optional, |
| 772 | cl::init(Val: false)); |
| 773 | } // namespace exportstream |
| 774 | } |
| 775 | |
| 776 | static ExitOnError ExitOnErr; |
| 777 | |
| 778 | static void yamlToPdb(StringRef Path) { |
| 779 | BumpPtrAllocator Allocator; |
| 780 | ErrorOr<std::unique_ptr<MemoryBuffer>> ErrorOrBuffer = |
| 781 | MemoryBuffer::getFileOrSTDIN(Filename: Path, /*IsText=*/false, |
| 782 | /*RequiresNullTerminator=*/false); |
| 783 | |
| 784 | if (ErrorOrBuffer.getError()) { |
| 785 | ExitOnErr(createFileError(F: Path, E: errorCodeToError(EC: ErrorOrBuffer.getError()))); |
| 786 | } |
| 787 | |
| 788 | std::unique_ptr<MemoryBuffer> &Buffer = ErrorOrBuffer.get(); |
| 789 | |
| 790 | llvm::yaml::Input In(Buffer->getBuffer()); |
| 791 | pdb::yaml::PdbObject YamlObj(Allocator); |
| 792 | In >> YamlObj; |
| 793 | |
| 794 | PDBFileBuilder Builder(Allocator); |
| 795 | |
| 796 | uint32_t BlockSize = 4096; |
| 797 | if (YamlObj.Headers) |
| 798 | BlockSize = YamlObj.Headers->SuperBlock.BlockSize; |
| 799 | ExitOnErr(Builder.initialize(BlockSize)); |
| 800 | // Add each of the reserved streams. We ignore stream metadata in the |
| 801 | // yaml, because we will reconstruct our own view of the streams. For |
| 802 | // example, the YAML may say that there were 20 streams in the original |
| 803 | // PDB, but maybe we only dump a subset of those 20 streams, so we will |
| 804 | // have fewer, and the ones we do have may end up with different indices |
| 805 | // than the ones in the original PDB. So we just start with a clean slate. |
| 806 | for (uint32_t I = 0; I < kSpecialStreamCount; ++I) |
| 807 | ExitOnErr(Builder.getMsfBuilder().addStream(Size: 0)); |
| 808 | |
| 809 | StringsAndChecksums Strings; |
| 810 | Strings.setStrings(std::make_shared<DebugStringTableSubsection>()); |
| 811 | |
| 812 | if (YamlObj.StringTable) { |
| 813 | for (auto S : *YamlObj.StringTable) |
| 814 | Strings.strings()->insert(S); |
| 815 | } |
| 816 | |
| 817 | pdb::yaml::PdbInfoStream DefaultInfoStream; |
| 818 | pdb::yaml::PdbDbiStream DefaultDbiStream; |
| 819 | pdb::yaml::PdbTpiStream DefaultTpiStream; |
| 820 | pdb::yaml::PdbTpiStream DefaultIpiStream; |
| 821 | |
| 822 | const auto &Info = YamlObj.PdbStream.value_or(u&: DefaultInfoStream); |
| 823 | |
| 824 | auto &InfoBuilder = Builder.getInfoBuilder(); |
| 825 | InfoBuilder.setAge(Info.Age); |
| 826 | InfoBuilder.setGuid(Info.Guid); |
| 827 | InfoBuilder.setSignature(Info.Signature); |
| 828 | InfoBuilder.setVersion(Info.Version); |
| 829 | for (auto F : Info.Features) |
| 830 | InfoBuilder.addFeature(Sig: F); |
| 831 | |
| 832 | const auto &Dbi = YamlObj.DbiStream.value_or(u&: DefaultDbiStream); |
| 833 | auto &DbiBuilder = Builder.getDbiBuilder(); |
| 834 | DbiBuilder.setAge(Dbi.Age); |
| 835 | DbiBuilder.setBuildNumber(Dbi.BuildNumber); |
| 836 | DbiBuilder.setFlags(Dbi.Flags); |
| 837 | DbiBuilder.setMachineType(Dbi.MachineType); |
| 838 | DbiBuilder.setPdbDllRbld(Dbi.PdbDllRbld); |
| 839 | DbiBuilder.setPdbDllVersion(Dbi.PdbDllVersion); |
| 840 | DbiBuilder.setVersionHeader(Dbi.VerHeader); |
| 841 | for (const auto &MI : Dbi.ModInfos) { |
| 842 | auto &ModiBuilder = ExitOnErr(DbiBuilder.addModuleInfo(ModuleName: MI.Mod)); |
| 843 | ModiBuilder.setObjFileName(MI.Obj); |
| 844 | |
| 845 | for (auto S : MI.SourceFiles) |
| 846 | ExitOnErr(DbiBuilder.addModuleSourceFile(Module&: ModiBuilder, File: S)); |
| 847 | if (MI.Modi) { |
| 848 | const auto &ModiStream = *MI.Modi; |
| 849 | for (const auto &Symbol : ModiStream.Symbols) { |
| 850 | ModiBuilder.addSymbol( |
| 851 | Symbol: Symbol.toCodeViewSymbol(Allocator, Container: CodeViewContainer::Pdb)); |
| 852 | } |
| 853 | } |
| 854 | |
| 855 | // Each module has its own checksum subsection, so scan for it every time. |
| 856 | Strings.setChecksums(nullptr); |
| 857 | CodeViewYAML::initializeStringsAndChecksums(Sections: MI.Subsections, SC&: Strings); |
| 858 | |
| 859 | auto CodeViewSubsections = ExitOnErr(CodeViewYAML::toCodeViewSubsectionList( |
| 860 | Allocator, Subsections: MI.Subsections, SC: Strings)); |
| 861 | for (auto &SS : CodeViewSubsections) { |
| 862 | ModiBuilder.addDebugSubsection(Subsection: SS); |
| 863 | } |
| 864 | } |
| 865 | |
| 866 | auto &TpiBuilder = Builder.getTpiBuilder(); |
| 867 | const auto &Tpi = YamlObj.TpiStream.value_or(u&: DefaultTpiStream); |
| 868 | TpiBuilder.setVersionHeader(Tpi.Version); |
| 869 | AppendingTypeTableBuilder TS(Allocator); |
| 870 | for (const auto &R : Tpi.Records) { |
| 871 | CVType Type = R.toCodeViewRecord(Serializer&: TS); |
| 872 | TpiBuilder.addTypeRecord(Type: Type.RecordData, Hash: std::nullopt); |
| 873 | } |
| 874 | |
| 875 | const auto &Ipi = YamlObj.IpiStream.value_or(u&: DefaultIpiStream); |
| 876 | auto &IpiBuilder = Builder.getIpiBuilder(); |
| 877 | IpiBuilder.setVersionHeader(Ipi.Version); |
| 878 | for (const auto &R : Ipi.Records) { |
| 879 | CVType Type = R.toCodeViewRecord(Serializer&: TS); |
| 880 | IpiBuilder.addTypeRecord(Type: Type.RecordData, Hash: std::nullopt); |
| 881 | } |
| 882 | |
| 883 | Builder.getStringTableBuilder().setStrings(*Strings.strings()); |
| 884 | |
| 885 | codeview::GUID IgnoredOutGuid; |
| 886 | ExitOnErr(Builder.commit(Filename: opts::yaml2pdb::YamlPdbOutputFile, Guid: &IgnoredOutGuid)); |
| 887 | } |
| 888 | |
| 889 | static PDBFile &loadPDB(StringRef Path, std::unique_ptr<IPDBSession> &Session) { |
| 890 | ExitOnErr(loadDataForPDB(Type: PDB_ReaderType::Native, Path, Session)); |
| 891 | |
| 892 | NativeSession *NS = static_cast<NativeSession *>(Session.get()); |
| 893 | return NS->getPDBFile(); |
| 894 | } |
| 895 | |
| 896 | static void pdb2Yaml(StringRef Path) { |
| 897 | std::unique_ptr<IPDBSession> Session; |
| 898 | auto &File = loadPDB(Path, Session); |
| 899 | |
| 900 | auto O = std::make_unique<YAMLOutputStyle>(args&: File); |
| 901 | |
| 902 | ExitOnErr(O->dump()); |
| 903 | } |
| 904 | |
| 905 | static void dumpRaw(StringRef Path) { |
| 906 | InputFile IF = ExitOnErr(InputFile::open(Path)); |
| 907 | |
| 908 | auto O = std::make_unique<DumpOutputStyle>(args&: IF); |
| 909 | ExitOnErr(O->dump()); |
| 910 | } |
| 911 | |
| 912 | static void dumpBytes(StringRef Path) { |
| 913 | std::unique_ptr<IPDBSession> Session; |
| 914 | auto &File = loadPDB(Path, Session); |
| 915 | |
| 916 | auto O = std::make_unique<BytesOutputStyle>(args&: File); |
| 917 | |
| 918 | ExitOnErr(O->dump()); |
| 919 | } |
| 920 | |
| 921 | bool opts::pretty::shouldDumpSymLevel(SymLevel Search) { |
| 922 | if (SymTypes.empty()) |
| 923 | return true; |
| 924 | if (llvm::is_contained(Range&: SymTypes, Element: Search)) |
| 925 | return true; |
| 926 | if (llvm::is_contained(Range&: SymTypes, Element: SymLevel::All)) |
| 927 | return true; |
| 928 | return false; |
| 929 | } |
| 930 | |
| 931 | uint32_t llvm::pdb::getTypeLength(const PDBSymbolData &Symbol) { |
| 932 | auto SymbolType = Symbol.getType(); |
| 933 | const IPDBRawSymbol &RawType = SymbolType->getRawSymbol(); |
| 934 | |
| 935 | return RawType.getLength(); |
| 936 | } |
| 937 | |
| 938 | bool opts::pretty::compareFunctionSymbols( |
| 939 | const std::unique_ptr<PDBSymbolFunc> &F1, |
| 940 | const std::unique_ptr<PDBSymbolFunc> &F2) { |
| 941 | assert(opts::pretty::SymbolOrder != opts::pretty::SymbolSortMode::None); |
| 942 | |
| 943 | if (opts::pretty::SymbolOrder == opts::pretty::SymbolSortMode::Name) |
| 944 | return F1->getName() < F2->getName(); |
| 945 | |
| 946 | // Note that we intentionally sort in descending order on length, since |
| 947 | // long functions are more interesting than short functions. |
| 948 | return F1->getLength() > F2->getLength(); |
| 949 | } |
| 950 | |
| 951 | bool opts::pretty::compareDataSymbols( |
| 952 | const std::unique_ptr<PDBSymbolData> &F1, |
| 953 | const std::unique_ptr<PDBSymbolData> &F2) { |
| 954 | assert(opts::pretty::SymbolOrder != opts::pretty::SymbolSortMode::None); |
| 955 | |
| 956 | if (opts::pretty::SymbolOrder == opts::pretty::SymbolSortMode::Name) |
| 957 | return F1->getName() < F2->getName(); |
| 958 | |
| 959 | // Note that we intentionally sort in descending order on length, since |
| 960 | // large types are more interesting than short ones. |
| 961 | return getTypeLength(Symbol: *F1) > getTypeLength(Symbol: *F2); |
| 962 | } |
| 963 | |
| 964 | static std::string stringOr(std::string Str, std::string IfEmpty) { |
| 965 | return (Str.empty()) ? IfEmpty : Str; |
| 966 | } |
| 967 | |
| 968 | static void dumpInjectedSources(LinePrinter &Printer, IPDBSession &Session) { |
| 969 | auto Sources = Session.getInjectedSources(); |
| 970 | if (!Sources || !Sources->getChildCount()) { |
| 971 | Printer.printLine(T: "There are no injected sources." ); |
| 972 | return; |
| 973 | } |
| 974 | |
| 975 | while (auto IS = Sources->getNext()) { |
| 976 | Printer.NewLine(); |
| 977 | std::string File = stringOr(Str: IS->getFileName(), IfEmpty: "<null>" ); |
| 978 | uint64_t Size = IS->getCodeByteSize(); |
| 979 | std::string Obj = stringOr(Str: IS->getObjectFileName(), IfEmpty: "<null>" ); |
| 980 | std::string VFName = stringOr(Str: IS->getVirtualFileName(), IfEmpty: "<null>" ); |
| 981 | uint32_t CRC = IS->getCrc32(); |
| 982 | |
| 983 | WithColor(Printer, PDB_ColorItem::Path).get() << File; |
| 984 | Printer << " (" ; |
| 985 | WithColor(Printer, PDB_ColorItem::LiteralValue).get() << Size; |
| 986 | Printer << " bytes): " ; |
| 987 | WithColor(Printer, PDB_ColorItem::Keyword).get() << "obj" ; |
| 988 | Printer << "=" ; |
| 989 | WithColor(Printer, PDB_ColorItem::Path).get() << Obj; |
| 990 | Printer << ", " ; |
| 991 | WithColor(Printer, PDB_ColorItem::Keyword).get() << "vname" ; |
| 992 | Printer << "=" ; |
| 993 | WithColor(Printer, PDB_ColorItem::Path).get() << VFName; |
| 994 | Printer << ", " ; |
| 995 | WithColor(Printer, PDB_ColorItem::Keyword).get() << "crc" ; |
| 996 | Printer << "=" ; |
| 997 | WithColor(Printer, PDB_ColorItem::LiteralValue).get() << CRC; |
| 998 | Printer << ", " ; |
| 999 | WithColor(Printer, PDB_ColorItem::Keyword).get() << "compression" ; |
| 1000 | Printer << "=" ; |
| 1001 | dumpPDBSourceCompression( |
| 1002 | OS&: WithColor(Printer, PDB_ColorItem::LiteralValue).get(), |
| 1003 | Compression: IS->getCompression()); |
| 1004 | |
| 1005 | if (!opts::pretty::ShowInjectedSourceContent) |
| 1006 | continue; |
| 1007 | |
| 1008 | // Set the indent level to 0 when printing file content. |
| 1009 | int Indent = Printer.getIndentLevel(); |
| 1010 | Printer.Unindent(Amount: Indent); |
| 1011 | |
| 1012 | if (IS->getCompression() == PDB_SourceCompression::None) |
| 1013 | Printer.printLine(T: IS->getCode()); |
| 1014 | else |
| 1015 | Printer.formatBinary(Label: "Compressed data" , |
| 1016 | Data: arrayRefFromStringRef(Input: IS->getCode()), |
| 1017 | /*StartOffset=*/0); |
| 1018 | |
| 1019 | // Re-indent back to the original level. |
| 1020 | Printer.Indent(Amount: Indent); |
| 1021 | } |
| 1022 | } |
| 1023 | |
| 1024 | template <typename OuterT, typename ChildT> |
| 1025 | void diaDumpChildren(PDBSymbol &Outer, PdbSymbolIdField Ids, |
| 1026 | PdbSymbolIdField Recurse) { |
| 1027 | OuterT *ConcreteOuter = dyn_cast<OuterT>(&Outer); |
| 1028 | if (!ConcreteOuter) |
| 1029 | return; |
| 1030 | |
| 1031 | auto Children = ConcreteOuter->template findAllChildren<ChildT>(); |
| 1032 | while (auto Child = Children->getNext()) { |
| 1033 | outs() << " {" ; |
| 1034 | Child->defaultDump(outs(), 4, Ids, Recurse); |
| 1035 | outs() << "\n }\n" ; |
| 1036 | } |
| 1037 | } |
| 1038 | |
| 1039 | static void dumpDia(StringRef Path) { |
| 1040 | std::unique_ptr<IPDBSession> Session; |
| 1041 | |
| 1042 | const auto ReaderType = |
| 1043 | opts::diadump::Native ? PDB_ReaderType::Native : PDB_ReaderType::DIA; |
| 1044 | ExitOnErr(loadDataForPDB(Type: ReaderType, Path, Session)); |
| 1045 | |
| 1046 | auto GlobalScope = Session->getGlobalScope(); |
| 1047 | |
| 1048 | std::vector<PDB_SymType> SymTypes; |
| 1049 | |
| 1050 | if (opts::diadump::Compilands) |
| 1051 | SymTypes.push_back(x: PDB_SymType::Compiland); |
| 1052 | if (opts::diadump::Enums) |
| 1053 | SymTypes.push_back(x: PDB_SymType::Enum); |
| 1054 | if (opts::diadump::Pointers) |
| 1055 | SymTypes.push_back(x: PDB_SymType::PointerType); |
| 1056 | if (opts::diadump::UDTs) |
| 1057 | SymTypes.push_back(x: PDB_SymType::UDT); |
| 1058 | if (opts::diadump::Funcsigs) |
| 1059 | SymTypes.push_back(x: PDB_SymType::FunctionSig); |
| 1060 | if (opts::diadump::Arrays) |
| 1061 | SymTypes.push_back(x: PDB_SymType::ArrayType); |
| 1062 | if (opts::diadump::VTShapes) |
| 1063 | SymTypes.push_back(x: PDB_SymType::VTableShape); |
| 1064 | if (opts::diadump::Typedefs) |
| 1065 | SymTypes.push_back(x: PDB_SymType::Typedef); |
| 1066 | PdbSymbolIdField Ids = opts::diadump::NoSymIndexIds ? PdbSymbolIdField::None |
| 1067 | : PdbSymbolIdField::All; |
| 1068 | |
| 1069 | PdbSymbolIdField Recurse = PdbSymbolIdField::None; |
| 1070 | if (opts::diadump::Recurse) |
| 1071 | Recurse = PdbSymbolIdField::All; |
| 1072 | if (!opts::diadump::ShowClassHierarchy) |
| 1073 | Ids &= ~(PdbSymbolIdField::ClassParent | PdbSymbolIdField::LexicalParent); |
| 1074 | |
| 1075 | for (PDB_SymType ST : SymTypes) { |
| 1076 | auto Children = GlobalScope->findAllChildren(Type: ST); |
| 1077 | while (auto Child = Children->getNext()) { |
| 1078 | outs() << "{" ; |
| 1079 | Child->defaultDump(OS&: outs(), Indent: 2, ShowFlags: Ids, RecurseFlags: Recurse); |
| 1080 | |
| 1081 | diaDumpChildren<PDBSymbolTypeEnum, PDBSymbolData>(Outer&: *Child, Ids, Recurse); |
| 1082 | outs() << "\n}\n" ; |
| 1083 | } |
| 1084 | } |
| 1085 | } |
| 1086 | |
| 1087 | static void dumpPretty(StringRef Path) { |
| 1088 | std::unique_ptr<IPDBSession> Session; |
| 1089 | |
| 1090 | const auto ReaderType = |
| 1091 | opts::pretty::Native ? PDB_ReaderType::Native : PDB_ReaderType::DIA; |
| 1092 | ExitOnErr(loadDataForPDB(Type: ReaderType, Path, Session)); |
| 1093 | |
| 1094 | if (opts::pretty::LoadAddress) |
| 1095 | Session->setLoadAddress(opts::pretty::LoadAddress); |
| 1096 | |
| 1097 | auto &Stream = outs(); |
| 1098 | const bool UseColor = opts::pretty::ColorOutput == cl::BOU_UNSET |
| 1099 | ? Stream.has_colors() |
| 1100 | : opts::pretty::ColorOutput == cl::BOU_TRUE; |
| 1101 | LinePrinter Printer(2, UseColor, Stream, opts::Filters); |
| 1102 | |
| 1103 | auto GlobalScope(Session->getGlobalScope()); |
| 1104 | if (!GlobalScope) |
| 1105 | return; |
| 1106 | std::string FileName(GlobalScope->getSymbolsFileName()); |
| 1107 | |
| 1108 | WithColor(Printer, PDB_ColorItem::None).get() << "Summary for " ; |
| 1109 | WithColor(Printer, PDB_ColorItem::Path).get() << FileName; |
| 1110 | Printer.Indent(); |
| 1111 | uint64_t FileSize = 0; |
| 1112 | |
| 1113 | Printer.NewLine(); |
| 1114 | WithColor(Printer, PDB_ColorItem::Identifier).get() << "Size" ; |
| 1115 | if (!sys::fs::file_size(Path: FileName, Result&: FileSize)) { |
| 1116 | Printer << ": " << FileSize << " bytes" ; |
| 1117 | } else { |
| 1118 | Printer << ": (Unable to obtain file size)" ; |
| 1119 | } |
| 1120 | |
| 1121 | Printer.NewLine(); |
| 1122 | WithColor(Printer, PDB_ColorItem::Identifier).get() << "Guid" ; |
| 1123 | Printer << ": " << GlobalScope->getGuid(); |
| 1124 | |
| 1125 | Printer.NewLine(); |
| 1126 | WithColor(Printer, PDB_ColorItem::Identifier).get() << "Age" ; |
| 1127 | Printer << ": " << GlobalScope->getAge(); |
| 1128 | |
| 1129 | Printer.NewLine(); |
| 1130 | WithColor(Printer, PDB_ColorItem::Identifier).get() << "Attributes" ; |
| 1131 | Printer << ": " ; |
| 1132 | if (GlobalScope->hasCTypes()) |
| 1133 | outs() << "HasCTypes " ; |
| 1134 | if (GlobalScope->hasPrivateSymbols()) |
| 1135 | outs() << "HasPrivateSymbols " ; |
| 1136 | Printer.Unindent(); |
| 1137 | |
| 1138 | if (!opts::pretty::WithName.empty()) { |
| 1139 | Printer.NewLine(); |
| 1140 | WithColor(Printer, PDB_ColorItem::SectionHeader).get() |
| 1141 | << "---SYMBOLS & TYPES BY NAME---" ; |
| 1142 | |
| 1143 | for (StringRef Name : opts::pretty::WithName) { |
| 1144 | auto Symbols = GlobalScope->findChildren( |
| 1145 | Type: PDB_SymType::None, Name, Flags: PDB_NameSearchFlags::NS_CaseSensitive); |
| 1146 | if (!Symbols || Symbols->getChildCount() == 0) { |
| 1147 | Printer.formatLine(Fmt: "[not found] - {0}" , Items&: Name); |
| 1148 | continue; |
| 1149 | } |
| 1150 | Printer.formatLine(Fmt: "[{0} occurrences] - {1}" , Items: Symbols->getChildCount(), |
| 1151 | Items&: Name); |
| 1152 | |
| 1153 | AutoIndent Indent(Printer); |
| 1154 | Printer.NewLine(); |
| 1155 | |
| 1156 | while (auto Symbol = Symbols->getNext()) { |
| 1157 | switch (Symbol->getSymTag()) { |
| 1158 | case PDB_SymType::Typedef: { |
| 1159 | TypedefDumper TD(Printer); |
| 1160 | std::unique_ptr<PDBSymbolTypeTypedef> T = |
| 1161 | llvm::unique_dyn_cast<PDBSymbolTypeTypedef>(Val: std::move(Symbol)); |
| 1162 | TD.start(Symbol: *T); |
| 1163 | break; |
| 1164 | } |
| 1165 | case PDB_SymType::Enum: { |
| 1166 | EnumDumper ED(Printer); |
| 1167 | std::unique_ptr<PDBSymbolTypeEnum> E = |
| 1168 | llvm::unique_dyn_cast<PDBSymbolTypeEnum>(Val: std::move(Symbol)); |
| 1169 | ED.start(Symbol: *E); |
| 1170 | break; |
| 1171 | } |
| 1172 | case PDB_SymType::UDT: { |
| 1173 | ClassDefinitionDumper CD(Printer); |
| 1174 | std::unique_ptr<PDBSymbolTypeUDT> C = |
| 1175 | llvm::unique_dyn_cast<PDBSymbolTypeUDT>(Val: std::move(Symbol)); |
| 1176 | CD.start(Class: *C); |
| 1177 | break; |
| 1178 | } |
| 1179 | case PDB_SymType::BaseClass: |
| 1180 | case PDB_SymType::Friend: { |
| 1181 | TypeDumper TD(Printer); |
| 1182 | Symbol->dump(Dumper&: TD); |
| 1183 | break; |
| 1184 | } |
| 1185 | case PDB_SymType::Function: { |
| 1186 | FunctionDumper FD(Printer); |
| 1187 | std::unique_ptr<PDBSymbolFunc> F = |
| 1188 | llvm::unique_dyn_cast<PDBSymbolFunc>(Val: std::move(Symbol)); |
| 1189 | FD.start(Symbol: *F, Pointer: FunctionDumper::PointerType::None); |
| 1190 | break; |
| 1191 | } |
| 1192 | case PDB_SymType::Data: { |
| 1193 | VariableDumper VD(Printer); |
| 1194 | std::unique_ptr<PDBSymbolData> D = |
| 1195 | llvm::unique_dyn_cast<PDBSymbolData>(Val: std::move(Symbol)); |
| 1196 | VD.start(Var: *D); |
| 1197 | break; |
| 1198 | } |
| 1199 | case PDB_SymType::PublicSymbol: { |
| 1200 | ExternalSymbolDumper ED(Printer); |
| 1201 | std::unique_ptr<PDBSymbolPublicSymbol> PS = |
| 1202 | llvm::unique_dyn_cast<PDBSymbolPublicSymbol>(Val: std::move(Symbol)); |
| 1203 | ED.dump(Symbol: *PS); |
| 1204 | break; |
| 1205 | } |
| 1206 | default: |
| 1207 | llvm_unreachable("Unexpected symbol tag!" ); |
| 1208 | } |
| 1209 | } |
| 1210 | } |
| 1211 | llvm::outs().flush(); |
| 1212 | } |
| 1213 | |
| 1214 | if (opts::pretty::Compilands) { |
| 1215 | Printer.NewLine(); |
| 1216 | WithColor(Printer, PDB_ColorItem::SectionHeader).get() |
| 1217 | << "---COMPILANDS---" ; |
| 1218 | auto Compilands = GlobalScope->findAllChildren<PDBSymbolCompiland>(); |
| 1219 | |
| 1220 | if (Compilands) { |
| 1221 | Printer.Indent(); |
| 1222 | CompilandDumper Dumper(Printer); |
| 1223 | CompilandDumpFlags options = CompilandDumper::Flags::None; |
| 1224 | if (opts::pretty::Lines) |
| 1225 | options = options | CompilandDumper::Flags::Lines; |
| 1226 | while (auto Compiland = Compilands->getNext()) |
| 1227 | Dumper.start(Symbol: *Compiland, flags: options); |
| 1228 | Printer.Unindent(); |
| 1229 | } |
| 1230 | } |
| 1231 | |
| 1232 | if (opts::pretty::Classes || opts::pretty::Enums || opts::pretty::Typedefs || |
| 1233 | opts::pretty::Funcsigs || opts::pretty::Pointers || |
| 1234 | opts::pretty::Arrays || opts::pretty::VTShapes) { |
| 1235 | Printer.NewLine(); |
| 1236 | WithColor(Printer, PDB_ColorItem::SectionHeader).get() << "---TYPES---" ; |
| 1237 | Printer.Indent(); |
| 1238 | TypeDumper Dumper(Printer); |
| 1239 | Dumper.start(Exe: *GlobalScope); |
| 1240 | Printer.Unindent(); |
| 1241 | } |
| 1242 | |
| 1243 | if (opts::pretty::Symbols) { |
| 1244 | Printer.NewLine(); |
| 1245 | WithColor(Printer, PDB_ColorItem::SectionHeader).get() << "---SYMBOLS---" ; |
| 1246 | if (auto Compilands = GlobalScope->findAllChildren<PDBSymbolCompiland>()) { |
| 1247 | Printer.Indent(); |
| 1248 | CompilandDumper Dumper(Printer); |
| 1249 | while (auto Compiland = Compilands->getNext()) |
| 1250 | Dumper.start(Symbol: *Compiland, flags: true); |
| 1251 | Printer.Unindent(); |
| 1252 | } |
| 1253 | } |
| 1254 | |
| 1255 | if (opts::pretty::Globals) { |
| 1256 | Printer.NewLine(); |
| 1257 | WithColor(Printer, PDB_ColorItem::SectionHeader).get() << "---GLOBALS---" ; |
| 1258 | Printer.Indent(); |
| 1259 | if (shouldDumpSymLevel(Search: opts::pretty::SymLevel::Functions)) { |
| 1260 | if (auto Functions = GlobalScope->findAllChildren<PDBSymbolFunc>()) { |
| 1261 | FunctionDumper Dumper(Printer); |
| 1262 | if (opts::pretty::SymbolOrder == opts::pretty::SymbolSortMode::None) { |
| 1263 | while (auto Function = Functions->getNext()) { |
| 1264 | Printer.NewLine(); |
| 1265 | Dumper.start(Symbol: *Function, Pointer: FunctionDumper::PointerType::None); |
| 1266 | } |
| 1267 | } else { |
| 1268 | std::vector<std::unique_ptr<PDBSymbolFunc>> Funcs; |
| 1269 | while (auto Func = Functions->getNext()) |
| 1270 | Funcs.push_back(x: std::move(Func)); |
| 1271 | llvm::sort(C&: Funcs, Comp: opts::pretty::compareFunctionSymbols); |
| 1272 | for (const auto &Func : Funcs) { |
| 1273 | Printer.NewLine(); |
| 1274 | Dumper.start(Symbol: *Func, Pointer: FunctionDumper::PointerType::None); |
| 1275 | } |
| 1276 | } |
| 1277 | } |
| 1278 | } |
| 1279 | if (shouldDumpSymLevel(Search: opts::pretty::SymLevel::Data)) { |
| 1280 | if (auto Vars = GlobalScope->findAllChildren<PDBSymbolData>()) { |
| 1281 | VariableDumper Dumper(Printer); |
| 1282 | if (opts::pretty::SymbolOrder == opts::pretty::SymbolSortMode::None) { |
| 1283 | while (auto Var = Vars->getNext()) |
| 1284 | Dumper.start(Var: *Var); |
| 1285 | } else { |
| 1286 | std::vector<std::unique_ptr<PDBSymbolData>> Datas; |
| 1287 | while (auto Var = Vars->getNext()) |
| 1288 | Datas.push_back(x: std::move(Var)); |
| 1289 | llvm::sort(C&: Datas, Comp: opts::pretty::compareDataSymbols); |
| 1290 | for (const auto &Var : Datas) |
| 1291 | Dumper.start(Var: *Var); |
| 1292 | } |
| 1293 | } |
| 1294 | } |
| 1295 | if (shouldDumpSymLevel(Search: opts::pretty::SymLevel::Thunks)) { |
| 1296 | if (auto Thunks = GlobalScope->findAllChildren<PDBSymbolThunk>()) { |
| 1297 | CompilandDumper Dumper(Printer); |
| 1298 | while (auto Thunk = Thunks->getNext()) |
| 1299 | Dumper.dump(Symbol: *Thunk); |
| 1300 | } |
| 1301 | } |
| 1302 | Printer.Unindent(); |
| 1303 | } |
| 1304 | if (opts::pretty::Externals) { |
| 1305 | Printer.NewLine(); |
| 1306 | WithColor(Printer, PDB_ColorItem::SectionHeader).get() << "---EXTERNALS---" ; |
| 1307 | Printer.Indent(); |
| 1308 | ExternalSymbolDumper Dumper(Printer); |
| 1309 | Dumper.start(Symbol: *GlobalScope); |
| 1310 | } |
| 1311 | if (opts::pretty::Lines) { |
| 1312 | Printer.NewLine(); |
| 1313 | } |
| 1314 | if (opts::pretty::InjectedSources) { |
| 1315 | Printer.NewLine(); |
| 1316 | WithColor(Printer, PDB_ColorItem::SectionHeader).get() |
| 1317 | << "---INJECTED SOURCES---" ; |
| 1318 | AutoIndent Indent1(Printer); |
| 1319 | dumpInjectedSources(Printer, Session&: *Session); |
| 1320 | } |
| 1321 | |
| 1322 | Printer.NewLine(); |
| 1323 | outs().flush(); |
| 1324 | } |
| 1325 | |
| 1326 | static void mergePdbs() { |
| 1327 | BumpPtrAllocator Allocator; |
| 1328 | MergingTypeTableBuilder MergedTpi(Allocator); |
| 1329 | MergingTypeTableBuilder MergedIpi(Allocator); |
| 1330 | |
| 1331 | // Create a Tpi and Ipi type table with all types from all input files. |
| 1332 | for (const auto &Path : opts::merge::InputFilenames) { |
| 1333 | std::unique_ptr<IPDBSession> Session; |
| 1334 | auto &File = loadPDB(Path, Session); |
| 1335 | SmallVector<TypeIndex, 128> TypeMap; |
| 1336 | SmallVector<TypeIndex, 128> IdMap; |
| 1337 | if (File.hasPDBTpiStream()) { |
| 1338 | auto &Tpi = ExitOnErr(File.getPDBTpiStream()); |
| 1339 | ExitOnErr( |
| 1340 | codeview::mergeTypeRecords(Dest&: MergedTpi, SourceToDest&: TypeMap, Types: Tpi.typeArray())); |
| 1341 | } |
| 1342 | if (File.hasPDBIpiStream()) { |
| 1343 | auto &Ipi = ExitOnErr(File.getPDBIpiStream()); |
| 1344 | ExitOnErr(codeview::mergeIdRecords(Dest&: MergedIpi, Types: TypeMap, SourceToDest&: IdMap, |
| 1345 | Ids: Ipi.typeArray())); |
| 1346 | } |
| 1347 | } |
| 1348 | |
| 1349 | // Then write the PDB. |
| 1350 | PDBFileBuilder Builder(Allocator); |
| 1351 | ExitOnErr(Builder.initialize(BlockSize: 4096)); |
| 1352 | // Add each of the reserved streams. We might not put any data in them, |
| 1353 | // but at least they have to be present. |
| 1354 | for (uint32_t I = 0; I < kSpecialStreamCount; ++I) |
| 1355 | ExitOnErr(Builder.getMsfBuilder().addStream(Size: 0)); |
| 1356 | |
| 1357 | auto &DestTpi = Builder.getTpiBuilder(); |
| 1358 | auto &DestIpi = Builder.getIpiBuilder(); |
| 1359 | MergedTpi.ForEachRecord(Func: [&DestTpi](TypeIndex TI, const CVType &Type) { |
| 1360 | DestTpi.addTypeRecord(Type: Type.RecordData, Hash: std::nullopt); |
| 1361 | }); |
| 1362 | MergedIpi.ForEachRecord(Func: [&DestIpi](TypeIndex TI, const CVType &Type) { |
| 1363 | DestIpi.addTypeRecord(Type: Type.RecordData, Hash: std::nullopt); |
| 1364 | }); |
| 1365 | Builder.getInfoBuilder().addFeature(Sig: PdbRaw_FeatureSig::VC140); |
| 1366 | |
| 1367 | SmallString<64> OutFile(opts::merge::PdbOutputFile); |
| 1368 | if (OutFile.empty()) { |
| 1369 | OutFile = opts::merge::InputFilenames[0]; |
| 1370 | llvm::sys::path::replace_extension(path&: OutFile, extension: "merged.pdb" ); |
| 1371 | } |
| 1372 | |
| 1373 | codeview::GUID IgnoredOutGuid; |
| 1374 | ExitOnErr(Builder.commit(Filename: OutFile, Guid: &IgnoredOutGuid)); |
| 1375 | } |
| 1376 | |
| 1377 | static void explain() { |
| 1378 | InputFile IF = |
| 1379 | ExitOnErr(InputFile::open(Path: opts::explain::InputFilename.front(), AllowUnknownFile: true)); |
| 1380 | |
| 1381 | for (uint64_t Off : opts::explain::Offsets) { |
| 1382 | auto O = std::make_unique<ExplainOutputStyle>(args&: IF, args&: Off); |
| 1383 | |
| 1384 | ExitOnErr(O->dump()); |
| 1385 | } |
| 1386 | } |
| 1387 | |
| 1388 | static void exportStream() { |
| 1389 | std::unique_ptr<IPDBSession> Session; |
| 1390 | PDBFile &File = loadPDB(Path: opts::exportstream::InputFilename.front(), Session); |
| 1391 | |
| 1392 | std::unique_ptr<MappedBlockStream> SourceStream; |
| 1393 | uint32_t Index = 0; |
| 1394 | bool Success = false; |
| 1395 | std::string OutFileName = opts::exportstream::OutputFile; |
| 1396 | |
| 1397 | if (!opts::exportstream::ForceName) { |
| 1398 | // First try to parse it as an integer, if it fails fall back to treating it |
| 1399 | // as a named stream. |
| 1400 | if (to_integer(S: opts::exportstream::Stream, Num&: Index)) { |
| 1401 | if (Index >= File.getNumStreams()) { |
| 1402 | errs() << "Error: " << Index << " is not a valid stream index.\n" ; |
| 1403 | exit(status: 1); |
| 1404 | } |
| 1405 | Success = true; |
| 1406 | outs() << "Dumping contents of stream index " << Index << " to file " |
| 1407 | << OutFileName << ".\n" ; |
| 1408 | } |
| 1409 | } |
| 1410 | |
| 1411 | if (!Success) { |
| 1412 | InfoStream &IS = cantFail(ValOrErr: File.getPDBInfoStream()); |
| 1413 | Index = ExitOnErr(IS.getNamedStreamIndex(Name: opts::exportstream::Stream)); |
| 1414 | outs() << "Dumping contents of stream '" << opts::exportstream::Stream |
| 1415 | << "' (index " << Index << ") to file " << OutFileName << ".\n" ; |
| 1416 | } |
| 1417 | |
| 1418 | SourceStream = File.createIndexedStream(SN: Index); |
| 1419 | auto OutFile = ExitOnErr( |
| 1420 | FileOutputBuffer::create(FilePath: OutFileName, Size: SourceStream->getLength())); |
| 1421 | FileBufferByteStream DestStream(std::move(OutFile), llvm::endianness::little); |
| 1422 | BinaryStreamWriter Writer(DestStream); |
| 1423 | ExitOnErr(Writer.writeStreamRef(Ref: *SourceStream)); |
| 1424 | ExitOnErr(DestStream.commit()); |
| 1425 | } |
| 1426 | |
| 1427 | static bool parseRange(StringRef Str, |
| 1428 | std::optional<opts::bytes::NumberRange> &Parsed) { |
| 1429 | if (Str.empty()) |
| 1430 | return true; |
| 1431 | |
| 1432 | llvm::Regex R("^([^-]+)(-([^-]+))?$" ); |
| 1433 | llvm::SmallVector<llvm::StringRef, 2> Matches; |
| 1434 | if (!R.match(String: Str, Matches: &Matches)) |
| 1435 | return false; |
| 1436 | |
| 1437 | Parsed.emplace(); |
| 1438 | if (!to_integer(S: Matches[1], Num&: Parsed->Min)) |
| 1439 | return false; |
| 1440 | |
| 1441 | if (!Matches[3].empty()) { |
| 1442 | Parsed->Max.emplace(); |
| 1443 | if (!to_integer(S: Matches[3], Num&: *Parsed->Max)) |
| 1444 | return false; |
| 1445 | } |
| 1446 | return true; |
| 1447 | } |
| 1448 | |
| 1449 | static void simplifyChunkList(llvm::cl::list<opts::ModuleSubsection> &Chunks) { |
| 1450 | // If this list contains "All" plus some other stuff, remove the other stuff |
| 1451 | // and just keep "All" in the list. |
| 1452 | if (!llvm::is_contained(Range&: Chunks, Element: opts::ModuleSubsection::All)) |
| 1453 | return; |
| 1454 | Chunks.reset(); |
| 1455 | Chunks.push_back(value: opts::ModuleSubsection::All); |
| 1456 | } |
| 1457 | |
| 1458 | int main(int Argc, const char **Argv) { |
| 1459 | InitLLVM X(Argc, Argv); |
| 1460 | ExitOnErr.setBanner("llvm-pdbutil: " ); |
| 1461 | |
| 1462 | cl::HideUnrelatedOptions( |
| 1463 | Categories: {&opts::TypeCategory, &opts::FilterCategory, &opts::OtherOptions}); |
| 1464 | cl::ParseCommandLineOptions(argc: Argc, argv: Argv, Overview: "LLVM PDB Dumper\n" ); |
| 1465 | |
| 1466 | if (opts::BytesSubcommand) { |
| 1467 | if (!parseRange(Str: opts::bytes::DumpBlockRangeOpt, |
| 1468 | Parsed&: opts::bytes::DumpBlockRange)) { |
| 1469 | errs() << "Argument '" << opts::bytes::DumpBlockRangeOpt |
| 1470 | << "' invalid format.\n" ; |
| 1471 | errs().flush(); |
| 1472 | exit(status: 1); |
| 1473 | } |
| 1474 | if (!parseRange(Str: opts::bytes::DumpByteRangeOpt, |
| 1475 | Parsed&: opts::bytes::DumpByteRange)) { |
| 1476 | errs() << "Argument '" << opts::bytes::DumpByteRangeOpt |
| 1477 | << "' invalid format.\n" ; |
| 1478 | errs().flush(); |
| 1479 | exit(status: 1); |
| 1480 | } |
| 1481 | } |
| 1482 | |
| 1483 | if (opts::DumpSubcommand) { |
| 1484 | if (opts::dump::RawAll) { |
| 1485 | opts::dump::DumpGlobals = true; |
| 1486 | opts::dump::DumpFpo = true; |
| 1487 | opts::dump::DumpInlineeLines = true; |
| 1488 | opts::dump::DumpIds = true; |
| 1489 | opts::dump::DumpIdExtras = true; |
| 1490 | opts::dump::DumpLines = true; |
| 1491 | opts::dump::DumpModules = true; |
| 1492 | opts::dump::DumpModuleFiles = true; |
| 1493 | opts::dump::DumpPublics = true; |
| 1494 | opts::dump::DumpSectionContribs = true; |
| 1495 | opts::dump::DumpSectionHeaders = true; |
| 1496 | opts::dump::DumpSectionMap = true; |
| 1497 | opts::dump::DumpStreams = true; |
| 1498 | opts::dump::DumpStreamBlocks = true; |
| 1499 | opts::dump::DumpStringTable = true; |
| 1500 | opts::dump::DumpStringTableDetails = true; |
| 1501 | opts::dump::DumpSummary = true; |
| 1502 | opts::dump::DumpSymbols = true; |
| 1503 | opts::dump::DumpSymbolStats = true; |
| 1504 | opts::dump::DumpTypes = true; |
| 1505 | opts::dump::DumpTypeExtras = true; |
| 1506 | opts::dump::DumpUdtStats = true; |
| 1507 | opts::dump::DumpXme = true; |
| 1508 | opts::dump::DumpXmi = true; |
| 1509 | } |
| 1510 | } |
| 1511 | if (opts::PdbToYamlSubcommand) { |
| 1512 | if (opts::pdb2yaml::All) { |
| 1513 | opts::pdb2yaml::StreamMetadata = true; |
| 1514 | opts::pdb2yaml::StreamDirectory = true; |
| 1515 | opts::pdb2yaml::PdbStream = true; |
| 1516 | opts::pdb2yaml::StringTable = true; |
| 1517 | opts::pdb2yaml::DbiStream = true; |
| 1518 | opts::pdb2yaml::TpiStream = true; |
| 1519 | opts::pdb2yaml::IpiStream = true; |
| 1520 | opts::pdb2yaml::PublicsStream = true; |
| 1521 | opts::pdb2yaml::DumpModules = true; |
| 1522 | opts::pdb2yaml::DumpModuleFiles = true; |
| 1523 | opts::pdb2yaml::DumpModuleSyms = true; |
| 1524 | opts::pdb2yaml::DumpModuleSubsections.push_back( |
| 1525 | value: opts::ModuleSubsection::All); |
| 1526 | } |
| 1527 | simplifyChunkList(Chunks&: opts::pdb2yaml::DumpModuleSubsections); |
| 1528 | |
| 1529 | if (opts::pdb2yaml::DumpModuleSyms || opts::pdb2yaml::DumpModuleFiles) |
| 1530 | opts::pdb2yaml::DumpModules = true; |
| 1531 | |
| 1532 | if (opts::pdb2yaml::DumpModules) |
| 1533 | opts::pdb2yaml::DbiStream = true; |
| 1534 | } |
| 1535 | |
| 1536 | llvm::sys::InitializeCOMRAII COM(llvm::sys::COMThreadingMode::MultiThreaded); |
| 1537 | |
| 1538 | // Initialize the filters for LinePrinter. |
| 1539 | auto propagate = [&](auto &Target, auto &Reference) { |
| 1540 | llvm::append_range(Target, Reference); |
| 1541 | }; |
| 1542 | |
| 1543 | propagate(opts::Filters.ExcludeTypes, opts::pretty::ExcludeTypes); |
| 1544 | propagate(opts::Filters.ExcludeTypes, opts::pretty::ExcludeTypes); |
| 1545 | propagate(opts::Filters.ExcludeSymbols, opts::pretty::ExcludeSymbols); |
| 1546 | propagate(opts::Filters.ExcludeCompilands, opts::pretty::ExcludeCompilands); |
| 1547 | propagate(opts::Filters.IncludeTypes, opts::pretty::IncludeTypes); |
| 1548 | propagate(opts::Filters.IncludeSymbols, opts::pretty::IncludeSymbols); |
| 1549 | propagate(opts::Filters.IncludeCompilands, opts::pretty::IncludeCompilands); |
| 1550 | opts::Filters.PaddingThreshold = opts::pretty::PaddingThreshold; |
| 1551 | opts::Filters.SizeThreshold = opts::pretty::SizeThreshold; |
| 1552 | opts::Filters.JustMyCode = opts::dump::JustMyCode; |
| 1553 | if (opts::dump::DumpModi.getNumOccurrences() > 0) { |
| 1554 | if (opts::dump::DumpModi.getNumOccurrences() != 1) { |
| 1555 | errs() << "argument '-modi' specified more than once.\n" ; |
| 1556 | errs().flush(); |
| 1557 | exit(status: 1); |
| 1558 | } |
| 1559 | opts::Filters.DumpModi = opts::dump::DumpModi; |
| 1560 | } |
| 1561 | if (opts::dump::DumpSymbolOffset) { |
| 1562 | if (opts::dump::DumpModi.getNumOccurrences() != 1) { |
| 1563 | errs() |
| 1564 | << "need to specify argument '-modi' when using '-symbol-offset'.\n" ; |
| 1565 | errs().flush(); |
| 1566 | exit(status: 1); |
| 1567 | } |
| 1568 | opts::Filters.SymbolOffset = opts::dump::DumpSymbolOffset; |
| 1569 | if (opts::dump::DumpParents) |
| 1570 | opts::Filters.ParentRecurseDepth = opts::dump::DumpParentDepth; |
| 1571 | if (opts::dump::DumpChildren) |
| 1572 | opts::Filters.ChildrenRecurseDepth = opts::dump::DumpChildrenDepth; |
| 1573 | } |
| 1574 | |
| 1575 | if (opts::PdbToYamlSubcommand) { |
| 1576 | pdb2Yaml(Path: opts::pdb2yaml::InputFilename.front()); |
| 1577 | } else if (opts::YamlToPdbSubcommand) { |
| 1578 | if (opts::yaml2pdb::YamlPdbOutputFile.empty()) { |
| 1579 | SmallString<16> OutputFilename(opts::yaml2pdb::InputFilename.getValue()); |
| 1580 | sys::path::replace_extension(path&: OutputFilename, extension: ".pdb" ); |
| 1581 | opts::yaml2pdb::YamlPdbOutputFile = std::string(OutputFilename); |
| 1582 | } |
| 1583 | yamlToPdb(Path: opts::yaml2pdb::InputFilename); |
| 1584 | } else if (opts::DiaDumpSubcommand) { |
| 1585 | llvm::for_each(Range&: opts::diadump::InputFilenames, F: dumpDia); |
| 1586 | } else if (opts::PrettySubcommand) { |
| 1587 | if (opts::pretty::Lines) |
| 1588 | opts::pretty::Compilands = true; |
| 1589 | |
| 1590 | if (opts::pretty::All) { |
| 1591 | opts::pretty::Compilands = true; |
| 1592 | opts::pretty::Symbols = true; |
| 1593 | opts::pretty::Globals = true; |
| 1594 | opts::pretty::Types = true; |
| 1595 | opts::pretty::Externals = true; |
| 1596 | opts::pretty::Lines = true; |
| 1597 | } |
| 1598 | |
| 1599 | if (opts::pretty::Types) { |
| 1600 | opts::pretty::Classes = true; |
| 1601 | opts::pretty::Typedefs = true; |
| 1602 | opts::pretty::Enums = true; |
| 1603 | opts::pretty::Pointers = true; |
| 1604 | opts::pretty::Funcsigs = true; |
| 1605 | } |
| 1606 | |
| 1607 | // When adding filters for excluded compilands and types, we need to |
| 1608 | // remember that these are regexes. So special characters such as * and \ |
| 1609 | // need to be escaped in the regex. In the case of a literal \, this means |
| 1610 | // it needs to be escaped again in the C++. So matching a single \ in the |
| 1611 | // input requires 4 \es in the C++. |
| 1612 | if (opts::pretty::ExcludeCompilerGenerated) { |
| 1613 | opts::Filters.ExcludeTypes.push_back(x: "__vc_attributes" ); |
| 1614 | opts::Filters.ExcludeCompilands.push_back(x: "\\* Linker \\*" ); |
| 1615 | } |
| 1616 | if (opts::pretty::ExcludeSystemLibraries) { |
| 1617 | opts::Filters.ExcludeCompilands.push_back( |
| 1618 | x: "f:\\\\binaries\\\\Intermediate\\\\vctools\\\\crt_bld" ); |
| 1619 | opts::Filters.ExcludeCompilands.push_back(x: "f:\\\\dd\\\\vctools\\\\crt" ); |
| 1620 | opts::Filters.ExcludeCompilands.push_back( |
| 1621 | x: "d:\\\\th.obj.x86fre\\\\minkernel" ); |
| 1622 | } |
| 1623 | llvm::for_each(Range&: opts::pretty::InputFilenames, F: dumpPretty); |
| 1624 | } else if (opts::DumpSubcommand) { |
| 1625 | llvm::for_each(Range&: opts::dump::InputFilenames, F: dumpRaw); |
| 1626 | } else if (opts::BytesSubcommand) { |
| 1627 | llvm::for_each(Range&: opts::bytes::InputFilenames, F: dumpBytes); |
| 1628 | } else if (opts::MergeSubcommand) { |
| 1629 | if (opts::merge::InputFilenames.size() < 2) { |
| 1630 | errs() << "merge subcommand requires at least 2 input files.\n" ; |
| 1631 | exit(status: 1); |
| 1632 | } |
| 1633 | mergePdbs(); |
| 1634 | } else if (opts::ExplainSubcommand) { |
| 1635 | explain(); |
| 1636 | } else if (opts::ExportSubcommand) { |
| 1637 | exportStream(); |
| 1638 | } |
| 1639 | |
| 1640 | outs().flush(); |
| 1641 | return 0; |
| 1642 | } |
| 1643 | |