| //===-- LLVMSymbolize.h ----------------------------------------- C++ -----===// |
| // |
| // The LLVM Compiler Infrastructure |
| // |
| // This file is distributed under the University of Illinois Open Source |
| // License. See LICENSE.TXT for details. |
| // |
| //===----------------------------------------------------------------------===// |
| // |
| // Header for LLVM symbolization library. |
| // |
| //===----------------------------------------------------------------------===// |
| #ifndef LLVM_TOOLS_LLVM_SYMBOLIZER_LLVMSYMBOLIZE_H |
| #define LLVM_TOOLS_LLVM_SYMBOLIZER_LLVMSYMBOLIZE_H |
| |
| #include "llvm/ADT/SmallVector.h" |
| #include "llvm/DebugInfo/DIContext.h" |
| #include "llvm/Object/MachOUniversal.h" |
| #include "llvm/Object/ObjectFile.h" |
| #include "llvm/Support/DataExtractor.h" |
| #include "llvm/Support/MemoryBuffer.h" |
| #include <map> |
| #include <memory> |
| #include <string> |
| |
| namespace llvm { |
| |
| typedef DILineInfoSpecifier::FunctionNameKind FunctionNameKind; |
| using namespace object; |
| |
| namespace symbolize { |
| |
| class ModuleInfo; |
| |
| class LLVMSymbolizer { |
| public: |
| struct Options { |
| bool UseSymbolTable : 1; |
| FunctionNameKind PrintFunctions; |
| bool PrintInlining : 1; |
| bool Demangle : 1; |
| std::string DefaultArch; |
| std::vector<std::string> DsymHints; |
| Options(bool UseSymbolTable = true, |
| FunctionNameKind PrintFunctions = FunctionNameKind::LinkageName, |
| bool PrintInlining = true, bool Demangle = true, |
| std::string DefaultArch = "") |
| : UseSymbolTable(UseSymbolTable), |
| PrintFunctions(PrintFunctions), PrintInlining(PrintInlining), |
| Demangle(Demangle), DefaultArch(DefaultArch) {} |
| }; |
| |
| LLVMSymbolizer(const Options &Opts = Options()) : Opts(Opts) {} |
| ~LLVMSymbolizer() { |
| flush(); |
| } |
| |
| // Returns the result of symbolization for module name/offset as |
| // a string (possibly containing newlines). |
| std::string |
| symbolizeCode(const std::string &ModuleName, uint64_t ModuleOffset); |
| std::string |
| symbolizeData(const std::string &ModuleName, uint64_t ModuleOffset); |
| void flush(); |
| static std::string DemangleName(const std::string &Name); |
| private: |
| typedef std::pair<ObjectFile*, ObjectFile*> ObjectPair; |
| |
| ModuleInfo *getOrCreateModuleInfo(const std::string &ModuleName); |
| ObjectFile *lookUpDsymFile(const std::string &Path, const MachOObjectFile *ExeObj, |
| const std::string &ArchName); |
| |
| /// \brief Returns pair of pointers to object and debug object. |
| ObjectPair getOrCreateObjects(const std::string &Path, |
| const std::string &ArchName); |
| /// \brief Returns a parsed object file for a given architecture in a |
| /// universal binary (or the binary itself if it is an object file). |
| ObjectFile *getObjectFileFromBinary(Binary *Bin, const std::string &ArchName); |
| |
| std::string printDILineInfo(DILineInfo LineInfo) const; |
| |
| // Owns all the parsed binaries and object files. |
| SmallVector<std::unique_ptr<Binary>, 4> ParsedBinariesAndObjects; |
| SmallVector<std::unique_ptr<MemoryBuffer>, 4> MemoryBuffers; |
| void addOwningBinary(OwningBinary<Binary> OwningBin) { |
| std::unique_ptr<Binary> Bin; |
| std::unique_ptr<MemoryBuffer> MemBuf; |
| std::tie(Bin, MemBuf) = OwningBin.takeBinary(); |
| ParsedBinariesAndObjects.push_back(std::move(Bin)); |
| MemoryBuffers.push_back(std::move(MemBuf)); |
| } |
| |
| // Owns module info objects. |
| std::map<std::string, ModuleInfo *> Modules; |
| std::map<std::pair<MachOUniversalBinary *, std::string>, ObjectFile *> |
| ObjectFileForArch; |
| std::map<std::pair<std::string, std::string>, ObjectPair> |
| ObjectPairForPathArch; |
| |
| Options Opts; |
| static const char kBadString[]; |
| }; |
| |
| class ModuleInfo { |
| public: |
| ModuleInfo(ObjectFile *Obj, DIContext *DICtx); |
| |
| DILineInfo symbolizeCode(uint64_t ModuleOffset, |
| const LLVMSymbolizer::Options &Opts) const; |
| DIInliningInfo symbolizeInlinedCode( |
| uint64_t ModuleOffset, const LLVMSymbolizer::Options &Opts) const; |
| bool symbolizeData(uint64_t ModuleOffset, std::string &Name, uint64_t &Start, |
| uint64_t &Size) const; |
| |
| private: |
| bool getNameFromSymbolTable(SymbolRef::Type Type, uint64_t Address, |
| std::string &Name, uint64_t &Addr, |
| uint64_t &Size) const; |
| // For big-endian PowerPC64 ELF, OpdAddress is the address of the .opd |
| // (function descriptor) section and OpdExtractor refers to its contents. |
| void addSymbol(const SymbolRef &Symbol, |
| DataExtractor *OpdExtractor = nullptr, |
| uint64_t OpdAddress = 0); |
| ObjectFile *Module; |
| std::unique_ptr<DIContext> DebugInfoContext; |
| |
| struct SymbolDesc { |
| uint64_t Addr; |
| // If size is 0, assume that symbol occupies the whole memory range up to |
| // the following symbol. |
| uint64_t Size; |
| friend bool operator<(const SymbolDesc &s1, const SymbolDesc &s2) { |
| return s1.Addr < s2.Addr; |
| } |
| }; |
| std::map<SymbolDesc, StringRef> Functions; |
| std::map<SymbolDesc, StringRef> Objects; |
| }; |
| |
| } // namespace symbolize |
| } // namespace llvm |
| |
| #endif |