use alloc::vec::Vec;
use std::ops::{Deref, DerefMut};
use std::{slice, usize};
use crate::common::{
DebugAbbrevOffset, DebugInfoOffset, DebugLineOffset, DebugMacinfoOffset, DebugMacroOffset,
DebugStrOffset, DebugTypeSignature, Encoding, Format, SectionId,
};
use crate::constants;
use crate::leb128::write::{sleb128_size, uleb128_size};
use crate::write::{
Abbreviation, AbbreviationTable, Address, AttributeSpecification, BaseId, DebugLineStrOffsets,
DebugStrOffsets, Error, Expression, FileId, LineProgram, LineStringId, LocationListId,
LocationListOffsets, LocationListTable, RangeListId, RangeListOffsets, RangeListTable,
Reference, Result, Section, Sections, StringId, Writer,
};
define_id!(UnitId, "An identifier for a unit in a `UnitTable`.");
define_id!(UnitEntryId, "An identifier for an entry in a `Unit`.");
#[derive(Debug, Default)]
pub struct UnitTable {
base_id: BaseId,
units: Vec<Unit>,
}
impl UnitTable {
#[inline]
pub fn add(&mut self, unit: Unit) -> UnitId {
let id = UnitId::new(self.base_id, self.units.len());
self.units.push(unit);
id
}
#[inline]
pub fn count(&self) -> usize {
self.units.len()
}
#[inline]
pub fn id(&self, index: usize) -> UnitId {
assert!(index < self.count());
UnitId::new(self.base_id, index)
}
#[inline]
pub fn get(&self, id: UnitId) -> &Unit {
debug_assert_eq!(self.base_id, id.base_id);
&self.units[id.index]
}
#[inline]
pub fn get_mut(&mut self, id: UnitId) -> &mut Unit {
debug_assert_eq!(self.base_id, id.base_id);
&mut self.units[id.index]
}
pub fn write<W: Writer>(
&mut self,
sections: &mut Sections<W>,
line_strings: &DebugLineStrOffsets,
strings: &DebugStrOffsets,
) -> Result<DebugInfoOffsets> {
let mut offsets = DebugInfoOffsets {
base_id: self.base_id,
units: Vec::new(),
};
for unit in &mut self.units {
let abbrev_offset = sections.debug_abbrev.offset();
let mut abbrevs = AbbreviationTable::default();
offsets.units.push(unit.write(
sections,
abbrev_offset,
&mut abbrevs,
line_strings,
strings,
)?);
abbrevs.write(&mut sections.debug_abbrev)?;
}
write_section_refs(
&mut sections.debug_info_refs,
&mut sections.debug_info.0,
&offsets,
)?;
write_section_refs(
&mut sections.debug_loc_refs,
&mut sections.debug_loc.0,
&offsets,
)?;
write_section_refs(
&mut sections.debug_loclists_refs,
&mut sections.debug_loclists.0,
&offsets,
)?;
Ok(offsets)
}
}
fn write_section_refs<W: Writer>(
references: &mut Vec<DebugInfoReference>,
w: &mut W,
offsets: &DebugInfoOffsets,
) -> Result<()> {
for r in references.drain(..) {
let entry_offset = offsets.entry(r.unit, r.entry).0;
debug_assert_ne!(entry_offset, 0);
w.write_offset_at(r.offset, entry_offset, SectionId::DebugInfo, r.size)?;
}
Ok(())
}
#[derive(Debug)]
pub struct Unit {
base_id: BaseId,
encoding: Encoding,
pub line_program: LineProgram,
pub ranges: RangeListTable,
pub locations: LocationListTable,
entries: Vec<DebuggingInformationEntry>,
root: UnitEntryId,
}
impl Unit {
pub fn new(encoding: Encoding, line_program: LineProgram) -> Self {
let base_id = BaseId::default();
let ranges = RangeListTable::default();
let locations = LocationListTable::default();
let mut entries = Vec::new();
let root = DebuggingInformationEntry::new(
base_id,
&mut entries,
None,
constants::DW_TAG_compile_unit,
);
Unit {
base_id,
encoding,
line_program,
ranges,
locations,
entries,
root,
}
}
#[inline]
pub fn encoding(&self) -> Encoding {
self.encoding
}
#[inline]
pub fn version(&self) -> u16 {
self.encoding.version
}
#[inline]
pub fn address_size(&self) -> u8 {
self.encoding.address_size
}
#[inline]
pub fn format(&self) -> Format {
self.encoding.format
}
#[inline]
pub fn count(&self) -> usize {
self.entries.len()
}
#[inline]
pub fn root(&self) -> UnitEntryId {
self.root
}
#[inline]
pub fn add(&mut self, parent: UnitEntryId, tag: constants::DwTag) -> UnitEntryId {
debug_assert_eq!(self.base_id, parent.base_id);
DebuggingInformationEntry::new(self.base_id, &mut self.entries, Some(parent), tag)
}
#[inline]
pub fn get(&self, id: UnitEntryId) -> &DebuggingInformationEntry {
debug_assert_eq!(self.base_id, id.base_id);
&self.entries[id.index]
}
#[inline]
pub fn get_mut(&mut self, id: UnitEntryId) -> &mut DebuggingInformationEntry {
debug_assert_eq!(self.base_id, id.base_id);
&mut self.entries[id.index]
}
fn line_program_in_use(&self) -> bool {
if self.line_program.is_none() {
return false;
}
if !self.line_program.is_empty() {
return true;
}
for entry in &self.entries {
for attr in &entry.attrs {
if let AttributeValue::FileIndex(Some(_)) = attr.value {
return true;
}
}
}
false
}
pub(crate) fn write<W: Writer>(
&mut self,
sections: &mut Sections<W>,
abbrev_offset: DebugAbbrevOffset,
abbrevs: &mut AbbreviationTable,
line_strings: &DebugLineStrOffsets,
strings: &DebugStrOffsets,
) -> Result<UnitOffsets> {
let line_program = if self.line_program_in_use() {
self.entries[self.root.index]
.set(constants::DW_AT_stmt_list, AttributeValue::LineProgramRef);
Some(self.line_program.write(
&mut sections.debug_line,
self.encoding,
line_strings,
strings,
)?)
} else {
self.entries[self.root.index].delete(constants::DW_AT_stmt_list);
None
};
let w = &mut sections.debug_info;
let mut offsets = UnitOffsets {
base_id: self.base_id,
unit: w.offset(),
entries: vec![EntryOffset::none(); self.entries.len()],
};
let length_offset = w.write_initial_length(self.format())?;
let length_base = w.len();
w.write_u16(self.version())?;
if 2 <= self.version() && self.version() <= 4 {
w.write_offset(
abbrev_offset.0,
SectionId::DebugAbbrev,
self.format().word_size(),
)?;
w.write_u8(self.address_size())?;
} else if self.version() == 5 {
w.write_u8(constants::DW_UT_compile.0)?;
w.write_u8(self.address_size())?;
w.write_offset(
abbrev_offset.0,
SectionId::DebugAbbrev,
self.format().word_size(),
)?;
} else {
return Err(Error::UnsupportedVersion(self.version()));
}
self.reorder_base_types();
let mut offset = w.len();
self.entries[self.root.index].calculate_offsets(
self,
&mut offset,
&mut offsets,
abbrevs,
)?;
let range_lists = self.ranges.write(sections, self.encoding)?;
let loc_lists = self
.locations
.write(sections, self.encoding, Some(&offsets))?;
let w = &mut sections.debug_info;
let mut unit_refs = Vec::new();
self.entries[self.root.index].write(
w,
&mut sections.debug_info_refs,
&mut unit_refs,
self,
&mut offsets,
line_program,
line_strings,
strings,
&range_lists,
&loc_lists,
)?;
let length = (w.len() - length_base) as u64;
w.write_initial_length_at(length_offset, length, self.format())?;
for (offset, entry) in unit_refs {
w.write_udata_at(
offset.0,
offsets.unit_offset(entry),
self.format().word_size(),
)?;
}
Ok(offsets)
}
fn reorder_base_types(&mut self) {
let root = &self.entries[self.root.index];
let mut root_children = Vec::with_capacity(root.children.len());
for entry in &root.children {
if self.entries[entry.index].tag == constants::DW_TAG_base_type {
root_children.push(*entry);
}
}
for entry in &root.children {
if self.entries[entry.index].tag != constants::DW_TAG_base_type {
root_children.push(*entry);
}
}
self.entries[self.root.index].children = root_children;
}
}
#[derive(Debug)]
pub struct DebuggingInformationEntry {
id: UnitEntryId,
parent: Option<UnitEntryId>,
tag: constants::DwTag,
sibling: bool,
attrs: Vec<Attribute>,
children: Vec<UnitEntryId>,
}
impl DebuggingInformationEntry {
#[allow(clippy::new_ret_no_self)]
fn new(
base_id: BaseId,
entries: &mut Vec<DebuggingInformationEntry>,
parent: Option<UnitEntryId>,
tag: constants::DwTag,
) -> UnitEntryId {
let id = UnitEntryId::new(base_id, entries.len());
entries.push(DebuggingInformationEntry {
id,
parent,
tag,
sibling: false,
attrs: Vec::new(),
children: Vec::new(),
});
if let Some(parent) = parent {
debug_assert_eq!(base_id, parent.base_id);
assert_ne!(parent, id);
entries[parent.index].children.push(id);
}
id
}
#[inline]
pub fn id(&self) -> UnitEntryId {
self.id
}
#[inline]
pub fn parent(&self) -> Option<UnitEntryId> {
self.parent
}
#[inline]
pub fn tag(&self) -> constants::DwTag {
self.tag
}
#[inline]
pub fn sibling(&self) -> bool {
self.sibling
}
#[inline]
pub fn set_sibling(&mut self, sibling: bool) {
self.sibling = sibling;
}
#[inline]
pub fn attrs(&self) -> slice::Iter<'_, Attribute> {
self.attrs.iter()
}
#[inline]
pub fn attrs_mut(&mut self) -> slice::IterMut<'_, Attribute> {
self.attrs.iter_mut()
}
pub fn get(&self, name: constants::DwAt) -> Option<&AttributeValue> {
self.attrs
.iter()
.find(|attr| attr.name == name)
.map(|attr| &attr.value)
}
pub fn get_mut(&mut self, name: constants::DwAt) -> Option<&mut AttributeValue> {
self.attrs
.iter_mut()
.find(|attr| attr.name == name)
.map(|attr| &mut attr.value)
}
pub fn set(&mut self, name: constants::DwAt, value: AttributeValue) {
assert_ne!(name, constants::DW_AT_sibling);
if let Some(attr) = self.attrs.iter_mut().find(|attr| attr.name == name) {
attr.value = value;
return;
}
self.attrs.push(Attribute { name, value });
}
pub fn delete(&mut self, name: constants::DwAt) {
self.attrs.retain(|x| x.name != name);
}
#[inline]
pub fn children(&self) -> slice::Iter<'_, UnitEntryId> {
self.children.iter()
}
pub fn delete_child(&mut self, id: UnitEntryId) {
self.children.retain(|&child| child != id);
}
fn abbreviation(&self, encoding: Encoding) -> Result<Abbreviation> {
let mut attrs = Vec::new();
if self.sibling && !self.children.is_empty() {
let form = match encoding.format {
Format::Dwarf32 => constants::DW_FORM_ref4,
Format::Dwarf64 => constants::DW_FORM_ref8,
};
attrs.push(AttributeSpecification::new(constants::DW_AT_sibling, form));
}
for attr in &self.attrs {
attrs.push(attr.specification(encoding)?);
}
Ok(Abbreviation::new(
self.tag,
!self.children.is_empty(),
attrs,
))
}
fn calculate_offsets(
&self,
unit: &Unit,
offset: &mut usize,
offsets: &mut UnitOffsets,
abbrevs: &mut AbbreviationTable,
) -> Result<()> {
offsets.entries[self.id.index].offset = DebugInfoOffset(*offset);
offsets.entries[self.id.index].abbrev = abbrevs.add(self.abbreviation(unit.encoding())?);
*offset += self.size(unit, offsets);
if !self.children.is_empty() {
for child in &self.children {
unit.entries[child.index].calculate_offsets(unit, offset, offsets, abbrevs)?;
}
*offset += 1;
}
Ok(())
}
fn size(&self, unit: &Unit, offsets: &UnitOffsets) -> usize {
let mut size = uleb128_size(offsets.abbrev(self.id));
if self.sibling && !self.children.is_empty() {
size += unit.format().word_size() as usize;
}
for attr in &self.attrs {
size += attr.value.size(unit, offsets);
}
size
}
fn write<W: Writer>(
&self,
w: &mut DebugInfo<W>,
debug_info_refs: &mut Vec<DebugInfoReference>,
unit_refs: &mut Vec<(DebugInfoOffset, UnitEntryId)>,
unit: &Unit,
offsets: &mut UnitOffsets,
line_program: Option<DebugLineOffset>,
line_strings: &DebugLineStrOffsets,
strings: &DebugStrOffsets,
range_lists: &RangeListOffsets,
loc_lists: &LocationListOffsets,
) -> Result<()> {
debug_assert_eq!(offsets.debug_info_offset(self.id), w.offset());
w.write_uleb128(offsets.abbrev(self.id))?;
let sibling_offset = if self.sibling && !self.children.is_empty() {
let offset = w.offset();
w.write_udata(0, unit.format().word_size())?;
Some(offset)
} else {
None
};
for attr in &self.attrs {
attr.value.write(
w,
debug_info_refs,
unit_refs,
unit,
offsets,
line_program,
line_strings,
strings,
range_lists,
loc_lists,
)?;
}
if !self.children.is_empty() {
for child in &self.children {
unit.entries[child.index].write(
w,
debug_info_refs,
unit_refs,
unit,
offsets,
line_program,
line_strings,
strings,
range_lists,
loc_lists,
)?;
}
w.write_u8(0)?;
}
if let Some(offset) = sibling_offset {
let next_offset = (w.offset().0 - offsets.unit.0) as u64;
w.write_udata_at(offset.0, next_offset, unit.format().word_size())?;
}
Ok(())
}
}
#[derive(Debug, Clone, PartialEq, Eq)]
pub struct Attribute {
name: constants::DwAt,
value: AttributeValue,
}
impl Attribute {
#[inline]
pub fn name(&self) -> constants::DwAt {
self.name
}
#[inline]
pub fn get(&self) -> &AttributeValue {
&self.value
}
#[inline]
pub fn set(&mut self, value: AttributeValue) {
self.value = value;
}
fn specification(&self, encoding: Encoding) -> Result<AttributeSpecification> {
Ok(AttributeSpecification::new(
self.name,
self.value.form(encoding)?,
))
}
}
#[derive(Debug, Clone, PartialEq, Eq)]
pub enum AttributeValue {
Address(Address),
Block(Vec<u8>),
Data1(u8),
Data2(u16),
Data4(u32),
Data8(u64),
Sdata(i64),
Udata(u64),
Exprloc(Expression),
Flag(bool),
FlagPresent,
UnitRef(UnitEntryId),
DebugInfoRef(Reference),
DebugInfoRefSup(DebugInfoOffset),
LineProgramRef,
LocationListRef(LocationListId),
DebugMacinfoRef(DebugMacinfoOffset),
DebugMacroRef(DebugMacroOffset),
RangeListRef(RangeListId),
DebugTypesRef(DebugTypeSignature),
StringRef(StringId),
DebugStrRefSup(DebugStrOffset),
LineStringRef(LineStringId),
String(Vec<u8>),
Encoding(constants::DwAte),
DecimalSign(constants::DwDs),
Endianity(constants::DwEnd),
Accessibility(constants::DwAccess),
Visibility(constants::DwVis),
Virtuality(constants::DwVirtuality),
Language(constants::DwLang),
AddressClass(constants::DwAddr),
IdentifierCase(constants::DwId),
CallingConvention(constants::DwCc),
Inline(constants::DwInl),
Ordering(constants::DwOrd),
FileIndex(Option<FileId>),
}
impl AttributeValue {
pub fn form(&self, encoding: Encoding) -> Result<constants::DwForm> {
let form = match *self {
AttributeValue::Address(_) => constants::DW_FORM_addr,
AttributeValue::Block(_) => constants::DW_FORM_block,
AttributeValue::Data1(_) => constants::DW_FORM_data1,
AttributeValue::Data2(_) => constants::DW_FORM_data2,
AttributeValue::Data4(_) => constants::DW_FORM_data4,
AttributeValue::Data8(_) => constants::DW_FORM_data8,
AttributeValue::Exprloc(_) => constants::DW_FORM_exprloc,
AttributeValue::Flag(_) => constants::DW_FORM_flag,
AttributeValue::FlagPresent => constants::DW_FORM_flag_present,
AttributeValue::UnitRef(_) => {
match encoding.format {
Format::Dwarf32 => constants::DW_FORM_ref4,
Format::Dwarf64 => constants::DW_FORM_ref8,
}
}
AttributeValue::DebugInfoRef(_) => constants::DW_FORM_ref_addr,
AttributeValue::DebugInfoRefSup(_) => {
match encoding.format {
Format::Dwarf32 => constants::DW_FORM_ref_sup4,
Format::Dwarf64 => constants::DW_FORM_ref_sup8,
}
}
AttributeValue::LineProgramRef
| AttributeValue::LocationListRef(_)
| AttributeValue::DebugMacinfoRef(_)
| AttributeValue::DebugMacroRef(_)
| AttributeValue::RangeListRef(_) => {
if encoding.version == 2 || encoding.version == 3 {
match encoding.format {
Format::Dwarf32 => constants::DW_FORM_data4,
Format::Dwarf64 => constants::DW_FORM_data8,
}
} else {
constants::DW_FORM_sec_offset
}
}
AttributeValue::DebugTypesRef(_) => constants::DW_FORM_ref_sig8,
AttributeValue::StringRef(_) => constants::DW_FORM_strp,
AttributeValue::DebugStrRefSup(_) => constants::DW_FORM_strp_sup,
AttributeValue::LineStringRef(_) => constants::DW_FORM_line_strp,
AttributeValue::String(_) => constants::DW_FORM_string,
AttributeValue::Encoding(_)
| AttributeValue::DecimalSign(_)
| AttributeValue::Endianity(_)
| AttributeValue::Accessibility(_)
| AttributeValue::Visibility(_)
| AttributeValue::Virtuality(_)
| AttributeValue::Language(_)
| AttributeValue::AddressClass(_)
| AttributeValue::IdentifierCase(_)
| AttributeValue::CallingConvention(_)
| AttributeValue::Inline(_)
| AttributeValue::Ordering(_)
| AttributeValue::FileIndex(_)
| AttributeValue::Udata(_) => constants::DW_FORM_udata,
AttributeValue::Sdata(_) => constants::DW_FORM_sdata,
};
Ok(form)
}
fn size(&self, unit: &Unit, offsets: &UnitOffsets) -> usize {
macro_rules! debug_assert_form {
($form:expr) => {
debug_assert_eq!(self.form(unit.encoding()).unwrap(), $form)
};
}
match *self {
AttributeValue::Address(_) => {
debug_assert_form!(constants::DW_FORM_addr);
unit.address_size() as usize
}
AttributeValue::Block(ref val) => {
debug_assert_form!(constants::DW_FORM_block);
uleb128_size(val.len() as u64) + val.len()
}
AttributeValue::Data1(_) => {
debug_assert_form!(constants::DW_FORM_data1);
1
}
AttributeValue::Data2(_) => {
debug_assert_form!(constants::DW_FORM_data2);
2
}
AttributeValue::Data4(_) => {
debug_assert_form!(constants::DW_FORM_data4);
4
}
AttributeValue::Data8(_) => {
debug_assert_form!(constants::DW_FORM_data8);
8
}
AttributeValue::Sdata(val) => {
debug_assert_form!(constants::DW_FORM_sdata);
sleb128_size(val)
}
AttributeValue::Udata(val) => {
debug_assert_form!(constants::DW_FORM_udata);
uleb128_size(val)
}
AttributeValue::Exprloc(ref val) => {
debug_assert_form!(constants::DW_FORM_exprloc);
let size = val.size(unit.encoding(), Some(offsets));
uleb128_size(size as u64) + size
}
AttributeValue::Flag(_) => {
debug_assert_form!(constants::DW_FORM_flag);
1
}
AttributeValue::FlagPresent => {
debug_assert_form!(constants::DW_FORM_flag_present);
0
}
AttributeValue::UnitRef(_) => {
match unit.format() {
Format::Dwarf32 => debug_assert_form!(constants::DW_FORM_ref4),
Format::Dwarf64 => debug_assert_form!(constants::DW_FORM_ref8),
}
unit.format().word_size() as usize
}
AttributeValue::DebugInfoRef(_) => {
debug_assert_form!(constants::DW_FORM_ref_addr);
if unit.version() == 2 {
unit.address_size() as usize
} else {
unit.format().word_size() as usize
}
}
AttributeValue::DebugInfoRefSup(_) => {
match unit.format() {
Format::Dwarf32 => debug_assert_form!(constants::DW_FORM_ref_sup4),
Format::Dwarf64 => debug_assert_form!(constants::DW_FORM_ref_sup8),
}
unit.format().word_size() as usize
}
AttributeValue::LineProgramRef => {
if unit.version() >= 4 {
debug_assert_form!(constants::DW_FORM_sec_offset);
}
unit.format().word_size() as usize
}
AttributeValue::LocationListRef(_) => {
if unit.version() >= 4 {
debug_assert_form!(constants::DW_FORM_sec_offset);
}
unit.format().word_size() as usize
}
AttributeValue::DebugMacinfoRef(_) => {
if unit.version() >= 4 {
debug_assert_form!(constants::DW_FORM_sec_offset);
}
unit.format().word_size() as usize
}
AttributeValue::DebugMacroRef(_) => {
if unit.version() >= 4 {
debug_assert_form!(constants::DW_FORM_sec_offset);
}
unit.format().word_size() as usize
}
AttributeValue::RangeListRef(_) => {
if unit.version() >= 4 {
debug_assert_form!(constants::DW_FORM_sec_offset);
}
unit.format().word_size() as usize
}
AttributeValue::DebugTypesRef(_) => {
debug_assert_form!(constants::DW_FORM_ref_sig8);
8
}
AttributeValue::StringRef(_) => {
debug_assert_form!(constants::DW_FORM_strp);
unit.format().word_size() as usize
}
AttributeValue::DebugStrRefSup(_) => {
debug_assert_form!(constants::DW_FORM_strp_sup);
unit.format().word_size() as usize
}
AttributeValue::LineStringRef(_) => {
debug_assert_form!(constants::DW_FORM_line_strp);
unit.format().word_size() as usize
}
AttributeValue::String(ref val) => {
debug_assert_form!(constants::DW_FORM_string);
val.len() + 1
}
AttributeValue::Encoding(val) => {
debug_assert_form!(constants::DW_FORM_udata);
uleb128_size(val.0 as u64)
}
AttributeValue::DecimalSign(val) => {
debug_assert_form!(constants::DW_FORM_udata);
uleb128_size(val.0 as u64)
}
AttributeValue::Endianity(val) => {
debug_assert_form!(constants::DW_FORM_udata);
uleb128_size(val.0 as u64)
}
AttributeValue::Accessibility(val) => {
debug_assert_form!(constants::DW_FORM_udata);
uleb128_size(val.0 as u64)
}
AttributeValue::Visibility(val) => {
debug_assert_form!(constants::DW_FORM_udata);
uleb128_size(val.0 as u64)
}
AttributeValue::Virtuality(val) => {
debug_assert_form!(constants::DW_FORM_udata);
uleb128_size(val.0 as u64)
}
AttributeValue::Language(val) => {
debug_assert_form!(constants::DW_FORM_udata);
uleb128_size(val.0 as u64)
}
AttributeValue::AddressClass(val) => {
debug_assert_form!(constants::DW_FORM_udata);
uleb128_size(val.0)
}
AttributeValue::IdentifierCase(val) => {
debug_assert_form!(constants::DW_FORM_udata);
uleb128_size(val.0 as u64)
}
AttributeValue::CallingConvention(val) => {
debug_assert_form!(constants::DW_FORM_udata);
uleb128_size(val.0 as u64)
}
AttributeValue::Inline(val) => {
debug_assert_form!(constants::DW_FORM_udata);
uleb128_size(val.0 as u64)
}
AttributeValue::Ordering(val) => {
debug_assert_form!(constants::DW_FORM_udata);
uleb128_size(val.0 as u64)
}
AttributeValue::FileIndex(val) => {
debug_assert_form!(constants::DW_FORM_udata);
uleb128_size(val.map(FileId::raw).unwrap_or(0))
}
}
}
fn write<W: Writer>(
&self,
w: &mut DebugInfo<W>,
debug_info_refs: &mut Vec<DebugInfoReference>,
unit_refs: &mut Vec<(DebugInfoOffset, UnitEntryId)>,
unit: &Unit,
offsets: &UnitOffsets,
line_program: Option<DebugLineOffset>,
line_strings: &DebugLineStrOffsets,
strings: &DebugStrOffsets,
range_lists: &RangeListOffsets,
loc_lists: &LocationListOffsets,
) -> Result<()> {
macro_rules! debug_assert_form {
($form:expr) => {
debug_assert_eq!(self.form(unit.encoding()).unwrap(), $form)
};
}
match *self {
AttributeValue::Address(val) => {
debug_assert_form!(constants::DW_FORM_addr);
w.write_address(val, unit.address_size())?;
}
AttributeValue::Block(ref val) => {
debug_assert_form!(constants::DW_FORM_block);
w.write_uleb128(val.len() as u64)?;
w.write(val)?;
}
AttributeValue::Data1(val) => {
debug_assert_form!(constants::DW_FORM_data1);
w.write_u8(val)?;
}
AttributeValue::Data2(val) => {
debug_assert_form!(constants::DW_FORM_data2);
w.write_u16(val)?;
}
AttributeValue::Data4(val) => {
debug_assert_form!(constants::DW_FORM_data4);
w.write_u32(val)?;
}
AttributeValue::Data8(val) => {
debug_assert_form!(constants::DW_FORM_data8);
w.write_u64(val)?;
}
AttributeValue::Sdata(val) => {
debug_assert_form!(constants::DW_FORM_sdata);
w.write_sleb128(val)?;
}
AttributeValue::Udata(val) => {
debug_assert_form!(constants::DW_FORM_udata);
w.write_uleb128(val)?;
}
AttributeValue::Exprloc(ref val) => {
debug_assert_form!(constants::DW_FORM_exprloc);
w.write_uleb128(val.size(unit.encoding(), Some(offsets)) as u64)?;
val.write(
&mut w.0,
Some(debug_info_refs),
unit.encoding(),
Some(offsets),
)?;
}
AttributeValue::Flag(val) => {
debug_assert_form!(constants::DW_FORM_flag);
w.write_u8(val as u8)?;
}
AttributeValue::FlagPresent => {
debug_assert_form!(constants::DW_FORM_flag_present);
}
AttributeValue::UnitRef(id) => {
match unit.format() {
Format::Dwarf32 => debug_assert_form!(constants::DW_FORM_ref4),
Format::Dwarf64 => debug_assert_form!(constants::DW_FORM_ref8),
}
unit_refs.push((w.offset(), id));
w.write_udata(0, unit.format().word_size())?;
}
AttributeValue::DebugInfoRef(reference) => {
debug_assert_form!(constants::DW_FORM_ref_addr);
let size = if unit.version() == 2 {
unit.address_size()
} else {
unit.format().word_size()
};
match reference {
Reference::Symbol(symbol) => w.write_reference(symbol, size)?,
Reference::Entry(unit, entry) => {
debug_info_refs.push(DebugInfoReference {
offset: w.len(),
unit,
entry,
size,
});
w.write_udata(0, size)?;
}
}
}
AttributeValue::DebugInfoRefSup(val) => {
match unit.format() {
Format::Dwarf32 => debug_assert_form!(constants::DW_FORM_ref_sup4),
Format::Dwarf64 => debug_assert_form!(constants::DW_FORM_ref_sup8),
}
w.write_udata(val.0 as u64, unit.format().word_size())?;
}
AttributeValue::LineProgramRef => {
if unit.version() >= 4 {
debug_assert_form!(constants::DW_FORM_sec_offset);
}
match line_program {
Some(line_program) => {
w.write_offset(
line_program.0,
SectionId::DebugLine,
unit.format().word_size(),
)?;
}
None => return Err(Error::InvalidAttributeValue),
}
}
AttributeValue::LocationListRef(val) => {
if unit.version() >= 4 {
debug_assert_form!(constants::DW_FORM_sec_offset);
}
let section = if unit.version() <= 4 {
SectionId::DebugLoc
} else {
SectionId::DebugLocLists
};
w.write_offset(loc_lists.get(val).0, section, unit.format().word_size())?;
}
AttributeValue::DebugMacinfoRef(val) => {
if unit.version() >= 4 {
debug_assert_form!(constants::DW_FORM_sec_offset);
}
w.write_offset(val.0, SectionId::DebugMacinfo, unit.format().word_size())?;
}
AttributeValue::DebugMacroRef(val) => {
if unit.version() >= 4 {
debug_assert_form!(constants::DW_FORM_sec_offset);
}
w.write_offset(val.0, SectionId::DebugMacro, unit.format().word_size())?;
}
AttributeValue::RangeListRef(val) => {
if unit.version() >= 4 {
debug_assert_form!(constants::DW_FORM_sec_offset);
}
let section = if unit.version() <= 4 {
SectionId::DebugRanges
} else {
SectionId::DebugRngLists
};
w.write_offset(range_lists.get(val).0, section, unit.format().word_size())?;
}
AttributeValue::DebugTypesRef(val) => {
debug_assert_form!(constants::DW_FORM_ref_sig8);
w.write_u64(val.0)?;
}
AttributeValue::StringRef(val) => {
debug_assert_form!(constants::DW_FORM_strp);
w.write_offset(
strings.get(val).0,
SectionId::DebugStr,
unit.format().word_size(),
)?;
}
AttributeValue::DebugStrRefSup(val) => {
debug_assert_form!(constants::DW_FORM_strp_sup);
w.write_udata(val.0 as u64, unit.format().word_size())?;
}
AttributeValue::LineStringRef(val) => {
debug_assert_form!(constants::DW_FORM_line_strp);
w.write_offset(
line_strings.get(val).0,
SectionId::DebugLineStr,
unit.format().word_size(),
)?;
}
AttributeValue::String(ref val) => {
debug_assert_form!(constants::DW_FORM_string);
w.write(val)?;
w.write_u8(0)?;
}
AttributeValue::Encoding(val) => {
debug_assert_form!(constants::DW_FORM_udata);
w.write_uleb128(u64::from(val.0))?;
}
AttributeValue::DecimalSign(val) => {
debug_assert_form!(constants::DW_FORM_udata);
w.write_uleb128(u64::from(val.0))?;
}
AttributeValue::Endianity(val) => {
debug_assert_form!(constants::DW_FORM_udata);
w.write_uleb128(u64::from(val.0))?;
}
AttributeValue::Accessibility(val) => {
debug_assert_form!(constants::DW_FORM_udata);
w.write_uleb128(u64::from(val.0))?;
}
AttributeValue::Visibility(val) => {
debug_assert_form!(constants::DW_FORM_udata);
w.write_uleb128(u64::from(val.0))?;
}
AttributeValue::Virtuality(val) => {
debug_assert_form!(constants::DW_FORM_udata);
w.write_uleb128(u64::from(val.0))?;
}
AttributeValue::Language(val) => {
debug_assert_form!(constants::DW_FORM_udata);
w.write_uleb128(u64::from(val.0))?;
}
AttributeValue::AddressClass(val) => {
debug_assert_form!(constants::DW_FORM_udata);
w.write_uleb128(val.0)?;
}
AttributeValue::IdentifierCase(val) => {
debug_assert_form!(constants::DW_FORM_udata);
w.write_uleb128(u64::from(val.0))?;
}
AttributeValue::CallingConvention(val) => {
debug_assert_form!(constants::DW_FORM_udata);
w.write_uleb128(u64::from(val.0))?;
}
AttributeValue::Inline(val) => {
debug_assert_form!(constants::DW_FORM_udata);
w.write_uleb128(u64::from(val.0))?;
}
AttributeValue::Ordering(val) => {
debug_assert_form!(constants::DW_FORM_udata);
w.write_uleb128(u64::from(val.0))?;
}
AttributeValue::FileIndex(val) => {
debug_assert_form!(constants::DW_FORM_udata);
w.write_uleb128(val.map(FileId::raw).unwrap_or(0))?;
}
}
Ok(())
}
}
define_section!(
DebugInfo,
DebugInfoOffset,
"A writable `.debug_info` section."
);
#[derive(Debug, Default)]
pub struct DebugInfoOffsets {
base_id: BaseId,
units: Vec<UnitOffsets>,
}
impl DebugInfoOffsets {
#[cfg(test)]
#[cfg(feature = "read")]
pub(crate) fn unit_offsets(&self, unit: UnitId) -> &UnitOffsets {
debug_assert_eq!(self.base_id, unit.base_id);
&self.units[unit.index]
}
#[inline]
pub fn unit(&self, unit: UnitId) -> DebugInfoOffset {
debug_assert_eq!(self.base_id, unit.base_id);
self.units[unit.index].unit
}
#[inline]
pub fn entry(&self, unit: UnitId, entry: UnitEntryId) -> DebugInfoOffset {
debug_assert_eq!(self.base_id, unit.base_id);
self.units[unit.index].debug_info_offset(entry)
}
}
#[derive(Debug)]
pub(crate) struct UnitOffsets {
base_id: BaseId,
unit: DebugInfoOffset,
entries: Vec<EntryOffset>,
}
impl UnitOffsets {
#[cfg(test)]
#[cfg(feature = "read")]
fn none() -> Self {
UnitOffsets {
base_id: BaseId::default(),
unit: DebugInfoOffset(0),
entries: Vec::new(),
}
}
#[inline]
pub(crate) fn debug_info_offset(&self, entry: UnitEntryId) -> DebugInfoOffset {
debug_assert_eq!(self.base_id, entry.base_id);
let offset = self.entries[entry.index].offset;
debug_assert_ne!(offset.0, 0);
offset
}
#[inline]
pub(crate) fn unit_offset(&self, entry: UnitEntryId) -> u64 {
let offset = self.debug_info_offset(entry);
(offset.0 - self.unit.0) as u64
}
#[inline]
pub(crate) fn abbrev(&self, entry: UnitEntryId) -> u64 {
debug_assert_eq!(self.base_id, entry.base_id);
self.entries[entry.index].abbrev
}
}
#[derive(Debug, Clone, Copy)]
pub(crate) struct EntryOffset {
offset: DebugInfoOffset,
abbrev: u64,
}
impl EntryOffset {
fn none() -> Self {
EntryOffset {
offset: DebugInfoOffset(0),
abbrev: 0,
}
}
}
#[derive(Debug, Clone, Copy)]
pub(crate) struct DebugInfoReference {
pub offset: usize,
pub size: u8,
pub unit: UnitId,
pub entry: UnitEntryId,
}
#[cfg(feature = "read")]
pub(crate) mod convert {
use super::*;
use crate::common::{DwoId, UnitSectionOffset};
use crate::read::{self, Reader};
use crate::write::{self, ConvertError, ConvertResult, LocationList, RangeList};
use std::collections::HashMap;
pub(crate) struct ConvertUnit<R: Reader<Offset = usize>> {
from_unit: read::Unit<R>,
base_id: BaseId,
encoding: Encoding,
entries: Vec<DebuggingInformationEntry>,
entry_offsets: Vec<read::UnitOffset>,
root: UnitEntryId,
}
pub(crate) struct ConvertUnitContext<'a, R: Reader<Offset = usize>> {
pub dwarf: &'a read::Dwarf<R>,
pub unit: &'a read::Unit<R>,
pub line_strings: &'a mut write::LineStringTable,
pub strings: &'a mut write::StringTable,
pub ranges: &'a mut write::RangeListTable,
pub locations: &'a mut write::LocationListTable,
pub convert_address: &'a dyn Fn(u64) -> Option<Address>,
pub base_address: Address,
pub line_program_offset: Option<DebugLineOffset>,
pub line_program_files: Vec<FileId>,
pub entry_ids: &'a HashMap<UnitSectionOffset, (UnitId, UnitEntryId)>,
}
impl UnitTable {
pub fn from<R: Reader<Offset = usize>>(
dwarf: &read::Dwarf<R>,
line_strings: &mut write::LineStringTable,
strings: &mut write::StringTable,
convert_address: &dyn Fn(u64) -> Option<Address>,
) -> ConvertResult<UnitTable> {
let base_id = BaseId::default();
let mut unit_entries = Vec::new();
let mut entry_ids = HashMap::new();
let mut from_units = dwarf.units();
while let Some(from_unit) = from_units.next()? {
let unit_id = UnitId::new(base_id, unit_entries.len());
unit_entries.push(Unit::convert_entries(
from_unit,
unit_id,
&mut entry_ids,
dwarf,
)?);
}
let mut units = Vec::new();
for unit_entries in unit_entries.drain(..) {
units.push(Unit::convert_attributes(
unit_entries,
&entry_ids,
dwarf,
line_strings,
strings,
convert_address,
)?);
}
Ok(UnitTable { base_id, units })
}
}
impl Unit {
pub(crate) fn convert_entries<R: Reader<Offset = usize>>(
from_header: read::UnitHeader<R>,
unit_id: UnitId,
entry_ids: &mut HashMap<UnitSectionOffset, (UnitId, UnitEntryId)>,
dwarf: &read::Dwarf<R>,
) -> ConvertResult<ConvertUnit<R>> {
match from_header.type_() {
read::UnitType::Compilation => (),
_ => return Err(ConvertError::UnsupportedUnitType),
}
let base_id = BaseId::default();
let from_unit = dwarf.unit(from_header)?;
let encoding = from_unit.encoding();
let mut entries = Vec::new();
let mut entry_offsets = Vec::new();
let mut from_tree = from_unit.entries_tree(None)?;
let from_root = from_tree.root()?;
let root = DebuggingInformationEntry::convert_entry(
from_root,
&from_unit,
base_id,
&mut entries,
&mut entry_offsets,
entry_ids,
None,
unit_id,
)?;
Ok(ConvertUnit {
from_unit,
base_id,
encoding,
entries,
entry_offsets,
root,
})
}
fn convert_attributes<R: Reader<Offset = usize>>(
unit: ConvertUnit<R>,
entry_ids: &HashMap<UnitSectionOffset, (UnitId, UnitEntryId)>,
dwarf: &read::Dwarf<R>,
line_strings: &mut write::LineStringTable,
strings: &mut write::StringTable,
convert_address: &dyn Fn(u64) -> Option<Address>,
) -> ConvertResult<Unit> {
let from_unit = unit.from_unit;
let base_address =
convert_address(from_unit.low_pc).ok_or(ConvertError::InvalidAddress)?;
let (line_program_offset, line_program, line_program_files) =
match from_unit.line_program {
Some(ref from_program) => {
let from_program = from_program.clone();
let line_program_offset = from_program.header().offset();
let (line_program, line_program_files) = LineProgram::from(
from_program,
dwarf,
line_strings,
strings,
convert_address,
)?;
(Some(line_program_offset), line_program, line_program_files)
}
None => (None, LineProgram::none(), Vec::new()),
};
let mut ranges = RangeListTable::default();
let mut locations = LocationListTable::default();
let mut context = ConvertUnitContext {
entry_ids,
dwarf,
unit: &from_unit,
line_strings,
strings,
ranges: &mut ranges,
locations: &mut locations,
convert_address,
base_address,
line_program_offset,
line_program_files,
};
let mut entries = unit.entries;
for entry in &mut entries {
entry.convert_attributes(&mut context, &unit.entry_offsets)?;
}
Ok(Unit {
base_id: unit.base_id,
encoding: unit.encoding,
line_program,
ranges,
locations,
entries,
root: unit.root,
})
}
}
impl DebuggingInformationEntry {
fn convert_entry<R: Reader<Offset = usize>>(
from: read::EntriesTreeNode<'_, '_, '_, R>,
from_unit: &read::Unit<R>,
base_id: BaseId,
entries: &mut Vec<DebuggingInformationEntry>,
entry_offsets: &mut Vec<read::UnitOffset>,
entry_ids: &mut HashMap<UnitSectionOffset, (UnitId, UnitEntryId)>,
parent: Option<UnitEntryId>,
unit_id: UnitId,
) -> ConvertResult<UnitEntryId> {
let from_entry = from.entry();
let id = DebuggingInformationEntry::new(base_id, entries, parent, from_entry.tag());
let offset = from_entry.offset();
entry_offsets.push(offset);
entry_ids.insert(offset.to_unit_section_offset(from_unit), (unit_id, id));
let mut from_children = from.children();
while let Some(from_child) = from_children.next()? {
DebuggingInformationEntry::convert_entry(
from_child,
from_unit,
base_id,
entries,
entry_offsets,
entry_ids,
Some(id),
unit_id,
)?;
}
Ok(id)
}
fn convert_attributes<R: Reader<Offset = usize>>(
&mut self,
context: &mut ConvertUnitContext<'_, R>,
entry_offsets: &[read::UnitOffset],
) -> ConvertResult<()> {
let offset = entry_offsets[self.id.index];
let from = context.unit.entry(offset)?;
let mut from_attrs = from.attrs();
while let Some(from_attr) = from_attrs.next()? {
if from_attr.name() == constants::DW_AT_sibling {
self.set_sibling(true);
} else if let Some(attr) = Attribute::from(context, &from_attr)? {
self.set(attr.name, attr.value);
}
}
Ok(())
}
}
impl Attribute {
pub(crate) fn from<R: Reader<Offset = usize>>(
context: &mut ConvertUnitContext<'_, R>,
from: &read::Attribute<R>,
) -> ConvertResult<Option<Attribute>> {
let value = AttributeValue::from(context, from.value())?;
Ok(value.map(|value| Attribute {
name: from.name(),
value,
}))
}
}
impl AttributeValue {
pub(crate) fn from<R: Reader<Offset = usize>>(
context: &mut ConvertUnitContext<'_, R>,
from: read::AttributeValue<R>,
) -> ConvertResult<Option<AttributeValue>> {
let to = match from {
read::AttributeValue::Addr(val) => match (context.convert_address)(val) {
Some(val) => AttributeValue::Address(val),
None => return Err(ConvertError::InvalidAddress),
},
read::AttributeValue::Block(r) => AttributeValue::Block(r.to_slice()?.into()),
read::AttributeValue::Data1(val) => AttributeValue::Data1(val),
read::AttributeValue::Data2(val) => AttributeValue::Data2(val),
read::AttributeValue::Data4(val) => AttributeValue::Data4(val),
read::AttributeValue::Data8(val) => AttributeValue::Data8(val),
read::AttributeValue::Sdata(val) => AttributeValue::Sdata(val),
read::AttributeValue::Udata(val) => AttributeValue::Udata(val),
read::AttributeValue::Exprloc(expression) => {
let expression = Expression::from(
expression,
context.unit.encoding(),
Some(context.dwarf),
Some(context.unit),
Some(context.entry_ids),
context.convert_address,
)?;
AttributeValue::Exprloc(expression)
}
read::AttributeValue::Flag(val) => AttributeValue::Flag(val),
read::AttributeValue::DebugAddrBase(_base) => {
return Ok(None);
}
read::AttributeValue::DebugAddrIndex(index) => {
let val = context.dwarf.address(context.unit, index)?;
match (context.convert_address)(val) {
Some(val) => AttributeValue::Address(val),
None => return Err(ConvertError::InvalidAddress),
}
}
read::AttributeValue::UnitRef(val) => {
if !context.unit.header.is_valid_offset(val) {
return Err(ConvertError::InvalidUnitRef);
}
let id = context
.entry_ids
.get(&val.to_unit_section_offset(context.unit))
.ok_or(ConvertError::InvalidUnitRef)?;
AttributeValue::UnitRef(id.1)
}
read::AttributeValue::DebugInfoRef(val) => {
let id = context
.entry_ids
.get(&UnitSectionOffset::DebugInfoOffset(val))
.ok_or(ConvertError::InvalidDebugInfoRef)?;
AttributeValue::DebugInfoRef(Reference::Entry(id.0, id.1))
}
read::AttributeValue::DebugInfoRefSup(val) => AttributeValue::DebugInfoRefSup(val),
read::AttributeValue::DebugLineRef(val) => {
if Some(val) == context.line_program_offset {
AttributeValue::LineProgramRef
} else {
return Err(ConvertError::InvalidLineRef);
}
}
read::AttributeValue::DebugMacinfoRef(val) => AttributeValue::DebugMacinfoRef(val),
read::AttributeValue::DebugMacroRef(val) => AttributeValue::DebugMacroRef(val),
read::AttributeValue::LocationListsRef(val) => {
let iter = context
.dwarf
.locations
.raw_locations(val, context.unit.encoding())?;
let loc_list = LocationList::from(iter, context)?;
let loc_id = context.locations.add(loc_list);
AttributeValue::LocationListRef(loc_id)
}
read::AttributeValue::DebugLocListsBase(_base) => {
return Ok(None);
}
read::AttributeValue::DebugLocListsIndex(index) => {
let offset = context.dwarf.locations_offset(context.unit, index)?;
let iter = context
.dwarf
.locations
.raw_locations(offset, context.unit.encoding())?;
let loc_list = LocationList::from(iter, context)?;
let loc_id = context.locations.add(loc_list);
AttributeValue::LocationListRef(loc_id)
}
read::AttributeValue::RangeListsRef(offset) => {
let offset = context.dwarf.ranges_offset_from_raw(context.unit, offset);
let iter = context.dwarf.raw_ranges(context.unit, offset)?;
let range_list = RangeList::from(iter, context)?;
let range_id = context.ranges.add(range_list);
AttributeValue::RangeListRef(range_id)
}
read::AttributeValue::DebugRngListsBase(_base) => {
return Ok(None);
}
read::AttributeValue::DebugRngListsIndex(index) => {
let offset = context.dwarf.ranges_offset(context.unit, index)?;
let iter = context
.dwarf
.ranges
.raw_ranges(offset, context.unit.encoding())?;
let range_list = RangeList::from(iter, context)?;
let range_id = context.ranges.add(range_list);
AttributeValue::RangeListRef(range_id)
}
read::AttributeValue::DebugTypesRef(val) => AttributeValue::DebugTypesRef(val),
read::AttributeValue::DebugStrRef(offset) => {
let r = context.dwarf.string(offset)?;
let id = context.strings.add(r.to_slice()?);
AttributeValue::StringRef(id)
}
read::AttributeValue::DebugStrRefSup(val) => AttributeValue::DebugStrRefSup(val),
read::AttributeValue::DebugStrOffsetsBase(_base) => {
return Ok(None);
}
read::AttributeValue::DebugStrOffsetsIndex(index) => {
let offset = context.dwarf.string_offset(context.unit, index)?;
let r = context.dwarf.string(offset)?;
let id = context.strings.add(r.to_slice()?);
AttributeValue::StringRef(id)
}
read::AttributeValue::DebugLineStrRef(offset) => {
let r = context.dwarf.line_string(offset)?;
let id = context.line_strings.add(r.to_slice()?);
AttributeValue::LineStringRef(id)
}
read::AttributeValue::String(r) => AttributeValue::String(r.to_slice()?.into()),
read::AttributeValue::Encoding(val) => AttributeValue::Encoding(val),
read::AttributeValue::DecimalSign(val) => AttributeValue::DecimalSign(val),
read::AttributeValue::Endianity(val) => AttributeValue::Endianity(val),
read::AttributeValue::Accessibility(val) => AttributeValue::Accessibility(val),
read::AttributeValue::Visibility(val) => AttributeValue::Visibility(val),
read::AttributeValue::Virtuality(val) => AttributeValue::Virtuality(val),
read::AttributeValue::Language(val) => AttributeValue::Language(val),
read::AttributeValue::AddressClass(val) => AttributeValue::AddressClass(val),
read::AttributeValue::IdentifierCase(val) => AttributeValue::IdentifierCase(val),
read::AttributeValue::CallingConvention(val) => {
AttributeValue::CallingConvention(val)
}
read::AttributeValue::Inline(val) => AttributeValue::Inline(val),
read::AttributeValue::Ordering(val) => AttributeValue::Ordering(val),
read::AttributeValue::FileIndex(val) => {
if val == 0 {
AttributeValue::FileIndex(None)
} else {
match context.line_program_files.get(val as usize) {
Some(id) => AttributeValue::FileIndex(Some(*id)),
None => return Err(ConvertError::InvalidFileIndex),
}
}
}
read::AttributeValue::SecOffset(_) => {
return Err(ConvertError::InvalidAttributeValue);
}
read::AttributeValue::DwoId(DwoId(val)) => AttributeValue::Udata(val),
};
Ok(Some(to))
}
}
}
#[cfg(test)]
#[cfg(feature = "read")]
mod tests {
use super::*;
use crate::common::{
DebugAddrBase, DebugLocListsBase, DebugRngListsBase, DebugStrOffsetsBase, LineEncoding,
};
use crate::constants;
use crate::read;
use crate::write::{
DebugLine, DebugLineStr, DebugStr, DwarfUnit, EndianVec, LineString, LineStringTable,
Location, LocationList, LocationListTable, Range, RangeList, RangeListOffsets,
RangeListTable, StringTable,
};
use crate::LittleEndian;
use std::collections::HashMap;
use std::mem;
use std::sync::Arc;
#[test]
fn test_unit_table() {
let mut strings = StringTable::default();
let mut units = UnitTable::default();
let unit_id1 = units.add(Unit::new(
Encoding {
version: 4,
address_size: 8,
format: Format::Dwarf32,
},
LineProgram::none(),
));
let unit2 = units.add(Unit::new(
Encoding {
version: 2,
address_size: 4,
format: Format::Dwarf64,
},
LineProgram::none(),
));
let unit3 = units.add(Unit::new(
Encoding {
version: 5,
address_size: 4,
format: Format::Dwarf32,
},
LineProgram::none(),
));
assert_eq!(units.count(), 3);
{
let unit1 = units.get_mut(unit_id1);
assert_eq!(unit1.version(), 4);
assert_eq!(unit1.address_size(), 8);
assert_eq!(unit1.format(), Format::Dwarf32);
assert_eq!(unit1.count(), 1);
let root_id = unit1.root();
assert_eq!(root_id, UnitEntryId::new(unit1.base_id, 0));
{
let root = unit1.get_mut(root_id);
assert_eq!(root.id(), root_id);
assert!(root.parent().is_none());
assert_eq!(root.tag(), constants::DW_TAG_compile_unit);
assert!(root.get(constants::DW_AT_producer).is_none());
assert!(root.get_mut(constants::DW_AT_producer).is_none());
let mut producer = AttributeValue::String(b"root"[..].into());
root.set(constants::DW_AT_producer, producer.clone());
assert_eq!(root.get(constants::DW_AT_producer), Some(&producer));
assert_eq!(root.get_mut(constants::DW_AT_producer), Some(&mut producer));
let mut attrs = root.attrs();
let attr = attrs.next().unwrap();
assert_eq!(attr.name(), constants::DW_AT_producer);
assert_eq!(attr.get(), &producer);
assert!(attrs.next().is_none());
}
let child1 = unit1.add(root_id, constants::DW_TAG_subprogram);
assert_eq!(child1, UnitEntryId::new(unit1.base_id, 1));
{
let child1 = unit1.get_mut(child1);
assert_eq!(child1.parent(), Some(root_id));
let tmp = AttributeValue::String(b"tmp"[..].into());
child1.set(constants::DW_AT_name, tmp.clone());
assert_eq!(child1.get(constants::DW_AT_name), Some(&tmp));
let name = AttributeValue::StringRef(strings.add(&b"child1"[..]));
{
let attr = child1.attrs_mut().next().unwrap();
assert_eq!(attr.name(), constants::DW_AT_name);
attr.set(name.clone());
}
assert_eq!(child1.get(constants::DW_AT_name), Some(&name));
}
let child2 = unit1.add(root_id, constants::DW_TAG_subprogram);
assert_eq!(child2, UnitEntryId::new(unit1.base_id, 2));
{
let child2 = unit1.get_mut(child2);
assert_eq!(child2.parent(), Some(root_id));
let tmp = AttributeValue::String(b"tmp"[..].into());
child2.set(constants::DW_AT_name, tmp.clone());
assert_eq!(child2.get(constants::DW_AT_name), Some(&tmp));
let name = AttributeValue::StringRef(strings.add(&b"child2"[..]));
child2.set(constants::DW_AT_name, name.clone());
assert_eq!(child2.get(constants::DW_AT_name), Some(&name));
}
{
let root = unit1.get(root_id);
assert_eq!(
root.children().cloned().collect::<Vec<_>>(),
vec![child1, child2]
);
}
}
{
let unit2 = units.get(unit2);
assert_eq!(unit2.version(), 2);
assert_eq!(unit2.address_size(), 4);
assert_eq!(unit2.format(), Format::Dwarf64);
assert_eq!(unit2.count(), 1);
let root = unit2.root();
assert_eq!(root, UnitEntryId::new(unit2.base_id, 0));
let root = unit2.get(root);
assert_eq!(root.id(), UnitEntryId::new(unit2.base_id, 0));
assert!(root.parent().is_none());
assert_eq!(root.tag(), constants::DW_TAG_compile_unit);
}
let mut sections = Sections::new(EndianVec::new(LittleEndian));
let debug_line_str_offsets = DebugLineStrOffsets::none();
let debug_str_offsets = strings.write(&mut sections.debug_str).unwrap();
units
.write(&mut sections, &debug_line_str_offsets, &debug_str_offsets)
.unwrap();
println!("{:?}", sections.debug_str);
println!("{:?}", sections.debug_info);
println!("{:?}", sections.debug_abbrev);
let dwarf = read::Dwarf {
debug_abbrev: read::DebugAbbrev::new(sections.debug_abbrev.slice(), LittleEndian),
debug_info: read::DebugInfo::new(sections.debug_info.slice(), LittleEndian),
debug_str: read::DebugStr::new(sections.debug_str.slice(), LittleEndian),
..Default::default()
};
let mut read_units = dwarf.units();
{
let read_unit1 = read_units.next().unwrap().unwrap();
let unit1 = units.get(unit_id1);
assert_eq!(unit1.version(), read_unit1.version());
assert_eq!(unit1.address_size(), read_unit1.address_size());
assert_eq!(unit1.format(), read_unit1.format());
let read_unit1 = dwarf.unit(read_unit1).unwrap();
let mut read_entries = read_unit1.entries();
let root = unit1.get(unit1.root());
{
let (depth, read_root) = read_entries.next_dfs().unwrap().unwrap();
assert_eq!(depth, 0);
assert_eq!(root.tag(), read_root.tag());
assert!(read_root.has_children());
let producer = match root.get(constants::DW_AT_producer).unwrap() {
AttributeValue::String(ref producer) => &**producer,
otherwise => panic!("unexpected {:?}", otherwise),
};
assert_eq!(producer, b"root");
let read_producer = read_root
.attr_value(constants::DW_AT_producer)
.unwrap()
.unwrap();
assert_eq!(
dwarf
.attr_string(&read_unit1, read_producer)
.unwrap()
.slice(),
producer
);
}
let mut children = root.children().cloned();
{
let child = children.next().unwrap();
assert_eq!(child, UnitEntryId::new(unit1.base_id, 1));
let child = unit1.get(child);
let (depth, read_child) = read_entries.next_dfs().unwrap().unwrap();
assert_eq!(depth, 1);
assert_eq!(child.tag(), read_child.tag());
assert!(!read_child.has_children());
let name = match child.get(constants::DW_AT_name).unwrap() {
AttributeValue::StringRef(name) => *name,
otherwise => panic!("unexpected {:?}", otherwise),
};
let name = strings.get(name);
assert_eq!(name, b"child1");
let read_name = read_child
.attr_value(constants::DW_AT_name)
.unwrap()
.unwrap();
assert_eq!(
dwarf.attr_string(&read_unit1, read_name).unwrap().slice(),
name
);
}
{
let child = children.next().unwrap();
assert_eq!(child, UnitEntryId::new(unit1.base_id, 2));
let child = unit1.get(child);
let (depth, read_child) = read_entries.next_dfs().unwrap().unwrap();
assert_eq!(depth, 0);
assert_eq!(child.tag(), read_child.tag());
assert!(!read_child.has_children());
let name = match child.get(constants::DW_AT_name).unwrap() {
AttributeValue::StringRef(name) => *name,
otherwise => panic!("unexpected {:?}", otherwise),
};
let name = strings.get(name);
assert_eq!(name, b"child2");
let read_name = read_child
.attr_value(constants::DW_AT_name)
.unwrap()
.unwrap();
assert_eq!(
dwarf.attr_string(&read_unit1, read_name).unwrap().slice(),
name
);
}
assert!(read_entries.next_dfs().unwrap().is_none());
}
{
let read_unit2 = read_units.next().unwrap().unwrap();
let unit2 = units.get(unit2);
assert_eq!(unit2.version(), read_unit2.version());
assert_eq!(unit2.address_size(), read_unit2.address_size());
assert_eq!(unit2.format(), read_unit2.format());
let abbrevs = dwarf.abbreviations(&read_unit2).unwrap();
let mut read_entries = read_unit2.entries(&abbrevs);
{
let root = unit2.get(unit2.root());
let (depth, read_root) = read_entries.next_dfs().unwrap().unwrap();
assert_eq!(depth, 0);
assert_eq!(root.tag(), read_root.tag());
assert!(!read_root.has_children());
}
assert!(read_entries.next_dfs().unwrap().is_none());
}
{
let read_unit3 = read_units.next().unwrap().unwrap();
let unit3 = units.get(unit3);
assert_eq!(unit3.version(), read_unit3.version());
assert_eq!(unit3.address_size(), read_unit3.address_size());
assert_eq!(unit3.format(), read_unit3.format());
let abbrevs = dwarf.abbreviations(&read_unit3).unwrap();
let mut read_entries = read_unit3.entries(&abbrevs);
{
let root = unit3.get(unit3.root());
let (depth, read_root) = read_entries.next_dfs().unwrap().unwrap();
assert_eq!(depth, 0);
assert_eq!(root.tag(), read_root.tag());
assert!(!read_root.has_children());
}
assert!(read_entries.next_dfs().unwrap().is_none());
}
assert!(read_units.next().unwrap().is_none());
let mut convert_line_strings = LineStringTable::default();
let mut convert_strings = StringTable::default();
let convert_units = UnitTable::from(
&dwarf,
&mut convert_line_strings,
&mut convert_strings,
&|address| Some(Address::Constant(address)),
)
.unwrap();
assert_eq!(convert_units.count(), units.count());
for i in 0..convert_units.count() {
let unit_id = units.id(i);
let unit = units.get(unit_id);
let convert_unit_id = convert_units.id(i);
let convert_unit = convert_units.get(convert_unit_id);
assert_eq!(convert_unit.version(), unit.version());
assert_eq!(convert_unit.address_size(), unit.address_size());
assert_eq!(convert_unit.format(), unit.format());
assert_eq!(convert_unit.count(), unit.count());
let root = unit.get(unit.root());
let convert_root = convert_unit.get(convert_unit.root());
assert_eq!(convert_root.tag(), root.tag());
for (convert_attr, attr) in convert_root.attrs().zip(root.attrs()) {
assert_eq!(convert_attr, attr);
}
}
}
#[test]
fn test_attribute_value() {
let mut strings = StringTable::default();
strings.add("string one");
let string_id = strings.add("string two");
let mut debug_str = DebugStr::from(EndianVec::new(LittleEndian));
let debug_str_offsets = strings.write(&mut debug_str).unwrap();
let read_debug_str = read::DebugStr::new(debug_str.slice(), LittleEndian);
let mut line_strings = LineStringTable::default();
line_strings.add("line string one");
let line_string_id = line_strings.add("line string two");
let mut debug_line_str = DebugLineStr::from(EndianVec::new(LittleEndian));
let debug_line_str_offsets = line_strings.write(&mut debug_line_str).unwrap();
let read_debug_line_str =
read::DebugLineStr::from(read::EndianSlice::new(debug_line_str.slice(), LittleEndian));
let data = vec![1, 2, 3, 4];
let read_data = read::EndianSlice::new(&[1, 2, 3, 4], LittleEndian);
let mut expression = Expression::new();
expression.op_constu(57);
let read_expression = read::Expression(read::EndianSlice::new(
&[constants::DW_OP_constu.0, 57],
LittleEndian,
));
let mut ranges = RangeListTable::default();
let range_id = ranges.add(RangeList(vec![Range::StartEnd {
begin: Address::Constant(0x1234),
end: Address::Constant(0x2345),
}]));
let mut locations = LocationListTable::default();
let loc_id = locations.add(LocationList(vec![Location::StartEnd {
begin: Address::Constant(0x1234),
end: Address::Constant(0x2345),
data: expression.clone(),
}]));
for &version in &[2, 3, 4, 5] {
for &address_size in &[4, 8] {
for &format in &[Format::Dwarf32, Format::Dwarf64] {
let encoding = Encoding {
format,
version,
address_size,
};
let mut sections = Sections::new(EndianVec::new(LittleEndian));
let range_list_offsets = ranges.write(&mut sections, encoding).unwrap();
let loc_list_offsets = locations.write(&mut sections, encoding, None).unwrap();
let read_debug_ranges =
read::DebugRanges::new(sections.debug_ranges.slice(), LittleEndian);
let read_debug_rnglists =
read::DebugRngLists::new(sections.debug_rnglists.slice(), LittleEndian);
let read_debug_loc =
read::DebugLoc::new(sections.debug_loc.slice(), LittleEndian);
let read_debug_loclists =
read::DebugLocLists::new(sections.debug_loclists.slice(), LittleEndian);
let mut units = UnitTable::default();
let unit = units.add(Unit::new(encoding, LineProgram::none()));
let unit = units.get(unit);
let encoding = Encoding {
format,
version,
address_size,
};
let from_unit = read::UnitHeader::new(
encoding,
0,
read::UnitType::Compilation,
DebugAbbrevOffset(0),
DebugInfoOffset(0).into(),
read::EndianSlice::new(&[], LittleEndian),
);
for (name, value, expect_value) in &[
(
constants::DW_AT_name,
AttributeValue::Address(Address::Constant(0x1234)),
read::AttributeValue::Addr(0x1234),
),
(
constants::DW_AT_name,
AttributeValue::Block(data.clone()),
read::AttributeValue::Block(read_data),
),
(
constants::DW_AT_name,
AttributeValue::Data1(0x12),
read::AttributeValue::Data1(0x12),
),
(
constants::DW_AT_name,
AttributeValue::Data2(0x1234),
read::AttributeValue::Data2(0x1234),
),
(
constants::DW_AT_name,
AttributeValue::Data4(0x1234),
read::AttributeValue::Data4(0x1234),
),
(
constants::DW_AT_name,
AttributeValue::Data8(0x1234),
read::AttributeValue::Data8(0x1234),
),
(
constants::DW_AT_name,
AttributeValue::Sdata(0x1234),
read::AttributeValue::Sdata(0x1234),
),
(
constants::DW_AT_name,
AttributeValue::Udata(0x1234),
read::AttributeValue::Udata(0x1234),
),
(
constants::DW_AT_name,
AttributeValue::Exprloc(expression.clone()),
read::AttributeValue::Exprloc(read_expression),
),
(
constants::DW_AT_name,
AttributeValue::Flag(false),
read::AttributeValue::Flag(false),
),
(
constants::DW_AT_name,
AttributeValue::DebugInfoRefSup(DebugInfoOffset(0x1234)),
read::AttributeValue::DebugInfoRefSup(DebugInfoOffset(0x1234)),
),
(
constants::DW_AT_location,
AttributeValue::LocationListRef(loc_id),
read::AttributeValue::SecOffset(loc_list_offsets.get(loc_id).0),
),
(
constants::DW_AT_macro_info,
AttributeValue::DebugMacinfoRef(DebugMacinfoOffset(0x1234)),
read::AttributeValue::SecOffset(0x1234),
),
(
constants::DW_AT_macros,
AttributeValue::DebugMacroRef(DebugMacroOffset(0x1234)),
read::AttributeValue::SecOffset(0x1234),
),
(
constants::DW_AT_ranges,
AttributeValue::RangeListRef(range_id),
read::AttributeValue::SecOffset(range_list_offsets.get(range_id).0),
),
(
constants::DW_AT_name,
AttributeValue::DebugTypesRef(DebugTypeSignature(0x1234)),
read::AttributeValue::DebugTypesRef(DebugTypeSignature(0x1234)),
),
(
constants::DW_AT_name,
AttributeValue::StringRef(string_id),
read::AttributeValue::DebugStrRef(debug_str_offsets.get(string_id)),
),
(
constants::DW_AT_name,
AttributeValue::DebugStrRefSup(DebugStrOffset(0x1234)),
read::AttributeValue::DebugStrRefSup(DebugStrOffset(0x1234)),
),
(
constants::DW_AT_name,
AttributeValue::LineStringRef(line_string_id),
read::AttributeValue::DebugLineStrRef(
debug_line_str_offsets.get(line_string_id),
),
),
(
constants::DW_AT_name,
AttributeValue::String(data.clone()),
read::AttributeValue::String(read_data),
),
(
constants::DW_AT_encoding,
AttributeValue::Encoding(constants::DwAte(0x12)),
read::AttributeValue::Udata(0x12),
),
(
constants::DW_AT_decimal_sign,
AttributeValue::DecimalSign(constants::DwDs(0x12)),
read::AttributeValue::Udata(0x12),
),
(
constants::DW_AT_endianity,
AttributeValue::Endianity(constants::DwEnd(0x12)),
read::AttributeValue::Udata(0x12),
),
(
constants::DW_AT_accessibility,
AttributeValue::Accessibility(constants::DwAccess(0x12)),
read::AttributeValue::Udata(0x12),
),
(
constants::DW_AT_visibility,
AttributeValue::Visibility(constants::DwVis(0x12)),
read::AttributeValue::Udata(0x12),
),
(
constants::DW_AT_virtuality,
AttributeValue::Virtuality(constants::DwVirtuality(0x12)),
read::AttributeValue::Udata(0x12),
),
(
constants::DW_AT_language,
AttributeValue::Language(constants::DwLang(0x12)),
read::AttributeValue::Udata(0x12),
),
(
constants::DW_AT_address_class,
AttributeValue::AddressClass(constants::DwAddr(0x12)),
read::AttributeValue::Udata(0x12),
),
(
constants::DW_AT_identifier_case,
AttributeValue::IdentifierCase(constants::DwId(0x12)),
read::AttributeValue::Udata(0x12),
),
(
constants::DW_AT_calling_convention,
AttributeValue::CallingConvention(constants::DwCc(0x12)),
read::AttributeValue::Udata(0x12),
),
(
constants::DW_AT_ordering,
AttributeValue::Ordering(constants::DwOrd(0x12)),
read::AttributeValue::Udata(0x12),
),
(
constants::DW_AT_inline,
AttributeValue::Inline(constants::DwInl(0x12)),
read::AttributeValue::Udata(0x12),
),
][..]
{
let form = value.form(encoding).unwrap();
let attr = Attribute {
name: *name,
value: value.clone(),
};
let offsets = UnitOffsets::none();
let line_program_offset = None;
let mut debug_info_refs = Vec::new();
let mut unit_refs = Vec::new();
let mut debug_info = DebugInfo::from(EndianVec::new(LittleEndian));
attr.value
.write(
&mut debug_info,
&mut debug_info_refs,
&mut unit_refs,
unit,
&offsets,
line_program_offset,
&debug_line_str_offsets,
&debug_str_offsets,
&range_list_offsets,
&loc_list_offsets,
)
.unwrap();
let spec = read::AttributeSpecification::new(*name, form, None);
let mut r = read::EndianSlice::new(debug_info.slice(), LittleEndian);
let read_attr = read::parse_attribute(&mut r, encoding, spec).unwrap();
let read_value = &read_attr.raw_value();
let read_value = unsafe {
mem::transmute::<
&read::AttributeValue<read::EndianSlice<'_, LittleEndian>>,
&read::AttributeValue<read::EndianSlice<'_, LittleEndian>>,
>(read_value)
};
assert_eq!(read_value, expect_value);
let dwarf = read::Dwarf {
debug_str: read_debug_str,
debug_line_str: read_debug_line_str,
ranges: read::RangeLists::new(read_debug_ranges, read_debug_rnglists),
locations: read::LocationLists::new(
read_debug_loc,
read_debug_loclists,
),
..Default::default()
};
let unit = read::Unit {
header: from_unit,
abbreviations: Arc::new(read::Abbreviations::default()),
name: None,
comp_dir: None,
low_pc: 0,
str_offsets_base: DebugStrOffsetsBase(0),
addr_base: DebugAddrBase(0),
loclists_base: DebugLocListsBase(0),
rnglists_base: DebugRngListsBase(0),
line_program: None,
dwo_id: None,
};
let mut context = convert::ConvertUnitContext {
dwarf: &dwarf,
unit: &unit,
line_strings: &mut line_strings,
strings: &mut strings,
ranges: &mut ranges,
locations: &mut locations,
convert_address: &|address| Some(Address::Constant(address)),
base_address: Address::Constant(0),
line_program_offset: None,
line_program_files: Vec::new(),
entry_ids: &HashMap::new(),
};
let convert_attr =
Attribute::from(&mut context, &read_attr).unwrap().unwrap();
assert_eq!(convert_attr, attr);
}
}
}
}
}
#[test]
fn test_unit_ref() {
let mut units = UnitTable::default();
let unit_id1 = units.add(Unit::new(
Encoding {
version: 4,
address_size: 8,
format: Format::Dwarf32,
},
LineProgram::none(),
));
assert_eq!(unit_id1, units.id(0));
let unit_id2 = units.add(Unit::new(
Encoding {
version: 2,
address_size: 4,
format: Format::Dwarf64,
},
LineProgram::none(),
));
assert_eq!(unit_id2, units.id(1));
let unit1_child1 = UnitEntryId::new(units.get(unit_id1).base_id, 1);
let unit1_child2 = UnitEntryId::new(units.get(unit_id1).base_id, 2);
let unit2_child1 = UnitEntryId::new(units.get(unit_id2).base_id, 1);
let unit2_child2 = UnitEntryId::new(units.get(unit_id2).base_id, 2);
{
let unit1 = units.get_mut(unit_id1);
let root = unit1.root();
let child_id1 = unit1.add(root, constants::DW_TAG_subprogram);
assert_eq!(child_id1, unit1_child1);
let child_id2 = unit1.add(root, constants::DW_TAG_subprogram);
assert_eq!(child_id2, unit1_child2);
{
let child1 = unit1.get_mut(child_id1);
child1.set(constants::DW_AT_type, AttributeValue::UnitRef(child_id2));
}
{
let child2 = unit1.get_mut(child_id2);
child2.set(
constants::DW_AT_type,
AttributeValue::DebugInfoRef(Reference::Entry(unit_id2, unit2_child1)),
);
}
}
{
let unit2 = units.get_mut(unit_id2);
let root = unit2.root();
let child_id1 = unit2.add(root, constants::DW_TAG_subprogram);
assert_eq!(child_id1, unit2_child1);
let child_id2 = unit2.add(root, constants::DW_TAG_subprogram);
assert_eq!(child_id2, unit2_child2);
{
let child1 = unit2.get_mut(child_id1);
child1.set(constants::DW_AT_type, AttributeValue::UnitRef(child_id2));
}
{
let child2 = unit2.get_mut(child_id2);
child2.set(
constants::DW_AT_type,
AttributeValue::DebugInfoRef(Reference::Entry(unit_id1, unit1_child1)),
);
}
}
let debug_line_str_offsets = DebugLineStrOffsets::none();
let debug_str_offsets = DebugStrOffsets::none();
let mut sections = Sections::new(EndianVec::new(LittleEndian));
let debug_info_offsets = units
.write(&mut sections, &debug_line_str_offsets, &debug_str_offsets)
.unwrap();
println!("{:?}", sections.debug_info);
println!("{:?}", sections.debug_abbrev);
let dwarf = read::Dwarf {
debug_abbrev: read::DebugAbbrev::new(sections.debug_abbrev.slice(), LittleEndian),
debug_info: read::DebugInfo::new(sections.debug_info.slice(), LittleEndian),
..Default::default()
};
let mut read_units = dwarf.units();
{
let read_unit1 = read_units.next().unwrap().unwrap();
assert_eq!(
read_unit1.offset(),
debug_info_offsets.unit(unit_id1).into()
);
let abbrevs = dwarf.abbreviations(&read_unit1).unwrap();
let mut read_entries = read_unit1.entries(&abbrevs);
{
let (_, _read_root) = read_entries.next_dfs().unwrap().unwrap();
}
{
let (_, read_child1) = read_entries.next_dfs().unwrap().unwrap();
let offset = debug_info_offsets
.entry(unit_id1, unit1_child2)
.to_unit_offset(&read_unit1)
.unwrap();
assert_eq!(
read_child1.attr_value(constants::DW_AT_type).unwrap(),
Some(read::AttributeValue::UnitRef(offset))
);
}
{
let (_, read_child2) = read_entries.next_dfs().unwrap().unwrap();
let offset = debug_info_offsets.entry(unit_id2, unit2_child1);
assert_eq!(
read_child2.attr_value(constants::DW_AT_type).unwrap(),
Some(read::AttributeValue::DebugInfoRef(offset))
);
}
}
{
let read_unit2 = read_units.next().unwrap().unwrap();
assert_eq!(
read_unit2.offset(),
debug_info_offsets.unit(unit_id2).into()
);
let abbrevs = dwarf.abbreviations(&read_unit2).unwrap();
let mut read_entries = read_unit2.entries(&abbrevs);
{
let (_, _read_root) = read_entries.next_dfs().unwrap().unwrap();
}
{
let (_, read_child1) = read_entries.next_dfs().unwrap().unwrap();
let offset = debug_info_offsets
.entry(unit_id2, unit2_child2)
.to_unit_offset(&read_unit2)
.unwrap();
assert_eq!(
read_child1.attr_value(constants::DW_AT_type).unwrap(),
Some(read::AttributeValue::UnitRef(offset))
);
}
{
let (_, read_child2) = read_entries.next_dfs().unwrap().unwrap();
let offset = debug_info_offsets.entry(unit_id1, unit1_child1);
assert_eq!(
read_child2.attr_value(constants::DW_AT_type).unwrap(),
Some(read::AttributeValue::DebugInfoRef(offset))
);
}
}
let mut convert_line_strings = LineStringTable::default();
let mut convert_strings = StringTable::default();
let convert_units = UnitTable::from(
&dwarf,
&mut convert_line_strings,
&mut convert_strings,
&|address| Some(Address::Constant(address)),
)
.unwrap();
assert_eq!(convert_units.count(), units.count());
for i in 0..convert_units.count() {
let unit = units.get(units.id(i));
let convert_unit = convert_units.get(convert_units.id(i));
assert_eq!(convert_unit.version(), unit.version());
assert_eq!(convert_unit.address_size(), unit.address_size());
assert_eq!(convert_unit.format(), unit.format());
assert_eq!(convert_unit.count(), unit.count());
let root = unit.get(unit.root());
let convert_root = convert_unit.get(convert_unit.root());
assert_eq!(convert_root.tag(), root.tag());
for (convert_attr, attr) in convert_root.attrs().zip(root.attrs()) {
assert_eq!(convert_attr, attr);
}
let child1 = unit.get(UnitEntryId::new(unit.base_id, 1));
let convert_child1 = convert_unit.get(UnitEntryId::new(convert_unit.base_id, 1));
assert_eq!(convert_child1.tag(), child1.tag());
for (convert_attr, attr) in convert_child1.attrs().zip(child1.attrs()) {
assert_eq!(convert_attr.name, attr.name);
match (convert_attr.value.clone(), attr.value.clone()) {
(
AttributeValue::DebugInfoRef(Reference::Entry(convert_unit, convert_entry)),
AttributeValue::DebugInfoRef(Reference::Entry(unit, entry)),
) => {
assert_eq!(convert_unit.index, unit.index);
assert_eq!(convert_entry.index, entry.index);
}
(AttributeValue::UnitRef(convert_id), AttributeValue::UnitRef(id)) => {
assert_eq!(convert_id.index, id.index);
}
(convert_value, value) => assert_eq!(convert_value, value),
}
}
let child2 = unit.get(UnitEntryId::new(unit.base_id, 2));
let convert_child2 = convert_unit.get(UnitEntryId::new(convert_unit.base_id, 2));
assert_eq!(convert_child2.tag(), child2.tag());
for (convert_attr, attr) in convert_child2.attrs().zip(child2.attrs()) {
assert_eq!(convert_attr.name, attr.name);
match (convert_attr.value.clone(), attr.value.clone()) {
(
AttributeValue::DebugInfoRef(Reference::Entry(convert_unit, convert_entry)),
AttributeValue::DebugInfoRef(Reference::Entry(unit, entry)),
) => {
assert_eq!(convert_unit.index, unit.index);
assert_eq!(convert_entry.index, entry.index);
}
(AttributeValue::UnitRef(convert_id), AttributeValue::UnitRef(id)) => {
assert_eq!(convert_id.index, id.index);
}
(convert_value, value) => assert_eq!(convert_value, value),
}
}
}
}
#[test]
fn test_sibling() {
fn add_child(
unit: &mut Unit,
parent: UnitEntryId,
tag: constants::DwTag,
name: &str,
) -> UnitEntryId {
let id = unit.add(parent, tag);
let child = unit.get_mut(id);
child.set(constants::DW_AT_name, AttributeValue::String(name.into()));
child.set_sibling(true);
id
}
fn add_children(units: &mut UnitTable, unit_id: UnitId) {
let unit = units.get_mut(unit_id);
let root = unit.root();
let child1 = add_child(unit, root, constants::DW_TAG_subprogram, "child1");
add_child(unit, child1, constants::DW_TAG_variable, "grandchild1");
add_child(unit, root, constants::DW_TAG_subprogram, "child2");
add_child(unit, root, constants::DW_TAG_subprogram, "child3");
}
fn next_child<R: read::Reader<Offset = usize>>(
entries: &mut read::EntriesCursor<'_, '_, R>,
) -> (read::UnitOffset, Option<read::UnitOffset>) {
let (_, entry) = entries.next_dfs().unwrap().unwrap();
let offset = entry.offset();
let sibling =
entry
.attr_value(constants::DW_AT_sibling)
.unwrap()
.map(|attr| match attr {
read::AttributeValue::UnitRef(offset) => offset,
_ => panic!("bad sibling value"),
});
(offset, sibling)
}
fn check_sibling<R: read::Reader<Offset = usize>>(
unit: &read::UnitHeader<R>,
debug_abbrev: &read::DebugAbbrev<R>,
) {
let abbrevs = unit.abbreviations(debug_abbrev).unwrap();
let mut entries = unit.entries(&abbrevs);
entries.next_dfs().unwrap().unwrap();
let (_, sibling1) = next_child(&mut entries);
entries.next_dfs().unwrap().unwrap();
let (offset2, sibling2) = next_child(&mut entries);
let (_, _) = next_child(&mut entries);
assert_eq!(sibling1, Some(offset2));
assert_eq!(sibling2, None);
}
let encoding = Encoding {
format: Format::Dwarf32,
version: 4,
address_size: 8,
};
let mut units = UnitTable::default();
let unit_id1 = units.add(Unit::new(encoding, LineProgram::none()));
add_children(&mut units, unit_id1);
let unit_id2 = units.add(Unit::new(encoding, LineProgram::none()));
add_children(&mut units, unit_id2);
let debug_line_str_offsets = DebugLineStrOffsets::none();
let debug_str_offsets = DebugStrOffsets::none();
let mut sections = Sections::new(EndianVec::new(LittleEndian));
units
.write(&mut sections, &debug_line_str_offsets, &debug_str_offsets)
.unwrap();
println!("{:?}", sections.debug_info);
println!("{:?}", sections.debug_abbrev);
let read_debug_info = read::DebugInfo::new(sections.debug_info.slice(), LittleEndian);
let read_debug_abbrev = read::DebugAbbrev::new(sections.debug_abbrev.slice(), LittleEndian);
let mut read_units = read_debug_info.units();
check_sibling(&read_units.next().unwrap().unwrap(), &read_debug_abbrev);
check_sibling(&read_units.next().unwrap().unwrap(), &read_debug_abbrev);
}
#[test]
fn test_line_ref() {
for &version in &[2, 3, 4, 5] {
for &address_size in &[4, 8] {
for &format in &[Format::Dwarf32, Format::Dwarf64] {
let encoding = Encoding {
format,
version,
address_size,
};
let mut line_program = LineProgram::new(
encoding,
LineEncoding::default(),
LineString::String(b"comp_dir".to_vec()),
LineString::String(b"comp_name".to_vec()),
None,
);
let dir = line_program.default_directory();
let file1 =
line_program.add_file(LineString::String(b"file1".to_vec()), dir, None);
let file2 =
line_program.add_file(LineString::String(b"file2".to_vec()), dir, None);
let line_strings = DebugLineStrOffsets::none();
let strings = DebugStrOffsets::none();
let mut debug_line = DebugLine::from(EndianVec::new(LittleEndian));
let line_program_offset = line_program
.write(&mut debug_line, encoding, &line_strings, &strings)
.unwrap();
let read_debug_line = read::DebugLine::new(debug_line.slice(), LittleEndian);
let read_line_program = read_debug_line
.program(
line_program_offset,
address_size,
Some(read::EndianSlice::new(b"comp_dir", LittleEndian)),
Some(read::EndianSlice::new(b"comp_name", LittleEndian)),
)
.unwrap();
let dwarf = read::Dwarf::default();
let mut convert_line_strings = LineStringTable::default();
let mut convert_strings = StringTable::default();
let (_, line_program_files) = LineProgram::from(
read_line_program,
&dwarf,
&mut convert_line_strings,
&mut convert_strings,
&|address| Some(Address::Constant(address)),
)
.unwrap();
let mut units = UnitTable::default();
let unit = units.add(Unit::new(encoding, LineProgram::none()));
let unit = units.get(unit);
let from_unit = read::UnitHeader::new(
encoding,
0,
read::UnitType::Compilation,
DebugAbbrevOffset(0),
DebugInfoOffset(0).into(),
read::EndianSlice::new(&[], LittleEndian),
);
for (name, value, expect_value) in &[
(
constants::DW_AT_stmt_list,
AttributeValue::LineProgramRef,
read::AttributeValue::SecOffset(line_program_offset.0),
),
(
constants::DW_AT_decl_file,
AttributeValue::FileIndex(Some(file1)),
read::AttributeValue::Udata(file1.raw()),
),
(
constants::DW_AT_decl_file,
AttributeValue::FileIndex(Some(file2)),
read::AttributeValue::Udata(file2.raw()),
),
][..]
{
let mut ranges = RangeListTable::default();
let mut locations = LocationListTable::default();
let mut strings = StringTable::default();
let mut line_strings = LineStringTable::default();
let form = value.form(encoding).unwrap();
let attr = Attribute {
name: *name,
value: value.clone(),
};
let mut debug_info_refs = Vec::new();
let mut unit_refs = Vec::new();
let mut debug_info = DebugInfo::from(EndianVec::new(LittleEndian));
let offsets = UnitOffsets::none();
let debug_line_str_offsets = DebugLineStrOffsets::none();
let debug_str_offsets = DebugStrOffsets::none();
let range_list_offsets = RangeListOffsets::none();
let loc_list_offsets = LocationListOffsets::none();
attr.value
.write(
&mut debug_info,
&mut debug_info_refs,
&mut unit_refs,
unit,
&offsets,
Some(line_program_offset),
&debug_line_str_offsets,
&debug_str_offsets,
&range_list_offsets,
&loc_list_offsets,
)
.unwrap();
let spec = read::AttributeSpecification::new(*name, form, None);
let mut r = read::EndianSlice::new(debug_info.slice(), LittleEndian);
let read_attr = read::parse_attribute(&mut r, encoding, spec).unwrap();
let read_value = &read_attr.raw_value();
let read_value = unsafe {
mem::transmute::<
&read::AttributeValue<read::EndianSlice<'_, LittleEndian>>,
&read::AttributeValue<read::EndianSlice<'_, LittleEndian>>,
>(read_value)
};
assert_eq!(read_value, expect_value);
let unit = read::Unit {
header: from_unit,
abbreviations: Arc::new(read::Abbreviations::default()),
name: None,
comp_dir: None,
low_pc: 0,
str_offsets_base: DebugStrOffsetsBase(0),
addr_base: DebugAddrBase(0),
loclists_base: DebugLocListsBase(0),
rnglists_base: DebugRngListsBase(0),
line_program: None,
dwo_id: None,
};
let mut context = convert::ConvertUnitContext {
dwarf: &dwarf,
unit: &unit,
line_strings: &mut line_strings,
strings: &mut strings,
ranges: &mut ranges,
locations: &mut locations,
convert_address: &|address| Some(Address::Constant(address)),
base_address: Address::Constant(0),
line_program_offset: Some(line_program_offset),
line_program_files: line_program_files.clone(),
entry_ids: &HashMap::new(),
};
let convert_attr =
Attribute::from(&mut context, &read_attr).unwrap().unwrap();
assert_eq!(convert_attr, attr);
}
}
}
}
}
#[test]
fn test_line_program_used() {
for used in [false, true] {
let encoding = Encoding {
format: Format::Dwarf32,
version: 5,
address_size: 8,
};
let line_program = LineProgram::new(
encoding,
LineEncoding::default(),
LineString::String(b"comp_dir".to_vec()),
LineString::String(b"comp_name".to_vec()),
None,
);
let mut unit = Unit::new(encoding, line_program);
let file_id = if used { Some(FileId::new(0)) } else { None };
let root = unit.root();
unit.get_mut(root).set(
constants::DW_AT_decl_file,
AttributeValue::FileIndex(file_id),
);
let mut units = UnitTable::default();
units.add(unit);
let debug_line_str_offsets = DebugLineStrOffsets::none();
let debug_str_offsets = DebugStrOffsets::none();
let mut sections = Sections::new(EndianVec::new(LittleEndian));
units
.write(&mut sections, &debug_line_str_offsets, &debug_str_offsets)
.unwrap();
assert_eq!(!used, sections.debug_line.slice().is_empty());
}
}
#[test]
fn test_delete_child() {
fn set_name(unit: &mut Unit, id: UnitEntryId, name: &str) {
let entry = unit.get_mut(id);
entry.set(constants::DW_AT_name, AttributeValue::String(name.into()));
}
fn check_name<R: read::Reader>(
entry: &read::DebuggingInformationEntry<'_, '_, R>,
debug_str: &read::DebugStr<R>,
name: &str,
) {
let name_attr = entry.attr(constants::DW_AT_name).unwrap().unwrap();
let entry_name = name_attr.string_value(debug_str).unwrap();
let entry_name_str = entry_name.to_string().unwrap();
assert_eq!(entry_name_str, name);
}
let encoding = Encoding {
format: Format::Dwarf32,
version: 4,
address_size: 8,
};
let mut dwarf = DwarfUnit::new(encoding);
let root = dwarf.unit.root();
let child1 = dwarf.unit.add(root, constants::DW_TAG_subprogram);
set_name(&mut dwarf.unit, child1, "child1");
let grandchild1 = dwarf.unit.add(child1, constants::DW_TAG_variable);
set_name(&mut dwarf.unit, grandchild1, "grandchild1");
let child2 = dwarf.unit.add(root, constants::DW_TAG_subprogram);
set_name(&mut dwarf.unit, child2, "child2");
dwarf.unit.get_mut(root).delete_child(child1);
let child3 = dwarf.unit.add(root, constants::DW_TAG_subprogram);
set_name(&mut dwarf.unit, child3, "child3");
let child4 = dwarf.unit.add(root, constants::DW_TAG_subprogram);
set_name(&mut dwarf.unit, child4, "child4");
let grandchild4 = dwarf.unit.add(child4, constants::DW_TAG_variable);
set_name(&mut dwarf.unit, grandchild4, "grandchild4");
dwarf.unit.get_mut(child4).delete_child(grandchild4);
let mut sections = Sections::new(EndianVec::new(LittleEndian));
dwarf.write(&mut sections).unwrap();
let read_debug_info = read::DebugInfo::new(sections.debug_info.slice(), LittleEndian);
let read_debug_abbrev = read::DebugAbbrev::new(sections.debug_abbrev.slice(), LittleEndian);
let read_debug_str = read::DebugStr::new(sections.debug_str.slice(), LittleEndian);
let read_unit = read_debug_info.units().next().unwrap().unwrap();
let abbrevs = read_unit.abbreviations(&read_debug_abbrev).unwrap();
let mut entries = read_unit.entries(&abbrevs);
entries.next_dfs().unwrap().unwrap();
let (_, read_child2) = entries.next_dfs().unwrap().unwrap();
check_name(read_child2, &read_debug_str, "child2");
let (_, read_child3) = entries.next_dfs().unwrap().unwrap();
check_name(read_child3, &read_debug_str, "child3");
let (_, read_child4) = entries.next_dfs().unwrap().unwrap();
check_name(read_child4, &read_debug_str, "child4");
assert!(entries.next_dfs().unwrap().is_none());
}
}