1061da546Spatrick //===-- SymbolFileDWARFDebugMap.h ------------------------------*- C++ -*-===// 2061da546Spatrick // 3061da546Spatrick // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 4061da546Spatrick // See https://llvm.org/LICENSE.txt for license information. 5061da546Spatrick // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 6061da546Spatrick // 7061da546Spatrick //===----------------------------------------------------------------------===// 8061da546Spatrick 9*dda28197Spatrick #ifndef LLDB_SOURCE_PLUGINS_SYMBOLFILE_DWARF_SYMBOLFILEDWARFDEBUGMAP_H 10*dda28197Spatrick #define LLDB_SOURCE_PLUGINS_SYMBOLFILE_DWARF_SYMBOLFILEDWARFDEBUGMAP_H 11061da546Spatrick 12061da546Spatrick #include "lldb/Symbol/SymbolFile.h" 13061da546Spatrick #include "lldb/Utility/RangeMap.h" 14061da546Spatrick #include "llvm/Support/Chrono.h" 15061da546Spatrick #include <bitset> 16061da546Spatrick #include <map> 17061da546Spatrick #include <vector> 18061da546Spatrick 19061da546Spatrick #include "UniqueDWARFASTType.h" 20061da546Spatrick 21061da546Spatrick class SymbolFileDWARF; 22061da546Spatrick class DWARFDebugAranges; 23061da546Spatrick class DWARFDeclContext; 24061da546Spatrick 25061da546Spatrick class SymbolFileDWARFDebugMap : public lldb_private::SymbolFile { 26061da546Spatrick /// LLVM RTTI support. 27061da546Spatrick static char ID; 28061da546Spatrick 29061da546Spatrick public: 30061da546Spatrick /// LLVM RTTI support. 31061da546Spatrick /// \{ 32061da546Spatrick bool isA(const void *ClassID) const override { 33061da546Spatrick return ClassID == &ID || SymbolFile::isA(ClassID); 34061da546Spatrick } 35061da546Spatrick static bool classof(const SymbolFile *obj) { return obj->isA(&ID); } 36061da546Spatrick /// \} 37061da546Spatrick 38061da546Spatrick // Static Functions 39061da546Spatrick static void Initialize(); 40061da546Spatrick 41061da546Spatrick static void Terminate(); 42061da546Spatrick 43061da546Spatrick static lldb_private::ConstString GetPluginNameStatic(); 44061da546Spatrick 45061da546Spatrick static const char *GetPluginDescriptionStatic(); 46061da546Spatrick 47061da546Spatrick static lldb_private::SymbolFile * 48061da546Spatrick CreateInstance(lldb::ObjectFileSP objfile_sp); 49061da546Spatrick 50061da546Spatrick // Constructors and Destructors 51061da546Spatrick SymbolFileDWARFDebugMap(lldb::ObjectFileSP objfile_sp); 52061da546Spatrick ~SymbolFileDWARFDebugMap() override; 53061da546Spatrick 54061da546Spatrick uint32_t CalculateAbilities() override; 55061da546Spatrick void InitializeObject() override; 56061da546Spatrick 57061da546Spatrick // Compile Unit function calls 58061da546Spatrick lldb::LanguageType 59061da546Spatrick ParseLanguage(lldb_private::CompileUnit &comp_unit) override; 60*dda28197Spatrick lldb_private::XcodeSDK 61*dda28197Spatrick ParseXcodeSDK(lldb_private::CompileUnit &comp_unit) override; 62061da546Spatrick size_t ParseFunctions(lldb_private::CompileUnit &comp_unit) override; 63061da546Spatrick bool ParseLineTable(lldb_private::CompileUnit &comp_unit) override; 64061da546Spatrick bool ParseDebugMacros(lldb_private::CompileUnit &comp_unit) override; 65061da546Spatrick 66061da546Spatrick bool ForEachExternalModule( 67061da546Spatrick lldb_private::CompileUnit &, llvm::DenseSet<lldb_private::SymbolFile *> &, 68061da546Spatrick llvm::function_ref<bool(lldb_private::Module &)>) override; 69061da546Spatrick 70061da546Spatrick bool ParseSupportFiles(lldb_private::CompileUnit &comp_unit, 71061da546Spatrick lldb_private::FileSpecList &support_files) override; 72061da546Spatrick 73061da546Spatrick bool ParseIsOptimized(lldb_private::CompileUnit &comp_unit) override; 74061da546Spatrick 75061da546Spatrick size_t ParseTypes(lldb_private::CompileUnit &comp_unit) override; 76061da546Spatrick 77061da546Spatrick bool ParseImportedModules( 78061da546Spatrick const lldb_private::SymbolContext &sc, 79061da546Spatrick std::vector<lldb_private::SourceModule> &imported_modules) override; 80061da546Spatrick size_t ParseBlocksRecursive(lldb_private::Function &func) override; 81061da546Spatrick size_t 82061da546Spatrick ParseVariablesForContext(const lldb_private::SymbolContext &sc) override; 83061da546Spatrick 84061da546Spatrick lldb_private::Type *ResolveTypeUID(lldb::user_id_t type_uid) override; 85061da546Spatrick llvm::Optional<ArrayInfo> GetDynamicArrayInfoForUID( 86061da546Spatrick lldb::user_id_t type_uid, 87061da546Spatrick const lldb_private::ExecutionContext *exe_ctx) override; 88061da546Spatrick 89061da546Spatrick lldb_private::CompilerDeclContext 90061da546Spatrick GetDeclContextForUID(lldb::user_id_t uid) override; 91061da546Spatrick lldb_private::CompilerDeclContext 92061da546Spatrick GetDeclContextContainingUID(lldb::user_id_t uid) override; 93061da546Spatrick void 94061da546Spatrick ParseDeclsForContext(lldb_private::CompilerDeclContext decl_ctx) override; 95061da546Spatrick 96061da546Spatrick bool CompleteType(lldb_private::CompilerType &compiler_type) override; 97061da546Spatrick uint32_t ResolveSymbolContext(const lldb_private::Address &so_addr, 98061da546Spatrick lldb::SymbolContextItem resolve_scope, 99061da546Spatrick lldb_private::SymbolContext &sc) override; 100061da546Spatrick uint32_t 101061da546Spatrick ResolveSymbolContext(const lldb_private::FileSpec &file_spec, uint32_t line, 102061da546Spatrick bool check_inlines, 103061da546Spatrick lldb::SymbolContextItem resolve_scope, 104061da546Spatrick lldb_private::SymbolContextList &sc_list) override; 105061da546Spatrick void 106061da546Spatrick FindGlobalVariables(lldb_private::ConstString name, 107*dda28197Spatrick const lldb_private::CompilerDeclContext &parent_decl_ctx, 108061da546Spatrick uint32_t max_matches, 109061da546Spatrick lldb_private::VariableList &variables) override; 110061da546Spatrick void FindGlobalVariables(const lldb_private::RegularExpression ®ex, 111061da546Spatrick uint32_t max_matches, 112061da546Spatrick lldb_private::VariableList &variables) override; 113061da546Spatrick void FindFunctions(lldb_private::ConstString name, 114*dda28197Spatrick const lldb_private::CompilerDeclContext &parent_decl_ctx, 115061da546Spatrick lldb::FunctionNameType name_type_mask, 116061da546Spatrick bool include_inlines, 117061da546Spatrick lldb_private::SymbolContextList &sc_list) override; 118061da546Spatrick void FindFunctions(const lldb_private::RegularExpression ®ex, 119061da546Spatrick bool include_inlines, 120061da546Spatrick lldb_private::SymbolContextList &sc_list) override; 121061da546Spatrick void 122061da546Spatrick FindTypes(lldb_private::ConstString name, 123*dda28197Spatrick const lldb_private::CompilerDeclContext &parent_decl_ctx, 124061da546Spatrick uint32_t max_matches, 125061da546Spatrick llvm::DenseSet<lldb_private::SymbolFile *> &searched_symbol_files, 126061da546Spatrick lldb_private::TypeMap &types) override; 127061da546Spatrick void 128061da546Spatrick FindTypes(llvm::ArrayRef<lldb_private::CompilerContext> context, 129061da546Spatrick lldb_private::LanguageSet languages, 130061da546Spatrick llvm::DenseSet<lldb_private::SymbolFile *> &searched_symbol_files, 131061da546Spatrick lldb_private::TypeMap &types) override; 132061da546Spatrick lldb_private::CompilerDeclContext FindNamespace( 133061da546Spatrick lldb_private::ConstString name, 134*dda28197Spatrick const lldb_private::CompilerDeclContext &parent_decl_ctx) override; 135061da546Spatrick void GetTypes(lldb_private::SymbolContextScope *sc_scope, 136061da546Spatrick lldb::TypeClass type_mask, 137061da546Spatrick lldb_private::TypeList &type_list) override; 138061da546Spatrick std::vector<std::unique_ptr<lldb_private::CallEdge>> 139061da546Spatrick ParseCallEdgesInFunction(lldb_private::UserID func_id) override; 140061da546Spatrick 141061da546Spatrick void DumpClangAST(lldb_private::Stream &s) override; 142061da546Spatrick 143061da546Spatrick // PluginInterface protocol 144061da546Spatrick lldb_private::ConstString GetPluginName() override; 145061da546Spatrick 146061da546Spatrick uint32_t GetPluginVersion() override; 147061da546Spatrick 148061da546Spatrick protected: 149061da546Spatrick enum { kHaveInitializedOSOs = (1 << 0), kNumFlags }; 150061da546Spatrick 151061da546Spatrick friend class DebugMapModule; 152061da546Spatrick friend class DWARFASTParserClang; 153061da546Spatrick friend class DWARFCompileUnit; 154061da546Spatrick friend class SymbolFileDWARF; 155061da546Spatrick struct OSOInfo { 156061da546Spatrick lldb::ModuleSP module_sp; 157061da546Spatrick 158061da546Spatrick OSOInfo() : module_sp() {} 159061da546Spatrick }; 160061da546Spatrick 161061da546Spatrick typedef std::shared_ptr<OSOInfo> OSOInfoSP; 162061da546Spatrick 163061da546Spatrick typedef lldb_private::RangeDataVector<lldb::addr_t, lldb::addr_t, 164061da546Spatrick lldb::addr_t> 165061da546Spatrick FileRangeMap; 166061da546Spatrick 167061da546Spatrick // Class specific types 168061da546Spatrick struct CompileUnitInfo { 169061da546Spatrick lldb_private::FileSpec so_file; 170061da546Spatrick lldb_private::ConstString oso_path; 171061da546Spatrick llvm::sys::TimePoint<> oso_mod_time; 172061da546Spatrick OSOInfoSP oso_sp; 173061da546Spatrick lldb::CompUnitSP compile_unit_sp; 174061da546Spatrick uint32_t first_symbol_index; 175061da546Spatrick uint32_t last_symbol_index; 176061da546Spatrick uint32_t first_symbol_id; 177061da546Spatrick uint32_t last_symbol_id; 178061da546Spatrick FileRangeMap file_range_map; 179061da546Spatrick bool file_range_map_valid; 180061da546Spatrick 181061da546Spatrick CompileUnitInfo() 182061da546Spatrick : so_file(), oso_path(), oso_mod_time(), oso_sp(), compile_unit_sp(), 183061da546Spatrick first_symbol_index(UINT32_MAX), last_symbol_index(UINT32_MAX), 184061da546Spatrick first_symbol_id(UINT32_MAX), last_symbol_id(UINT32_MAX), 185061da546Spatrick file_range_map(), file_range_map_valid(false) {} 186061da546Spatrick 187061da546Spatrick const FileRangeMap &GetFileRangeMap(SymbolFileDWARFDebugMap *exe_symfile); 188061da546Spatrick }; 189061da546Spatrick 190061da546Spatrick // Protected Member Functions 191061da546Spatrick void InitOSO(); 192061da546Spatrick 193061da546Spatrick uint32_t CalculateNumCompileUnits() override; 194061da546Spatrick lldb::CompUnitSP ParseCompileUnitAtIndex(uint32_t index) override; 195061da546Spatrick 196061da546Spatrick static uint32_t GetOSOIndexFromUserID(lldb::user_id_t uid) { 197061da546Spatrick return (uint32_t)((uid >> 32ull) - 1ull); 198061da546Spatrick } 199061da546Spatrick 200061da546Spatrick static SymbolFileDWARF *GetSymbolFileAsSymbolFileDWARF(SymbolFile *sym_file); 201061da546Spatrick 202061da546Spatrick bool GetFileSpecForSO(uint32_t oso_idx, lldb_private::FileSpec &file_spec); 203061da546Spatrick 204061da546Spatrick CompileUnitInfo *GetCompUnitInfo(const lldb_private::SymbolContext &sc); 205061da546Spatrick CompileUnitInfo *GetCompUnitInfo(const lldb_private::CompileUnit &comp_unit); 206061da546Spatrick 207061da546Spatrick size_t GetCompUnitInfosForModule(const lldb_private::Module *oso_module, 208061da546Spatrick std::vector<CompileUnitInfo *> &cu_infos); 209061da546Spatrick 210061da546Spatrick lldb_private::Module * 211061da546Spatrick GetModuleByCompUnitInfo(CompileUnitInfo *comp_unit_info); 212061da546Spatrick 213061da546Spatrick lldb_private::Module *GetModuleByOSOIndex(uint32_t oso_idx); 214061da546Spatrick 215061da546Spatrick lldb_private::ObjectFile * 216061da546Spatrick GetObjectFileByCompUnitInfo(CompileUnitInfo *comp_unit_info); 217061da546Spatrick 218061da546Spatrick lldb_private::ObjectFile *GetObjectFileByOSOIndex(uint32_t oso_idx); 219061da546Spatrick 220061da546Spatrick uint32_t GetCompUnitInfoIndex(const CompileUnitInfo *comp_unit_info); 221061da546Spatrick 222061da546Spatrick SymbolFileDWARF *GetSymbolFile(const lldb_private::SymbolContext &sc); 223061da546Spatrick SymbolFileDWARF *GetSymbolFile(const lldb_private::CompileUnit &comp_unit); 224061da546Spatrick 225061da546Spatrick SymbolFileDWARF *GetSymbolFileByCompUnitInfo(CompileUnitInfo *comp_unit_info); 226061da546Spatrick 227061da546Spatrick SymbolFileDWARF *GetSymbolFileByOSOIndex(uint32_t oso_idx); 228061da546Spatrick 229061da546Spatrick // If closure returns "false", iteration continues. If it returns 230061da546Spatrick // "true", iteration terminates. 231061da546Spatrick void ForEachSymbolFile(std::function<bool(SymbolFileDWARF *)> closure) { 232061da546Spatrick for (uint32_t oso_idx = 0, num_oso_idxs = m_compile_unit_infos.size(); 233061da546Spatrick oso_idx < num_oso_idxs; ++oso_idx) { 234061da546Spatrick if (SymbolFileDWARF *oso_dwarf = GetSymbolFileByOSOIndex(oso_idx)) { 235061da546Spatrick if (closure(oso_dwarf)) 236061da546Spatrick return; 237061da546Spatrick } 238061da546Spatrick } 239061da546Spatrick } 240061da546Spatrick 241061da546Spatrick CompileUnitInfo *GetCompileUnitInfoForSymbolWithIndex(uint32_t symbol_idx, 242061da546Spatrick uint32_t *oso_idx_ptr); 243061da546Spatrick 244061da546Spatrick CompileUnitInfo *GetCompileUnitInfoForSymbolWithID(lldb::user_id_t symbol_id, 245061da546Spatrick uint32_t *oso_idx_ptr); 246061da546Spatrick 247061da546Spatrick static int 248061da546Spatrick SymbolContainsSymbolWithIndex(uint32_t *symbol_idx_ptr, 249061da546Spatrick const CompileUnitInfo *comp_unit_info); 250061da546Spatrick 251061da546Spatrick static int SymbolContainsSymbolWithID(lldb::user_id_t *symbol_idx_ptr, 252061da546Spatrick const CompileUnitInfo *comp_unit_info); 253061da546Spatrick 254061da546Spatrick void PrivateFindGlobalVariables( 255061da546Spatrick lldb_private::ConstString name, 256*dda28197Spatrick const lldb_private::CompilerDeclContext &parent_decl_ctx, 257061da546Spatrick const std::vector<uint32_t> &name_symbol_indexes, uint32_t max_matches, 258061da546Spatrick lldb_private::VariableList &variables); 259061da546Spatrick 260061da546Spatrick void SetCompileUnit(SymbolFileDWARF *oso_dwarf, 261061da546Spatrick const lldb::CompUnitSP &cu_sp); 262061da546Spatrick 263061da546Spatrick lldb::CompUnitSP GetCompileUnit(SymbolFileDWARF *oso_dwarf); 264061da546Spatrick 265061da546Spatrick CompileUnitInfo *GetCompileUnitInfo(SymbolFileDWARF *oso_dwarf); 266061da546Spatrick 267061da546Spatrick lldb::TypeSP 268061da546Spatrick FindDefinitionTypeForDWARFDeclContext(const DWARFDeclContext &die_decl_ctx); 269061da546Spatrick 270061da546Spatrick bool Supports_DW_AT_APPLE_objc_complete_type(SymbolFileDWARF *skip_dwarf_oso); 271061da546Spatrick 272061da546Spatrick lldb::TypeSP FindCompleteObjCDefinitionTypeForDIE( 273061da546Spatrick const DWARFDIE &die, lldb_private::ConstString type_name, 274061da546Spatrick bool must_be_implementation); 275061da546Spatrick 276061da546Spatrick UniqueDWARFASTTypeMap &GetUniqueDWARFASTTypeMap() { 277061da546Spatrick return m_unique_ast_type_map; 278061da546Spatrick } 279061da546Spatrick 280061da546Spatrick // OSOEntry 281061da546Spatrick class OSOEntry { 282061da546Spatrick public: 283061da546Spatrick OSOEntry() 284061da546Spatrick : m_exe_sym_idx(UINT32_MAX), m_oso_file_addr(LLDB_INVALID_ADDRESS) {} 285061da546Spatrick 286061da546Spatrick OSOEntry(uint32_t exe_sym_idx, lldb::addr_t oso_file_addr) 287061da546Spatrick : m_exe_sym_idx(exe_sym_idx), m_oso_file_addr(oso_file_addr) {} 288061da546Spatrick 289061da546Spatrick uint32_t GetExeSymbolIndex() const { return m_exe_sym_idx; } 290061da546Spatrick 291061da546Spatrick bool operator<(const OSOEntry &rhs) const { 292061da546Spatrick return m_exe_sym_idx < rhs.m_exe_sym_idx; 293061da546Spatrick } 294061da546Spatrick 295061da546Spatrick lldb::addr_t GetOSOFileAddress() const { return m_oso_file_addr; } 296061da546Spatrick 297061da546Spatrick void SetOSOFileAddress(lldb::addr_t oso_file_addr) { 298061da546Spatrick m_oso_file_addr = oso_file_addr; 299061da546Spatrick } 300061da546Spatrick 301061da546Spatrick protected: 302061da546Spatrick uint32_t m_exe_sym_idx; 303061da546Spatrick lldb::addr_t m_oso_file_addr; 304061da546Spatrick }; 305061da546Spatrick 306061da546Spatrick typedef lldb_private::RangeDataVector<lldb::addr_t, lldb::addr_t, OSOEntry> 307061da546Spatrick DebugMap; 308061da546Spatrick 309061da546Spatrick // Member Variables 310061da546Spatrick std::bitset<kNumFlags> m_flags; 311061da546Spatrick std::vector<CompileUnitInfo> m_compile_unit_infos; 312061da546Spatrick std::vector<uint32_t> m_func_indexes; // Sorted by address 313061da546Spatrick std::vector<uint32_t> m_glob_indexes; 314061da546Spatrick std::map<std::pair<lldb_private::ConstString, llvm::sys::TimePoint<>>, 315061da546Spatrick OSOInfoSP> 316061da546Spatrick m_oso_map; 317061da546Spatrick UniqueDWARFASTTypeMap m_unique_ast_type_map; 318061da546Spatrick lldb_private::LazyBool m_supports_DW_AT_APPLE_objc_complete_type; 319061da546Spatrick DebugMap m_debug_map; 320061da546Spatrick 321061da546Spatrick // When an object file from the debug map gets parsed in 322061da546Spatrick // SymbolFileDWARF, it needs to tell the debug map about the object 323061da546Spatrick // files addresses by calling this function once for each N_FUN, 324061da546Spatrick // N_GSYM and N_STSYM and after all entries in the debug map have 325061da546Spatrick // been matched up, FinalizeOSOFileRanges() should be called. 326061da546Spatrick bool AddOSOFileRange(CompileUnitInfo *cu_info, lldb::addr_t exe_file_addr, 327061da546Spatrick lldb::addr_t exe_byte_size, lldb::addr_t oso_file_addr, 328061da546Spatrick lldb::addr_t oso_byte_size); 329061da546Spatrick 330061da546Spatrick // Called after calling AddOSOFileRange() for each object file debug 331061da546Spatrick // map entry to finalize the info for the unlinked compile unit. 332061da546Spatrick void FinalizeOSOFileRanges(CompileUnitInfo *cu_info); 333061da546Spatrick 334061da546Spatrick /// Convert \a addr from a .o file address, to an executable address. 335061da546Spatrick /// 336061da546Spatrick /// \param[in] addr 337061da546Spatrick /// A section offset address from a .o file 338061da546Spatrick /// 339061da546Spatrick /// \return 340061da546Spatrick /// Returns true if \a addr was converted to be an executable 341061da546Spatrick /// section/offset address, false otherwise. 342061da546Spatrick bool LinkOSOAddress(lldb_private::Address &addr); 343061da546Spatrick 344061da546Spatrick /// Convert a .o file "file address" to an executable "file address". 345061da546Spatrick /// 346061da546Spatrick /// \param[in] oso_symfile 347061da546Spatrick /// The DWARF symbol file that contains \a oso_file_addr 348061da546Spatrick /// 349061da546Spatrick /// \param[in] oso_file_addr 350061da546Spatrick /// A .o file "file address" to convert. 351061da546Spatrick /// 352061da546Spatrick /// \return 353061da546Spatrick /// LLDB_INVALID_ADDRESS if \a oso_file_addr is not in the 354061da546Spatrick /// linked executable, otherwise a valid "file address" from the 355061da546Spatrick /// linked executable that contains the debug map. 356061da546Spatrick lldb::addr_t LinkOSOFileAddress(SymbolFileDWARF *oso_symfile, 357061da546Spatrick lldb::addr_t oso_file_addr); 358061da546Spatrick 359061da546Spatrick /// Given a line table full of lines with "file addresses" that are 360061da546Spatrick /// for a .o file represented by \a oso_symfile, link a new line table 361061da546Spatrick /// and return it. 362061da546Spatrick /// 363061da546Spatrick /// \param[in] oso_symfile 364061da546Spatrick /// The DWARF symbol file that produced the \a line_table 365061da546Spatrick /// 366061da546Spatrick /// \param[in] line_table 367061da546Spatrick /// A pointer to the line table. 368061da546Spatrick /// 369061da546Spatrick /// \return 370061da546Spatrick /// Returns a valid line table full of linked addresses, or NULL 371061da546Spatrick /// if none of the line table addresses exist in the main 372061da546Spatrick /// executable. 373061da546Spatrick lldb_private::LineTable * 374061da546Spatrick LinkOSOLineTable(SymbolFileDWARF *oso_symfile, 375061da546Spatrick lldb_private::LineTable *line_table); 376061da546Spatrick 377061da546Spatrick size_t AddOSOARanges(SymbolFileDWARF *dwarf2Data, 378061da546Spatrick DWARFDebugAranges *debug_aranges); 379061da546Spatrick }; 380061da546Spatrick 381*dda28197Spatrick #endif // LLDB_SOURCE_PLUGINS_SYMBOLFILE_DWARF_SYMBOLFILEDWARFDEBUGMAP_H 382