//===-- DWARFAbbreviationDeclaration.cpp ------------------------*- C++ -*-===//
//
// The LLVM Compiler Infrastructure
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
#include "DWARFAbbreviationDeclaration.h"
#include "lldb/Core/dwarf.h"
#include "DWARFFormValue.h"
using namespace lldb_private;
DWARFAbbreviationDeclaration::DWARFAbbreviationDeclaration() :
m_code (InvalidCode),
m_tag (0),
m_has_children (0),
m_attributes()
{
}
DWARFAbbreviationDeclaration::DWARFAbbreviationDeclaration(dw_tag_t tag, uint8_t has_children) :
m_code (InvalidCode),
m_tag (tag),
m_has_children (has_children),
m_attributes()
{
}
bool
DWARFAbbreviationDeclaration::Extract(const DataExtractor& data, lldb::offset_t* offset_ptr)
{
return Extract(data, offset_ptr, data.GetULEB128(offset_ptr));
}
bool
DWARFAbbreviationDeclaration::Extract(const DataExtractor& data, lldb::offset_t *offset_ptr, dw_uleb128_t code)
{
m_code = code;
m_attributes.clear();
if (m_code)
{
m_tag = data.GetULEB128(offset_ptr);
m_has_children = data.GetU8(offset_ptr);
while (data.ValidOffset(*offset_ptr))
{
dw_attr_t attr = data.GetULEB128(offset_ptr);
dw_form_t form = data.GetULEB128(offset_ptr);
if (attr && form)
m_attributes.push_back(DWARFAttribute(attr, form));
else
break;
}
return m_tag != 0;
}
else
{
m_tag = 0;
m_has_children = 0;
}
return false;
}
void
DWARFAbbreviationDeclaration::Dump(Stream *s) const
{
// *ostrm_ptr << std::setfill(' ') << std::dec << '[' << std::setw(3) << std::right << m_code << ']' << ' ' << std::setw(30) << std::left << DW_TAG_value_to_name(m_tag) << DW_CHILDREN_value_to_name(m_has_children) << std::endl;
//
// DWARFAttribute::const_iterator pos;
//
// for (pos = m_attributes.begin(); pos != m_attributes.end(); ++pos)
// *ostrm_ptr << " " << std::setw(29) << std::left << DW_AT_value_to_name(pos->attr()) << ' ' << DW_FORM_value_to_name(pos->form()) << std::endl;
//
// *ostrm_ptr << std::endl;
}
bool
DWARFAbbreviationDeclaration::IsValid()
{
return m_code != 0 && m_tag != 0;
}
void
DWARFAbbreviationDeclaration::CopyExcludingAddressAttributes(const DWARFAbbreviationDeclaration& abbr_decl, const uint32_t idx)
{
m_code = abbr_decl.Code(); // Invalidate the code since that can't be copied safely.
m_tag = abbr_decl.Tag();
m_has_children = abbr_decl.HasChildren();
const DWARFAttribute::collection& attributes = abbr_decl.Attributes();
const uint32_t num_abbr_decl_attributes = attributes.size();
dw_attr_t attr;
dw_form_t form;
uint32_t i;
for (i = 0; i < num_abbr_decl_attributes; ++i)
{
attributes[i].get(attr, form);
switch (attr)
{
case DW_AT_location:
case DW_AT_frame_base:
// Only add these if they are location expressions (have a single
// value) and not location lists (have a lists of location
// expressions which are only valid over specific address ranges)
if (DWARFFormValue::IsBlockForm(form))
m_attributes.push_back(DWARFAttribute(attr, form));
break;
case DW_AT_low_pc:
case DW_AT_high_pc:
case DW_AT_ranges:
case DW_AT_entry_pc:
// Don't add these attributes
if (i >= idx)
break;
// Fall through and add attribute
default:
// Add anything that isn't address related
m_attributes.push_back(DWARFAttribute(attr, form));
break;
}
}
}
void
DWARFAbbreviationDeclaration::CopyChangingStringToStrp(
const DWARFAbbreviationDeclaration& abbr_decl,
const DataExtractor& debug_info_data,
dw_offset_t debug_info_offset,
const DWARFCompileUnit* cu,
const uint32_t strp_min_len
)
{
m_code = InvalidCode;
m_tag = abbr_decl.Tag();
m_has_children = abbr_decl.HasChildren();
const DWARFAttribute::collection& attributes = abbr_decl.Attributes();
const uint32_t num_abbr_decl_attributes = attributes.size();
dw_attr_t attr;
dw_form_t form;
uint32_t i;
lldb::offset_t offset = debug_info_offset;
for (i = 0; i < num_abbr_decl_attributes; ++i)
{
attributes[i].get(attr, form);
dw_offset_t attr_offset = offset;
DWARFFormValue::SkipValue(form, debug_info_data, &offset, cu);
if (form == DW_FORM_string && ((offset - attr_offset) >= strp_min_len))
m_attributes.push_back(DWARFAttribute(attr, DW_FORM_strp));
else
m_attributes.push_back(DWARFAttribute(attr, form));
}
}
uint32_t
DWARFAbbreviationDeclaration::FindAttributeIndex(dw_attr_t attr) const
{
uint32_t i;
const uint32_t kNumAttributes = m_attributes.size();
for (i = 0; i < kNumAttributes; ++i)
{
if (m_attributes[i].get_attr() == attr)
return i;
}
return DW_INVALID_INDEX;
}
bool
DWARFAbbreviationDeclaration::operator == (const DWARFAbbreviationDeclaration& rhs) const
{
return Tag() == rhs.Tag()
&& HasChildren() == rhs.HasChildren()
&& Attributes() == rhs.Attributes();
}
#if 0
DWARFAbbreviationDeclaration::Append(BinaryStreamBuf& out_buff) const
{
out_buff.Append32_as_ULEB128(Code());
out_buff.Append32_as_ULEB128(Tag());
out_buff.Append8(HasChildren());
const uint32_t kNumAttributes = m_attributes.size();
for (uint32_t i = 0; i < kNumAttributes; ++i)
{
out_buff.Append32_as_ULEB128(m_attributes[i].attr());
out_buff.Append32_as_ULEB128(m_attributes[i].form());
}
out_buff.Append8(0); // Output a zero for attr (faster than using Append32_as_ULEB128)
out_buff.Append8(0); // Output a zero for attr (faster than using Append32_as_ULEB128)
}
#endif // 0