You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
202 lines
6.6 KiB
202 lines
6.6 KiB
//===-- DWARFDebugInfo.cpp ------------------------------------------------===//
|
|
//
|
|
// 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 <algorithm>
|
|
#include <set>
|
|
|
|
#include "lldb/Host/PosixApi.h"
|
|
#include "lldb/Symbol/ObjectFile.h"
|
|
#include "lldb/Utility/RegularExpression.h"
|
|
#include "lldb/Utility/Stream.h"
|
|
#include "llvm/Support/Casting.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(SymbolFileDWARF &dwarf,
|
|
lldb_private::DWARFContext &context)
|
|
: m_dwarf(dwarf), m_context(context), m_units(), m_cu_aranges_up() {}
|
|
|
|
llvm::Expected<DWARFDebugAranges &> DWARFDebugInfo::GetCompileUnitAranges() {
|
|
if (m_cu_aranges_up)
|
|
return *m_cu_aranges_up;
|
|
|
|
m_cu_aranges_up = std::make_unique<DWARFDebugAranges>();
|
|
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<dw_offset_t> 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 DWARFDebugInfo::ParseUnitsFor(DIERef::Section section) {
|
|
DWARFDataExtractor data = section == DIERef::Section::DebugTypes
|
|
? m_context.getOrLoadDebugTypesData()
|
|
: m_context.getOrLoadDebugInfoData();
|
|
const llvm::DWARFUnitIndex *index = nullptr;
|
|
if (m_context.isDwo())
|
|
index = &llvm::getDWARFUnitIndex(m_context.GetAsLLVM(),
|
|
section == DIERef::Section::DebugTypes
|
|
? llvm::DW_SECT_EXT_TYPES
|
|
: llvm::DW_SECT_INFO);
|
|
lldb::offset_t offset = 0;
|
|
while (data.ValidOffset(offset)) {
|
|
llvm::Expected<DWARFUnitSP> unit_sp = DWARFUnit::extract(
|
|
m_dwarf, m_units.size(), data, section, &offset, index);
|
|
|
|
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);
|
|
m_units.push_back(*unit_sp);
|
|
offset = (*unit_sp)->GetNextUnitOffset();
|
|
|
|
if (auto *type_unit = llvm::dyn_cast<DWARFTypeUnit>(unit_sp->get())) {
|
|
m_type_hash_to_unit_index.emplace_back(type_unit->GetTypeHash(),
|
|
unit_sp.get()->GetID());
|
|
}
|
|
}
|
|
}
|
|
|
|
void DWARFDebugInfo::ParseUnitHeadersIfNeeded() {
|
|
llvm::call_once(m_units_once_flag, [&] {
|
|
ParseUnitsFor(DIERef::Section::DebugInfo);
|
|
ParseUnitsFor(DIERef::Section::DebugTypes);
|
|
llvm::sort(m_type_hash_to_unit_index, llvm::less_first());
|
|
});
|
|
}
|
|
|
|
size_t DWARFDebugInfo::GetNumUnits() {
|
|
ParseUnitHeadersIfNeeded();
|
|
return m_units.size();
|
|
}
|
|
|
|
DWARFUnit *DWARFDebugInfo::GetUnitAtIndex(size_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<DIERef::Section, dw_offset_t> &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) {
|
|
return GetUnitContainingDIEOffset(die_ref.section(), die_ref.die_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;
|
|
}
|
|
|
|
DWARFTypeUnit *DWARFDebugInfo::GetTypeUnitForHash(uint64_t hash) {
|
|
auto pos = llvm::lower_bound(m_type_hash_to_unit_index,
|
|
std::make_pair(hash, 0u), llvm::less_first());
|
|
if (pos == m_type_hash_to_unit_index.end() || pos->first != hash)
|
|
return nullptr;
|
|
return llvm::cast<DWARFTypeUnit>(GetUnitAtIndex(pos->second));
|
|
}
|
|
|
|
bool DWARFDebugInfo::ContainsTypeUnits() {
|
|
ParseUnitHeadersIfNeeded();
|
|
return !m_type_hash_to_unit_index.empty();
|
|
}
|
|
|
|
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->GetNonSkeletonUnit().GetDIE(die_ref.die_offset());
|
|
return DWARFDIE(); // Not found
|
|
}
|