diff options
Diffstat (limited to 'llvm/tools')
-rw-r--r-- | llvm/tools/llvm-pdbdump/llvm-pdbdump.cpp | 139 |
1 files changed, 13 insertions, 126 deletions
diff --git a/llvm/tools/llvm-pdbdump/llvm-pdbdump.cpp b/llvm/tools/llvm-pdbdump/llvm-pdbdump.cpp index 78a2155cf51..c9c78a97110 100644 --- a/llvm/tools/llvm-pdbdump/llvm-pdbdump.cpp +++ b/llvm/tools/llvm-pdbdump/llvm-pdbdump.cpp @@ -36,6 +36,7 @@ #include "llvm/DebugInfo/PDB/PDBSymbolFunc.h" #include "llvm/DebugInfo/PDB/PDBSymbolThunk.h" #include "llvm/DebugInfo/PDB/Raw/PDBFile.h" +#include "llvm/DebugInfo/PDB/Raw/PDBInfoStream.h" #include "llvm/DebugInfo/PDB/Raw/PDBStream.h" #include "llvm/DebugInfo/PDB/Raw/RawSession.h" #include "llvm/Support/CommandLine.h" @@ -233,135 +234,21 @@ static void dumpStructure(RawSession &RS) { } } - // Stream 1 starts with the following header: - // uint32_t Version; - // uint32_t Signature; - // uint32_t Age; - // GUID Guid; - PDBStream Stream1(1, File); - uint32_t Version; - uint32_t Signature; - uint32_t Age; - PDB_UniqueId Guid; - - Stream1.readInteger(Version); - outs() << "Version: " << Version << '\n'; - // PDB's with versions before PDBImpvVC70 might not have the Guid field, we - // don't support them. - if (Version < 20000404) - reportError("", std::make_error_code(std::errc::not_supported)); - - // This appears to be the time the PDB was last opened by an MSVC tool? - // It is definitely a timestamp of some sort. - Stream1.readInteger(Signature); - outs() << "Signature: "; - outs().write_hex(Signature) << '\n'; - - // This appears to be a number which is used to determine that the PDB is kept - // in sync with the EXE. - Stream1.readInteger(Age); - outs() << "Age: " << Age << '\n'; - - // I'm not sure what the purpose of the GUID is. - Stream1.readObject(&Guid); - outs() << "Guid: " << Guid << '\n'; - - // This is some sort of weird string-set/hash table encoded in the stream. - // It starts with the number of bytes in the table. - uint32_t NumberOfBytes; - Stream1.readInteger(NumberOfBytes); - outs() << "NumberOfBytes: " << NumberOfBytes << '\n'; - - // Following that field is the starting offset of strings in the name table. - uint32_t StringsOffset = Stream1.getOffset(); - Stream1.setOffset(StringsOffset + NumberOfBytes); - - // This appears to be equivalent to the total number of strings *actually* - // in the name table. - uint32_t HashSize; - Stream1.readInteger(HashSize); - outs() << "HashSize: " << HashSize << '\n'; - - // This appears to be an upper bound on the number of strings in the name - // table. - uint32_t MaxNumberOfStrings; - Stream1.readInteger(MaxNumberOfStrings); - outs() << "MaxNumberOfStrings: " << MaxNumberOfStrings << '\n'; - - // This appears to be a hash table which uses bitfields to determine whether - // or not a bucket is 'present'. - uint32_t NumPresentWords; - Stream1.readInteger(NumPresentWords); - outs() << "NumPresentWords: " << NumPresentWords << '\n'; - - // Store all the 'present' bits in a vector for later processing. - SmallVector<uint32_t, 1> PresentWords; - for (uint32_t I = 0; I != NumPresentWords; ++I) { - uint32_t Word; - Stream1.readInteger(Word); - PresentWords.push_back(Word); - outs() << "Word: " << Word << '\n'; - } - - // This appears to be a hash table which uses bitfields to determine whether - // or not a bucket is 'deleted'. - uint32_t NumDeletedWords; - Stream1.readInteger(NumDeletedWords); - outs() << "NumDeletedWords: " << NumDeletedWords << '\n'; - - // Store all the 'deleted' bits in a vector for later processing. - SmallVector<uint32_t, 1> DeletedWords; - for (uint32_t I = 0; I != NumDeletedWords; ++I) { - uint32_t Word; - Stream1.readInteger(Word); - DeletedWords.push_back(Word); - outs() << "Word: " << Word << '\n'; - } + PDBInfoStream InfoStream(File); + if (auto EC = InfoStream.reload()) + reportError("", EC); - BitVector Present(MaxNumberOfStrings, false); - if (!PresentWords.empty()) - Present.setBitsInMask(PresentWords.data(), PresentWords.size()); - BitVector Deleted(MaxNumberOfStrings, false); - if (!DeletedWords.empty()) - Deleted.setBitsInMask(DeletedWords.data(), DeletedWords.size()); - - StringMap<uint32_t> NamedStreams; - for (uint32_t I = 0; I < MaxNumberOfStrings; ++I) { - if (!Present.test(I)) - continue; - - // For all present entries, dump out their mapping. - - // This appears to be an offset relative to the start of the strings. - // It tells us where the null-terminated string begins. - uint32_t NameOffset; - Stream1.readInteger(NameOffset); - outs() << "NameOffset: " << NameOffset << '\n'; - - // This appears to be a stream number into the stream directory. - uint32_t NameIndex; - Stream1.readInteger(NameIndex); - outs() << "NameIndex: " << NameIndex << '\n'; - - // Compute the offset of the start of the string relative to the stream. - uint32_t StringOffset = StringsOffset + NameOffset; - uint32_t OldOffset = Stream1.getOffset(); - // Pump out our c-string from the stream. - std::string Str; - Stream1.setOffset(StringOffset); - Stream1.readZeroString(Str); - outs() << "String: " << Str << "\n\n"; - - Stream1.setOffset(OldOffset); - // Add this to a string-map from name to stream number. - NamedStreams.insert({Str, NameIndex}); - } + outs() << "Version: " << InfoStream.getVersion() << '\n'; + outs() << "Signature: "; + outs().write_hex(InfoStream.getSignature()) << '\n'; + outs() << "Age: " << InfoStream.getAge() << '\n'; + outs() << "Guid: " << InfoStream.getGuid() << '\n'; // Let's try to dump out the named stream "/names". - auto NameI = NamedStreams.find("/names"); - if (NameI != NamedStreams.end()) { - PDBStream NameStream(NameI->second, File); - outs() << "NameStream: " << NameI->second << '\n'; + uint32_t NameStreamIndex = InfoStream.getNamedStreamIndex("/names"); + if (NameStreamIndex != 0) { + PDBStream NameStream(NameStreamIndex, File); + outs() << "NameStream: " << NameStreamIndex << '\n'; // The name stream appears to start with a signature and version. uint32_t NameStreamSignature; |