|  | //===- StreamUtil.cpp - PDB stream utilities --------------------*- C++ -*-===// | 
|  | // | 
|  | //                     The LLVM Compiler Infrastructure | 
|  | // | 
|  | // This file is distributed under the University of Illinois Open Source | 
|  | // License. See LICENSE.TXT for details. | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | #include "StreamUtil.h" | 
|  | #include "FormatUtil.h" | 
|  |  | 
|  | #include "llvm/ADT/DenseMap.h" | 
|  | #include "llvm/ADT/DenseMapInfo.h" | 
|  | #include "llvm/DebugInfo/PDB/Native/DbiModuleDescriptor.h" | 
|  | #include "llvm/DebugInfo/PDB/Native/DbiModuleList.h" | 
|  | #include "llvm/DebugInfo/PDB/Native/DbiStream.h" | 
|  | #include "llvm/DebugInfo/PDB/Native/InfoStream.h" | 
|  | #include "llvm/DebugInfo/PDB/Native/PDBFile.h" | 
|  | #include "llvm/DebugInfo/PDB/Native/TpiStream.h" | 
|  |  | 
|  | using namespace llvm; | 
|  | using namespace llvm::pdb; | 
|  |  | 
|  | std::string StreamInfo::getLongName() const { | 
|  | if (Purpose == StreamPurpose::NamedStream) | 
|  | return formatv("Named Stream \"{0}\"", Name).str(); | 
|  | if (Purpose == StreamPurpose::ModuleStream) | 
|  | return formatv("Module \"{0}\"", Name).str(); | 
|  | return Name; | 
|  | } | 
|  |  | 
|  | StreamInfo StreamInfo::createStream(StreamPurpose Purpose, StringRef Name, | 
|  | uint32_t StreamIndex) { | 
|  | StreamInfo Result; | 
|  | Result.Name = Name; | 
|  | Result.StreamIndex = StreamIndex; | 
|  | Result.Purpose = Purpose; | 
|  | return Result; | 
|  | } | 
|  |  | 
|  | StreamInfo StreamInfo::createModuleStream(StringRef Module, | 
|  | uint32_t StreamIndex, uint32_t Modi) { | 
|  | StreamInfo Result; | 
|  | Result.Name = Module; | 
|  | Result.StreamIndex = StreamIndex; | 
|  | Result.ModuleIndex = Modi; | 
|  | Result.Purpose = StreamPurpose::ModuleStream; | 
|  | return Result; | 
|  | } | 
|  |  | 
|  | static inline StreamInfo stream(StreamPurpose Purpose, StringRef Label, | 
|  | uint32_t Idx) { | 
|  | return StreamInfo::createStream(Purpose, Label, Idx); | 
|  | } | 
|  |  | 
|  | static inline StreamInfo moduleStream(StringRef Label, uint32_t StreamIdx, | 
|  | uint32_t Modi) { | 
|  | return StreamInfo::createModuleStream(Label, StreamIdx, Modi); | 
|  | } | 
|  |  | 
|  | struct IndexedModuleDescriptor { | 
|  | uint32_t Modi; | 
|  | DbiModuleDescriptor Descriptor; | 
|  | }; | 
|  |  | 
|  | void llvm::pdb::discoverStreamPurposes(PDBFile &File, | 
|  | SmallVectorImpl<StreamInfo> &Streams) { | 
|  | // It's OK if we fail to load some of these streams, we still attempt to print | 
|  | // what we can. | 
|  | auto Dbi = File.getPDBDbiStream(); | 
|  | auto Tpi = File.getPDBTpiStream(); | 
|  | auto Ipi = File.getPDBIpiStream(); | 
|  | auto Info = File.getPDBInfoStream(); | 
|  |  | 
|  | uint32_t StreamCount = File.getNumStreams(); | 
|  | DenseMap<uint16_t, IndexedModuleDescriptor> ModStreams; | 
|  | DenseMap<uint16_t, std::string> NamedStreams; | 
|  |  | 
|  | if (Dbi) { | 
|  | const DbiModuleList &Modules = Dbi->modules(); | 
|  | for (uint32_t I = 0; I < Modules.getModuleCount(); ++I) { | 
|  | IndexedModuleDescriptor IMD; | 
|  | IMD.Modi = I; | 
|  | IMD.Descriptor = Modules.getModuleDescriptor(I); | 
|  | uint16_t SN = IMD.Descriptor.getModuleStreamIndex(); | 
|  | if (SN != kInvalidStreamIndex) | 
|  | ModStreams[SN] = IMD; | 
|  | } | 
|  | } | 
|  | if (Info) { | 
|  | for (auto &NSE : Info->named_streams()) { | 
|  | if (NSE.second != kInvalidStreamIndex) | 
|  | NamedStreams[NSE.second] = NSE.first(); | 
|  | } | 
|  | } | 
|  |  | 
|  | Streams.resize(StreamCount); | 
|  | for (uint16_t StreamIdx = 0; StreamIdx < StreamCount; ++StreamIdx) { | 
|  | if (StreamIdx == OldMSFDirectory) | 
|  | Streams[StreamIdx] = | 
|  | stream(StreamPurpose::Other, "Old MSF Directory", StreamIdx); | 
|  | else if (StreamIdx == StreamPDB) | 
|  | Streams[StreamIdx] = stream(StreamPurpose::PDB, "PDB Stream", StreamIdx); | 
|  | else if (StreamIdx == StreamDBI) | 
|  | Streams[StreamIdx] = stream(StreamPurpose::DBI, "DBI Stream", StreamIdx); | 
|  | else if (StreamIdx == StreamTPI) | 
|  | Streams[StreamIdx] = stream(StreamPurpose::TPI, "TPI Stream", StreamIdx); | 
|  | else if (StreamIdx == StreamIPI) | 
|  | Streams[StreamIdx] = stream(StreamPurpose::IPI, "IPI Stream", StreamIdx); | 
|  | else if (Dbi && StreamIdx == Dbi->getGlobalSymbolStreamIndex()) | 
|  | Streams[StreamIdx] = | 
|  | stream(StreamPurpose::GlobalHash, "Global Symbol Hash", StreamIdx); | 
|  | else if (Dbi && StreamIdx == Dbi->getPublicSymbolStreamIndex()) | 
|  | Streams[StreamIdx] = | 
|  | stream(StreamPurpose::PublicHash, "Public Symbol Hash", StreamIdx); | 
|  | else if (Dbi && StreamIdx == Dbi->getSymRecordStreamIndex()) | 
|  | Streams[StreamIdx] = | 
|  | stream(StreamPurpose::Symbols, "Symbol Records", StreamIdx); | 
|  | else if (Tpi && StreamIdx == Tpi->getTypeHashStreamIndex()) | 
|  | Streams[StreamIdx] = | 
|  | stream(StreamPurpose::TpiHash, "TPI Hash", StreamIdx); | 
|  | else if (Tpi && StreamIdx == Tpi->getTypeHashStreamAuxIndex()) | 
|  | Streams[StreamIdx] = | 
|  | stream(StreamPurpose::Other, "TPI Aux Hash", StreamIdx); | 
|  | else if (Ipi && StreamIdx == Ipi->getTypeHashStreamIndex()) | 
|  | Streams[StreamIdx] = | 
|  | stream(StreamPurpose::IpiHash, "IPI Hash", StreamIdx); | 
|  | else if (Ipi && StreamIdx == Ipi->getTypeHashStreamAuxIndex()) | 
|  | Streams[StreamIdx] = | 
|  | stream(StreamPurpose::Other, "IPI Aux Hash", StreamIdx); | 
|  | else if (Dbi && | 
|  | StreamIdx == Dbi->getDebugStreamIndex(DbgHeaderType::Exception)) | 
|  | Streams[StreamIdx] = | 
|  | stream(StreamPurpose::Other, "Exception Data", StreamIdx); | 
|  | else if (Dbi && StreamIdx == Dbi->getDebugStreamIndex(DbgHeaderType::Fixup)) | 
|  | Streams[StreamIdx] = | 
|  | stream(StreamPurpose::Other, "Fixup Data", StreamIdx); | 
|  | else if (Dbi && StreamIdx == Dbi->getDebugStreamIndex(DbgHeaderType::FPO)) | 
|  | Streams[StreamIdx] = stream(StreamPurpose::Other, "FPO Data", StreamIdx); | 
|  | else if (Dbi && | 
|  | StreamIdx == Dbi->getDebugStreamIndex(DbgHeaderType::NewFPO)) | 
|  | Streams[StreamIdx] = | 
|  | stream(StreamPurpose::Other, "New FPO Data", StreamIdx); | 
|  | else if (Dbi && | 
|  | StreamIdx == Dbi->getDebugStreamIndex(DbgHeaderType::OmapFromSrc)) | 
|  | Streams[StreamIdx] = | 
|  | stream(StreamPurpose::Other, "Omap From Source Data", StreamIdx); | 
|  | else if (Dbi && | 
|  | StreamIdx == Dbi->getDebugStreamIndex(DbgHeaderType::OmapToSrc)) | 
|  | Streams[StreamIdx] = | 
|  | stream(StreamPurpose::Other, "Omap To Source Data", StreamIdx); | 
|  | else if (Dbi && StreamIdx == Dbi->getDebugStreamIndex(DbgHeaderType::Pdata)) | 
|  | Streams[StreamIdx] = stream(StreamPurpose::Other, "Pdata", StreamIdx); | 
|  | else if (Dbi && | 
|  | StreamIdx == Dbi->getDebugStreamIndex(DbgHeaderType::SectionHdr)) | 
|  | Streams[StreamIdx] = | 
|  | stream(StreamPurpose::Other, "Section Header Data", StreamIdx); | 
|  | else if (Dbi && | 
|  | StreamIdx == | 
|  | Dbi->getDebugStreamIndex(DbgHeaderType::SectionHdrOrig)) | 
|  | Streams[StreamIdx] = stream(StreamPurpose::Other, | 
|  | "Section Header Original Data", StreamIdx); | 
|  | else if (Dbi && | 
|  | StreamIdx == Dbi->getDebugStreamIndex(DbgHeaderType::TokenRidMap)) | 
|  | Streams[StreamIdx] = | 
|  | stream(StreamPurpose::Other, "Token Rid Data", StreamIdx); | 
|  | else if (Dbi && StreamIdx == Dbi->getDebugStreamIndex(DbgHeaderType::Xdata)) | 
|  | Streams[StreamIdx] = stream(StreamPurpose::Other, "Xdata", StreamIdx); | 
|  | else { | 
|  | auto ModIter = ModStreams.find(StreamIdx); | 
|  | auto NSIter = NamedStreams.find(StreamIdx); | 
|  | if (ModIter != ModStreams.end()) { | 
|  | Streams[StreamIdx] = | 
|  | moduleStream(ModIter->second.Descriptor.getModuleName(), StreamIdx, | 
|  | ModIter->second.Modi); | 
|  | } else if (NSIter != NamedStreams.end()) { | 
|  | Streams[StreamIdx] = | 
|  | stream(StreamPurpose::NamedStream, NSIter->second, StreamIdx); | 
|  | } else { | 
|  | Streams[StreamIdx] = stream(StreamPurpose::Other, "???", StreamIdx); | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | // Consume errors from missing streams. | 
|  | if (!Dbi) | 
|  | consumeError(Dbi.takeError()); | 
|  | if (!Tpi) | 
|  | consumeError(Tpi.takeError()); | 
|  | if (!Ipi) | 
|  | consumeError(Ipi.takeError()); | 
|  | if (!Info) | 
|  | consumeError(Info.takeError()); | 
|  | } |