15ffd83dbSDimitry Andric //===-- SymbolFileDWARFDebugMap.cpp ---------------------------------------===// 20b57cec5SDimitry Andric // 30b57cec5SDimitry Andric // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. 40b57cec5SDimitry Andric // See https://llvm.org/LICENSE.txt for license information. 50b57cec5SDimitry Andric // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception 60b57cec5SDimitry Andric // 70b57cec5SDimitry Andric //===----------------------------------------------------------------------===// 80b57cec5SDimitry Andric 90b57cec5SDimitry Andric #include "SymbolFileDWARFDebugMap.h" 10bdd1243dSDimitry Andric #include "DWARFCompileUnit.h" 110b57cec5SDimitry Andric #include "DWARFDebugAranges.h" 12bdd1243dSDimitry Andric #include "DWARFDebugInfo.h" 130b57cec5SDimitry Andric 140b57cec5SDimitry Andric #include "lldb/Core/Module.h" 150b57cec5SDimitry Andric #include "lldb/Core/ModuleList.h" 160b57cec5SDimitry Andric #include "lldb/Core/PluginManager.h" 170b57cec5SDimitry Andric #include "lldb/Core/Section.h" 180b57cec5SDimitry Andric #include "lldb/Host/FileSystem.h" 190b57cec5SDimitry Andric #include "lldb/Utility/RangeMap.h" 200b57cec5SDimitry Andric #include "lldb/Utility/RegularExpression.h" 21bdd1243dSDimitry Andric #include "lldb/Utility/StreamString.h" 225f757f3fSDimitry Andric #include "lldb/Utility/StructuredData.h" 235f757f3fSDimitry Andric #include "lldb/Utility/Timer.h" 240b57cec5SDimitry Andric 250b57cec5SDimitry Andric //#define DEBUG_OSO_DMAP // DO NOT CHECKIN WITH THIS NOT COMMENTED OUT 260b57cec5SDimitry Andric 270b57cec5SDimitry Andric #include "lldb/Symbol/CompileUnit.h" 280b57cec5SDimitry Andric #include "lldb/Symbol/LineTable.h" 290b57cec5SDimitry Andric #include "lldb/Symbol/ObjectFile.h" 300b57cec5SDimitry Andric #include "lldb/Symbol/SymbolVendor.h" 310b57cec5SDimitry Andric #include "lldb/Symbol/TypeMap.h" 320b57cec5SDimitry Andric #include "lldb/Symbol/VariableList.h" 3306c3fb27SDimitry Andric #include "llvm/ADT/STLExtras.h" 340b57cec5SDimitry Andric #include "llvm/Support/ScopedPrinter.h" 350b57cec5SDimitry Andric 36bdd1243dSDimitry Andric #include "lldb/Target/StackFrame.h" 37bdd1243dSDimitry Andric 380b57cec5SDimitry Andric #include "LogChannelDWARF.h" 390b57cec5SDimitry Andric #include "SymbolFileDWARF.h" 40*0fca6ea1SDimitry Andric #include "lldb/lldb-private-enumerations.h" 410b57cec5SDimitry Andric 420b57cec5SDimitry Andric #include <memory> 43bdd1243dSDimitry Andric #include <optional> 440b57cec5SDimitry Andric 450b57cec5SDimitry Andric using namespace lldb; 460b57cec5SDimitry Andric using namespace lldb_private; 475f757f3fSDimitry Andric using namespace lldb_private::plugin::dwarf; 480b57cec5SDimitry Andric 49480093f4SDimitry Andric char SymbolFileDWARFDebugMap::ID; 50480093f4SDimitry Andric 510b57cec5SDimitry Andric // Subclass lldb_private::Module so we can intercept the 520b57cec5SDimitry Andric // "Module::GetObjectFile()" (so we can fixup the object file sections) and 539dba64beSDimitry Andric // also for "Module::GetSymbolFile()" (so we can fixup the symbol file id. 540b57cec5SDimitry Andric 550b57cec5SDimitry Andric const SymbolFileDWARFDebugMap::FileRangeMap & 560b57cec5SDimitry Andric SymbolFileDWARFDebugMap::CompileUnitInfo::GetFileRangeMap( 570b57cec5SDimitry Andric SymbolFileDWARFDebugMap *exe_symfile) { 580b57cec5SDimitry Andric if (file_range_map_valid) 590b57cec5SDimitry Andric return file_range_map; 600b57cec5SDimitry Andric 610b57cec5SDimitry Andric file_range_map_valid = true; 620b57cec5SDimitry Andric 630b57cec5SDimitry Andric Module *oso_module = exe_symfile->GetModuleByCompUnitInfo(this); 640b57cec5SDimitry Andric if (!oso_module) 650b57cec5SDimitry Andric return file_range_map; 660b57cec5SDimitry Andric 670b57cec5SDimitry Andric ObjectFile *oso_objfile = oso_module->GetObjectFile(); 680b57cec5SDimitry Andric if (!oso_objfile) 690b57cec5SDimitry Andric return file_range_map; 700b57cec5SDimitry Andric 711fd87a68SDimitry Andric Log *log = GetLog(DWARFLog::DebugMap); 729dba64beSDimitry Andric LLDB_LOGF( 739dba64beSDimitry Andric log, 740b57cec5SDimitry Andric "%p: SymbolFileDWARFDebugMap::CompileUnitInfo::GetFileRangeMap ('%s')", 750b57cec5SDimitry Andric static_cast<void *>(this), 760b57cec5SDimitry Andric oso_module->GetSpecificationDescription().c_str()); 770b57cec5SDimitry Andric 780b57cec5SDimitry Andric std::vector<SymbolFileDWARFDebugMap::CompileUnitInfo *> cu_infos; 790b57cec5SDimitry Andric if (exe_symfile->GetCompUnitInfosForModule(oso_module, cu_infos)) { 800b57cec5SDimitry Andric for (auto comp_unit_info : cu_infos) { 810b57cec5SDimitry Andric Symtab *exe_symtab = exe_symfile->GetObjectFile()->GetSymtab(); 820b57cec5SDimitry Andric ModuleSP oso_module_sp(oso_objfile->GetModule()); 830b57cec5SDimitry Andric Symtab *oso_symtab = oso_objfile->GetSymtab(); 840b57cec5SDimitry Andric 850b57cec5SDimitry Andric /// const uint32_t fun_resolve_flags = SymbolContext::Module | 860b57cec5SDimitry Andric /// eSymbolContextCompUnit | eSymbolContextFunction; 870b57cec5SDimitry Andric // SectionList *oso_sections = oso_objfile->Sections(); 880b57cec5SDimitry Andric // Now we need to make sections that map from zero based object file 890b57cec5SDimitry Andric // addresses to where things ended up in the main executable. 900b57cec5SDimitry Andric 910b57cec5SDimitry Andric assert(comp_unit_info->first_symbol_index != UINT32_MAX); 920b57cec5SDimitry Andric // End index is one past the last valid symbol index 930b57cec5SDimitry Andric const uint32_t oso_end_idx = comp_unit_info->last_symbol_index + 1; 940b57cec5SDimitry Andric for (uint32_t idx = comp_unit_info->first_symbol_index + 950b57cec5SDimitry Andric 2; // Skip the N_SO and N_OSO 960b57cec5SDimitry Andric idx < oso_end_idx; ++idx) { 970b57cec5SDimitry Andric Symbol *exe_symbol = exe_symtab->SymbolAtIndex(idx); 980b57cec5SDimitry Andric if (exe_symbol) { 990b57cec5SDimitry Andric if (!exe_symbol->IsDebug()) 1000b57cec5SDimitry Andric continue; 1010b57cec5SDimitry Andric 1020b57cec5SDimitry Andric switch (exe_symbol->GetType()) { 1030b57cec5SDimitry Andric default: 1040b57cec5SDimitry Andric break; 1050b57cec5SDimitry Andric 1060b57cec5SDimitry Andric case eSymbolTypeCode: { 1070b57cec5SDimitry Andric // For each N_FUN, or function that we run into in the debug map we 1080b57cec5SDimitry Andric // make a new section that we add to the sections found in the .o 1090b57cec5SDimitry Andric // file. This new section has the file address set to what the 1100b57cec5SDimitry Andric // addresses are in the .o file, and the load address is adjusted 1110b57cec5SDimitry Andric // to match where it ended up in the final executable! We do this 1120b57cec5SDimitry Andric // before we parse any dwarf info so that when it goes get parsed 1130b57cec5SDimitry Andric // all section/offset addresses that get registered will resolve 1140b57cec5SDimitry Andric // correctly to the new addresses in the main executable. 1150b57cec5SDimitry Andric 1160b57cec5SDimitry Andric // First we find the original symbol in the .o file's symbol table 1170b57cec5SDimitry Andric Symbol *oso_fun_symbol = oso_symtab->FindFirstSymbolWithNameAndType( 1185ffd83dbSDimitry Andric exe_symbol->GetMangled().GetName(Mangled::ePreferMangled), 1190b57cec5SDimitry Andric eSymbolTypeCode, Symtab::eDebugNo, Symtab::eVisibilityAny); 1200b57cec5SDimitry Andric if (oso_fun_symbol) { 1210b57cec5SDimitry Andric // Add the inverse OSO file address to debug map entry mapping 1220b57cec5SDimitry Andric exe_symfile->AddOSOFileRange( 1230b57cec5SDimitry Andric this, exe_symbol->GetAddressRef().GetFileAddress(), 1240b57cec5SDimitry Andric exe_symbol->GetByteSize(), 1250b57cec5SDimitry Andric oso_fun_symbol->GetAddressRef().GetFileAddress(), 1260b57cec5SDimitry Andric oso_fun_symbol->GetByteSize()); 1270b57cec5SDimitry Andric } 1280b57cec5SDimitry Andric } break; 1290b57cec5SDimitry Andric 1300b57cec5SDimitry Andric case eSymbolTypeData: { 1310b57cec5SDimitry Andric // For each N_GSYM we remap the address for the global by making a 1320b57cec5SDimitry Andric // new section that we add to the sections found in the .o file. 1330b57cec5SDimitry Andric // This new section has the file address set to what the addresses 1340b57cec5SDimitry Andric // are in the .o file, and the load address is adjusted to match 1350b57cec5SDimitry Andric // where it ended up in the final executable! We do this before we 1360b57cec5SDimitry Andric // parse any dwarf info so that when it goes get parsed all 1370b57cec5SDimitry Andric // section/offset addresses that get registered will resolve 1380b57cec5SDimitry Andric // correctly to the new addresses in the main executable. We 1390b57cec5SDimitry Andric // initially set the section size to be 1 byte, but will need to 1400b57cec5SDimitry Andric // fix up these addresses further after all globals have been 1410b57cec5SDimitry Andric // parsed to span the gaps, or we can find the global variable 1420b57cec5SDimitry Andric // sizes from the DWARF info as we are parsing. 1430b57cec5SDimitry Andric 1440b57cec5SDimitry Andric // Next we find the non-stab entry that corresponds to the N_GSYM 1450b57cec5SDimitry Andric // in the .o file 1460b57cec5SDimitry Andric Symbol *oso_gsym_symbol = 1470b57cec5SDimitry Andric oso_symtab->FindFirstSymbolWithNameAndType( 1485ffd83dbSDimitry Andric exe_symbol->GetMangled().GetName(Mangled::ePreferMangled), 1490b57cec5SDimitry Andric eSymbolTypeData, Symtab::eDebugNo, Symtab::eVisibilityAny); 1500b57cec5SDimitry Andric if (exe_symbol && oso_gsym_symbol && exe_symbol->ValueIsAddress() && 1510b57cec5SDimitry Andric oso_gsym_symbol->ValueIsAddress()) { 1520b57cec5SDimitry Andric // Add the inverse OSO file address to debug map entry mapping 1530b57cec5SDimitry Andric exe_symfile->AddOSOFileRange( 1540b57cec5SDimitry Andric this, exe_symbol->GetAddressRef().GetFileAddress(), 1550b57cec5SDimitry Andric exe_symbol->GetByteSize(), 1560b57cec5SDimitry Andric oso_gsym_symbol->GetAddressRef().GetFileAddress(), 1570b57cec5SDimitry Andric oso_gsym_symbol->GetByteSize()); 1580b57cec5SDimitry Andric } 1590b57cec5SDimitry Andric } break; 1600b57cec5SDimitry Andric } 1610b57cec5SDimitry Andric } 1620b57cec5SDimitry Andric } 1630b57cec5SDimitry Andric 1640b57cec5SDimitry Andric exe_symfile->FinalizeOSOFileRanges(this); 1650b57cec5SDimitry Andric // We don't need the symbols anymore for the .o files 1660b57cec5SDimitry Andric oso_objfile->ClearSymtab(); 1670b57cec5SDimitry Andric } 1680b57cec5SDimitry Andric } 1690b57cec5SDimitry Andric return file_range_map; 1700b57cec5SDimitry Andric } 1710b57cec5SDimitry Andric 1725f757f3fSDimitry Andric namespace lldb_private::plugin { 1735f757f3fSDimitry Andric namespace dwarf { 1740b57cec5SDimitry Andric class DebugMapModule : public Module { 1750b57cec5SDimitry Andric public: 1760b57cec5SDimitry Andric DebugMapModule(const ModuleSP &exe_module_sp, uint32_t cu_idx, 1770b57cec5SDimitry Andric const FileSpec &file_spec, const ArchSpec &arch, 1785f757f3fSDimitry Andric ConstString object_name, off_t object_offset, 1790b57cec5SDimitry Andric const llvm::sys::TimePoint<> object_mod_time) 1800b57cec5SDimitry Andric : Module(file_spec, arch, object_name, object_offset, object_mod_time), 1810b57cec5SDimitry Andric m_exe_module_wp(exe_module_sp), m_cu_idx(cu_idx) {} 1820b57cec5SDimitry Andric 1830b57cec5SDimitry Andric ~DebugMapModule() override = default; 1840b57cec5SDimitry Andric 1859dba64beSDimitry Andric SymbolFile * 1869dba64beSDimitry Andric GetSymbolFile(bool can_create = true, 1870b57cec5SDimitry Andric lldb_private::Stream *feedback_strm = nullptr) override { 1880b57cec5SDimitry Andric // Scope for locker 1890b57cec5SDimitry Andric if (m_symfile_up.get() || !can_create) 1909dba64beSDimitry Andric return m_symfile_up ? m_symfile_up->GetSymbolFile() : nullptr; 1910b57cec5SDimitry Andric 1920b57cec5SDimitry Andric ModuleSP exe_module_sp(m_exe_module_wp.lock()); 1930b57cec5SDimitry Andric if (exe_module_sp) { 1940b57cec5SDimitry Andric // Now get the object file outside of a locking scope 1950b57cec5SDimitry Andric ObjectFile *oso_objfile = GetObjectFile(); 1960b57cec5SDimitry Andric if (oso_objfile) { 1970b57cec5SDimitry Andric std::lock_guard<std::recursive_mutex> guard(m_mutex); 1989dba64beSDimitry Andric if (SymbolFile *symfile = 1999dba64beSDimitry Andric Module::GetSymbolFile(can_create, feedback_strm)) { 2000b57cec5SDimitry Andric // Set a pointer to this class to set our OSO DWARF file know that 2010b57cec5SDimitry Andric // the DWARF is being used along with a debug map and that it will 2020b57cec5SDimitry Andric // have the remapped sections that we do below. 2030b57cec5SDimitry Andric SymbolFileDWARF *oso_symfile = 2049dba64beSDimitry Andric SymbolFileDWARFDebugMap::GetSymbolFileAsSymbolFileDWARF(symfile); 2050b57cec5SDimitry Andric 2060b57cec5SDimitry Andric if (!oso_symfile) 2070b57cec5SDimitry Andric return nullptr; 2080b57cec5SDimitry Andric 2090b57cec5SDimitry Andric ObjectFile *exe_objfile = exe_module_sp->GetObjectFile(); 2109dba64beSDimitry Andric SymbolFile *exe_symfile = exe_module_sp->GetSymbolFile(); 2110b57cec5SDimitry Andric 2129dba64beSDimitry Andric if (exe_objfile && exe_symfile) { 2130b57cec5SDimitry Andric oso_symfile->SetDebugMapModule(exe_module_sp); 2140b57cec5SDimitry Andric // Set the ID of the symbol file DWARF to the index of the OSO 2150b57cec5SDimitry Andric // shifted left by 32 bits to provide a unique prefix for any 2160b57cec5SDimitry Andric // UserID's that get created in the symbol file. 21706c3fb27SDimitry Andric oso_symfile->SetFileIndex((uint64_t)m_cu_idx); 2180b57cec5SDimitry Andric } 2199dba64beSDimitry Andric return symfile; 2200b57cec5SDimitry Andric } 2210b57cec5SDimitry Andric } 2220b57cec5SDimitry Andric } 2230b57cec5SDimitry Andric return nullptr; 2240b57cec5SDimitry Andric } 2250b57cec5SDimitry Andric 2260b57cec5SDimitry Andric protected: 2270b57cec5SDimitry Andric ModuleWP m_exe_module_wp; 2280b57cec5SDimitry Andric const uint32_t m_cu_idx; 2290b57cec5SDimitry Andric }; 2305f757f3fSDimitry Andric } // namespace dwarf 2315f757f3fSDimitry Andric } // namespace lldb_private::plugin 2320b57cec5SDimitry Andric 2330b57cec5SDimitry Andric void SymbolFileDWARFDebugMap::Initialize() { 2340b57cec5SDimitry Andric PluginManager::RegisterPlugin(GetPluginNameStatic(), 2350b57cec5SDimitry Andric GetPluginDescriptionStatic(), CreateInstance); 2360b57cec5SDimitry Andric } 2370b57cec5SDimitry Andric 2380b57cec5SDimitry Andric void SymbolFileDWARFDebugMap::Terminate() { 2390b57cec5SDimitry Andric PluginManager::UnregisterPlugin(CreateInstance); 2400b57cec5SDimitry Andric } 2410b57cec5SDimitry Andric 242349cc55cSDimitry Andric llvm::StringRef SymbolFileDWARFDebugMap::GetPluginDescriptionStatic() { 2430b57cec5SDimitry Andric return "DWARF and DWARF3 debug symbol file reader (debug map)."; 2440b57cec5SDimitry Andric } 2450b57cec5SDimitry Andric 2469dba64beSDimitry Andric SymbolFile *SymbolFileDWARFDebugMap::CreateInstance(ObjectFileSP objfile_sp) { 2479dba64beSDimitry Andric return new SymbolFileDWARFDebugMap(std::move(objfile_sp)); 2480b57cec5SDimitry Andric } 2490b57cec5SDimitry Andric 2509dba64beSDimitry Andric SymbolFileDWARFDebugMap::SymbolFileDWARFDebugMap(ObjectFileSP objfile_sp) 25181ad6265SDimitry Andric : SymbolFileCommon(std::move(objfile_sp)), m_flags(), m_compile_unit_infos(), 2529dba64beSDimitry Andric m_func_indexes(), m_glob_indexes(), 2530b57cec5SDimitry Andric m_supports_DW_AT_APPLE_objc_complete_type(eLazyBoolCalculate) {} 2540b57cec5SDimitry Andric 255fe6060f1SDimitry Andric SymbolFileDWARFDebugMap::~SymbolFileDWARFDebugMap() = default; 2560b57cec5SDimitry Andric 2570b57cec5SDimitry Andric void SymbolFileDWARFDebugMap::InitializeObject() {} 2580b57cec5SDimitry Andric 2590b57cec5SDimitry Andric void SymbolFileDWARFDebugMap::InitOSO() { 2600b57cec5SDimitry Andric if (m_flags.test(kHaveInitializedOSOs)) 2610b57cec5SDimitry Andric return; 2620b57cec5SDimitry Andric 2630b57cec5SDimitry Andric m_flags.set(kHaveInitializedOSOs); 2640b57cec5SDimitry Andric 2650b57cec5SDimitry Andric // If the object file has been stripped, there is no sense in looking further 2660b57cec5SDimitry Andric // as all of the debug symbols for the debug map will not be available 2679dba64beSDimitry Andric if (m_objfile_sp->IsStripped()) 2680b57cec5SDimitry Andric return; 2690b57cec5SDimitry Andric 2700b57cec5SDimitry Andric // Also make sure the file type is some sort of executable. Core files, debug 2710b57cec5SDimitry Andric // info files (dSYM), object files (.o files), and stub libraries all can 2729dba64beSDimitry Andric switch (m_objfile_sp->GetType()) { 2730b57cec5SDimitry Andric case ObjectFile::eTypeInvalid: 2740b57cec5SDimitry Andric case ObjectFile::eTypeCoreFile: 2750b57cec5SDimitry Andric case ObjectFile::eTypeDebugInfo: 2760b57cec5SDimitry Andric case ObjectFile::eTypeObjectFile: 2770b57cec5SDimitry Andric case ObjectFile::eTypeStubLibrary: 2780b57cec5SDimitry Andric case ObjectFile::eTypeUnknown: 2790b57cec5SDimitry Andric case ObjectFile::eTypeJIT: 2800b57cec5SDimitry Andric return; 2810b57cec5SDimitry Andric 2820b57cec5SDimitry Andric case ObjectFile::eTypeExecutable: 2830b57cec5SDimitry Andric case ObjectFile::eTypeDynamicLinker: 2840b57cec5SDimitry Andric case ObjectFile::eTypeSharedLibrary: 2850b57cec5SDimitry Andric break; 2860b57cec5SDimitry Andric } 2870b57cec5SDimitry Andric 2880b57cec5SDimitry Andric // In order to get the abilities of this plug-in, we look at the list of 2890b57cec5SDimitry Andric // N_OSO entries (object files) from the symbol table and make sure that 2900b57cec5SDimitry Andric // these files exist and also contain valid DWARF. If we get any of that then 2910b57cec5SDimitry Andric // we return the abilities of the first N_OSO's DWARF. 2920b57cec5SDimitry Andric 2939dba64beSDimitry Andric Symtab *symtab = m_objfile_sp->GetSymtab(); 29406c3fb27SDimitry Andric if (!symtab) 29506c3fb27SDimitry Andric return; 29606c3fb27SDimitry Andric 2971fd87a68SDimitry Andric Log *log = GetLog(DWARFLog::DebugMap); 2980b57cec5SDimitry Andric 2990b57cec5SDimitry Andric std::vector<uint32_t> oso_indexes; 3000b57cec5SDimitry Andric // When a mach-o symbol is encoded, the n_type field is encoded in bits 3010b57cec5SDimitry Andric // 23:16, and the n_desc field is encoded in bits 15:0. 3020b57cec5SDimitry Andric // 3030b57cec5SDimitry Andric // To find all N_OSO entries that are part of the DWARF + debug map we find 3040b57cec5SDimitry Andric // only object file symbols with the flags value as follows: bits 23:16 == 3050b57cec5SDimitry Andric // 0x66 (N_OSO) bits 15: 0 == 0x0001 (specifies this is a debug map object 3060b57cec5SDimitry Andric // file) 3070b57cec5SDimitry Andric const uint32_t k_oso_symbol_flags_value = 0x660001u; 3080b57cec5SDimitry Andric 3090b57cec5SDimitry Andric const uint32_t oso_index_count = 3100b57cec5SDimitry Andric symtab->AppendSymbolIndexesWithTypeAndFlagsValue( 3110b57cec5SDimitry Andric eSymbolTypeObjectFile, k_oso_symbol_flags_value, oso_indexes); 3120b57cec5SDimitry Andric 31306c3fb27SDimitry Andric if (oso_index_count == 0) 31406c3fb27SDimitry Andric return; 31506c3fb27SDimitry Andric 3160b57cec5SDimitry Andric symtab->AppendSymbolIndexesWithType(eSymbolTypeCode, Symtab::eDebugYes, 31706c3fb27SDimitry Andric Symtab::eVisibilityAny, m_func_indexes); 3180b57cec5SDimitry Andric symtab->AppendSymbolIndexesWithType(eSymbolTypeData, Symtab::eDebugYes, 31906c3fb27SDimitry Andric Symtab::eVisibilityAny, m_glob_indexes); 3200b57cec5SDimitry Andric 3210b57cec5SDimitry Andric symtab->SortSymbolIndexesByValue(m_func_indexes, true); 3220b57cec5SDimitry Andric symtab->SortSymbolIndexesByValue(m_glob_indexes, true); 3230b57cec5SDimitry Andric 32406c3fb27SDimitry Andric for (uint32_t sym_idx : 32506c3fb27SDimitry Andric llvm::concat<uint32_t>(m_func_indexes, m_glob_indexes)) { 3260b57cec5SDimitry Andric const Symbol *symbol = symtab->SymbolAtIndex(sym_idx); 3270b57cec5SDimitry Andric lldb::addr_t file_addr = symbol->GetAddressRef().GetFileAddress(); 3280b57cec5SDimitry Andric lldb::addr_t byte_size = symbol->GetByteSize(); 32906c3fb27SDimitry Andric DebugMap::Entry debug_map_entry(file_addr, byte_size, 33006c3fb27SDimitry Andric OSOEntry(sym_idx, LLDB_INVALID_ADDRESS)); 3310b57cec5SDimitry Andric m_debug_map.Append(debug_map_entry); 3320b57cec5SDimitry Andric } 3330b57cec5SDimitry Andric m_debug_map.Sort(); 3340b57cec5SDimitry Andric 3350b57cec5SDimitry Andric m_compile_unit_infos.resize(oso_index_count); 3360b57cec5SDimitry Andric 3370b57cec5SDimitry Andric for (uint32_t i = 0; i < oso_index_count; ++i) { 3380b57cec5SDimitry Andric const uint32_t so_idx = oso_indexes[i] - 1; 3390b57cec5SDimitry Andric const uint32_t oso_idx = oso_indexes[i]; 3400b57cec5SDimitry Andric const Symbol *so_symbol = symtab->SymbolAtIndex(so_idx); 3410b57cec5SDimitry Andric const Symbol *oso_symbol = symtab->SymbolAtIndex(oso_idx); 3420b57cec5SDimitry Andric if (so_symbol && oso_symbol && 3430b57cec5SDimitry Andric so_symbol->GetType() == eSymbolTypeSourceFile && 3440b57cec5SDimitry Andric oso_symbol->GetType() == eSymbolTypeObjectFile) { 34506c3fb27SDimitry Andric m_compile_unit_infos[i].so_file.SetFile(so_symbol->GetName().AsCString(), 34606c3fb27SDimitry Andric FileSpec::Style::native); 3470b57cec5SDimitry Andric m_compile_unit_infos[i].oso_path = oso_symbol->GetName(); 3480b57cec5SDimitry Andric m_compile_unit_infos[i].oso_mod_time = 3490b57cec5SDimitry Andric llvm::sys::toTimePoint(oso_symbol->GetIntegerValue(0)); 3500b57cec5SDimitry Andric uint32_t sibling_idx = so_symbol->GetSiblingIndex(); 3510b57cec5SDimitry Andric // The sibling index can't be less that or equal to the current index 3520b57cec5SDimitry Andric // "i" 35306c3fb27SDimitry Andric if (sibling_idx <= i || sibling_idx == UINT32_MAX) { 3549dba64beSDimitry Andric m_objfile_sp->GetModule()->ReportError( 355bdd1243dSDimitry Andric "N_SO in symbol with UID {0} has invalid sibling in debug " 356bdd1243dSDimitry Andric "map, " 3570b57cec5SDimitry Andric "please file a bug and attach the binary listed in this error", 3580b57cec5SDimitry Andric so_symbol->GetID()); 3590b57cec5SDimitry Andric } else { 3600b57cec5SDimitry Andric const Symbol *last_symbol = symtab->SymbolAtIndex(sibling_idx - 1); 3610b57cec5SDimitry Andric m_compile_unit_infos[i].first_symbol_index = so_idx; 3620b57cec5SDimitry Andric m_compile_unit_infos[i].last_symbol_index = sibling_idx - 1; 3630b57cec5SDimitry Andric m_compile_unit_infos[i].first_symbol_id = so_symbol->GetID(); 3640b57cec5SDimitry Andric m_compile_unit_infos[i].last_symbol_id = last_symbol->GetID(); 3650b57cec5SDimitry Andric 3669dba64beSDimitry Andric LLDB_LOGF(log, "Initialized OSO 0x%8.8x: file=%s", i, 3670b57cec5SDimitry Andric oso_symbol->GetName().GetCString()); 3680b57cec5SDimitry Andric } 3690b57cec5SDimitry Andric } else { 3700b57cec5SDimitry Andric if (oso_symbol == nullptr) 3719dba64beSDimitry Andric m_objfile_sp->GetModule()->ReportError( 372bdd1243dSDimitry Andric "N_OSO symbol[{0}] can't be found, please file a bug and " 373bdd1243dSDimitry Andric "attach " 3740b57cec5SDimitry Andric "the binary listed in this error", 3750b57cec5SDimitry Andric oso_idx); 3760b57cec5SDimitry Andric else if (so_symbol == nullptr) 3779dba64beSDimitry Andric m_objfile_sp->GetModule()->ReportError( 378bdd1243dSDimitry Andric "N_SO not found for N_OSO symbol[{0}], please file a bug and " 3790b57cec5SDimitry Andric "attach the binary listed in this error", 3800b57cec5SDimitry Andric oso_idx); 3810b57cec5SDimitry Andric else if (so_symbol->GetType() != eSymbolTypeSourceFile) 3829dba64beSDimitry Andric m_objfile_sp->GetModule()->ReportError( 383bdd1243dSDimitry Andric "N_SO has incorrect symbol type ({0}) for N_OSO " 384bdd1243dSDimitry Andric "symbol[{1}], " 3850b57cec5SDimitry Andric "please file a bug and attach the binary listed in this error", 3860b57cec5SDimitry Andric so_symbol->GetType(), oso_idx); 3870b57cec5SDimitry Andric else if (oso_symbol->GetType() != eSymbolTypeSourceFile) 3889dba64beSDimitry Andric m_objfile_sp->GetModule()->ReportError( 389bdd1243dSDimitry Andric "N_OSO has incorrect symbol type ({0}) for N_OSO " 390bdd1243dSDimitry Andric "symbol[{1}], " 3910b57cec5SDimitry Andric "please file a bug and attach the binary listed in this error", 3920b57cec5SDimitry Andric oso_symbol->GetType(), oso_idx); 3930b57cec5SDimitry Andric } 3940b57cec5SDimitry Andric } 3950b57cec5SDimitry Andric } 3960b57cec5SDimitry Andric 3970b57cec5SDimitry Andric Module *SymbolFileDWARFDebugMap::GetModuleByOSOIndex(uint32_t oso_idx) { 3980b57cec5SDimitry Andric const uint32_t cu_count = GetNumCompileUnits(); 3990b57cec5SDimitry Andric if (oso_idx < cu_count) 4000b57cec5SDimitry Andric return GetModuleByCompUnitInfo(&m_compile_unit_infos[oso_idx]); 4010b57cec5SDimitry Andric return nullptr; 4020b57cec5SDimitry Andric } 4030b57cec5SDimitry Andric 4040b57cec5SDimitry Andric Module *SymbolFileDWARFDebugMap::GetModuleByCompUnitInfo( 4050b57cec5SDimitry Andric CompileUnitInfo *comp_unit_info) { 4060b57cec5SDimitry Andric if (!comp_unit_info->oso_sp) { 4070b57cec5SDimitry Andric auto pos = m_oso_map.find( 4080b57cec5SDimitry Andric {comp_unit_info->oso_path, comp_unit_info->oso_mod_time}); 4090b57cec5SDimitry Andric if (pos != m_oso_map.end()) { 4100b57cec5SDimitry Andric comp_unit_info->oso_sp = pos->second; 4110b57cec5SDimitry Andric } else { 4120b57cec5SDimitry Andric ObjectFile *obj_file = GetObjectFile(); 4130b57cec5SDimitry Andric comp_unit_info->oso_sp = std::make_shared<OSOInfo>(); 4140b57cec5SDimitry Andric m_oso_map[{comp_unit_info->oso_path, comp_unit_info->oso_mod_time}] = 4150b57cec5SDimitry Andric comp_unit_info->oso_sp; 4160b57cec5SDimitry Andric const char *oso_path = comp_unit_info->oso_path.GetCString(); 4170b57cec5SDimitry Andric FileSpec oso_file(oso_path); 4180b57cec5SDimitry Andric ConstString oso_object; 4190b57cec5SDimitry Andric if (FileSystem::Instance().Exists(oso_file)) { 4200b57cec5SDimitry Andric // The modification time returned by the FS can have a higher precision 4210b57cec5SDimitry Andric // than the one from the CU. 4220b57cec5SDimitry Andric auto oso_mod_time = std::chrono::time_point_cast<std::chrono::seconds>( 4230b57cec5SDimitry Andric FileSystem::Instance().GetModificationTime(oso_file)); 4249dba64beSDimitry Andric // A timestamp of 0 means that the linker was in deterministic mode. In 4259dba64beSDimitry Andric // that case, we should skip the check against the filesystem last 4269dba64beSDimitry Andric // modification timestamp, since it will never match. 4279dba64beSDimitry Andric if (comp_unit_info->oso_mod_time != llvm::sys::TimePoint<>() && 4289dba64beSDimitry Andric oso_mod_time != comp_unit_info->oso_mod_time) { 429bdd1243dSDimitry Andric comp_unit_info->oso_load_error.SetErrorStringWithFormat( 430bdd1243dSDimitry Andric "debug map object file \"%s\" changed (actual: 0x%8.8x, debug " 431bdd1243dSDimitry Andric "map: 0x%8.8x) since this executable was linked, debug info " 432bdd1243dSDimitry Andric "will not be loaded", oso_file.GetPath().c_str(), 433bdd1243dSDimitry Andric (uint32_t)llvm::sys::toTimeT(oso_mod_time), 434bdd1243dSDimitry Andric (uint32_t)llvm::sys::toTimeT(comp_unit_info->oso_mod_time)); 4350b57cec5SDimitry Andric obj_file->GetModule()->ReportError( 436bdd1243dSDimitry Andric "{0}", comp_unit_info->oso_load_error.AsCString()); 4370b57cec5SDimitry Andric return nullptr; 4380b57cec5SDimitry Andric } 4390b57cec5SDimitry Andric 4400b57cec5SDimitry Andric } else { 4410b57cec5SDimitry Andric const bool must_exist = true; 4420b57cec5SDimitry Andric 4430b57cec5SDimitry Andric if (!ObjectFile::SplitArchivePathWithObject(oso_path, oso_file, 4440b57cec5SDimitry Andric oso_object, must_exist)) { 445bdd1243dSDimitry Andric comp_unit_info->oso_load_error.SetErrorStringWithFormat( 446bdd1243dSDimitry Andric "debug map object file \"%s\" containing debug info does not " 447bdd1243dSDimitry Andric "exist, debug info will not be loaded", 448bdd1243dSDimitry Andric comp_unit_info->oso_path.GetCString()); 4490b57cec5SDimitry Andric return nullptr; 4500b57cec5SDimitry Andric } 4510b57cec5SDimitry Andric } 4520b57cec5SDimitry Andric // Always create a new module for .o files. Why? Because we use the debug 4530b57cec5SDimitry Andric // map, to add new sections to each .o file and even though a .o file 4540b57cec5SDimitry Andric // might not have changed, the sections that get added to the .o file can 4550b57cec5SDimitry Andric // change. 4560b57cec5SDimitry Andric ArchSpec oso_arch; 4570b57cec5SDimitry Andric // Only adopt the architecture from the module (not the vendor or OS) 4580b57cec5SDimitry Andric // since .o files for "i386-apple-ios" will historically show up as "i386 4590b57cec5SDimitry Andric // -apple-macosx" due to the lack of a LC_VERSION_MIN_MACOSX or 4600b57cec5SDimitry Andric // LC_VERSION_MIN_IPHONEOS load command... 4619dba64beSDimitry Andric oso_arch.SetTriple(m_objfile_sp->GetModule() 4620b57cec5SDimitry Andric ->GetArchitecture() 4630b57cec5SDimitry Andric .GetTriple() 4640b57cec5SDimitry Andric .getArchName() 4650b57cec5SDimitry Andric .str() 4660b57cec5SDimitry Andric .c_str()); 4670b57cec5SDimitry Andric comp_unit_info->oso_sp->module_sp = std::make_shared<DebugMapModule>( 4680b57cec5SDimitry Andric obj_file->GetModule(), GetCompUnitInfoIndex(comp_unit_info), oso_file, 4695f757f3fSDimitry Andric oso_arch, oso_object, 0, 4700b57cec5SDimitry Andric oso_object ? comp_unit_info->oso_mod_time : llvm::sys::TimePoint<>()); 471bdd1243dSDimitry Andric 47206c3fb27SDimitry Andric if (oso_object && !comp_unit_info->oso_sp->module_sp->GetObjectFile() && 47306c3fb27SDimitry Andric FileSystem::Instance().Exists(oso_file)) { 474bdd1243dSDimitry Andric // If we are loading a .o file from a .a file the "oso_object" will 475bdd1243dSDimitry Andric // have a valid value name and if the .a file exists, either the .o 476bdd1243dSDimitry Andric // file didn't exist in the .a file or the mod time didn't match. 477bdd1243dSDimitry Andric comp_unit_info->oso_load_error.SetErrorStringWithFormat( 478bdd1243dSDimitry Andric "\"%s\" object from the \"%s\" archive: " 479bdd1243dSDimitry Andric "either the .o file doesn't exist in the archive or the " 480bdd1243dSDimitry Andric "modification time (0x%8.8x) of the .o file doesn't match", 481bdd1243dSDimitry Andric oso_object.AsCString(), oso_file.GetPath().c_str(), 482bdd1243dSDimitry Andric (uint32_t)llvm::sys::toTimeT(comp_unit_info->oso_mod_time)); 483bdd1243dSDimitry Andric } 484bdd1243dSDimitry Andric } 4850b57cec5SDimitry Andric } 4860b57cec5SDimitry Andric if (comp_unit_info->oso_sp) 4870b57cec5SDimitry Andric return comp_unit_info->oso_sp->module_sp.get(); 4880b57cec5SDimitry Andric return nullptr; 4890b57cec5SDimitry Andric } 4900b57cec5SDimitry Andric 4910b57cec5SDimitry Andric bool SymbolFileDWARFDebugMap::GetFileSpecForSO(uint32_t oso_idx, 4920b57cec5SDimitry Andric FileSpec &file_spec) { 4930b57cec5SDimitry Andric if (oso_idx < m_compile_unit_infos.size()) { 4940b57cec5SDimitry Andric if (m_compile_unit_infos[oso_idx].so_file) { 4950b57cec5SDimitry Andric file_spec = m_compile_unit_infos[oso_idx].so_file; 4960b57cec5SDimitry Andric return true; 4970b57cec5SDimitry Andric } 4980b57cec5SDimitry Andric } 4990b57cec5SDimitry Andric return false; 5000b57cec5SDimitry Andric } 5010b57cec5SDimitry Andric 5020b57cec5SDimitry Andric ObjectFile *SymbolFileDWARFDebugMap::GetObjectFileByOSOIndex(uint32_t oso_idx) { 5030b57cec5SDimitry Andric Module *oso_module = GetModuleByOSOIndex(oso_idx); 5040b57cec5SDimitry Andric if (oso_module) 5050b57cec5SDimitry Andric return oso_module->GetObjectFile(); 5060b57cec5SDimitry Andric return nullptr; 5070b57cec5SDimitry Andric } 5080b57cec5SDimitry Andric 5090b57cec5SDimitry Andric SymbolFileDWARF * 5100b57cec5SDimitry Andric SymbolFileDWARFDebugMap::GetSymbolFile(const SymbolContext &sc) { 5110b57cec5SDimitry Andric return GetSymbolFile(*sc.comp_unit); 5120b57cec5SDimitry Andric } 5130b57cec5SDimitry Andric 5140b57cec5SDimitry Andric SymbolFileDWARF * 5150b57cec5SDimitry Andric SymbolFileDWARFDebugMap::GetSymbolFile(const CompileUnit &comp_unit) { 5160b57cec5SDimitry Andric CompileUnitInfo *comp_unit_info = GetCompUnitInfo(comp_unit); 5170b57cec5SDimitry Andric if (comp_unit_info) 5180b57cec5SDimitry Andric return GetSymbolFileByCompUnitInfo(comp_unit_info); 5190b57cec5SDimitry Andric return nullptr; 5200b57cec5SDimitry Andric } 5210b57cec5SDimitry Andric 5220b57cec5SDimitry Andric ObjectFile *SymbolFileDWARFDebugMap::GetObjectFileByCompUnitInfo( 5230b57cec5SDimitry Andric CompileUnitInfo *comp_unit_info) { 5240b57cec5SDimitry Andric Module *oso_module = GetModuleByCompUnitInfo(comp_unit_info); 5250b57cec5SDimitry Andric if (oso_module) 5260b57cec5SDimitry Andric return oso_module->GetObjectFile(); 5270b57cec5SDimitry Andric return nullptr; 5280b57cec5SDimitry Andric } 5290b57cec5SDimitry Andric 5300b57cec5SDimitry Andric uint32_t SymbolFileDWARFDebugMap::GetCompUnitInfoIndex( 5310b57cec5SDimitry Andric const CompileUnitInfo *comp_unit_info) { 5320b57cec5SDimitry Andric if (!m_compile_unit_infos.empty()) { 5330b57cec5SDimitry Andric const CompileUnitInfo *first_comp_unit_info = &m_compile_unit_infos.front(); 5340b57cec5SDimitry Andric const CompileUnitInfo *last_comp_unit_info = &m_compile_unit_infos.back(); 5350b57cec5SDimitry Andric if (first_comp_unit_info <= comp_unit_info && 5360b57cec5SDimitry Andric comp_unit_info <= last_comp_unit_info) 5370b57cec5SDimitry Andric return comp_unit_info - first_comp_unit_info; 5380b57cec5SDimitry Andric } 5390b57cec5SDimitry Andric return UINT32_MAX; 5400b57cec5SDimitry Andric } 5410b57cec5SDimitry Andric 5420b57cec5SDimitry Andric SymbolFileDWARF * 5430b57cec5SDimitry Andric SymbolFileDWARFDebugMap::GetSymbolFileByOSOIndex(uint32_t oso_idx) { 5440b57cec5SDimitry Andric unsigned size = m_compile_unit_infos.size(); 5450b57cec5SDimitry Andric if (oso_idx < size) 5460b57cec5SDimitry Andric return GetSymbolFileByCompUnitInfo(&m_compile_unit_infos[oso_idx]); 5470b57cec5SDimitry Andric return nullptr; 5480b57cec5SDimitry Andric } 5490b57cec5SDimitry Andric 5500b57cec5SDimitry Andric SymbolFileDWARF * 5510b57cec5SDimitry Andric SymbolFileDWARFDebugMap::GetSymbolFileAsSymbolFileDWARF(SymbolFile *sym_file) { 5520b57cec5SDimitry Andric if (sym_file && 5530b57cec5SDimitry Andric sym_file->GetPluginName() == SymbolFileDWARF::GetPluginNameStatic()) 5545ffd83dbSDimitry Andric return static_cast<SymbolFileDWARF *>(sym_file); 5550b57cec5SDimitry Andric return nullptr; 5560b57cec5SDimitry Andric } 5570b57cec5SDimitry Andric 5580b57cec5SDimitry Andric SymbolFileDWARF *SymbolFileDWARFDebugMap::GetSymbolFileByCompUnitInfo( 5590b57cec5SDimitry Andric CompileUnitInfo *comp_unit_info) { 5609dba64beSDimitry Andric if (Module *oso_module = GetModuleByCompUnitInfo(comp_unit_info)) 5619dba64beSDimitry Andric return GetSymbolFileAsSymbolFileDWARF(oso_module->GetSymbolFile()); 5620b57cec5SDimitry Andric return nullptr; 5630b57cec5SDimitry Andric } 5640b57cec5SDimitry Andric 5650b57cec5SDimitry Andric uint32_t SymbolFileDWARFDebugMap::CalculateAbilities() { 5660b57cec5SDimitry Andric // In order to get the abilities of this plug-in, we look at the list of 5670b57cec5SDimitry Andric // N_OSO entries (object files) from the symbol table and make sure that 5680b57cec5SDimitry Andric // these files exist and also contain valid DWARF. If we get any of that then 5690b57cec5SDimitry Andric // we return the abilities of the first N_OSO's DWARF. 5700b57cec5SDimitry Andric 5710b57cec5SDimitry Andric const uint32_t oso_index_count = GetNumCompileUnits(); 5720b57cec5SDimitry Andric if (oso_index_count > 0) { 5730b57cec5SDimitry Andric InitOSO(); 5740b57cec5SDimitry Andric if (!m_compile_unit_infos.empty()) { 5750b57cec5SDimitry Andric return SymbolFile::CompileUnits | SymbolFile::Functions | 5760b57cec5SDimitry Andric SymbolFile::Blocks | SymbolFile::GlobalVariables | 5770b57cec5SDimitry Andric SymbolFile::LocalVariables | SymbolFile::VariableTypes | 5780b57cec5SDimitry Andric SymbolFile::LineTables; 5790b57cec5SDimitry Andric } 5800b57cec5SDimitry Andric } 5810b57cec5SDimitry Andric return 0; 5820b57cec5SDimitry Andric } 5830b57cec5SDimitry Andric 5849dba64beSDimitry Andric uint32_t SymbolFileDWARFDebugMap::CalculateNumCompileUnits() { 5850b57cec5SDimitry Andric InitOSO(); 5860b57cec5SDimitry Andric return m_compile_unit_infos.size(); 5870b57cec5SDimitry Andric } 5880b57cec5SDimitry Andric 5890b57cec5SDimitry Andric CompUnitSP SymbolFileDWARFDebugMap::ParseCompileUnitAtIndex(uint32_t cu_idx) { 5900b57cec5SDimitry Andric CompUnitSP comp_unit_sp; 5910b57cec5SDimitry Andric const uint32_t cu_count = GetNumCompileUnits(); 5920b57cec5SDimitry Andric 5930b57cec5SDimitry Andric if (cu_idx < cu_count) { 594bdd1243dSDimitry Andric auto &cu_info = m_compile_unit_infos[cu_idx]; 595bdd1243dSDimitry Andric Module *oso_module = GetModuleByCompUnitInfo(&cu_info); 5960b57cec5SDimitry Andric if (oso_module) { 5970b57cec5SDimitry Andric FileSpec so_file_spec; 5980b57cec5SDimitry Andric if (GetFileSpecForSO(cu_idx, so_file_spec)) { 5990b57cec5SDimitry Andric // User zero as the ID to match the compile unit at offset zero in each 600bdd1243dSDimitry Andric // .o file. 6010b57cec5SDimitry Andric lldb::user_id_t cu_id = 0; 6027a6dacacSDimitry Andric cu_info.compile_units_sps.push_back(std::make_shared<CompileUnit>( 6037a6dacacSDimitry Andric m_objfile_sp->GetModule(), nullptr, 6047a6dacacSDimitry Andric std::make_shared<SupportFile>(so_file_spec), cu_id, 605bdd1243dSDimitry Andric eLanguageTypeUnknown, eLazyBoolCalculate)); 606bdd1243dSDimitry Andric cu_info.id_to_index_map.insert({0, 0}); 607bdd1243dSDimitry Andric SetCompileUnitAtIndex(cu_idx, cu_info.compile_units_sps[0]); 608bdd1243dSDimitry Andric // If there's a symbol file also register all the extra compile units. 609bdd1243dSDimitry Andric if (SymbolFileDWARF *oso_symfile = 610bdd1243dSDimitry Andric GetSymbolFileByCompUnitInfo(&cu_info)) { 611bdd1243dSDimitry Andric auto num_dwarf_units = oso_symfile->DebugInfo().GetNumUnits(); 612bdd1243dSDimitry Andric for (size_t i = 0; i < num_dwarf_units; ++i) { 613bdd1243dSDimitry Andric auto *dwarf_unit = oso_symfile->DebugInfo().GetUnitAtIndex(i); 614bdd1243dSDimitry Andric if (auto *dwarf_cu = llvm::dyn_cast<DWARFCompileUnit>(dwarf_unit)) { 615bdd1243dSDimitry Andric // The "main" one was already registered. 616bdd1243dSDimitry Andric if (dwarf_cu->GetID() == 0) 617bdd1243dSDimitry Andric continue; 618bdd1243dSDimitry Andric cu_info.compile_units_sps.push_back(std::make_shared<CompileUnit>( 6197a6dacacSDimitry Andric m_objfile_sp->GetModule(), nullptr, 6207a6dacacSDimitry Andric std::make_shared<SupportFile>(so_file_spec), 621bdd1243dSDimitry Andric dwarf_cu->GetID(), eLanguageTypeUnknown, eLazyBoolCalculate)); 622bdd1243dSDimitry Andric cu_info.id_to_index_map.insert( 623bdd1243dSDimitry Andric {dwarf_cu->GetID(), cu_info.compile_units_sps.size() - 1}); 6240b57cec5SDimitry Andric } 6250b57cec5SDimitry Andric } 6260b57cec5SDimitry Andric } 627bdd1243dSDimitry Andric } 628bdd1243dSDimitry Andric } 629bdd1243dSDimitry Andric if (!cu_info.compile_units_sps.empty()) 630bdd1243dSDimitry Andric comp_unit_sp = cu_info.compile_units_sps[0]; 6310b57cec5SDimitry Andric } 6320b57cec5SDimitry Andric 6330b57cec5SDimitry Andric return comp_unit_sp; 6340b57cec5SDimitry Andric } 6350b57cec5SDimitry Andric 6360b57cec5SDimitry Andric SymbolFileDWARFDebugMap::CompileUnitInfo * 6370b57cec5SDimitry Andric SymbolFileDWARFDebugMap::GetCompUnitInfo(const SymbolContext &sc) { 6380b57cec5SDimitry Andric return GetCompUnitInfo(*sc.comp_unit); 6390b57cec5SDimitry Andric } 6400b57cec5SDimitry Andric 6410b57cec5SDimitry Andric SymbolFileDWARFDebugMap::CompileUnitInfo * 6420b57cec5SDimitry Andric SymbolFileDWARFDebugMap::GetCompUnitInfo(const CompileUnit &comp_unit) { 6430b57cec5SDimitry Andric const uint32_t cu_count = GetNumCompileUnits(); 6440b57cec5SDimitry Andric for (uint32_t i = 0; i < cu_count; ++i) { 645bdd1243dSDimitry Andric auto &id_to_index_map = m_compile_unit_infos[i].id_to_index_map; 646bdd1243dSDimitry Andric 647bdd1243dSDimitry Andric auto it = id_to_index_map.find(comp_unit.GetID()); 648bdd1243dSDimitry Andric if (it != id_to_index_map.end() && 649bdd1243dSDimitry Andric &comp_unit == 650bdd1243dSDimitry Andric m_compile_unit_infos[i].compile_units_sps[it->getSecond()].get()) 6510b57cec5SDimitry Andric return &m_compile_unit_infos[i]; 6520b57cec5SDimitry Andric } 6530b57cec5SDimitry Andric return nullptr; 6540b57cec5SDimitry Andric } 6550b57cec5SDimitry Andric 6560b57cec5SDimitry Andric size_t SymbolFileDWARFDebugMap::GetCompUnitInfosForModule( 6570b57cec5SDimitry Andric const lldb_private::Module *module, 6580b57cec5SDimitry Andric std::vector<CompileUnitInfo *> &cu_infos) { 6590b57cec5SDimitry Andric const uint32_t cu_count = GetNumCompileUnits(); 6600b57cec5SDimitry Andric for (uint32_t i = 0; i < cu_count; ++i) { 6610b57cec5SDimitry Andric if (module == GetModuleByCompUnitInfo(&m_compile_unit_infos[i])) 6620b57cec5SDimitry Andric cu_infos.push_back(&m_compile_unit_infos[i]); 6630b57cec5SDimitry Andric } 6640b57cec5SDimitry Andric return cu_infos.size(); 6650b57cec5SDimitry Andric } 6660b57cec5SDimitry Andric 6670b57cec5SDimitry Andric lldb::LanguageType 6680b57cec5SDimitry Andric SymbolFileDWARFDebugMap::ParseLanguage(CompileUnit &comp_unit) { 6699dba64beSDimitry Andric std::lock_guard<std::recursive_mutex> guard(GetModuleMutex()); 6700b57cec5SDimitry Andric SymbolFileDWARF *oso_dwarf = GetSymbolFile(comp_unit); 6710b57cec5SDimitry Andric if (oso_dwarf) 6720b57cec5SDimitry Andric return oso_dwarf->ParseLanguage(comp_unit); 6730b57cec5SDimitry Andric return eLanguageTypeUnknown; 6740b57cec5SDimitry Andric } 6750b57cec5SDimitry Andric 6765ffd83dbSDimitry Andric XcodeSDK SymbolFileDWARFDebugMap::ParseXcodeSDK(CompileUnit &comp_unit) { 6775ffd83dbSDimitry Andric std::lock_guard<std::recursive_mutex> guard(GetModuleMutex()); 6785ffd83dbSDimitry Andric SymbolFileDWARF *oso_dwarf = GetSymbolFile(comp_unit); 6795ffd83dbSDimitry Andric if (oso_dwarf) 6805ffd83dbSDimitry Andric return oso_dwarf->ParseXcodeSDK(comp_unit); 6815ffd83dbSDimitry Andric return {}; 6825ffd83dbSDimitry Andric } 6835ffd83dbSDimitry Andric 68406c3fb27SDimitry Andric llvm::SmallSet<lldb::LanguageType, 4> 68506c3fb27SDimitry Andric SymbolFileDWARFDebugMap::ParseAllLanguages( 68606c3fb27SDimitry Andric lldb_private::CompileUnit &comp_unit) { 68706c3fb27SDimitry Andric llvm::SmallSet<lldb::LanguageType, 4> langs; 68806c3fb27SDimitry Andric auto *info = GetCompUnitInfo(comp_unit); 68906c3fb27SDimitry Andric for (auto &comp_unit : info->compile_units_sps) { 69006c3fb27SDimitry Andric langs.insert(comp_unit->GetLanguage()); 69106c3fb27SDimitry Andric } 69206c3fb27SDimitry Andric return langs; 69306c3fb27SDimitry Andric } 69406c3fb27SDimitry Andric 6950b57cec5SDimitry Andric size_t SymbolFileDWARFDebugMap::ParseFunctions(CompileUnit &comp_unit) { 6969dba64beSDimitry Andric std::lock_guard<std::recursive_mutex> guard(GetModuleMutex()); 6970b57cec5SDimitry Andric SymbolFileDWARF *oso_dwarf = GetSymbolFile(comp_unit); 6980b57cec5SDimitry Andric if (oso_dwarf) 6990b57cec5SDimitry Andric return oso_dwarf->ParseFunctions(comp_unit); 7000b57cec5SDimitry Andric return 0; 7010b57cec5SDimitry Andric } 7020b57cec5SDimitry Andric 7030b57cec5SDimitry Andric bool SymbolFileDWARFDebugMap::ParseLineTable(CompileUnit &comp_unit) { 7049dba64beSDimitry Andric std::lock_guard<std::recursive_mutex> guard(GetModuleMutex()); 7050b57cec5SDimitry Andric SymbolFileDWARF *oso_dwarf = GetSymbolFile(comp_unit); 7060b57cec5SDimitry Andric if (oso_dwarf) 7070b57cec5SDimitry Andric return oso_dwarf->ParseLineTable(comp_unit); 7080b57cec5SDimitry Andric return false; 7090b57cec5SDimitry Andric } 7100b57cec5SDimitry Andric 7110b57cec5SDimitry Andric bool SymbolFileDWARFDebugMap::ParseDebugMacros(CompileUnit &comp_unit) { 7129dba64beSDimitry Andric std::lock_guard<std::recursive_mutex> guard(GetModuleMutex()); 7130b57cec5SDimitry Andric SymbolFileDWARF *oso_dwarf = GetSymbolFile(comp_unit); 7140b57cec5SDimitry Andric if (oso_dwarf) 7150b57cec5SDimitry Andric return oso_dwarf->ParseDebugMacros(comp_unit); 7160b57cec5SDimitry Andric return false; 7170b57cec5SDimitry Andric } 7180b57cec5SDimitry Andric 719480093f4SDimitry Andric bool SymbolFileDWARFDebugMap::ForEachExternalModule( 720480093f4SDimitry Andric CompileUnit &comp_unit, 721480093f4SDimitry Andric llvm::DenseSet<lldb_private::SymbolFile *> &visited_symbol_files, 722480093f4SDimitry Andric llvm::function_ref<bool(Module &)> f) { 7239dba64beSDimitry Andric std::lock_guard<std::recursive_mutex> guard(GetModuleMutex()); 7249dba64beSDimitry Andric SymbolFileDWARF *oso_dwarf = GetSymbolFile(comp_unit); 7259dba64beSDimitry Andric if (oso_dwarf) 726480093f4SDimitry Andric return oso_dwarf->ForEachExternalModule(comp_unit, visited_symbol_files, f); 727480093f4SDimitry Andric return false; 7289dba64beSDimitry Andric } 7299dba64beSDimitry Andric 7301db9f3b2SDimitry Andric bool SymbolFileDWARFDebugMap::ParseSupportFiles( 7311db9f3b2SDimitry Andric CompileUnit &comp_unit, SupportFileList &support_files) { 7329dba64beSDimitry Andric std::lock_guard<std::recursive_mutex> guard(GetModuleMutex()); 7330b57cec5SDimitry Andric SymbolFileDWARF *oso_dwarf = GetSymbolFile(comp_unit); 7340b57cec5SDimitry Andric if (oso_dwarf) 7350b57cec5SDimitry Andric return oso_dwarf->ParseSupportFiles(comp_unit, support_files); 7360b57cec5SDimitry Andric return false; 7370b57cec5SDimitry Andric } 7380b57cec5SDimitry Andric 7390b57cec5SDimitry Andric bool SymbolFileDWARFDebugMap::ParseIsOptimized(CompileUnit &comp_unit) { 7409dba64beSDimitry Andric std::lock_guard<std::recursive_mutex> guard(GetModuleMutex()); 7410b57cec5SDimitry Andric SymbolFileDWARF *oso_dwarf = GetSymbolFile(comp_unit); 7420b57cec5SDimitry Andric if (oso_dwarf) 7430b57cec5SDimitry Andric return oso_dwarf->ParseIsOptimized(comp_unit); 7440b57cec5SDimitry Andric return false; 7450b57cec5SDimitry Andric } 7460b57cec5SDimitry Andric 7470b57cec5SDimitry Andric bool SymbolFileDWARFDebugMap::ParseImportedModules( 7480b57cec5SDimitry Andric const SymbolContext &sc, std::vector<SourceModule> &imported_modules) { 7499dba64beSDimitry Andric std::lock_guard<std::recursive_mutex> guard(GetModuleMutex()); 7500b57cec5SDimitry Andric SymbolFileDWARF *oso_dwarf = GetSymbolFile(sc); 7510b57cec5SDimitry Andric if (oso_dwarf) 7520b57cec5SDimitry Andric return oso_dwarf->ParseImportedModules(sc, imported_modules); 7530b57cec5SDimitry Andric return false; 7540b57cec5SDimitry Andric } 7550b57cec5SDimitry Andric 7560b57cec5SDimitry Andric size_t SymbolFileDWARFDebugMap::ParseBlocksRecursive(Function &func) { 7579dba64beSDimitry Andric std::lock_guard<std::recursive_mutex> guard(GetModuleMutex()); 7580b57cec5SDimitry Andric CompileUnit *comp_unit = func.GetCompileUnit(); 7590b57cec5SDimitry Andric if (!comp_unit) 7600b57cec5SDimitry Andric return 0; 7610b57cec5SDimitry Andric 7620b57cec5SDimitry Andric SymbolFileDWARF *oso_dwarf = GetSymbolFile(*comp_unit); 7630b57cec5SDimitry Andric if (oso_dwarf) 7640b57cec5SDimitry Andric return oso_dwarf->ParseBlocksRecursive(func); 7650b57cec5SDimitry Andric return 0; 7660b57cec5SDimitry Andric } 7670b57cec5SDimitry Andric 7680b57cec5SDimitry Andric size_t SymbolFileDWARFDebugMap::ParseTypes(CompileUnit &comp_unit) { 7699dba64beSDimitry Andric std::lock_guard<std::recursive_mutex> guard(GetModuleMutex()); 7700b57cec5SDimitry Andric SymbolFileDWARF *oso_dwarf = GetSymbolFile(comp_unit); 7710b57cec5SDimitry Andric if (oso_dwarf) 7720b57cec5SDimitry Andric return oso_dwarf->ParseTypes(comp_unit); 7730b57cec5SDimitry Andric return 0; 7740b57cec5SDimitry Andric } 7750b57cec5SDimitry Andric 7760b57cec5SDimitry Andric size_t 7770b57cec5SDimitry Andric SymbolFileDWARFDebugMap::ParseVariablesForContext(const SymbolContext &sc) { 7789dba64beSDimitry Andric std::lock_guard<std::recursive_mutex> guard(GetModuleMutex()); 7790b57cec5SDimitry Andric SymbolFileDWARF *oso_dwarf = GetSymbolFile(sc); 7800b57cec5SDimitry Andric if (oso_dwarf) 7810b57cec5SDimitry Andric return oso_dwarf->ParseVariablesForContext(sc); 7820b57cec5SDimitry Andric return 0; 7830b57cec5SDimitry Andric } 7840b57cec5SDimitry Andric 7850b57cec5SDimitry Andric Type *SymbolFileDWARFDebugMap::ResolveTypeUID(lldb::user_id_t type_uid) { 7869dba64beSDimitry Andric std::lock_guard<std::recursive_mutex> guard(GetModuleMutex()); 7870b57cec5SDimitry Andric const uint64_t oso_idx = GetOSOIndexFromUserID(type_uid); 7880b57cec5SDimitry Andric SymbolFileDWARF *oso_dwarf = GetSymbolFileByOSOIndex(oso_idx); 7890b57cec5SDimitry Andric if (oso_dwarf) 7900b57cec5SDimitry Andric return oso_dwarf->ResolveTypeUID(type_uid); 7910b57cec5SDimitry Andric return nullptr; 7920b57cec5SDimitry Andric } 7930b57cec5SDimitry Andric 794bdd1243dSDimitry Andric std::optional<SymbolFile::ArrayInfo> 7950b57cec5SDimitry Andric SymbolFileDWARFDebugMap::GetDynamicArrayInfoForUID( 7960b57cec5SDimitry Andric lldb::user_id_t type_uid, const lldb_private::ExecutionContext *exe_ctx) { 7970b57cec5SDimitry Andric const uint64_t oso_idx = GetOSOIndexFromUserID(type_uid); 7980b57cec5SDimitry Andric SymbolFileDWARF *oso_dwarf = GetSymbolFileByOSOIndex(oso_idx); 7990b57cec5SDimitry Andric if (oso_dwarf) 8000b57cec5SDimitry Andric return oso_dwarf->GetDynamicArrayInfoForUID(type_uid, exe_ctx); 801bdd1243dSDimitry Andric return std::nullopt; 8020b57cec5SDimitry Andric } 8030b57cec5SDimitry Andric 8040b57cec5SDimitry Andric bool SymbolFileDWARFDebugMap::CompleteType(CompilerType &compiler_type) { 8050b57cec5SDimitry Andric bool success = false; 8060b57cec5SDimitry Andric if (compiler_type) { 807*0fca6ea1SDimitry Andric ForEachSymbolFile([&](SymbolFileDWARF *oso_dwarf) { 8085f757f3fSDimitry Andric if (oso_dwarf->HasForwardDeclForCompilerType(compiler_type)) { 8090b57cec5SDimitry Andric oso_dwarf->CompleteType(compiler_type); 8100b57cec5SDimitry Andric success = true; 811*0fca6ea1SDimitry Andric return IterationAction::Stop; 8120b57cec5SDimitry Andric } 813*0fca6ea1SDimitry Andric return IterationAction::Continue; 8140b57cec5SDimitry Andric }); 8150b57cec5SDimitry Andric } 8160b57cec5SDimitry Andric return success; 8170b57cec5SDimitry Andric } 8180b57cec5SDimitry Andric 8190b57cec5SDimitry Andric uint32_t 8200b57cec5SDimitry Andric SymbolFileDWARFDebugMap::ResolveSymbolContext(const Address &exe_so_addr, 8210b57cec5SDimitry Andric SymbolContextItem resolve_scope, 8220b57cec5SDimitry Andric SymbolContext &sc) { 8239dba64beSDimitry Andric std::lock_guard<std::recursive_mutex> guard(GetModuleMutex()); 8240b57cec5SDimitry Andric uint32_t resolved_flags = 0; 8259dba64beSDimitry Andric Symtab *symtab = m_objfile_sp->GetSymtab(); 8260b57cec5SDimitry Andric if (symtab) { 8270b57cec5SDimitry Andric const addr_t exe_file_addr = exe_so_addr.GetFileAddress(); 8280b57cec5SDimitry Andric 8290b57cec5SDimitry Andric const DebugMap::Entry *debug_map_entry = 8300b57cec5SDimitry Andric m_debug_map.FindEntryThatContains(exe_file_addr); 8310b57cec5SDimitry Andric if (debug_map_entry) { 8320b57cec5SDimitry Andric 8330b57cec5SDimitry Andric sc.symbol = 8340b57cec5SDimitry Andric symtab->SymbolAtIndex(debug_map_entry->data.GetExeSymbolIndex()); 8350b57cec5SDimitry Andric 8360b57cec5SDimitry Andric if (sc.symbol != nullptr) { 8370b57cec5SDimitry Andric resolved_flags |= eSymbolContextSymbol; 8380b57cec5SDimitry Andric 8390b57cec5SDimitry Andric uint32_t oso_idx = 0; 8400b57cec5SDimitry Andric CompileUnitInfo *comp_unit_info = 8410b57cec5SDimitry Andric GetCompileUnitInfoForSymbolWithID(sc.symbol->GetID(), &oso_idx); 8420b57cec5SDimitry Andric if (comp_unit_info) { 8430b57cec5SDimitry Andric comp_unit_info->GetFileRangeMap(this); 8440b57cec5SDimitry Andric Module *oso_module = GetModuleByCompUnitInfo(comp_unit_info); 8450b57cec5SDimitry Andric if (oso_module) { 8460b57cec5SDimitry Andric lldb::addr_t oso_file_addr = 8470b57cec5SDimitry Andric exe_file_addr - debug_map_entry->GetRangeBase() + 8480b57cec5SDimitry Andric debug_map_entry->data.GetOSOFileAddress(); 8490b57cec5SDimitry Andric Address oso_so_addr; 8500b57cec5SDimitry Andric if (oso_module->ResolveFileAddress(oso_file_addr, oso_so_addr)) { 851*0fca6ea1SDimitry Andric if (SymbolFile *sym_file = oso_module->GetSymbolFile()) { 852*0fca6ea1SDimitry Andric resolved_flags |= sym_file->ResolveSymbolContext( 8530b57cec5SDimitry Andric oso_so_addr, resolve_scope, sc); 854*0fca6ea1SDimitry Andric } else { 855*0fca6ea1SDimitry Andric ObjectFile *obj_file = GetObjectFile(); 856*0fca6ea1SDimitry Andric LLDB_LOG(GetLog(DWARFLog::DebugMap), 857*0fca6ea1SDimitry Andric "Failed to get symfile for OSO: {0} in module: {1}", 858*0fca6ea1SDimitry Andric oso_module->GetFileSpec(), 859*0fca6ea1SDimitry Andric obj_file ? obj_file->GetFileSpec() 860*0fca6ea1SDimitry Andric : FileSpec("unknown")); 861*0fca6ea1SDimitry Andric } 8620b57cec5SDimitry Andric } 8630b57cec5SDimitry Andric } 8640b57cec5SDimitry Andric } 8650b57cec5SDimitry Andric } 8660b57cec5SDimitry Andric } 8670b57cec5SDimitry Andric } 8680b57cec5SDimitry Andric return resolved_flags; 8690b57cec5SDimitry Andric } 8700b57cec5SDimitry Andric 8710b57cec5SDimitry Andric uint32_t SymbolFileDWARFDebugMap::ResolveSymbolContext( 872fe6060f1SDimitry Andric const SourceLocationSpec &src_location_spec, 8730b57cec5SDimitry Andric SymbolContextItem resolve_scope, SymbolContextList &sc_list) { 8749dba64beSDimitry Andric std::lock_guard<std::recursive_mutex> guard(GetModuleMutex()); 8750b57cec5SDimitry Andric const uint32_t initial = sc_list.GetSize(); 8760b57cec5SDimitry Andric const uint32_t cu_count = GetNumCompileUnits(); 8770b57cec5SDimitry Andric 8780b57cec5SDimitry Andric for (uint32_t i = 0; i < cu_count; ++i) { 8790b57cec5SDimitry Andric // If we are checking for inlines, then we need to look through all compile 8800b57cec5SDimitry Andric // units no matter if "file_spec" matches. 881fe6060f1SDimitry Andric bool resolve = src_location_spec.GetCheckInlines(); 8820b57cec5SDimitry Andric 8830b57cec5SDimitry Andric if (!resolve) { 8840b57cec5SDimitry Andric FileSpec so_file_spec; 885480093f4SDimitry Andric if (GetFileSpecForSO(i, so_file_spec)) 886fe6060f1SDimitry Andric resolve = 887fe6060f1SDimitry Andric FileSpec::Match(src_location_spec.GetFileSpec(), so_file_spec); 8880b57cec5SDimitry Andric } 8890b57cec5SDimitry Andric if (resolve) { 8900b57cec5SDimitry Andric SymbolFileDWARF *oso_dwarf = GetSymbolFileByOSOIndex(i); 8910b57cec5SDimitry Andric if (oso_dwarf) 892fe6060f1SDimitry Andric oso_dwarf->ResolveSymbolContext(src_location_spec, resolve_scope, 893fe6060f1SDimitry Andric sc_list); 8940b57cec5SDimitry Andric } 8950b57cec5SDimitry Andric } 8960b57cec5SDimitry Andric return sc_list.GetSize() - initial; 8970b57cec5SDimitry Andric } 8980b57cec5SDimitry Andric 8999dba64beSDimitry Andric void SymbolFileDWARFDebugMap::PrivateFindGlobalVariables( 9005ffd83dbSDimitry Andric ConstString name, const CompilerDeclContext &parent_decl_ctx, 9010b57cec5SDimitry Andric const std::vector<uint32_t> 9020b57cec5SDimitry Andric &indexes, // Indexes into the symbol table that match "name" 9030b57cec5SDimitry Andric uint32_t max_matches, VariableList &variables) { 9040b57cec5SDimitry Andric const size_t match_count = indexes.size(); 9050b57cec5SDimitry Andric for (size_t i = 0; i < match_count; ++i) { 9060b57cec5SDimitry Andric uint32_t oso_idx; 9070b57cec5SDimitry Andric CompileUnitInfo *comp_unit_info = 9080b57cec5SDimitry Andric GetCompileUnitInfoForSymbolWithIndex(indexes[i], &oso_idx); 9090b57cec5SDimitry Andric if (comp_unit_info) { 9100b57cec5SDimitry Andric SymbolFileDWARF *oso_dwarf = GetSymbolFileByOSOIndex(oso_idx); 9110b57cec5SDimitry Andric if (oso_dwarf) { 9129dba64beSDimitry Andric oso_dwarf->FindGlobalVariables(name, parent_decl_ctx, max_matches, 9139dba64beSDimitry Andric variables); 9140b57cec5SDimitry Andric if (variables.GetSize() > max_matches) 9150b57cec5SDimitry Andric break; 9160b57cec5SDimitry Andric } 9170b57cec5SDimitry Andric } 9180b57cec5SDimitry Andric } 9190b57cec5SDimitry Andric } 9200b57cec5SDimitry Andric 9219dba64beSDimitry Andric void SymbolFileDWARFDebugMap::FindGlobalVariables( 9225ffd83dbSDimitry Andric ConstString name, const CompilerDeclContext &parent_decl_ctx, 9230b57cec5SDimitry Andric uint32_t max_matches, VariableList &variables) { 9249dba64beSDimitry Andric std::lock_guard<std::recursive_mutex> guard(GetModuleMutex()); 9250b57cec5SDimitry Andric uint32_t total_matches = 0; 9260b57cec5SDimitry Andric 927*0fca6ea1SDimitry Andric ForEachSymbolFile([&](SymbolFileDWARF *oso_dwarf) { 9289dba64beSDimitry Andric const uint32_t old_size = variables.GetSize(); 9299dba64beSDimitry Andric oso_dwarf->FindGlobalVariables(name, parent_decl_ctx, max_matches, 9309dba64beSDimitry Andric variables); 9319dba64beSDimitry Andric const uint32_t oso_matches = variables.GetSize() - old_size; 9320b57cec5SDimitry Andric if (oso_matches > 0) { 9330b57cec5SDimitry Andric total_matches += oso_matches; 9340b57cec5SDimitry Andric 9350b57cec5SDimitry Andric // Are we getting all matches? 9360b57cec5SDimitry Andric if (max_matches == UINT32_MAX) 937*0fca6ea1SDimitry Andric return IterationAction::Continue; // Yep, continue getting everything 9380b57cec5SDimitry Andric 9390b57cec5SDimitry Andric // If we have found enough matches, lets get out 9400b57cec5SDimitry Andric if (max_matches >= total_matches) 941*0fca6ea1SDimitry Andric return IterationAction::Stop; 9420b57cec5SDimitry Andric 9430b57cec5SDimitry Andric // Update the max matches for any subsequent calls to find globals in any 9440b57cec5SDimitry Andric // other object files with DWARF 9450b57cec5SDimitry Andric max_matches -= oso_matches; 9460b57cec5SDimitry Andric } 9470b57cec5SDimitry Andric 948*0fca6ea1SDimitry Andric return IterationAction::Continue; 9490b57cec5SDimitry Andric }); 9500b57cec5SDimitry Andric } 9510b57cec5SDimitry Andric 9529dba64beSDimitry Andric void SymbolFileDWARFDebugMap::FindGlobalVariables( 9539dba64beSDimitry Andric const RegularExpression ®ex, uint32_t max_matches, 9540b57cec5SDimitry Andric VariableList &variables) { 9559dba64beSDimitry Andric std::lock_guard<std::recursive_mutex> guard(GetModuleMutex()); 9560b57cec5SDimitry Andric uint32_t total_matches = 0; 957*0fca6ea1SDimitry Andric ForEachSymbolFile([&](SymbolFileDWARF *oso_dwarf) { 9589dba64beSDimitry Andric const uint32_t old_size = variables.GetSize(); 9590b57cec5SDimitry Andric oso_dwarf->FindGlobalVariables(regex, max_matches, variables); 9609dba64beSDimitry Andric 9619dba64beSDimitry Andric const uint32_t oso_matches = variables.GetSize() - old_size; 9620b57cec5SDimitry Andric if (oso_matches > 0) { 9630b57cec5SDimitry Andric total_matches += oso_matches; 9640b57cec5SDimitry Andric 9650b57cec5SDimitry Andric // Are we getting all matches? 9660b57cec5SDimitry Andric if (max_matches == UINT32_MAX) 967*0fca6ea1SDimitry Andric return IterationAction::Continue; // Yep, continue getting everything 9680b57cec5SDimitry Andric 9690b57cec5SDimitry Andric // If we have found enough matches, lets get out 9700b57cec5SDimitry Andric if (max_matches >= total_matches) 971*0fca6ea1SDimitry Andric return IterationAction::Stop; 9720b57cec5SDimitry Andric 9730b57cec5SDimitry Andric // Update the max matches for any subsequent calls to find globals in any 9740b57cec5SDimitry Andric // other object files with DWARF 9750b57cec5SDimitry Andric max_matches -= oso_matches; 9760b57cec5SDimitry Andric } 9770b57cec5SDimitry Andric 978*0fca6ea1SDimitry Andric return IterationAction::Continue; 9790b57cec5SDimitry Andric }); 9800b57cec5SDimitry Andric } 9810b57cec5SDimitry Andric 9820b57cec5SDimitry Andric int SymbolFileDWARFDebugMap::SymbolContainsSymbolWithIndex( 9830b57cec5SDimitry Andric uint32_t *symbol_idx_ptr, const CompileUnitInfo *comp_unit_info) { 9840b57cec5SDimitry Andric const uint32_t symbol_idx = *symbol_idx_ptr; 9850b57cec5SDimitry Andric 9860b57cec5SDimitry Andric if (symbol_idx < comp_unit_info->first_symbol_index) 9870b57cec5SDimitry Andric return -1; 9880b57cec5SDimitry Andric 9890b57cec5SDimitry Andric if (symbol_idx <= comp_unit_info->last_symbol_index) 9900b57cec5SDimitry Andric return 0; 9910b57cec5SDimitry Andric 9920b57cec5SDimitry Andric return 1; 9930b57cec5SDimitry Andric } 9940b57cec5SDimitry Andric 9950b57cec5SDimitry Andric int SymbolFileDWARFDebugMap::SymbolContainsSymbolWithID( 9960b57cec5SDimitry Andric user_id_t *symbol_idx_ptr, const CompileUnitInfo *comp_unit_info) { 9970b57cec5SDimitry Andric const user_id_t symbol_id = *symbol_idx_ptr; 9980b57cec5SDimitry Andric 9990b57cec5SDimitry Andric if (symbol_id < comp_unit_info->first_symbol_id) 10000b57cec5SDimitry Andric return -1; 10010b57cec5SDimitry Andric 10020b57cec5SDimitry Andric if (symbol_id <= comp_unit_info->last_symbol_id) 10030b57cec5SDimitry Andric return 0; 10040b57cec5SDimitry Andric 10050b57cec5SDimitry Andric return 1; 10060b57cec5SDimitry Andric } 10070b57cec5SDimitry Andric 10080b57cec5SDimitry Andric SymbolFileDWARFDebugMap::CompileUnitInfo * 10090b57cec5SDimitry Andric SymbolFileDWARFDebugMap::GetCompileUnitInfoForSymbolWithIndex( 10100b57cec5SDimitry Andric uint32_t symbol_idx, uint32_t *oso_idx_ptr) { 10110b57cec5SDimitry Andric const uint32_t oso_index_count = m_compile_unit_infos.size(); 10120b57cec5SDimitry Andric CompileUnitInfo *comp_unit_info = nullptr; 10130b57cec5SDimitry Andric if (oso_index_count) { 10140b57cec5SDimitry Andric comp_unit_info = (CompileUnitInfo *)bsearch( 10150b57cec5SDimitry Andric &symbol_idx, &m_compile_unit_infos[0], m_compile_unit_infos.size(), 10160b57cec5SDimitry Andric sizeof(CompileUnitInfo), 10170b57cec5SDimitry Andric (ComparisonFunction)SymbolContainsSymbolWithIndex); 10180b57cec5SDimitry Andric } 10190b57cec5SDimitry Andric 10200b57cec5SDimitry Andric if (oso_idx_ptr) { 10210b57cec5SDimitry Andric if (comp_unit_info != nullptr) 10220b57cec5SDimitry Andric *oso_idx_ptr = comp_unit_info - &m_compile_unit_infos[0]; 10230b57cec5SDimitry Andric else 10240b57cec5SDimitry Andric *oso_idx_ptr = UINT32_MAX; 10250b57cec5SDimitry Andric } 10260b57cec5SDimitry Andric return comp_unit_info; 10270b57cec5SDimitry Andric } 10280b57cec5SDimitry Andric 10290b57cec5SDimitry Andric SymbolFileDWARFDebugMap::CompileUnitInfo * 10300b57cec5SDimitry Andric SymbolFileDWARFDebugMap::GetCompileUnitInfoForSymbolWithID( 10310b57cec5SDimitry Andric user_id_t symbol_id, uint32_t *oso_idx_ptr) { 10320b57cec5SDimitry Andric const uint32_t oso_index_count = m_compile_unit_infos.size(); 10330b57cec5SDimitry Andric CompileUnitInfo *comp_unit_info = nullptr; 10340b57cec5SDimitry Andric if (oso_index_count) { 10350b57cec5SDimitry Andric comp_unit_info = (CompileUnitInfo *)::bsearch( 10360b57cec5SDimitry Andric &symbol_id, &m_compile_unit_infos[0], m_compile_unit_infos.size(), 10370b57cec5SDimitry Andric sizeof(CompileUnitInfo), 10380b57cec5SDimitry Andric (ComparisonFunction)SymbolContainsSymbolWithID); 10390b57cec5SDimitry Andric } 10400b57cec5SDimitry Andric 10410b57cec5SDimitry Andric if (oso_idx_ptr) { 10420b57cec5SDimitry Andric if (comp_unit_info != nullptr) 10430b57cec5SDimitry Andric *oso_idx_ptr = comp_unit_info - &m_compile_unit_infos[0]; 10440b57cec5SDimitry Andric else 10450b57cec5SDimitry Andric *oso_idx_ptr = UINT32_MAX; 10460b57cec5SDimitry Andric } 10470b57cec5SDimitry Andric return comp_unit_info; 10480b57cec5SDimitry Andric } 10490b57cec5SDimitry Andric 10500b57cec5SDimitry Andric static void RemoveFunctionsWithModuleNotEqualTo(const ModuleSP &module_sp, 10510b57cec5SDimitry Andric SymbolContextList &sc_list, 10520b57cec5SDimitry Andric uint32_t start_idx) { 10530b57cec5SDimitry Andric // We found functions in .o files. Not all functions in the .o files will 10540b57cec5SDimitry Andric // have made it into the final output file. The ones that did make it into 10550b57cec5SDimitry Andric // the final output file will have a section whose module matches the module 10560b57cec5SDimitry Andric // from the ObjectFile for this SymbolFile. When the modules don't match, 10570b57cec5SDimitry Andric // then we have something that was in a .o file, but doesn't map to anything 10580b57cec5SDimitry Andric // in the final executable. 10590b57cec5SDimitry Andric uint32_t i = start_idx; 10600b57cec5SDimitry Andric while (i < sc_list.GetSize()) { 10610b57cec5SDimitry Andric SymbolContext sc; 10620b57cec5SDimitry Andric sc_list.GetContextAtIndex(i, sc); 10630b57cec5SDimitry Andric if (sc.function) { 10640b57cec5SDimitry Andric const SectionSP section_sp( 10650b57cec5SDimitry Andric sc.function->GetAddressRange().GetBaseAddress().GetSection()); 10660b57cec5SDimitry Andric if (section_sp->GetModule() != module_sp) { 10670b57cec5SDimitry Andric sc_list.RemoveContextAtIndex(i); 10680b57cec5SDimitry Andric continue; 10690b57cec5SDimitry Andric } 10700b57cec5SDimitry Andric } 10710b57cec5SDimitry Andric ++i; 10720b57cec5SDimitry Andric } 10730b57cec5SDimitry Andric } 10740b57cec5SDimitry Andric 10759dba64beSDimitry Andric void SymbolFileDWARFDebugMap::FindFunctions( 1076bdd1243dSDimitry Andric const Module::LookupInfo &lookup_info, 1077bdd1243dSDimitry Andric const CompilerDeclContext &parent_decl_ctx, bool include_inlines, 10780b57cec5SDimitry Andric SymbolContextList &sc_list) { 10799dba64beSDimitry Andric std::lock_guard<std::recursive_mutex> guard(GetModuleMutex()); 1080e8d8bef9SDimitry Andric LLDB_SCOPED_TIMERF("SymbolFileDWARFDebugMap::FindFunctions (name = %s)", 1081bdd1243dSDimitry Andric lookup_info.GetLookupName().GetCString()); 10820b57cec5SDimitry Andric 1083*0fca6ea1SDimitry Andric ForEachSymbolFile([&](SymbolFileDWARF *oso_dwarf) { 10840b57cec5SDimitry Andric uint32_t sc_idx = sc_list.GetSize(); 1085bdd1243dSDimitry Andric oso_dwarf->FindFunctions(lookup_info, parent_decl_ctx, include_inlines, 1086bdd1243dSDimitry Andric sc_list); 10879dba64beSDimitry Andric if (!sc_list.IsEmpty()) { 10889dba64beSDimitry Andric RemoveFunctionsWithModuleNotEqualTo(m_objfile_sp->GetModule(), sc_list, 10890b57cec5SDimitry Andric sc_idx); 10900b57cec5SDimitry Andric } 1091*0fca6ea1SDimitry Andric return IterationAction::Continue; 10920b57cec5SDimitry Andric }); 10930b57cec5SDimitry Andric } 10940b57cec5SDimitry Andric 10959dba64beSDimitry Andric void SymbolFileDWARFDebugMap::FindFunctions(const RegularExpression ®ex, 10960b57cec5SDimitry Andric bool include_inlines, 10970b57cec5SDimitry Andric SymbolContextList &sc_list) { 10989dba64beSDimitry Andric std::lock_guard<std::recursive_mutex> guard(GetModuleMutex()); 1099e8d8bef9SDimitry Andric LLDB_SCOPED_TIMERF("SymbolFileDWARFDebugMap::FindFunctions (regex = '%s')", 11000b57cec5SDimitry Andric regex.GetText().str().c_str()); 11010b57cec5SDimitry Andric 1102*0fca6ea1SDimitry Andric ForEachSymbolFile([&](SymbolFileDWARF *oso_dwarf) { 11030b57cec5SDimitry Andric uint32_t sc_idx = sc_list.GetSize(); 11040b57cec5SDimitry Andric 11059dba64beSDimitry Andric oso_dwarf->FindFunctions(regex, include_inlines, sc_list); 11069dba64beSDimitry Andric if (!sc_list.IsEmpty()) { 11079dba64beSDimitry Andric RemoveFunctionsWithModuleNotEqualTo(m_objfile_sp->GetModule(), sc_list, 11080b57cec5SDimitry Andric sc_idx); 11090b57cec5SDimitry Andric } 1110*0fca6ea1SDimitry Andric return IterationAction::Continue; 11110b57cec5SDimitry Andric }); 11120b57cec5SDimitry Andric } 11130b57cec5SDimitry Andric 11149dba64beSDimitry Andric void SymbolFileDWARFDebugMap::GetTypes(SymbolContextScope *sc_scope, 11150b57cec5SDimitry Andric lldb::TypeClass type_mask, 11160b57cec5SDimitry Andric TypeList &type_list) { 11179dba64beSDimitry Andric std::lock_guard<std::recursive_mutex> guard(GetModuleMutex()); 1118e8d8bef9SDimitry Andric LLDB_SCOPED_TIMERF("SymbolFileDWARFDebugMap::GetTypes (type_mask = 0x%8.8x)", 11190b57cec5SDimitry Andric type_mask); 11200b57cec5SDimitry Andric 11210b57cec5SDimitry Andric SymbolFileDWARF *oso_dwarf = nullptr; 11220b57cec5SDimitry Andric if (sc_scope) { 11230b57cec5SDimitry Andric SymbolContext sc; 11240b57cec5SDimitry Andric sc_scope->CalculateSymbolContext(&sc); 11250b57cec5SDimitry Andric 11260b57cec5SDimitry Andric CompileUnitInfo *cu_info = GetCompUnitInfo(sc); 11270b57cec5SDimitry Andric if (cu_info) { 11280b57cec5SDimitry Andric oso_dwarf = GetSymbolFileByCompUnitInfo(cu_info); 11290b57cec5SDimitry Andric if (oso_dwarf) 11300b57cec5SDimitry Andric oso_dwarf->GetTypes(sc_scope, type_mask, type_list); 11310b57cec5SDimitry Andric } 11320b57cec5SDimitry Andric } else { 1133*0fca6ea1SDimitry Andric ForEachSymbolFile([&](SymbolFileDWARF *oso_dwarf) { 11340b57cec5SDimitry Andric oso_dwarf->GetTypes(sc_scope, type_mask, type_list); 1135*0fca6ea1SDimitry Andric return IterationAction::Continue; 11360b57cec5SDimitry Andric }); 11370b57cec5SDimitry Andric } 11380b57cec5SDimitry Andric } 11390b57cec5SDimitry Andric 1140480093f4SDimitry Andric std::vector<std::unique_ptr<lldb_private::CallEdge>> 114106c3fb27SDimitry Andric SymbolFileDWARFDebugMap::ParseCallEdgesInFunction( 114206c3fb27SDimitry Andric lldb_private::UserID func_id) { 11430b57cec5SDimitry Andric uint32_t oso_idx = GetOSOIndexFromUserID(func_id.GetID()); 11440b57cec5SDimitry Andric SymbolFileDWARF *oso_dwarf = GetSymbolFileByOSOIndex(oso_idx); 11450b57cec5SDimitry Andric if (oso_dwarf) 11460b57cec5SDimitry Andric return oso_dwarf->ParseCallEdgesInFunction(func_id); 11470b57cec5SDimitry Andric return {}; 11480b57cec5SDimitry Andric } 11490b57cec5SDimitry Andric 1150*0fca6ea1SDimitry Andric DWARFDIE SymbolFileDWARFDebugMap::FindDefinitionDIE(const DWARFDIE &die) { 1151*0fca6ea1SDimitry Andric DWARFDIE result; 1152*0fca6ea1SDimitry Andric ForEachSymbolFile([&](SymbolFileDWARF *oso_dwarf) { 1153*0fca6ea1SDimitry Andric result = oso_dwarf->FindDefinitionDIE(die); 1154*0fca6ea1SDimitry Andric return result ? IterationAction::Stop : IterationAction::Continue; 11550b57cec5SDimitry Andric }); 1156*0fca6ea1SDimitry Andric return result; 11570b57cec5SDimitry Andric } 11580b57cec5SDimitry Andric 11590b57cec5SDimitry Andric bool SymbolFileDWARFDebugMap::Supports_DW_AT_APPLE_objc_complete_type( 11600b57cec5SDimitry Andric SymbolFileDWARF *skip_dwarf_oso) { 11610b57cec5SDimitry Andric if (m_supports_DW_AT_APPLE_objc_complete_type == eLazyBoolCalculate) { 11620b57cec5SDimitry Andric m_supports_DW_AT_APPLE_objc_complete_type = eLazyBoolNo; 1163*0fca6ea1SDimitry Andric ForEachSymbolFile([&](SymbolFileDWARF *oso_dwarf) { 11640b57cec5SDimitry Andric if (skip_dwarf_oso != oso_dwarf && 11650b57cec5SDimitry Andric oso_dwarf->Supports_DW_AT_APPLE_objc_complete_type(nullptr)) { 11660b57cec5SDimitry Andric m_supports_DW_AT_APPLE_objc_complete_type = eLazyBoolYes; 1167*0fca6ea1SDimitry Andric return IterationAction::Stop; 11680b57cec5SDimitry Andric } 1169*0fca6ea1SDimitry Andric return IterationAction::Continue; 11700b57cec5SDimitry Andric }); 11710b57cec5SDimitry Andric } 11720b57cec5SDimitry Andric return m_supports_DW_AT_APPLE_objc_complete_type == eLazyBoolYes; 11730b57cec5SDimitry Andric } 11740b57cec5SDimitry Andric 11750b57cec5SDimitry Andric TypeSP SymbolFileDWARFDebugMap::FindCompleteObjCDefinitionTypeForDIE( 11760b57cec5SDimitry Andric const DWARFDIE &die, ConstString type_name, 11770b57cec5SDimitry Andric bool must_be_implementation) { 11780b57cec5SDimitry Andric // If we have a debug map, we will have an Objective-C symbol whose name is 11790b57cec5SDimitry Andric // the type name and whose type is eSymbolTypeObjCClass. If we can find that 11800b57cec5SDimitry Andric // symbol and find its containing parent, we can locate the .o file that will 11810b57cec5SDimitry Andric // contain the implementation definition since it will be scoped inside the 11820b57cec5SDimitry Andric // N_SO and we can then locate the SymbolFileDWARF that corresponds to that 11830b57cec5SDimitry Andric // N_SO. 11840b57cec5SDimitry Andric SymbolFileDWARF *oso_dwarf = nullptr; 11850b57cec5SDimitry Andric TypeSP type_sp; 11869dba64beSDimitry Andric ObjectFile *module_objfile = m_objfile_sp->GetModule()->GetObjectFile(); 11870b57cec5SDimitry Andric if (module_objfile) { 11880b57cec5SDimitry Andric Symtab *symtab = module_objfile->GetSymtab(); 11890b57cec5SDimitry Andric if (symtab) { 11900b57cec5SDimitry Andric Symbol *objc_class_symbol = symtab->FindFirstSymbolWithNameAndType( 11910b57cec5SDimitry Andric type_name, eSymbolTypeObjCClass, Symtab::eDebugAny, 11920b57cec5SDimitry Andric Symtab::eVisibilityAny); 11930b57cec5SDimitry Andric if (objc_class_symbol) { 11940b57cec5SDimitry Andric // Get the N_SO symbol that contains the objective C class symbol as 11950b57cec5SDimitry Andric // this should be the .o file that contains the real definition... 11960b57cec5SDimitry Andric const Symbol *source_file_symbol = symtab->GetParent(objc_class_symbol); 11970b57cec5SDimitry Andric 11980b57cec5SDimitry Andric if (source_file_symbol && 11990b57cec5SDimitry Andric source_file_symbol->GetType() == eSymbolTypeSourceFile) { 12000b57cec5SDimitry Andric const uint32_t source_file_symbol_idx = 12010b57cec5SDimitry Andric symtab->GetIndexForSymbol(source_file_symbol); 12020b57cec5SDimitry Andric if (source_file_symbol_idx != UINT32_MAX) { 12030b57cec5SDimitry Andric CompileUnitInfo *compile_unit_info = 12040b57cec5SDimitry Andric GetCompileUnitInfoForSymbolWithIndex(source_file_symbol_idx, 12050b57cec5SDimitry Andric nullptr); 12060b57cec5SDimitry Andric if (compile_unit_info) { 12070b57cec5SDimitry Andric oso_dwarf = GetSymbolFileByCompUnitInfo(compile_unit_info); 12080b57cec5SDimitry Andric if (oso_dwarf) { 12090b57cec5SDimitry Andric TypeSP type_sp(oso_dwarf->FindCompleteObjCDefinitionTypeForDIE( 12100b57cec5SDimitry Andric die, type_name, must_be_implementation)); 12110b57cec5SDimitry Andric if (type_sp) { 12120b57cec5SDimitry Andric return type_sp; 12130b57cec5SDimitry Andric } 12140b57cec5SDimitry Andric } 12150b57cec5SDimitry Andric } 12160b57cec5SDimitry Andric } 12170b57cec5SDimitry Andric } 12180b57cec5SDimitry Andric } 12190b57cec5SDimitry Andric } 12200b57cec5SDimitry Andric } 12210b57cec5SDimitry Andric 12220b57cec5SDimitry Andric // Only search all .o files for the definition if we don't need the 12230b57cec5SDimitry Andric // implementation because otherwise, with a valid debug map we should have 12240b57cec5SDimitry Andric // the ObjC class symbol and the code above should have found it. 12250b57cec5SDimitry Andric if (!must_be_implementation) { 12260b57cec5SDimitry Andric TypeSP type_sp; 12270b57cec5SDimitry Andric 1228*0fca6ea1SDimitry Andric ForEachSymbolFile([&](SymbolFileDWARF *oso_dwarf) { 12290b57cec5SDimitry Andric type_sp = oso_dwarf->FindCompleteObjCDefinitionTypeForDIE( 12300b57cec5SDimitry Andric die, type_name, must_be_implementation); 1231*0fca6ea1SDimitry Andric return type_sp ? IterationAction::Stop : IterationAction::Continue; 12320b57cec5SDimitry Andric }); 12330b57cec5SDimitry Andric 12340b57cec5SDimitry Andric return type_sp; 12350b57cec5SDimitry Andric } 12360b57cec5SDimitry Andric return TypeSP(); 12370b57cec5SDimitry Andric } 12380b57cec5SDimitry Andric 12395f757f3fSDimitry Andric void SymbolFileDWARFDebugMap::FindTypes(const TypeQuery &query, 12405f757f3fSDimitry Andric TypeResults &results) { 12419dba64beSDimitry Andric std::lock_guard<std::recursive_mutex> guard(GetModuleMutex()); 1242*0fca6ea1SDimitry Andric ForEachSymbolFile([&](SymbolFileDWARF *oso_dwarf) { 12435f757f3fSDimitry Andric oso_dwarf->FindTypes(query, results); 1244*0fca6ea1SDimitry Andric return results.Done(query) ? IterationAction::Stop 1245*0fca6ea1SDimitry Andric : IterationAction::Continue; 1246480093f4SDimitry Andric }); 1247480093f4SDimitry Andric } 1248480093f4SDimitry Andric 12490b57cec5SDimitry Andric CompilerDeclContext SymbolFileDWARFDebugMap::FindNamespace( 125006c3fb27SDimitry Andric lldb_private::ConstString name, const CompilerDeclContext &parent_decl_ctx, 125106c3fb27SDimitry Andric bool only_root_namespaces) { 12529dba64beSDimitry Andric std::lock_guard<std::recursive_mutex> guard(GetModuleMutex()); 12530b57cec5SDimitry Andric CompilerDeclContext matching_namespace; 12540b57cec5SDimitry Andric 1255*0fca6ea1SDimitry Andric ForEachSymbolFile([&](SymbolFileDWARF *oso_dwarf) { 125606c3fb27SDimitry Andric matching_namespace = 125706c3fb27SDimitry Andric oso_dwarf->FindNamespace(name, parent_decl_ctx, only_root_namespaces); 12580b57cec5SDimitry Andric 1259*0fca6ea1SDimitry Andric return matching_namespace ? IterationAction::Stop 1260*0fca6ea1SDimitry Andric : IterationAction::Continue; 12610b57cec5SDimitry Andric }); 12620b57cec5SDimitry Andric 12630b57cec5SDimitry Andric return matching_namespace; 12640b57cec5SDimitry Andric } 12650b57cec5SDimitry Andric 12660b57cec5SDimitry Andric void SymbolFileDWARFDebugMap::DumpClangAST(Stream &s) { 1267*0fca6ea1SDimitry Andric ForEachSymbolFile([&s](SymbolFileDWARF *oso_dwarf) { 12680b57cec5SDimitry Andric oso_dwarf->DumpClangAST(s); 12695ffd83dbSDimitry Andric // The underlying assumption is that DumpClangAST(...) will obtain the 12705ffd83dbSDimitry Andric // AST from the underlying TypeSystem and therefore we only need to do 12715ffd83dbSDimitry Andric // this once and can stop after the first iteration hence we return true. 1272*0fca6ea1SDimitry Andric return IterationAction::Stop; 12730b57cec5SDimitry Andric }); 12740b57cec5SDimitry Andric } 12750b57cec5SDimitry Andric 12765f757f3fSDimitry Andric bool SymbolFileDWARFDebugMap::GetSeparateDebugInfo( 12775f757f3fSDimitry Andric lldb_private::StructuredData::Dictionary &d, bool errors_only) { 12785f757f3fSDimitry Andric StructuredData::Array separate_debug_info_files; 12795f757f3fSDimitry Andric const uint32_t cu_count = GetNumCompileUnits(); 12805f757f3fSDimitry Andric for (uint32_t cu_idx = 0; cu_idx < cu_count; ++cu_idx) { 12815f757f3fSDimitry Andric const auto &info = m_compile_unit_infos[cu_idx]; 12825f757f3fSDimitry Andric StructuredData::DictionarySP oso_data = 12835f757f3fSDimitry Andric std::make_shared<StructuredData::Dictionary>(); 12845f757f3fSDimitry Andric oso_data->AddStringItem("so_file", info.so_file.GetPath()); 12855f757f3fSDimitry Andric oso_data->AddStringItem("oso_path", info.oso_path); 12865f757f3fSDimitry Andric oso_data->AddIntegerItem("oso_mod_time", 12875f757f3fSDimitry Andric (uint32_t)llvm::sys::toTimeT(info.oso_mod_time)); 12885f757f3fSDimitry Andric 12895f757f3fSDimitry Andric bool loaded_successfully = false; 12905f757f3fSDimitry Andric if (GetModuleByOSOIndex(cu_idx)) { 12915f757f3fSDimitry Andric // If we have a valid pointer to the module, we successfully 12925f757f3fSDimitry Andric // loaded the oso if there are no load errors. 12935f757f3fSDimitry Andric if (!info.oso_load_error.Fail()) { 12945f757f3fSDimitry Andric loaded_successfully = true; 12955f757f3fSDimitry Andric } 12965f757f3fSDimitry Andric } 12975f757f3fSDimitry Andric if (!loaded_successfully) { 12985f757f3fSDimitry Andric oso_data->AddStringItem("error", info.oso_load_error.AsCString()); 12995f757f3fSDimitry Andric } 13005f757f3fSDimitry Andric oso_data->AddBooleanItem("loaded", loaded_successfully); 13015f757f3fSDimitry Andric if (!errors_only || oso_data->HasKey("error")) 13025f757f3fSDimitry Andric separate_debug_info_files.AddItem(oso_data); 13035f757f3fSDimitry Andric } 13045f757f3fSDimitry Andric 13055f757f3fSDimitry Andric d.AddStringItem("type", "oso"); 13065f757f3fSDimitry Andric d.AddStringItem("symfile", GetMainObjectFile()->GetFileSpec().GetPath()); 13075f757f3fSDimitry Andric d.AddItem("separate-debug-info-files", 13085f757f3fSDimitry Andric std::make_shared<StructuredData::Array>( 13095f757f3fSDimitry Andric std::move(separate_debug_info_files))); 13105f757f3fSDimitry Andric return true; 13115f757f3fSDimitry Andric } 13125f757f3fSDimitry Andric 13130b57cec5SDimitry Andric lldb::CompUnitSP 1314bdd1243dSDimitry Andric SymbolFileDWARFDebugMap::GetCompileUnit(SymbolFileDWARF *oso_dwarf, DWARFCompileUnit &dwarf_cu) { 13150b57cec5SDimitry Andric if (oso_dwarf) { 13160b57cec5SDimitry Andric const uint32_t cu_count = GetNumCompileUnits(); 13170b57cec5SDimitry Andric for (uint32_t cu_idx = 0; cu_idx < cu_count; ++cu_idx) { 13180b57cec5SDimitry Andric SymbolFileDWARF *oso_symfile = 13190b57cec5SDimitry Andric GetSymbolFileByCompUnitInfo(&m_compile_unit_infos[cu_idx]); 13200b57cec5SDimitry Andric if (oso_symfile == oso_dwarf) { 1321bdd1243dSDimitry Andric if (m_compile_unit_infos[cu_idx].compile_units_sps.empty()) 13220b57cec5SDimitry Andric ParseCompileUnitAtIndex(cu_idx); 13230b57cec5SDimitry Andric 1324bdd1243dSDimitry Andric auto &id_to_index_map = m_compile_unit_infos[cu_idx].id_to_index_map; 1325bdd1243dSDimitry Andric auto it = id_to_index_map.find(dwarf_cu.GetID()); 1326bdd1243dSDimitry Andric if (it != id_to_index_map.end()) 1327bdd1243dSDimitry Andric return m_compile_unit_infos[cu_idx] 1328bdd1243dSDimitry Andric .compile_units_sps[it->getSecond()]; 13290b57cec5SDimitry Andric } 13300b57cec5SDimitry Andric } 13310b57cec5SDimitry Andric } 13320b57cec5SDimitry Andric llvm_unreachable("this shouldn't happen"); 13330b57cec5SDimitry Andric } 13340b57cec5SDimitry Andric 13350b57cec5SDimitry Andric SymbolFileDWARFDebugMap::CompileUnitInfo * 13360b57cec5SDimitry Andric SymbolFileDWARFDebugMap::GetCompileUnitInfo(SymbolFileDWARF *oso_dwarf) { 13370b57cec5SDimitry Andric if (oso_dwarf) { 13380b57cec5SDimitry Andric const uint32_t cu_count = GetNumCompileUnits(); 13390b57cec5SDimitry Andric for (uint32_t cu_idx = 0; cu_idx < cu_count; ++cu_idx) { 13400b57cec5SDimitry Andric SymbolFileDWARF *oso_symfile = 13410b57cec5SDimitry Andric GetSymbolFileByCompUnitInfo(&m_compile_unit_infos[cu_idx]); 13420b57cec5SDimitry Andric if (oso_symfile == oso_dwarf) { 13430b57cec5SDimitry Andric return &m_compile_unit_infos[cu_idx]; 13440b57cec5SDimitry Andric } 13450b57cec5SDimitry Andric } 13460b57cec5SDimitry Andric } 13470b57cec5SDimitry Andric return nullptr; 13480b57cec5SDimitry Andric } 13490b57cec5SDimitry Andric 13500b57cec5SDimitry Andric void SymbolFileDWARFDebugMap::SetCompileUnit(SymbolFileDWARF *oso_dwarf, 13510b57cec5SDimitry Andric const CompUnitSP &cu_sp) { 13520b57cec5SDimitry Andric if (oso_dwarf) { 13530b57cec5SDimitry Andric const uint32_t cu_count = GetNumCompileUnits(); 13540b57cec5SDimitry Andric for (uint32_t cu_idx = 0; cu_idx < cu_count; ++cu_idx) { 13550b57cec5SDimitry Andric SymbolFileDWARF *oso_symfile = 13560b57cec5SDimitry Andric GetSymbolFileByCompUnitInfo(&m_compile_unit_infos[cu_idx]); 13570b57cec5SDimitry Andric if (oso_symfile == oso_dwarf) { 1358bdd1243dSDimitry Andric if (!m_compile_unit_infos[cu_idx].compile_units_sps.empty()) { 1359bdd1243dSDimitry Andric assert(m_compile_unit_infos[cu_idx].compile_units_sps[0].get() == 13600b57cec5SDimitry Andric cu_sp.get()); 13610b57cec5SDimitry Andric } else { 1362bdd1243dSDimitry Andric assert(cu_sp->GetID() == 0 && 1363bdd1243dSDimitry Andric "Setting first compile unit but with id different than 0!"); 1364bdd1243dSDimitry Andric auto &compile_units_sps = m_compile_unit_infos[cu_idx].compile_units_sps; 1365bdd1243dSDimitry Andric compile_units_sps.push_back(cu_sp); 1366bdd1243dSDimitry Andric m_compile_unit_infos[cu_idx].id_to_index_map.insert( 1367bdd1243dSDimitry Andric {cu_sp->GetID(), compile_units_sps.size() - 1}); 1368bdd1243dSDimitry Andric 13699dba64beSDimitry Andric SetCompileUnitAtIndex(cu_idx, cu_sp); 13700b57cec5SDimitry Andric } 13710b57cec5SDimitry Andric } 13720b57cec5SDimitry Andric } 13730b57cec5SDimitry Andric } 13740b57cec5SDimitry Andric } 13750b57cec5SDimitry Andric 13760b57cec5SDimitry Andric CompilerDeclContext 13770b57cec5SDimitry Andric SymbolFileDWARFDebugMap::GetDeclContextForUID(lldb::user_id_t type_uid) { 13780b57cec5SDimitry Andric const uint64_t oso_idx = GetOSOIndexFromUserID(type_uid); 13795f757f3fSDimitry Andric if (SymbolFileDWARF *oso_dwarf = GetSymbolFileByOSOIndex(oso_idx)) 13800b57cec5SDimitry Andric return oso_dwarf->GetDeclContextForUID(type_uid); 13815f757f3fSDimitry Andric return {}; 13820b57cec5SDimitry Andric } 13830b57cec5SDimitry Andric 13840b57cec5SDimitry Andric CompilerDeclContext 13850b57cec5SDimitry Andric SymbolFileDWARFDebugMap::GetDeclContextContainingUID(lldb::user_id_t type_uid) { 13860b57cec5SDimitry Andric const uint64_t oso_idx = GetOSOIndexFromUserID(type_uid); 13875f757f3fSDimitry Andric if (SymbolFileDWARF *oso_dwarf = GetSymbolFileByOSOIndex(oso_idx)) 13880b57cec5SDimitry Andric return oso_dwarf->GetDeclContextContainingUID(type_uid); 13895f757f3fSDimitry Andric return {}; 13905f757f3fSDimitry Andric } 13915f757f3fSDimitry Andric 13925f757f3fSDimitry Andric std::vector<CompilerContext> 13935f757f3fSDimitry Andric SymbolFileDWARFDebugMap::GetCompilerContextForUID(lldb::user_id_t type_uid) { 13945f757f3fSDimitry Andric const uint64_t oso_idx = GetOSOIndexFromUserID(type_uid); 13955f757f3fSDimitry Andric if (SymbolFileDWARF *oso_dwarf = GetSymbolFileByOSOIndex(oso_idx)) 13965f757f3fSDimitry Andric return oso_dwarf->GetCompilerContextForUID(type_uid); 13975f757f3fSDimitry Andric return {}; 13980b57cec5SDimitry Andric } 13990b57cec5SDimitry Andric 14000b57cec5SDimitry Andric void SymbolFileDWARFDebugMap::ParseDeclsForContext( 14010b57cec5SDimitry Andric lldb_private::CompilerDeclContext decl_ctx) { 1402*0fca6ea1SDimitry Andric ForEachSymbolFile([&](SymbolFileDWARF *oso_dwarf) { 14030b57cec5SDimitry Andric oso_dwarf->ParseDeclsForContext(decl_ctx); 1404*0fca6ea1SDimitry Andric return IterationAction::Continue; 14050b57cec5SDimitry Andric }); 14060b57cec5SDimitry Andric } 14070b57cec5SDimitry Andric 14080b57cec5SDimitry Andric bool SymbolFileDWARFDebugMap::AddOSOFileRange(CompileUnitInfo *cu_info, 14090b57cec5SDimitry Andric lldb::addr_t exe_file_addr, 14100b57cec5SDimitry Andric lldb::addr_t exe_byte_size, 14110b57cec5SDimitry Andric lldb::addr_t oso_file_addr, 14120b57cec5SDimitry Andric lldb::addr_t oso_byte_size) { 14130b57cec5SDimitry Andric const uint32_t debug_map_idx = 14140b57cec5SDimitry Andric m_debug_map.FindEntryIndexThatContains(exe_file_addr); 14150b57cec5SDimitry Andric if (debug_map_idx != UINT32_MAX) { 14160b57cec5SDimitry Andric DebugMap::Entry *debug_map_entry = 14170b57cec5SDimitry Andric m_debug_map.FindEntryThatContains(exe_file_addr); 14180b57cec5SDimitry Andric debug_map_entry->data.SetOSOFileAddress(oso_file_addr); 14190b57cec5SDimitry Andric addr_t range_size = std::min<addr_t>(exe_byte_size, oso_byte_size); 14200b57cec5SDimitry Andric if (range_size == 0) { 14210b57cec5SDimitry Andric range_size = std::max<addr_t>(exe_byte_size, oso_byte_size); 14220b57cec5SDimitry Andric if (range_size == 0) 14230b57cec5SDimitry Andric range_size = 1; 14240b57cec5SDimitry Andric } 14250b57cec5SDimitry Andric cu_info->file_range_map.Append( 14260b57cec5SDimitry Andric FileRangeMap::Entry(oso_file_addr, range_size, exe_file_addr)); 14270b57cec5SDimitry Andric return true; 14280b57cec5SDimitry Andric } 14290b57cec5SDimitry Andric return false; 14300b57cec5SDimitry Andric } 14310b57cec5SDimitry Andric 14320b57cec5SDimitry Andric void SymbolFileDWARFDebugMap::FinalizeOSOFileRanges(CompileUnitInfo *cu_info) { 14330b57cec5SDimitry Andric cu_info->file_range_map.Sort(); 14340b57cec5SDimitry Andric #if defined(DEBUG_OSO_DMAP) 14350b57cec5SDimitry Andric const FileRangeMap &oso_file_range_map = cu_info->GetFileRangeMap(this); 14360b57cec5SDimitry Andric const size_t n = oso_file_range_map.GetSize(); 14370b57cec5SDimitry Andric printf("SymbolFileDWARFDebugMap::FinalizeOSOFileRanges (cu_info = %p) %s\n", 14380b57cec5SDimitry Andric cu_info, cu_info->oso_sp->module_sp->GetFileSpec().GetPath().c_str()); 14390b57cec5SDimitry Andric for (size_t i = 0; i < n; ++i) { 14400b57cec5SDimitry Andric const FileRangeMap::Entry &entry = oso_file_range_map.GetEntryRef(i); 14410b57cec5SDimitry Andric printf("oso [0x%16.16" PRIx64 " - 0x%16.16" PRIx64 14420b57cec5SDimitry Andric ") ==> exe [0x%16.16" PRIx64 " - 0x%16.16" PRIx64 ")\n", 14430b57cec5SDimitry Andric entry.GetRangeBase(), entry.GetRangeEnd(), entry.data, 14440b57cec5SDimitry Andric entry.data + entry.GetByteSize()); 14450b57cec5SDimitry Andric } 14460b57cec5SDimitry Andric #endif 14470b57cec5SDimitry Andric } 14480b57cec5SDimitry Andric 14490b57cec5SDimitry Andric lldb::addr_t 14500b57cec5SDimitry Andric SymbolFileDWARFDebugMap::LinkOSOFileAddress(SymbolFileDWARF *oso_symfile, 14510b57cec5SDimitry Andric lldb::addr_t oso_file_addr) { 14520b57cec5SDimitry Andric CompileUnitInfo *cu_info = GetCompileUnitInfo(oso_symfile); 14530b57cec5SDimitry Andric if (cu_info) { 14540b57cec5SDimitry Andric const FileRangeMap::Entry *oso_range_entry = 14550b57cec5SDimitry Andric cu_info->GetFileRangeMap(this).FindEntryThatContains(oso_file_addr); 14560b57cec5SDimitry Andric if (oso_range_entry) { 14570b57cec5SDimitry Andric const DebugMap::Entry *debug_map_entry = 14580b57cec5SDimitry Andric m_debug_map.FindEntryThatContains(oso_range_entry->data); 14590b57cec5SDimitry Andric if (debug_map_entry) { 14600b57cec5SDimitry Andric const lldb::addr_t offset = 14610b57cec5SDimitry Andric oso_file_addr - oso_range_entry->GetRangeBase(); 14620b57cec5SDimitry Andric const lldb::addr_t exe_file_addr = 14630b57cec5SDimitry Andric debug_map_entry->GetRangeBase() + offset; 14640b57cec5SDimitry Andric return exe_file_addr; 14650b57cec5SDimitry Andric } 14660b57cec5SDimitry Andric } 14670b57cec5SDimitry Andric } 14680b57cec5SDimitry Andric return LLDB_INVALID_ADDRESS; 14690b57cec5SDimitry Andric } 14700b57cec5SDimitry Andric 14710b57cec5SDimitry Andric bool SymbolFileDWARFDebugMap::LinkOSOAddress(Address &addr) { 14720b57cec5SDimitry Andric // Make sure this address hasn't been fixed already 14730b57cec5SDimitry Andric Module *exe_module = GetObjectFile()->GetModule().get(); 14740b57cec5SDimitry Andric Module *addr_module = addr.GetModule().get(); 14750b57cec5SDimitry Andric if (addr_module == exe_module) 14760b57cec5SDimitry Andric return true; // Address is already in terms of the main executable module 14770b57cec5SDimitry Andric 14789dba64beSDimitry Andric CompileUnitInfo *cu_info = GetCompileUnitInfo( 14799dba64beSDimitry Andric GetSymbolFileAsSymbolFileDWARF(addr_module->GetSymbolFile())); 14800b57cec5SDimitry Andric if (cu_info) { 14810b57cec5SDimitry Andric const lldb::addr_t oso_file_addr = addr.GetFileAddress(); 14820b57cec5SDimitry Andric const FileRangeMap::Entry *oso_range_entry = 14830b57cec5SDimitry Andric cu_info->GetFileRangeMap(this).FindEntryThatContains(oso_file_addr); 14840b57cec5SDimitry Andric if (oso_range_entry) { 14850b57cec5SDimitry Andric const DebugMap::Entry *debug_map_entry = 14860b57cec5SDimitry Andric m_debug_map.FindEntryThatContains(oso_range_entry->data); 14870b57cec5SDimitry Andric if (debug_map_entry) { 14880b57cec5SDimitry Andric const lldb::addr_t offset = 14890b57cec5SDimitry Andric oso_file_addr - oso_range_entry->GetRangeBase(); 14900b57cec5SDimitry Andric const lldb::addr_t exe_file_addr = 14910b57cec5SDimitry Andric debug_map_entry->GetRangeBase() + offset; 14920b57cec5SDimitry Andric return exe_module->ResolveFileAddress(exe_file_addr, addr); 14930b57cec5SDimitry Andric } 14940b57cec5SDimitry Andric } 14950b57cec5SDimitry Andric } 14960b57cec5SDimitry Andric return true; 14970b57cec5SDimitry Andric } 14980b57cec5SDimitry Andric 14990b57cec5SDimitry Andric LineTable *SymbolFileDWARFDebugMap::LinkOSOLineTable(SymbolFileDWARF *oso_dwarf, 15000b57cec5SDimitry Andric LineTable *line_table) { 15010b57cec5SDimitry Andric CompileUnitInfo *cu_info = GetCompileUnitInfo(oso_dwarf); 15020b57cec5SDimitry Andric if (cu_info) 15030b57cec5SDimitry Andric return line_table->LinkLineTable(cu_info->GetFileRangeMap(this)); 15040b57cec5SDimitry Andric return nullptr; 15050b57cec5SDimitry Andric } 15060b57cec5SDimitry Andric 15070b57cec5SDimitry Andric size_t 15080b57cec5SDimitry Andric SymbolFileDWARFDebugMap::AddOSOARanges(SymbolFileDWARF *dwarf2Data, 15090b57cec5SDimitry Andric DWARFDebugAranges *debug_aranges) { 15100b57cec5SDimitry Andric size_t num_line_entries_added = 0; 15110b57cec5SDimitry Andric if (debug_aranges && dwarf2Data) { 15120b57cec5SDimitry Andric CompileUnitInfo *compile_unit_info = GetCompileUnitInfo(dwarf2Data); 15130b57cec5SDimitry Andric if (compile_unit_info) { 15140b57cec5SDimitry Andric const FileRangeMap &file_range_map = 15150b57cec5SDimitry Andric compile_unit_info->GetFileRangeMap(this); 15160b57cec5SDimitry Andric for (size_t idx = 0; idx < file_range_map.GetSize(); idx++) { 15170b57cec5SDimitry Andric const FileRangeMap::Entry *entry = file_range_map.GetEntryAtIndex(idx); 15180b57cec5SDimitry Andric if (entry) { 151906c3fb27SDimitry Andric debug_aranges->AppendRange(*dwarf2Data->GetFileIndex(), 152006c3fb27SDimitry Andric entry->GetRangeBase(), 15210b57cec5SDimitry Andric entry->GetRangeEnd()); 15220b57cec5SDimitry Andric num_line_entries_added++; 15230b57cec5SDimitry Andric } 15240b57cec5SDimitry Andric } 15250b57cec5SDimitry Andric } 15260b57cec5SDimitry Andric } 15270b57cec5SDimitry Andric return num_line_entries_added; 15280b57cec5SDimitry Andric } 1529349cc55cSDimitry Andric 153081ad6265SDimitry Andric ModuleList SymbolFileDWARFDebugMap::GetDebugInfoModules() { 153181ad6265SDimitry Andric ModuleList oso_modules; 1532*0fca6ea1SDimitry Andric ForEachSymbolFile([&](SymbolFileDWARF *oso_dwarf) { 1533349cc55cSDimitry Andric ObjectFile *oso_objfile = oso_dwarf->GetObjectFile(); 1534349cc55cSDimitry Andric if (oso_objfile) { 1535349cc55cSDimitry Andric ModuleSP module_sp = oso_objfile->GetModule(); 153681ad6265SDimitry Andric if (module_sp) 153781ad6265SDimitry Andric oso_modules.Append(module_sp); 1538349cc55cSDimitry Andric } 1539*0fca6ea1SDimitry Andric return IterationAction::Continue; 1540349cc55cSDimitry Andric }); 154181ad6265SDimitry Andric return oso_modules; 1542349cc55cSDimitry Andric } 1543bdd1243dSDimitry Andric 1544bdd1243dSDimitry Andric Status SymbolFileDWARFDebugMap::CalculateFrameVariableError(StackFrame &frame) { 1545bdd1243dSDimitry Andric std::lock_guard<std::recursive_mutex> guard(GetModuleMutex()); 1546bdd1243dSDimitry Andric 1547bdd1243dSDimitry Andric // We need to make sure that our PC value from the frame matches the module 1548bdd1243dSDimitry Andric // for this object file since we will lookup the PC file address in the debug 1549bdd1243dSDimitry Andric // map below. 1550bdd1243dSDimitry Andric Address pc_addr = frame.GetFrameCodeAddress(); 1551bdd1243dSDimitry Andric if (pc_addr.GetModule() == m_objfile_sp->GetModule()) { 1552bdd1243dSDimitry Andric Symtab *symtab = m_objfile_sp->GetSymtab(); 1553bdd1243dSDimitry Andric if (symtab) { 1554bdd1243dSDimitry Andric const DebugMap::Entry *debug_map_entry = 1555bdd1243dSDimitry Andric m_debug_map.FindEntryThatContains(pc_addr.GetFileAddress()); 1556bdd1243dSDimitry Andric if (debug_map_entry) { 1557bdd1243dSDimitry Andric Symbol *symbol = 1558bdd1243dSDimitry Andric symtab->SymbolAtIndex(debug_map_entry->data.GetExeSymbolIndex()); 1559bdd1243dSDimitry Andric if (symbol) { 1560bdd1243dSDimitry Andric uint32_t oso_idx = 0; 1561bdd1243dSDimitry Andric CompileUnitInfo *comp_unit_info = 1562bdd1243dSDimitry Andric GetCompileUnitInfoForSymbolWithID(symbol->GetID(), &oso_idx); 1563bdd1243dSDimitry Andric if (comp_unit_info) { 1564bdd1243dSDimitry Andric Module *oso_module = GetModuleByCompUnitInfo(comp_unit_info); 1565bdd1243dSDimitry Andric if (oso_module) { 1566bdd1243dSDimitry Andric // Check the .o file's DWARF in case it has an error to display. 1567bdd1243dSDimitry Andric SymbolFile *oso_sym_file = oso_module->GetSymbolFile(); 1568bdd1243dSDimitry Andric if (oso_sym_file) 1569bdd1243dSDimitry Andric return oso_sym_file->GetFrameVariableError(frame); 1570bdd1243dSDimitry Andric } 1571bdd1243dSDimitry Andric // If we don't have a valid OSO module here, then something went 1572bdd1243dSDimitry Andric // wrong as we have a symbol for the address in the debug map, but 1573bdd1243dSDimitry Andric // we weren't able to open the .o file. Display an appropriate 1574bdd1243dSDimitry Andric // error 1575bdd1243dSDimitry Andric if (comp_unit_info->oso_load_error.Fail()) 1576bdd1243dSDimitry Andric return comp_unit_info->oso_load_error; 1577bdd1243dSDimitry Andric else 1578bdd1243dSDimitry Andric return Status("unable to load debug map object file \"%s\" " 1579bdd1243dSDimitry Andric "exist, debug info will not be loaded", 1580bdd1243dSDimitry Andric comp_unit_info->oso_path.GetCString()); 1581bdd1243dSDimitry Andric } 1582bdd1243dSDimitry Andric } 1583bdd1243dSDimitry Andric } 1584bdd1243dSDimitry Andric } 1585bdd1243dSDimitry Andric } 1586bdd1243dSDimitry Andric return Status(); 1587bdd1243dSDimitry Andric } 158806c3fb27SDimitry Andric 158906c3fb27SDimitry Andric void SymbolFileDWARFDebugMap::GetCompileOptions( 159006c3fb27SDimitry Andric std::unordered_map<lldb::CompUnitSP, lldb_private::Args> &args) { 159106c3fb27SDimitry Andric 1592*0fca6ea1SDimitry Andric ForEachSymbolFile([&](SymbolFileDWARF *oso_dwarf) { 159306c3fb27SDimitry Andric oso_dwarf->GetCompileOptions(args); 1594*0fca6ea1SDimitry Andric return IterationAction::Continue; 159506c3fb27SDimitry Andric }); 159606c3fb27SDimitry Andric } 1597