//===-- DWARFDebugInfo.cpp --------------------------------------*- C++ -*-===// // // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. // See https://llvm.org/LICENSE.txt for license information. // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception // //===----------------------------------------------------------------------===// #include "SymbolFileDWARF.h" #include #include #include "lldb/Host/PosixApi.h" #include "lldb/Symbol/ObjectFile.h" #include "lldb/Utility/RegularExpression.h" #include "lldb/Utility/Stream.h" #include "DWARFCompileUnit.h" #include "DWARFContext.h" #include "DWARFDebugAranges.h" #include "DWARFDebugInfo.h" #include "DWARFDebugInfoEntry.h" #include "DWARFFormValue.h" #include "DWARFTypeUnit.h" using namespace lldb; using namespace lldb_private; using namespace std; // Constructor DWARFDebugInfo::DWARFDebugInfo(lldb_private::DWARFContext &context) : m_dwarf2Data(nullptr), m_context(context), m_units(), m_cu_aranges_up() {} // SetDwarfData void DWARFDebugInfo::SetDwarfData(SymbolFileDWARF *dwarf2Data) { m_dwarf2Data = dwarf2Data; m_units.clear(); } llvm::Expected DWARFDebugInfo::GetCompileUnitAranges() { if (m_cu_aranges_up) return *m_cu_aranges_up; assert(m_dwarf2Data); m_cu_aranges_up = llvm::make_unique(); const DWARFDataExtractor &debug_aranges_data = m_context.getOrLoadArangesData(); if (llvm::Error error = m_cu_aranges_up->extract(debug_aranges_data)) return std::move(error); // Make a list of all CUs represented by the arange data in the file. std::set cus_with_data; for (size_t n = 0; n < m_cu_aranges_up->GetNumRanges(); n++) { dw_offset_t offset = m_cu_aranges_up->OffsetAtIndex(n); if (offset != DW_INVALID_OFFSET) cus_with_data.insert(offset); } // Manually build arange data for everything that wasn't in the // .debug_aranges table. const size_t num_units = GetNumUnits(); for (size_t idx = 0; idx < num_units; ++idx) { DWARFUnit *cu = GetUnitAtIndex(idx); dw_offset_t offset = cu->GetOffset(); if (cus_with_data.find(offset) == cus_with_data.end()) cu->BuildAddressRangeTable(m_cu_aranges_up.get()); } const bool minimize = true; m_cu_aranges_up->Sort(minimize); return *m_cu_aranges_up; } void Parse(SymbolFileDWARF *dwarf, const DWARFDataExtractor &data, DIERef::Section section, std::vector &units) { lldb::offset_t offset = 0; while (data.ValidOffset(offset)) { llvm::Expected unit_sp = DWARFUnit::extract(dwarf, units.size(), data, section, &offset); if (!unit_sp) { // FIXME: Propagate this error up. llvm::consumeError(unit_sp.takeError()); return; } // If it didn't return an error, then it should be returning a valid Unit. assert(*unit_sp); units.push_back(*unit_sp); offset = (*unit_sp)->GetNextUnitOffset(); } } void DWARFDebugInfo::ParseUnitHeadersIfNeeded() { if (!m_units.empty()) return; if (!m_dwarf2Data) return; Parse(m_dwarf2Data, m_context.getOrLoadDebugInfoData(), DIERef::Section::DebugInfo, m_units); Parse(m_dwarf2Data, m_context.getOrLoadDebugTypesData(), DIERef::Section::DebugTypes, m_units); } size_t DWARFDebugInfo::GetNumUnits() { ParseUnitHeadersIfNeeded(); return m_units.size(); } DWARFUnit *DWARFDebugInfo::GetUnitAtIndex(user_id_t idx) { DWARFUnit *cu = nullptr; if (idx < GetNumUnits()) cu = m_units[idx].get(); return cu; } uint32_t DWARFDebugInfo::FindUnitIndex(DIERef::Section section, dw_offset_t offset) { ParseUnitHeadersIfNeeded(); // llvm::lower_bound is not used as for DIE offsets it would still return // index +1 and GetOffset() returning index itself would be a special case. auto pos = llvm::upper_bound( m_units, std::make_pair(section, offset), [](const std::pair &lhs, const DWARFUnitSP &rhs) { return lhs < std::make_pair(rhs->GetDebugSection(), rhs->GetOffset()); }); uint32_t idx = std::distance(m_units.begin(), pos); if (idx == 0) return DW_INVALID_OFFSET; return idx - 1; } DWARFUnit *DWARFDebugInfo::GetUnitAtOffset(DIERef::Section section, dw_offset_t cu_offset, uint32_t *idx_ptr) { uint32_t idx = FindUnitIndex(section, cu_offset); DWARFUnit *result = GetUnitAtIndex(idx); if (result && result->GetOffset() != cu_offset) { result = nullptr; idx = DW_INVALID_INDEX; } if (idx_ptr) *idx_ptr = idx; return result; } DWARFUnit *DWARFDebugInfo::GetUnit(const DIERef &die_ref) { if (die_ref.cu_offset == DW_INVALID_OFFSET) return GetUnitContainingDIEOffset(die_ref.section, die_ref.die_offset); else return GetUnitAtOffset(die_ref.section, die_ref.cu_offset); } DWARFUnit * DWARFDebugInfo::GetUnitContainingDIEOffset(DIERef::Section section, dw_offset_t die_offset) { uint32_t idx = FindUnitIndex(section, die_offset); DWARFUnit *result = GetUnitAtIndex(idx); if (result && !result->ContainsDIEOffset(die_offset)) return nullptr; return result; } DWARFDIE DWARFDebugInfo::GetDIEForDIEOffset(DIERef::Section section, dw_offset_t die_offset) { DWARFUnit *cu = GetUnitContainingDIEOffset(section, die_offset); if (cu) return cu->GetDIE(die_offset); return DWARFDIE(); } // GetDIE() // // Get the DIE (Debug Information Entry) with the specified offset. DWARFDIE DWARFDebugInfo::GetDIE(const DIERef &die_ref) { DWARFUnit *cu = GetUnit(die_ref); if (cu) return cu->GetDIE(die_ref.die_offset); return DWARFDIE(); // Not found }