xref: /freebsd-src/contrib/llvm-project/lldb/source/Plugins/SymbolFile/DWARF/SymbolFileDWARFDebugMap.cpp (revision 0fca6ea1d4eea4c934cfff25ac9ee8ad6fe95583)
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 &regex, 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 &regex,
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