//===- PDBSymbolCompiland.cpp - compiland details ---------------*- C++ -*-===// // // The LLVM Compiler Infrastructure // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// #include "llvm/DebugInfo/PDB/IPDBSession.h" #include "llvm/DebugInfo/PDB/IPDBSourceFile.h" #include "llvm/DebugInfo/PDB/PDBSymbolCompiland.h" #include "llvm/DebugInfo/PDB/PDBSymbolCompilandDetails.h" #include "llvm/DebugInfo/PDB/PDBSymbolCompilandEnv.h" #include "llvm/DebugInfo/PDB/PDBSymDumper.h" #include "llvm/ADT/StringSwitch.h" #include "llvm/Support/Path.h" #include <utility> using namespace llvm; using namespace llvm::pdb; PDBSymbolCompiland::PDBSymbolCompiland(const IPDBSession &PDBSession, std::unique_ptr<IPDBRawSymbol> Symbol) : PDBSymbol(PDBSession, std::move(Symbol)) { assert(RawSymbol->getSymTag() == PDB_SymType::Compiland); } void PDBSymbolCompiland::dump(PDBSymDumper &Dumper) const { Dumper.dump(*this); } std::string PDBSymbolCompiland::getSourceFileName() const { return sys::path::filename(getSourceFileFullPath()).str(); } std::string PDBSymbolCompiland::getSourceFileFullPath() const { std::string SourceFileFullPath; // RecordedResult could be the basename, relative path or full path of the // source file. Usually it is retrieved and recorded from the command that // compiles this compiland. // // cmd FileName -> RecordedResult = .\\FileName // cmd (Path)\\FileName -> RecordedResult = (Path)\\FileName // std::string RecordedResult = RawSymbol->getSourceFileName(); if (RecordedResult.empty()) { if (auto Envs = findAllChildren<PDBSymbolCompilandEnv>()) { std::string EnvWorkingDir, EnvSrc; while (auto Env = Envs->getNext()) { std::string Var = Env->getName(); if (Var == "cwd") { EnvWorkingDir = Env->getValue(); continue; } if (Var == "src") { EnvSrc = Env->getValue(); if (sys::path::is_absolute(EnvSrc)) return EnvSrc; RecordedResult = EnvSrc; continue; } } if (!EnvWorkingDir.empty() && !EnvSrc.empty()) { auto Len = EnvWorkingDir.length(); if (EnvWorkingDir[Len - 1] != '/' && EnvWorkingDir[Len - 1] != '\\') { std::string Path = EnvWorkingDir + "\\" + EnvSrc; std::replace(Path.begin(), Path.end(), '/', '\\'); // We will return it as full path if we can't find a better one. if (sys::path::is_absolute(Path)) SourceFileFullPath = Path; } } } } if (!RecordedResult.empty()) { if (sys::path::is_absolute(RecordedResult)) return RecordedResult; // This searches name that has same basename as the one in RecordedResult. auto OneSrcFile = Session.findOneSourceFile( this, RecordedResult, PDB_NameSearchFlags::NS_CaseInsensitive); if (OneSrcFile) return OneSrcFile->getFileName(); } // At this point, we have to walk through all source files of this compiland, // and determine the right source file if any that is used to generate this // compiland based on language indicated in compilanddetails language field. auto Details = findOneChild<PDBSymbolCompilandDetails>(); PDB_Lang Lang = Details ? Details->getLanguage() : PDB_Lang::Cpp; auto SrcFiles = Session.getSourceFilesForCompiland(*this); if (SrcFiles) { bool LangC = (Lang == PDB_Lang::Cpp || Lang == PDB_Lang::C); while (auto File = SrcFiles->getNext()) { std::string FileName = File->getFileName(); auto file_extension = sys::path::extension(FileName); if (StringSwitch<bool>(file_extension.lower()) .Case(".cpp", LangC) .Case(".c", LangC) .Case(".cc", LangC) .Case(".cxx", LangC) .Case(".asm", Lang == PDB_Lang::Masm) .Default(false)) return File->getFileName(); } } return SourceFileFullPath; }