#include "DWARFCompileUnit.h"
#include "DWARFDebugAranges.h"
#include "SymbolFileDWARFDebugMap.h"
#include "lldb/Symbol/CompileUnit.h"
#include "lldb/Symbol/LineTable.h"
#include "lldb/Utility/Stream.h"
using namespace lldb;
using namespace lldb_private;
using namespace lldb_private::plugin::dwarf;
void DWARFCompileUnit::Dump(Stream *s) const {
s->Format(
"{0:x16}: Compile Unit: length = {1:x8}, version = {2:x}, "
"abbr_offset = {3:x8}, addr_size = {4:x2} (next CU at "
"[{5:x16}])\n",
GetOffset(), GetLength(), GetVersion(), (uint32_t)GetAbbrevOffset(),
GetAddressByteSize(), GetNextUnitOffset());
}
void DWARFCompileUnit::BuildAddressRangeTable(
DWARFDebugAranges *debug_aranges) {
size_t num_debug_aranges = debug_aranges->GetNumRanges();
const DWARFDebugInfoEntry *die = GetUnitDIEPtrOnly();
const dw_offset_t cu_offset = GetOffset();
if (die) {
DWARFRangeList ranges =
die->GetAttributeAddressRanges(this, true);
#ifdef MS_DEBUGGER
if (!ranges.IsEmpty()) {
auto last_range_end = ranges.GetEntryRef(0).GetRangeBase();
for (const DWARFRangeList::Entry &range : ranges) {
if (range.GetRangeBase() >= last_range_end) {
debug_aranges->AppendRange(cu_offset, range.GetRangeBase(),
range.GetRangeEnd());
last_range_end = range.GetRangeEnd();
}
}
}
#else
for (const DWARFRangeList::Entry &range : ranges)
debug_aranges->AppendRange(cu_offset, range.GetRangeBase(),
range.GetRangeEnd());
#endif
if (!ranges.IsEmpty())
return;
}
if (debug_aranges->GetNumRanges() == num_debug_aranges) {
SymbolContext sc;
sc.comp_unit = m_dwarf.GetCompUnitForDWARFCompUnit(*this);
if (sc.comp_unit) {
SymbolFileDWARFDebugMap *debug_map_sym_file =
m_dwarf.GetDebugMapSymfile();
if (debug_map_sym_file) {
auto *cu_info =
debug_map_sym_file->GetCompileUnitInfo(&GetSymbolFileDWARF());
if (cu_info->compile_units_sps.empty())
debug_map_sym_file->AddOSOARanges(&m_dwarf, debug_aranges);
}
}
}
if (debug_aranges->GetNumRanges() == num_debug_aranges) {
SymbolContext sc;
sc.comp_unit = m_dwarf.GetCompUnitForDWARFCompUnit(*this);
if (sc.comp_unit) {
if (LineTable *line_table = sc.comp_unit->GetLineTable()) {
LineTable::FileAddressRanges file_ranges;
const bool append = true;
const size_t num_ranges =
line_table->GetContiguousFileAddressRanges(file_ranges, append);
for (uint32_t idx = 0; idx < num_ranges; ++idx) {
const LineTable::FileAddressRanges::Entry &range =
file_ranges.GetEntryRef(idx);
debug_aranges->AppendRange(GetOffset(), range.GetRangeBase(),
range.GetRangeEnd());
}
}
}
}
}
DWARFCompileUnit &DWARFCompileUnit::GetNonSkeletonUnit() {
return llvm::cast<DWARFCompileUnit>(DWARFUnit::GetNonSkeletonUnit());
}
DWARFDIE DWARFCompileUnit::LookupAddress(const dw_addr_t address) {
if (DIE()) {
const DWARFDebugAranges &func_aranges = GetFunctionAranges();
if (!func_aranges.IsEmpty())
return GetDIE(func_aranges.FindAddress(address));
}
return DWARFDIE();
}