//===- NativeSession.cpp - Native implementation of IPDBSession -*- C++ -*-===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// #include "llvm/DebugInfo/PDB/Native/NativeSession.h" #include "llvm/BinaryFormat/Magic.h" #include "llvm/DebugInfo/MSF/MSFCommon.h" #include "llvm/DebugInfo/MSF/MappedBlockStream.h" #include "llvm/DebugInfo/PDB/IPDBEnumChildren.h" #include "llvm/DebugInfo/PDB/IPDBSourceFile.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/ISectionContribVisitor.h" #include "llvm/DebugInfo/PDB/Native/ModuleDebugStream.h" #include "llvm/DebugInfo/PDB/Native/NativeEnumInjectedSources.h" #include "llvm/DebugInfo/PDB/Native/NativeExeSymbol.h" #include "llvm/DebugInfo/PDB/Native/PDBFile.h" #include "llvm/DebugInfo/PDB/Native/RawConstants.h" #include "llvm/DebugInfo/PDB/Native/RawError.h" #include "llvm/DebugInfo/PDB/Native/RawTypes.h" #include "llvm/DebugInfo/PDB/Native/SymbolCache.h" #include "llvm/DebugInfo/PDB/PDBSymbol.h" #include "llvm/DebugInfo/PDB/PDBSymbolCompiland.h" #include "llvm/DebugInfo/PDB/PDBSymbolExe.h" #include "llvm/Object/Binary.h" #include "llvm/Object/COFF.h" #include "llvm/Support/Allocator.h" #include "llvm/Support/BinaryByteStream.h" #include "llvm/Support/BinaryStreamArray.h" #include "llvm/Support/Error.h" #include "llvm/Support/ErrorOr.h" #include "llvm/Support/MemoryBuffer.h" #include "llvm/Support/Path.h" #include #include #include #include using namespace llvm; using namespace llvm::msf; using namespace llvm::pdb; namespace llvm { namespace codeview { union DebugInfo; } } // namespace llvm static DbiStream *getDbiStreamPtr(PDBFile &File) { Expected DbiS = File.getPDBDbiStream(); if (DbiS) return &DbiS.get(); consumeError(DbiS.takeError()); return nullptr; } NativeSession::NativeSession(std::unique_ptr PdbFile, std::unique_ptr Allocator) : Pdb(std::move(PdbFile)), Allocator(std::move(Allocator)), Cache(*this, getDbiStreamPtr(*Pdb)), AddrToModuleIndex(IMapAllocator) {} NativeSession::~NativeSession() = default; Error NativeSession::createFromPdb(std::unique_ptr Buffer, std::unique_ptr &Session) { StringRef Path = Buffer->getBufferIdentifier(); auto Stream = std::make_unique( std::move(Buffer), llvm::support::little); auto Allocator = std::make_unique(); auto File = std::make_unique(Path, std::move(Stream), *Allocator); if (auto EC = File->parseFileHeaders()) return EC; if (auto EC = File->parseStreamData()) return EC; Session = std::make_unique(std::move(File), std::move(Allocator)); return Error::success(); } static Expected> loadPdbFile(StringRef PdbPath, std::unique_ptr &Allocator) { ErrorOr> ErrorOrBuffer = MemoryBuffer::getFile(PdbPath, /*IsText=*/false, /*RequiresNullTerminator=*/false); if (!ErrorOrBuffer) return make_error(ErrorOrBuffer.getError()); std::unique_ptr Buffer = std::move(*ErrorOrBuffer); PdbPath = Buffer->getBufferIdentifier(); file_magic Magic; auto EC = identify_magic(PdbPath, Magic); if (EC || Magic != file_magic::pdb) return make_error(EC); auto Stream = std::make_unique(std::move(Buffer), llvm::support::little); auto File = std::make_unique(PdbPath, std::move(Stream), *Allocator); if (auto EC = File->parseFileHeaders()) return std::move(EC); if (auto EC = File->parseStreamData()) return std::move(EC); return std::move(File); } Error NativeSession::createFromPdbPath(StringRef PdbPath, std::unique_ptr &Session) { auto Allocator = std::make_unique(); auto PdbFile = loadPdbFile(PdbPath, Allocator); if (!PdbFile) return PdbFile.takeError(); Session = std::make_unique(std::move(PdbFile.get()), std::move(Allocator)); return Error::success(); } static Expected getPdbPathFromExe(StringRef ExePath) { Expected> BinaryFile = object::createBinary(ExePath); if (!BinaryFile) return BinaryFile.takeError(); const object::COFFObjectFile *ObjFile = dyn_cast(BinaryFile->getBinary()); if (!ObjFile) return make_error(raw_error_code::invalid_format); StringRef PdbPath; const llvm::codeview::DebugInfo *PdbInfo = nullptr; if (Error E = ObjFile->getDebugPDBInfo(PdbInfo, PdbPath)) return std::move(E); return std::string(PdbPath); } Error NativeSession::createFromExe(StringRef ExePath, std::unique_ptr &Session) { Expected PdbPath = getPdbPathFromExe(ExePath); if (!PdbPath) return PdbPath.takeError(); file_magic Magic; auto EC = identify_magic(PdbPath.get(), Magic); if (EC || Magic != file_magic::pdb) return make_error(EC); auto Allocator = std::make_unique(); auto File = loadPdbFile(PdbPath.get(), Allocator); if (!File) return File.takeError(); Session = std::make_unique(std::move(File.get()), std::move(Allocator)); return Error::success(); } Expected NativeSession::searchForPdb(const PdbSearchOptions &Opts) { Expected PathOrErr = getPdbPathFromExe(Opts.ExePath); if (!PathOrErr) return PathOrErr.takeError(); StringRef PathFromExe = PathOrErr.get(); sys::path::Style Style = PathFromExe.startswith("/") ? sys::path::Style::posix : sys::path::Style::windows; StringRef PdbName = sys::path::filename(PathFromExe, Style); // Check if pdb exists in the executable directory. SmallString<128> PdbPath = StringRef(Opts.ExePath); sys::path::remove_filename(PdbPath); sys::path::append(PdbPath, PdbName); auto Allocator = std::make_unique(); if (auto File = loadPdbFile(PdbPath, Allocator)) return std::string(PdbPath); else consumeError(File.takeError()); // Check path that was in the executable. if (auto File = loadPdbFile(PathFromExe, Allocator)) return std::string(PathFromExe); else return File.takeError(); return make_error("PDB not found"); } uint64_t NativeSession::getLoadAddress() const { return LoadAddress; } bool NativeSession::setLoadAddress(uint64_t Address) { LoadAddress = Address; return true; } std::unique_ptr NativeSession::getGlobalScope() { return PDBSymbol::createAs(*this, getNativeGlobalScope()); } std::unique_ptr NativeSession::getSymbolById(SymIndexId SymbolId) const { return Cache.getSymbolById(SymbolId); } bool NativeSession::addressForVA(uint64_t VA, uint32_t &Section, uint32_t &Offset) const { uint32_t RVA = VA - getLoadAddress(); return addressForRVA(RVA, Section, Offset); } bool NativeSession::addressForRVA(uint32_t RVA, uint32_t &Section, uint32_t &Offset) const { Section = 0; Offset = 0; auto Dbi = Pdb->getPDBDbiStream(); if (!Dbi) return false; if ((int32_t)RVA < 0) return true; Offset = RVA; for (; Section < Dbi->getSectionHeaders().size(); ++Section) { auto &Sec = Dbi->getSectionHeaders()[Section]; if (RVA < Sec.VirtualAddress) return true; Offset = RVA - Sec.VirtualAddress; } return true; } std::unique_ptr NativeSession::findSymbolByAddress(uint64_t Address, PDB_SymType Type) { uint32_t Section; uint32_t Offset; addressForVA(Address, Section, Offset); return findSymbolBySectOffset(Section, Offset, Type); } std::unique_ptr NativeSession::findSymbolByRVA(uint32_t RVA, PDB_SymType Type) { uint32_t Section; uint32_t Offset; addressForRVA(RVA, Section, Offset); return findSymbolBySectOffset(Section, Offset, Type); } std::unique_ptr NativeSession::findSymbolBySectOffset(uint32_t Sect, uint32_t Offset, PDB_SymType Type) { if (AddrToModuleIndex.empty()) parseSectionContribs(); return Cache.findSymbolBySectOffset(Sect, Offset, Type); } std::unique_ptr NativeSession::findLineNumbers(const PDBSymbolCompiland &Compiland, const IPDBSourceFile &File) const { return nullptr; } std::unique_ptr NativeSession::findLineNumbersByAddress(uint64_t Address, uint32_t Length) const { return Cache.findLineNumbersByVA(Address, Length); } std::unique_ptr NativeSession::findLineNumbersByRVA(uint32_t RVA, uint32_t Length) const { return Cache.findLineNumbersByVA(getLoadAddress() + RVA, Length); } std::unique_ptr NativeSession::findLineNumbersBySectOffset(uint32_t Section, uint32_t Offset, uint32_t Length) const { uint64_t VA = getVAFromSectOffset(Section, Offset); return Cache.findLineNumbersByVA(VA, Length); } std::unique_ptr NativeSession::findSourceFiles(const PDBSymbolCompiland *Compiland, StringRef Pattern, PDB_NameSearchFlags Flags) const { return nullptr; } std::unique_ptr NativeSession::findOneSourceFile(const PDBSymbolCompiland *Compiland, StringRef Pattern, PDB_NameSearchFlags Flags) const { return nullptr; } std::unique_ptr> NativeSession::findCompilandsForSourceFile(StringRef Pattern, PDB_NameSearchFlags Flags) const { return nullptr; } std::unique_ptr NativeSession::findOneCompilandForSourceFile(StringRef Pattern, PDB_NameSearchFlags Flags) const { return nullptr; } std::unique_ptr NativeSession::getAllSourceFiles() const { return nullptr; } std::unique_ptr NativeSession::getSourceFilesForCompiland( const PDBSymbolCompiland &Compiland) const { return nullptr; } std::unique_ptr NativeSession::getSourceFileById(uint32_t FileId) const { return Cache.getSourceFileById(FileId); } std::unique_ptr NativeSession::getDebugStreams() const { return nullptr; } std::unique_ptr NativeSession::getEnumTables() const { return nullptr; } std::unique_ptr NativeSession::getInjectedSources() const { auto ISS = Pdb->getInjectedSourceStream(); if (!ISS) { consumeError(ISS.takeError()); return nullptr; } auto Strings = Pdb->getStringTable(); if (!Strings) { consumeError(Strings.takeError()); return nullptr; } return std::make_unique(*Pdb, *ISS, *Strings); } std::unique_ptr NativeSession::getSectionContribs() const { return nullptr; } std::unique_ptr NativeSession::getFrameData() const { return nullptr; } void NativeSession::initializeExeSymbol() { if (ExeSymbol == 0) ExeSymbol = Cache.createSymbol(); } NativeExeSymbol &NativeSession::getNativeGlobalScope() const { const_cast(*this).initializeExeSymbol(); return Cache.getNativeSymbolById(ExeSymbol); } uint32_t NativeSession::getRVAFromSectOffset(uint32_t Section, uint32_t Offset) const { if (Section <= 0) return 0; auto Dbi = getDbiStreamPtr(*Pdb); if (!Dbi) return 0; uint32_t MaxSection = Dbi->getSectionHeaders().size(); if (Section > MaxSection + 1) Section = MaxSection + 1; auto &Sec = Dbi->getSectionHeaders()[Section - 1]; return Sec.VirtualAddress + Offset; } uint64_t NativeSession::getVAFromSectOffset(uint32_t Section, uint32_t Offset) const { return LoadAddress + getRVAFromSectOffset(Section, Offset); } bool NativeSession::moduleIndexForVA(uint64_t VA, uint16_t &ModuleIndex) const { ModuleIndex = 0; auto Iter = AddrToModuleIndex.find(VA); if (Iter == AddrToModuleIndex.end()) return false; ModuleIndex = Iter.value(); return true; } bool NativeSession::moduleIndexForSectOffset(uint32_t Sect, uint32_t Offset, uint16_t &ModuleIndex) const { ModuleIndex = 0; auto Iter = AddrToModuleIndex.find(getVAFromSectOffset(Sect, Offset)); if (Iter == AddrToModuleIndex.end()) return false; ModuleIndex = Iter.value(); return true; } void NativeSession::parseSectionContribs() { auto Dbi = Pdb->getPDBDbiStream(); if (!Dbi) return; class Visitor : public ISectionContribVisitor { NativeSession &Session; IMap &AddrMap; public: Visitor(NativeSession &Session, IMap &AddrMap) : Session(Session), AddrMap(AddrMap) {} void visit(const SectionContrib &C) override { if (C.Size == 0) return; uint64_t VA = Session.getVAFromSectOffset(C.ISect, C.Off); uint64_t End = VA + C.Size; // Ignore overlapping sections based on the assumption that a valid // PDB file should not have overlaps. if (!AddrMap.overlaps(VA, End)) AddrMap.insert(VA, End, C.Imod); } void visit(const SectionContrib2 &C) override { visit(C.Base); } }; Visitor V(*this, AddrToModuleIndex); Dbi->visitSectionContributions(V); } Expected NativeSession::getModuleDebugStream(uint32_t Index) const { auto *Dbi = getDbiStreamPtr(*Pdb); assert(Dbi && "Dbi stream not present"); DbiModuleDescriptor Modi = Dbi->modules().getModuleDescriptor(Index); uint16_t ModiStream = Modi.getModuleStreamIndex(); if (ModiStream == kInvalidStreamIndex) return make_error("Module stream not present"); std::unique_ptr ModStreamData = Pdb->createIndexedStream(ModiStream); ModuleDebugStreamRef ModS(Modi, std::move(ModStreamData)); if (auto EC = ModS.reload()) return std::move(EC); return std::move(ModS); }