123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471 |
- //===- 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 <algorithm>
- #include <cassert>
- #include <memory>
- #include <utility>
- 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<DbiStream &> DbiS = File.getPDBDbiStream();
- if (DbiS)
- return &DbiS.get();
- consumeError(DbiS.takeError());
- return nullptr;
- }
- NativeSession::NativeSession(std::unique_ptr<PDBFile> PdbFile,
- std::unique_ptr<BumpPtrAllocator> Allocator)
- : Pdb(std::move(PdbFile)), Allocator(std::move(Allocator)),
- Cache(*this, getDbiStreamPtr(*Pdb)), AddrToModuleIndex(IMapAllocator) {}
- NativeSession::~NativeSession() = default;
- Error NativeSession::createFromPdb(std::unique_ptr<MemoryBuffer> Buffer,
- std::unique_ptr<IPDBSession> &Session) {
- StringRef Path = Buffer->getBufferIdentifier();
- auto Stream = std::make_unique<MemoryBufferByteStream>(
- std::move(Buffer), llvm::support::little);
- auto Allocator = std::make_unique<BumpPtrAllocator>();
- auto File = std::make_unique<PDBFile>(Path, std::move(Stream), *Allocator);
- if (auto EC = File->parseFileHeaders())
- return EC;
- if (auto EC = File->parseStreamData())
- return EC;
- Session =
- std::make_unique<NativeSession>(std::move(File), std::move(Allocator));
- return Error::success();
- }
- static Expected<std::unique_ptr<PDBFile>>
- loadPdbFile(StringRef PdbPath, std::unique_ptr<BumpPtrAllocator> &Allocator) {
- ErrorOr<std::unique_ptr<MemoryBuffer>> ErrorOrBuffer =
- MemoryBuffer::getFile(PdbPath, /*IsText=*/false,
- /*RequiresNullTerminator=*/false);
- if (!ErrorOrBuffer)
- return make_error<RawError>(ErrorOrBuffer.getError());
- std::unique_ptr<llvm::MemoryBuffer> 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<RawError>(EC);
- auto Stream = std::make_unique<MemoryBufferByteStream>(std::move(Buffer),
- llvm::support::little);
- auto File = std::make_unique<PDBFile>(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<IPDBSession> &Session) {
- auto Allocator = std::make_unique<BumpPtrAllocator>();
- auto PdbFile = loadPdbFile(PdbPath, Allocator);
- if (!PdbFile)
- return PdbFile.takeError();
- Session = std::make_unique<NativeSession>(std::move(PdbFile.get()),
- std::move(Allocator));
- return Error::success();
- }
- static Expected<std::string> getPdbPathFromExe(StringRef ExePath) {
- Expected<object::OwningBinary<object::Binary>> BinaryFile =
- object::createBinary(ExePath);
- if (!BinaryFile)
- return BinaryFile.takeError();
- const object::COFFObjectFile *ObjFile =
- dyn_cast<object::COFFObjectFile>(BinaryFile->getBinary());
- if (!ObjFile)
- return make_error<RawError>(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<IPDBSession> &Session) {
- Expected<std::string> 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<RawError>(EC);
- auto Allocator = std::make_unique<BumpPtrAllocator>();
- auto File = loadPdbFile(PdbPath.get(), Allocator);
- if (!File)
- return File.takeError();
- Session = std::make_unique<NativeSession>(std::move(File.get()),
- std::move(Allocator));
- return Error::success();
- }
- Expected<std::string>
- NativeSession::searchForPdb(const PdbSearchOptions &Opts) {
- Expected<std::string> 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<BumpPtrAllocator>();
- 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<RawError>("PDB not found");
- }
- uint64_t NativeSession::getLoadAddress() const { return LoadAddress; }
- bool NativeSession::setLoadAddress(uint64_t Address) {
- LoadAddress = Address;
- return true;
- }
- std::unique_ptr<PDBSymbolExe> NativeSession::getGlobalScope() {
- return PDBSymbol::createAs<PDBSymbolExe>(*this, getNativeGlobalScope());
- }
- std::unique_ptr<PDBSymbol>
- 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<PDBSymbol>
- 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<PDBSymbol> 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<PDBSymbol>
- NativeSession::findSymbolBySectOffset(uint32_t Sect, uint32_t Offset,
- PDB_SymType Type) {
- if (AddrToModuleIndex.empty())
- parseSectionContribs();
- return Cache.findSymbolBySectOffset(Sect, Offset, Type);
- }
- std::unique_ptr<IPDBEnumLineNumbers>
- NativeSession::findLineNumbers(const PDBSymbolCompiland &Compiland,
- const IPDBSourceFile &File) const {
- return nullptr;
- }
- std::unique_ptr<IPDBEnumLineNumbers>
- NativeSession::findLineNumbersByAddress(uint64_t Address,
- uint32_t Length) const {
- return Cache.findLineNumbersByVA(Address, Length);
- }
- std::unique_ptr<IPDBEnumLineNumbers>
- NativeSession::findLineNumbersByRVA(uint32_t RVA, uint32_t Length) const {
- return Cache.findLineNumbersByVA(getLoadAddress() + RVA, Length);
- }
- std::unique_ptr<IPDBEnumLineNumbers>
- 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<IPDBEnumSourceFiles>
- NativeSession::findSourceFiles(const PDBSymbolCompiland *Compiland,
- StringRef Pattern,
- PDB_NameSearchFlags Flags) const {
- return nullptr;
- }
- std::unique_ptr<IPDBSourceFile>
- NativeSession::findOneSourceFile(const PDBSymbolCompiland *Compiland,
- StringRef Pattern,
- PDB_NameSearchFlags Flags) const {
- return nullptr;
- }
- std::unique_ptr<IPDBEnumChildren<PDBSymbolCompiland>>
- NativeSession::findCompilandsForSourceFile(StringRef Pattern,
- PDB_NameSearchFlags Flags) const {
- return nullptr;
- }
- std::unique_ptr<PDBSymbolCompiland>
- NativeSession::findOneCompilandForSourceFile(StringRef Pattern,
- PDB_NameSearchFlags Flags) const {
- return nullptr;
- }
- std::unique_ptr<IPDBEnumSourceFiles> NativeSession::getAllSourceFiles() const {
- return nullptr;
- }
- std::unique_ptr<IPDBEnumSourceFiles> NativeSession::getSourceFilesForCompiland(
- const PDBSymbolCompiland &Compiland) const {
- return nullptr;
- }
- std::unique_ptr<IPDBSourceFile>
- NativeSession::getSourceFileById(uint32_t FileId) const {
- return Cache.getSourceFileById(FileId);
- }
- std::unique_ptr<IPDBEnumDataStreams> NativeSession::getDebugStreams() const {
- return nullptr;
- }
- std::unique_ptr<IPDBEnumTables> NativeSession::getEnumTables() const {
- return nullptr;
- }
- std::unique_ptr<IPDBEnumInjectedSources>
- 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<NativeEnumInjectedSources>(*Pdb, *ISS, *Strings);
- }
- std::unique_ptr<IPDBEnumSectionContribs>
- NativeSession::getSectionContribs() const {
- return nullptr;
- }
- std::unique_ptr<IPDBEnumFrameData>
- NativeSession::getFrameData() const {
- return nullptr;
- }
- void NativeSession::initializeExeSymbol() {
- if (ExeSymbol == 0)
- ExeSymbol = Cache.createSymbol<NativeExeSymbol>();
- }
- NativeExeSymbol &NativeSession::getNativeGlobalScope() const {
- const_cast<NativeSession &>(*this).initializeExeSymbol();
- return Cache.getNativeSymbolById<NativeExeSymbol>(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<ModuleDebugStreamRef>
- 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<RawError>("Module stream not present");
- std::unique_ptr<msf::MappedBlockStream> ModStreamData =
- Pdb->createIndexedStream(ModiStream);
- ModuleDebugStreamRef ModS(Modi, std::move(ModStreamData));
- if (auto EC = ModS.reload())
- return std::move(EC);
- return std::move(ModS);
- }
|