diff options
Diffstat (limited to 'vendor/gimli/examples')
-rw-r--r-- | vendor/gimli/examples/dwarf-validate.rs | 267 | ||||
-rw-r--r-- | vendor/gimli/examples/dwarfdump.rs | 2369 | ||||
-rw-r--r-- | vendor/gimli/examples/simple.rs | 67 | ||||
-rw-r--r-- | vendor/gimli/examples/simple_line.rs | 106 |
4 files changed, 0 insertions, 2809 deletions
diff --git a/vendor/gimli/examples/dwarf-validate.rs b/vendor/gimli/examples/dwarf-validate.rs deleted file mode 100644 index 54d8f3a1d..000000000 --- a/vendor/gimli/examples/dwarf-validate.rs +++ /dev/null @@ -1,267 +0,0 @@ -// Allow clippy lints when building without clippy. -#![allow(unknown_lints)] - -use gimli::{AttributeValue, UnitHeader}; -use object::{Object, ObjectSection}; -use rayon::prelude::*; -use std::borrow::{Borrow, Cow}; -use std::env; -use std::fs; -use std::io::{self, BufWriter, Write}; -use std::iter::Iterator; -use std::path::{Path, PathBuf}; -use std::process; -use std::sync::Mutex; -use typed_arena::Arena; - -trait Reader: gimli::Reader<Offset = usize> + Send + Sync { - type SyncSendEndian: gimli::Endianity + Send + Sync; -} - -impl<'input, Endian> Reader for gimli::EndianSlice<'input, Endian> -where - Endian: gimli::Endianity + Send + Sync, -{ - type SyncSendEndian = Endian; -} - -struct ErrorWriter<W: Write + Send> { - inner: Mutex<(W, usize)>, - path: PathBuf, -} - -impl<W: Write + Send> ErrorWriter<W> { - #[allow(clippy::needless_pass_by_value)] - fn error(&self, s: String) { - let mut lock = self.inner.lock().unwrap(); - writeln!(&mut lock.0, "DWARF error in {}: {}", self.path.display(), s).unwrap(); - lock.1 += 1; - } -} - -fn main() { - let mut w = BufWriter::new(io::stdout()); - let mut errors = 0; - for arg in env::args_os().skip(1) { - let path = Path::new(&arg); - let file = match fs::File::open(&path) { - Ok(file) => file, - Err(err) => { - eprintln!("Failed to open file '{}': {}", path.display(), err); - errors += 1; - continue; - } - }; - let file = match unsafe { memmap2::Mmap::map(&file) } { - Ok(mmap) => mmap, - Err(err) => { - eprintln!("Failed to map file '{}': {}", path.display(), &err); - errors += 1; - continue; - } - }; - let file = match object::File::parse(&*file) { - Ok(file) => file, - Err(err) => { - eprintln!("Failed to parse file '{}': {}", path.display(), err); - errors += 1; - continue; - } - }; - - let endian = if file.is_little_endian() { - gimli::RunTimeEndian::Little - } else { - gimli::RunTimeEndian::Big - }; - let mut error_writer = ErrorWriter { - inner: Mutex::new((&mut w, 0)), - path: path.to_owned(), - }; - validate_file(&mut error_writer, &file, endian); - errors += error_writer.inner.into_inner().unwrap().1; - } - // Flush any errors. - drop(w); - if errors > 0 { - process::exit(1); - } -} - -fn validate_file<W, Endian>(w: &mut ErrorWriter<W>, file: &object::File, endian: Endian) -where - W: Write + Send, - Endian: gimli::Endianity + Send + Sync, -{ - let arena = Arena::new(); - - fn load_section<'a, 'file, 'input, S, Endian>( - arena: &'a Arena<Cow<'file, [u8]>>, - file: &'file object::File<'input>, - endian: Endian, - ) -> S - where - S: gimli::Section<gimli::EndianSlice<'a, Endian>>, - Endian: gimli::Endianity + Send + Sync, - 'file: 'input, - 'a: 'file, - { - let data = match file.section_by_name(S::section_name()) { - Some(ref section) => section - .uncompressed_data() - .unwrap_or(Cow::Borrowed(&[][..])), - None => Cow::Borrowed(&[][..]), - }; - let data_ref = (*arena.alloc(data)).borrow(); - S::from(gimli::EndianSlice::new(data_ref, endian)) - } - - // Variables representing sections of the file. The type of each is inferred from its use in the - // validate_info function below. - let debug_abbrev = &load_section(&arena, file, endian); - let debug_info = &load_section(&arena, file, endian); - - validate_info(w, debug_info, debug_abbrev); -} - -struct UnitSummary { - // True if we successfully parsed all the DIEs and attributes in the compilation unit - internally_valid: bool, - offset: gimli::DebugInfoOffset, - die_offsets: Vec<gimli::UnitOffset>, - global_die_references: Vec<(gimli::UnitOffset, gimli::DebugInfoOffset)>, -} - -fn validate_info<W, R>( - w: &mut ErrorWriter<W>, - debug_info: &gimli::DebugInfo<R>, - debug_abbrev: &gimli::DebugAbbrev<R>, -) where - W: Write + Send, - R: Reader, -{ - let mut units = Vec::new(); - let mut units_iter = debug_info.units(); - let mut last_offset = 0; - loop { - let u = match units_iter.next() { - Err(err) => { - w.error(format!( - "Can't read unit header at offset {:#x}, stopping reading units: {}", - last_offset, err - )); - break; - } - Ok(None) => break, - Ok(Some(u)) => u, - }; - last_offset = u.offset().as_debug_info_offset().unwrap().0 + u.length_including_self(); - units.push(u); - } - let process_unit = |unit: UnitHeader<R>| -> UnitSummary { - let unit_offset = unit.offset().as_debug_info_offset().unwrap(); - let mut ret = UnitSummary { - internally_valid: false, - offset: unit_offset, - die_offsets: Vec::new(), - global_die_references: Vec::new(), - }; - let abbrevs = match unit.abbreviations(debug_abbrev) { - Ok(abbrevs) => abbrevs, - Err(err) => { - w.error(format!( - "Invalid abbrevs for unit {:#x}: {}", - unit_offset.0, &err - )); - return ret; - } - }; - let mut entries = unit.entries(&abbrevs); - let mut unit_refs = Vec::new(); - loop { - let (_, entry) = match entries.next_dfs() { - Err(err) => { - w.error(format!( - "Invalid DIE for unit {:#x}: {}", - unit_offset.0, &err - )); - return ret; - } - Ok(None) => break, - Ok(Some(entry)) => entry, - }; - ret.die_offsets.push(entry.offset()); - - let mut attrs = entry.attrs(); - loop { - let attr = match attrs.next() { - Err(err) => { - w.error(format!( - "Invalid attribute for unit {:#x} at DIE {:#x}: {}", - unit_offset.0, - entry.offset().0, - &err - )); - return ret; - } - Ok(None) => break, - Ok(Some(attr)) => attr, - }; - match attr.value() { - AttributeValue::UnitRef(offset) => { - unit_refs.push((entry.offset(), offset)); - } - AttributeValue::DebugInfoRef(offset) => { - ret.global_die_references.push((entry.offset(), offset)); - } - _ => (), - } - } - } - ret.internally_valid = true; - ret.die_offsets.shrink_to_fit(); - ret.global_die_references.shrink_to_fit(); - - // Check intra-unit references - for (from, to) in unit_refs { - if ret.die_offsets.binary_search(&to).is_err() { - w.error(format!( - "Invalid intra-unit reference in unit {:#x} from DIE {:#x} to {:#x}", - unit_offset.0, from.0, to.0 - )); - } - } - - ret - }; - let processed_units = units.into_par_iter().map(process_unit).collect::<Vec<_>>(); - - let check_unit = |summary: &UnitSummary| { - if !summary.internally_valid { - return; - } - for &(from, to) in summary.global_die_references.iter() { - let u = match processed_units.binary_search_by_key(&to, |v| v.offset) { - Ok(i) => &processed_units[i], - Err(i) => { - if i > 0 { - &processed_units[i - 1] - } else { - w.error(format!("Invalid cross-unit reference in unit {:#x} from DIE {:#x} to global DIE {:#x}: no unit found", - summary.offset.0, from.0, to.0)); - continue; - } - } - }; - if !u.internally_valid { - continue; - } - let to_offset = gimli::UnitOffset(to.0 - u.offset.0); - if u.die_offsets.binary_search(&to_offset).is_err() { - w.error(format!("Invalid cross-unit reference in unit {:#x} from DIE {:#x} to global DIE {:#x}: unit at {:#x} contains no DIE {:#x}", - summary.offset.0, from.0, to.0, u.offset.0, to_offset.0)); - } - } - }; - processed_units.par_iter().for_each(check_unit); -} diff --git a/vendor/gimli/examples/dwarfdump.rs b/vendor/gimli/examples/dwarfdump.rs deleted file mode 100644 index a11f298f8..000000000 --- a/vendor/gimli/examples/dwarfdump.rs +++ /dev/null @@ -1,2369 +0,0 @@ -// Allow clippy lints when building without clippy. -#![allow(unknown_lints)] - -use fallible_iterator::FallibleIterator; -use gimli::{Section, UnitHeader, UnitOffset, UnitSectionOffset, UnitType, UnwindSection}; -use object::{Object, ObjectSection, ObjectSymbol}; -use regex::bytes::Regex; -use std::borrow::{Borrow, Cow}; -use std::cmp::min; -use std::collections::HashMap; -use std::env; -use std::fmt::{self, Debug}; -use std::fs; -use std::io; -use std::io::{BufWriter, Write}; -use std::iter::Iterator; -use std::mem; -use std::process; -use std::result; -use std::sync::{Condvar, Mutex}; -use typed_arena::Arena; - -#[derive(Debug, Clone, Copy, PartialEq, Eq)] -pub enum Error { - GimliError(gimli::Error), - ObjectError(object::read::Error), - IoError, -} - -impl fmt::Display for Error { - #[inline] - fn fmt(&self, f: &mut fmt::Formatter) -> ::std::result::Result<(), fmt::Error> { - Debug::fmt(self, f) - } -} - -fn writeln_error<W: Write, R: Reader>( - w: &mut W, - dwarf: &gimli::Dwarf<R>, - err: Error, - msg: &str, -) -> io::Result<()> { - writeln!( - w, - "{}: {}", - msg, - match err { - Error::GimliError(err) => dwarf.format_error(err), - Error::ObjectError(err) => - format!("{}:{:?}", "An object error occurred while reading", err), - Error::IoError => "An I/O error occurred while writing.".to_string(), - } - ) -} - -impl From<gimli::Error> for Error { - fn from(err: gimli::Error) -> Self { - Error::GimliError(err) - } -} - -impl From<io::Error> for Error { - fn from(_: io::Error) -> Self { - Error::IoError - } -} - -impl From<object::read::Error> for Error { - fn from(err: object::read::Error) -> Self { - Error::ObjectError(err) - } -} - -pub type Result<T> = result::Result<T, Error>; - -fn parallel_output<W, II, F>(w: &mut W, max_workers: usize, iter: II, f: F) -> Result<()> -where - W: Write + Send, - F: Sync + Fn(II::Item, &mut Vec<u8>) -> Result<()>, - II: IntoIterator, - II::IntoIter: Send, -{ - struct ParallelOutputState<I, W> { - iterator: I, - current_worker: usize, - result: Result<()>, - w: W, - } - - let state = Mutex::new(ParallelOutputState { - iterator: iter.into_iter().fuse(), - current_worker: 0, - result: Ok(()), - w, - }); - let workers = min(max_workers, num_cpus::get()); - let mut condvars = Vec::new(); - for _ in 0..workers { - condvars.push(Condvar::new()); - } - { - let state_ref = &state; - let f_ref = &f; - let condvars_ref = &condvars; - crossbeam::scope(|scope| { - for i in 0..workers { - scope.spawn(move |_| { - let mut v = Vec::new(); - let mut lock = state_ref.lock().unwrap(); - while lock.current_worker != i { - lock = condvars_ref[i].wait(lock).unwrap(); - } - loop { - let item = if lock.result.is_ok() { - lock.iterator.next() - } else { - None - }; - lock.current_worker = (i + 1) % workers; - condvars_ref[lock.current_worker].notify_one(); - mem::drop(lock); - - let ret = if let Some(item) = item { - v.clear(); - f_ref(item, &mut v) - } else { - return; - }; - - lock = state_ref.lock().unwrap(); - while lock.current_worker != i { - lock = condvars_ref[i].wait(lock).unwrap(); - } - if lock.result.is_ok() { - let ret2 = lock.w.write_all(&v); - if ret.is_err() { - lock.result = ret; - } else { - lock.result = ret2.map_err(Error::from); - } - } - } - }); - } - }) - .unwrap(); - } - state.into_inner().unwrap().result -} - -trait Reader: gimli::Reader<Offset = usize> + Send + Sync {} - -impl<'input, Endian> Reader for gimli::EndianSlice<'input, Endian> where - Endian: gimli::Endianity + Send + Sync -{ -} - -type RelocationMap = HashMap<usize, object::Relocation>; - -fn add_relocations( - relocations: &mut RelocationMap, - file: &object::File, - section: &object::Section, -) { - for (offset64, mut relocation) in section.relocations() { - let offset = offset64 as usize; - if offset as u64 != offset64 { - continue; - } - let offset = offset as usize; - match relocation.kind() { - object::RelocationKind::Absolute => { - match relocation.target() { - object::RelocationTarget::Symbol(symbol_idx) => { - match file.symbol_by_index(symbol_idx) { - Ok(symbol) => { - let addend = - symbol.address().wrapping_add(relocation.addend() as u64); - relocation.set_addend(addend as i64); - } - Err(_) => { - eprintln!( - "Relocation with invalid symbol for section {} at offset 0x{:08x}", - section.name().unwrap(), - offset - ); - } - } - } - _ => {} - } - if relocations.insert(offset, relocation).is_some() { - eprintln!( - "Multiple relocations for section {} at offset 0x{:08x}", - section.name().unwrap(), - offset - ); - } - } - _ => { - eprintln!( - "Unsupported relocation for section {} at offset 0x{:08x}", - section.name().unwrap(), - offset - ); - } - } - } -} - -/// Apply relocations to addresses and offsets during parsing, -/// instead of requiring the data to be fully relocated prior -/// to parsing. -/// -/// Pros -/// - allows readonly buffers, we don't need to implement writing of values back to buffers -/// - potentially allows us to handle addresses and offsets differently -/// - potentially allows us to add metadata from the relocation (eg symbol names) -/// Cons -/// - maybe incomplete -#[derive(Debug, Clone)] -struct Relocate<'a, R: gimli::Reader<Offset = usize>> { - relocations: &'a RelocationMap, - section: R, - reader: R, -} - -impl<'a, R: gimli::Reader<Offset = usize>> Relocate<'a, R> { - fn relocate(&self, offset: usize, value: u64) -> u64 { - if let Some(relocation) = self.relocations.get(&offset) { - match relocation.kind() { - object::RelocationKind::Absolute => { - if relocation.has_implicit_addend() { - // Use the explicit addend too, because it may have the symbol value. - return value.wrapping_add(relocation.addend() as u64); - } else { - return relocation.addend() as u64; - } - } - _ => {} - } - }; - value - } -} - -impl<'a, R: gimli::Reader<Offset = usize>> gimli::Reader for Relocate<'a, R> { - type Endian = R::Endian; - type Offset = R::Offset; - - fn read_address(&mut self, address_size: u8) -> gimli::Result<u64> { - let offset = self.reader.offset_from(&self.section); - let value = self.reader.read_address(address_size)?; - Ok(self.relocate(offset, value)) - } - - fn read_length(&mut self, format: gimli::Format) -> gimli::Result<usize> { - let offset = self.reader.offset_from(&self.section); - let value = self.reader.read_length(format)?; - <usize as gimli::ReaderOffset>::from_u64(self.relocate(offset, value as u64)) - } - - fn read_offset(&mut self, format: gimli::Format) -> gimli::Result<usize> { - let offset = self.reader.offset_from(&self.section); - let value = self.reader.read_offset(format)?; - <usize as gimli::ReaderOffset>::from_u64(self.relocate(offset, value as u64)) - } - - fn read_sized_offset(&mut self, size: u8) -> gimli::Result<usize> { - let offset = self.reader.offset_from(&self.section); - let value = self.reader.read_sized_offset(size)?; - <usize as gimli::ReaderOffset>::from_u64(self.relocate(offset, value as u64)) - } - - #[inline] - fn split(&mut self, len: Self::Offset) -> gimli::Result<Self> { - let mut other = self.clone(); - other.reader.truncate(len)?; - self.reader.skip(len)?; - Ok(other) - } - - // All remaining methods simply delegate to `self.reader`. - - #[inline] - fn endian(&self) -> Self::Endian { - self.reader.endian() - } - - #[inline] - fn len(&self) -> Self::Offset { - self.reader.len() - } - - #[inline] - fn empty(&mut self) { - self.reader.empty() - } - - #[inline] - fn truncate(&mut self, len: Self::Offset) -> gimli::Result<()> { - self.reader.truncate(len) - } - - #[inline] - fn offset_from(&self, base: &Self) -> Self::Offset { - self.reader.offset_from(&base.reader) - } - - #[inline] - fn offset_id(&self) -> gimli::ReaderOffsetId { - self.reader.offset_id() - } - - #[inline] - fn lookup_offset_id(&self, id: gimli::ReaderOffsetId) -> Option<Self::Offset> { - self.reader.lookup_offset_id(id) - } - - #[inline] - fn find(&self, byte: u8) -> gimli::Result<Self::Offset> { - self.reader.find(byte) - } - - #[inline] - fn skip(&mut self, len: Self::Offset) -> gimli::Result<()> { - self.reader.skip(len) - } - - #[inline] - fn to_slice(&self) -> gimli::Result<Cow<[u8]>> { - self.reader.to_slice() - } - - #[inline] - fn to_string(&self) -> gimli::Result<Cow<str>> { - self.reader.to_string() - } - - #[inline] - fn to_string_lossy(&self) -> gimli::Result<Cow<str>> { - self.reader.to_string_lossy() - } - - #[inline] - fn read_slice(&mut self, buf: &mut [u8]) -> gimli::Result<()> { - self.reader.read_slice(buf) - } -} - -impl<'a, R: Reader> Reader for Relocate<'a, R> {} - -#[derive(Default)] -struct Flags<'a> { - eh_frame: bool, - goff: bool, - info: bool, - line: bool, - pubnames: bool, - pubtypes: bool, - aranges: bool, - dwo: bool, - dwp: bool, - dwo_parent: Option<object::File<'a>>, - sup: Option<object::File<'a>>, - raw: bool, - match_units: Option<Regex>, -} - -fn print_usage(opts: &getopts::Options) -> ! { - let brief = format!("Usage: {} <options> <file>", env::args().next().unwrap()); - write!(&mut io::stderr(), "{}", opts.usage(&brief)).ok(); - process::exit(1); -} - -fn main() { - let mut opts = getopts::Options::new(); - opts.optflag( - "", - "eh-frame", - "print .eh-frame exception handling frame information", - ); - opts.optflag("G", "", "show global die offsets"); - opts.optflag("i", "", "print .debug_info and .debug_types sections"); - opts.optflag("l", "", "print .debug_line section"); - opts.optflag("p", "", "print .debug_pubnames section"); - opts.optflag("r", "", "print .debug_aranges section"); - opts.optflag("y", "", "print .debug_pubtypes section"); - opts.optflag( - "", - "dwo", - "print the .dwo versions of the selected sections", - ); - opts.optflag( - "", - "dwp", - "print the .dwp versions of the selected sections", - ); - opts.optopt( - "", - "dwo-parent", - "use the specified file as the parent of the dwo or dwp (e.g. for .debug_addr)", - "library path", - ); - opts.optflag("", "raw", "print raw data values"); - opts.optopt( - "u", - "match-units", - "print compilation units whose output matches a regex", - "REGEX", - ); - opts.optopt("", "sup", "path to supplementary object file", "PATH"); - - let matches = match opts.parse(env::args().skip(1)) { - Ok(m) => m, - Err(e) => { - writeln!(&mut io::stderr(), "{:?}\n", e).ok(); - print_usage(&opts); - } - }; - if matches.free.is_empty() { - print_usage(&opts); - } - - let mut all = true; - let mut flags = Flags::default(); - if matches.opt_present("eh-frame") { - flags.eh_frame = true; - all = false; - } - if matches.opt_present("G") { - flags.goff = true; - } - if matches.opt_present("i") { - flags.info = true; - all = false; - } - if matches.opt_present("l") { - flags.line = true; - all = false; - } - if matches.opt_present("p") { - flags.pubnames = true; - all = false; - } - if matches.opt_present("y") { - flags.pubtypes = true; - all = false; - } - if matches.opt_present("r") { - flags.aranges = true; - all = false; - } - if matches.opt_present("dwo") { - flags.dwo = true; - } - if matches.opt_present("dwp") { - flags.dwp = true; - } - if matches.opt_present("raw") { - flags.raw = true; - } - if all { - // .eh_frame is excluded even when printing all information. - // cosmetic flags like -G must be set explicitly too. - flags.info = true; - flags.line = true; - flags.pubnames = true; - flags.pubtypes = true; - flags.aranges = true; - } - flags.match_units = if let Some(r) = matches.opt_str("u") { - match Regex::new(&r) { - Ok(r) => Some(r), - Err(e) => { - eprintln!("Invalid regular expression {}: {}", r, e); - process::exit(1); - } - } - } else { - None - }; - - let arena_mmap = Arena::new(); - let load_file = |path| { - let file = match fs::File::open(&path) { - Ok(file) => file, - Err(err) => { - eprintln!("Failed to open file '{}': {}", path, err); - process::exit(1); - } - }; - let mmap = match unsafe { memmap2::Mmap::map(&file) } { - Ok(mmap) => mmap, - Err(err) => { - eprintln!("Failed to map file '{}': {}", path, err); - process::exit(1); - } - }; - let mmap_ref = (*arena_mmap.alloc(mmap)).borrow(); - match object::File::parse(&**mmap_ref) { - Ok(file) => Some(file), - Err(err) => { - eprintln!("Failed to parse file '{}': {}", path, err); - process::exit(1); - } - } - }; - - flags.sup = matches.opt_str("sup").and_then(load_file); - flags.dwo_parent = matches.opt_str("dwo-parent").and_then(load_file); - if flags.dwo_parent.is_some() && !flags.dwo && !flags.dwp { - eprintln!("--dwo-parent also requires --dwo or --dwp"); - process::exit(1); - } - if flags.dwo_parent.is_none() && flags.dwp { - eprintln!("--dwp also requires --dwo-parent"); - process::exit(1); - } - - for file_path in &matches.free { - if matches.free.len() != 1 { - println!("{}", file_path); - println!(); - } - - let file = match fs::File::open(&file_path) { - Ok(file) => file, - Err(err) => { - eprintln!("Failed to open file '{}': {}", file_path, err); - continue; - } - }; - let file = match unsafe { memmap2::Mmap::map(&file) } { - Ok(mmap) => mmap, - Err(err) => { - eprintln!("Failed to map file '{}': {}", file_path, err); - continue; - } - }; - let file = match object::File::parse(&*file) { - Ok(file) => file, - Err(err) => { - eprintln!("Failed to parse file '{}': {}", file_path, err); - continue; - } - }; - - let endian = if file.is_little_endian() { - gimli::RunTimeEndian::Little - } else { - gimli::RunTimeEndian::Big - }; - let ret = dump_file(&file, endian, &flags); - match ret { - Ok(_) => (), - Err(err) => eprintln!("Failed to dump '{}': {}", file_path, err,), - } - } -} - -fn empty_file_section<'input, 'arena, Endian: gimli::Endianity>( - endian: Endian, - arena_relocations: &'arena Arena<RelocationMap>, -) -> Relocate<'arena, gimli::EndianSlice<'arena, Endian>> { - let reader = gimli::EndianSlice::new(&[], endian); - let section = reader; - let relocations = RelocationMap::default(); - let relocations = (*arena_relocations.alloc(relocations)).borrow(); - Relocate { - relocations, - section, - reader, - } -} - -fn load_file_section<'input, 'arena, Endian: gimli::Endianity>( - id: gimli::SectionId, - file: &object::File<'input>, - endian: Endian, - is_dwo: bool, - arena_data: &'arena Arena<Cow<'input, [u8]>>, - arena_relocations: &'arena Arena<RelocationMap>, -) -> Result<Relocate<'arena, gimli::EndianSlice<'arena, Endian>>> { - let mut relocations = RelocationMap::default(); - let name = if is_dwo { - id.dwo_name() - } else if file.format() == object::BinaryFormat::Xcoff { - id.xcoff_name() - } else { - Some(id.name()) - }; - - let data = match name.and_then(|name| file.section_by_name(&name)) { - Some(ref section) => { - // DWO sections never have relocations, so don't bother. - if !is_dwo { - add_relocations(&mut relocations, file, section); - } - section.uncompressed_data()? - } - // Use a non-zero capacity so that `ReaderOffsetId`s are unique. - None => Cow::Owned(Vec::with_capacity(1)), - }; - let data_ref = (*arena_data.alloc(data)).borrow(); - let reader = gimli::EndianSlice::new(data_ref, endian); - let section = reader; - let relocations = (*arena_relocations.alloc(relocations)).borrow(); - Ok(Relocate { - relocations, - section, - reader, - }) -} - -fn dump_file<Endian>(file: &object::File, endian: Endian, flags: &Flags) -> Result<()> -where - Endian: gimli::Endianity + Send + Sync, -{ - let arena_data = Arena::new(); - let arena_relocations = Arena::new(); - - let dwo_parent = if let Some(dwo_parent_file) = flags.dwo_parent.as_ref() { - let mut load_dwo_parent_section = |id: gimli::SectionId| -> Result<_> { - load_file_section( - id, - dwo_parent_file, - endian, - false, - &arena_data, - &arena_relocations, - ) - }; - Some(gimli::Dwarf::load(&mut load_dwo_parent_section)?) - } else { - None - }; - let dwo_parent = dwo_parent.as_ref(); - - let dwo_parent_units = if let Some(dwo_parent) = dwo_parent { - Some( - match dwo_parent - .units() - .map(|unit_header| dwo_parent.unit(unit_header)) - .filter_map(|unit| Ok(unit.dwo_id.map(|dwo_id| (dwo_id, unit)))) - .collect() - { - Ok(units) => units, - Err(err) => { - eprintln!("Failed to process --dwo-parent units: {}", err); - return Ok(()); - } - }, - ) - } else { - None - }; - let dwo_parent_units = dwo_parent_units.as_ref(); - - let mut load_section = |id: gimli::SectionId| -> Result<_> { - load_file_section( - id, - file, - endian, - flags.dwo || flags.dwp, - &arena_data, - &arena_relocations, - ) - }; - - let w = &mut BufWriter::new(io::stdout()); - if flags.dwp { - let empty = empty_file_section(endian, &arena_relocations); - let dwp = gimli::DwarfPackage::load(&mut load_section, empty)?; - dump_dwp(w, &dwp, dwo_parent.unwrap(), dwo_parent_units, flags)?; - w.flush()?; - return Ok(()); - } - - let mut dwarf = gimli::Dwarf::load(&mut load_section)?; - if flags.dwo { - if let Some(dwo_parent) = dwo_parent { - dwarf.make_dwo(&dwo_parent); - } else { - dwarf.file_type = gimli::DwarfFileType::Dwo; - } - } - - if let Some(sup_file) = flags.sup.as_ref() { - let mut load_sup_section = |id: gimli::SectionId| -> Result<_> { - // Note: we really only need the `.debug_str` section, - // but for now we load them all. - load_file_section(id, sup_file, endian, false, &arena_data, &arena_relocations) - }; - dwarf.load_sup(&mut load_sup_section)?; - } - - if flags.eh_frame { - let eh_frame = gimli::EhFrame::load(&mut load_section).unwrap(); - dump_eh_frame(w, file, eh_frame)?; - } - if flags.info { - dump_info(w, &dwarf, dwo_parent_units, flags)?; - dump_types(w, &dwarf, dwo_parent_units, flags)?; - } - if flags.line { - dump_line(w, &dwarf)?; - } - if flags.pubnames { - let debug_pubnames = &gimli::Section::load(&mut load_section).unwrap(); - dump_pubnames(w, debug_pubnames, &dwarf.debug_info)?; - } - if flags.aranges { - let debug_aranges = &gimli::Section::load(&mut load_section).unwrap(); - dump_aranges(w, debug_aranges)?; - } - if flags.pubtypes { - let debug_pubtypes = &gimli::Section::load(&mut load_section).unwrap(); - dump_pubtypes(w, debug_pubtypes, &dwarf.debug_info)?; - } - w.flush()?; - Ok(()) -} - -fn dump_eh_frame<R: Reader, W: Write>( - w: &mut W, - file: &object::File, - mut eh_frame: gimli::EhFrame<R>, -) -> Result<()> { - // TODO: this might be better based on the file format. - let address_size = file - .architecture() - .address_size() - .map(|w| w.bytes()) - .unwrap_or(mem::size_of::<usize>() as u8); - eh_frame.set_address_size(address_size); - - fn register_name_none(_: gimli::Register) -> Option<&'static str> { - None - } - let arch_register_name = match file.architecture() { - object::Architecture::Arm | object::Architecture::Aarch64 => gimli::Arm::register_name, - object::Architecture::I386 => gimli::X86::register_name, - object::Architecture::X86_64 => gimli::X86_64::register_name, - _ => register_name_none, - }; - let register_name = &|register| match arch_register_name(register) { - Some(name) => Cow::Borrowed(name), - None => Cow::Owned(format!("{}", register.0)), - }; - - let mut bases = gimli::BaseAddresses::default(); - if let Some(section) = file.section_by_name(".eh_frame_hdr") { - bases = bases.set_eh_frame_hdr(section.address()); - } - if let Some(section) = file.section_by_name(".eh_frame") { - bases = bases.set_eh_frame(section.address()); - } - if let Some(section) = file.section_by_name(".text") { - bases = bases.set_text(section.address()); - } - if let Some(section) = file.section_by_name(".got") { - bases = bases.set_got(section.address()); - } - - // TODO: Print "__eh_frame" here on macOS, and more generally use the - // section that we're actually looking at, which is what the canonical - // dwarfdump does. - writeln!( - w, - "Exception handling frame information for section .eh_frame" - )?; - - let mut cies = HashMap::new(); - - let mut entries = eh_frame.entries(&bases); - loop { - match entries.next()? { - None => return Ok(()), - Some(gimli::CieOrFde::Cie(cie)) => { - writeln!(w)?; - writeln!(w, "{:#010x}: CIE", cie.offset())?; - writeln!(w, " length: {:#010x}", cie.entry_len())?; - // TODO: CIE_id - writeln!(w, " version: {:#04x}", cie.version())?; - // TODO: augmentation - writeln!(w, " code_align: {}", cie.code_alignment_factor())?; - writeln!(w, " data_align: {}", cie.data_alignment_factor())?; - writeln!( - w, - " ra_register: {}", - register_name(cie.return_address_register()) - )?; - if let Some(encoding) = cie.lsda_encoding() { - writeln!( - w, - " lsda_encoding: {}/{}", - encoding.application(), - encoding.format() - )?; - } - if let Some((encoding, personality)) = cie.personality_with_encoding() { - write!( - w, - " personality: {}/{} ", - encoding.application(), - encoding.format() - )?; - dump_pointer(w, personality)?; - writeln!(w)?; - } - if let Some(encoding) = cie.fde_address_encoding() { - writeln!( - w, - " fde_encoding: {}/{}", - encoding.application(), - encoding.format() - )?; - } - let instructions = cie.instructions(&eh_frame, &bases); - dump_cfi_instructions(w, instructions, true, register_name)?; - writeln!(w)?; - } - Some(gimli::CieOrFde::Fde(partial)) => { - let mut offset = None; - let fde = partial.parse(|_, bases, o| { - offset = Some(o); - cies.entry(o) - .or_insert_with(|| eh_frame.cie_from_offset(bases, o)) - .clone() - })?; - - writeln!(w)?; - writeln!(w, "{:#010x}: FDE", fde.offset())?; - writeln!(w, " length: {:#010x}", fde.entry_len())?; - writeln!(w, " CIE_pointer: {:#010x}", offset.unwrap().0)?; - // TODO: symbolicate the start address like the canonical dwarfdump does. - writeln!(w, " start_addr: {:#018x}", fde.initial_address())?; - writeln!( - w, - " range_size: {:#018x} (end_addr = {:#018x})", - fde.len(), - fde.initial_address() + fde.len() - )?; - if let Some(lsda) = fde.lsda() { - write!(w, " lsda: ")?; - dump_pointer(w, lsda)?; - writeln!(w)?; - } - let instructions = fde.instructions(&eh_frame, &bases); - dump_cfi_instructions(w, instructions, false, register_name)?; - writeln!(w)?; - } - } - } -} - -fn dump_pointer<W: Write>(w: &mut W, p: gimli::Pointer) -> Result<()> { - match p { - gimli::Pointer::Direct(p) => { - write!(w, "{:#018x}", p)?; - } - gimli::Pointer::Indirect(p) => { - write!(w, "({:#018x})", p)?; - } - } - Ok(()) -} - -#[allow(clippy::unneeded_field_pattern)] -fn dump_cfi_instructions<R: Reader, W: Write>( - w: &mut W, - mut insns: gimli::CallFrameInstructionIter<R>, - is_initial: bool, - register_name: &dyn Fn(gimli::Register) -> Cow<'static, str>, -) -> Result<()> { - use gimli::CallFrameInstruction::*; - - // TODO: we need to actually evaluate these instructions as we iterate them - // so we can print the initialized state for CIEs, and each unwind row's - // registers for FDEs. - // - // TODO: We should print DWARF expressions for the CFI instructions that - // embed DWARF expressions within themselves. - - if !is_initial { - writeln!(w, " Instructions:")?; - } - - loop { - match insns.next() { - Err(e) => { - writeln!(w, "Failed to decode CFI instruction: {}", e)?; - return Ok(()); - } - Ok(None) => { - if is_initial { - writeln!(w, " Instructions: Init State:")?; - } - return Ok(()); - } - Ok(Some(op)) => match op { - SetLoc { address } => { - writeln!(w, " DW_CFA_set_loc ({:#x})", address)?; - } - AdvanceLoc { delta } => { - writeln!(w, " DW_CFA_advance_loc ({})", delta)?; - } - DefCfa { register, offset } => { - writeln!( - w, - " DW_CFA_def_cfa ({}, {})", - register_name(register), - offset - )?; - } - DefCfaSf { - register, - factored_offset, - } => { - writeln!( - w, - " DW_CFA_def_cfa_sf ({}, {})", - register_name(register), - factored_offset - )?; - } - DefCfaRegister { register } => { - writeln!( - w, - " DW_CFA_def_cfa_register ({})", - register_name(register) - )?; - } - DefCfaOffset { offset } => { - writeln!(w, " DW_CFA_def_cfa_offset ({})", offset)?; - } - DefCfaOffsetSf { factored_offset } => { - writeln!( - w, - " DW_CFA_def_cfa_offset_sf ({})", - factored_offset - )?; - } - DefCfaExpression { expression: _ } => { - writeln!(w, " DW_CFA_def_cfa_expression (...)")?; - } - Undefined { register } => { - writeln!( - w, - " DW_CFA_undefined ({})", - register_name(register) - )?; - } - SameValue { register } => { - writeln!( - w, - " DW_CFA_same_value ({})", - register_name(register) - )?; - } - Offset { - register, - factored_offset, - } => { - writeln!( - w, - " DW_CFA_offset ({}, {})", - register_name(register), - factored_offset - )?; - } - OffsetExtendedSf { - register, - factored_offset, - } => { - writeln!( - w, - " DW_CFA_offset_extended_sf ({}, {})", - register_name(register), - factored_offset - )?; - } - ValOffset { - register, - factored_offset, - } => { - writeln!( - w, - " DW_CFA_val_offset ({}, {})", - register_name(register), - factored_offset - )?; - } - ValOffsetSf { - register, - factored_offset, - } => { - writeln!( - w, - " DW_CFA_val_offset_sf ({}, {})", - register_name(register), - factored_offset - )?; - } - Register { - dest_register, - src_register, - } => { - writeln!( - w, - " DW_CFA_register ({}, {})", - register_name(dest_register), - register_name(src_register) - )?; - } - Expression { - register, - expression: _, - } => { - writeln!( - w, - " DW_CFA_expression ({}, ...)", - register_name(register) - )?; - } - ValExpression { - register, - expression: _, - } => { - writeln!( - w, - " DW_CFA_val_expression ({}, ...)", - register_name(register) - )?; - } - Restore { register } => { - writeln!( - w, - " DW_CFA_restore ({})", - register_name(register) - )?; - } - RememberState => { - writeln!(w, " DW_CFA_remember_state")?; - } - RestoreState => { - writeln!(w, " DW_CFA_restore_state")?; - } - ArgsSize { size } => { - writeln!(w, " DW_CFA_GNU_args_size ({})", size)?; - } - Nop => { - writeln!(w, " DW_CFA_nop")?; - } - }, - } - } -} - -fn dump_dwp<R: Reader, W: Write + Send>( - w: &mut W, - dwp: &gimli::DwarfPackage<R>, - dwo_parent: &gimli::Dwarf<R>, - dwo_parent_units: Option<&HashMap<gimli::DwoId, gimli::Unit<R>>>, - flags: &Flags, -) -> Result<()> -where - R::Endian: Send + Sync, -{ - if dwp.cu_index.unit_count() != 0 { - writeln!( - w, - "\n.debug_cu_index: version = {}, sections = {}, units = {}, slots = {}", - dwp.cu_index.version(), - dwp.cu_index.section_count(), - dwp.cu_index.unit_count(), - dwp.cu_index.slot_count(), - )?; - for i in 1..=dwp.cu_index.unit_count() { - writeln!(w, "\nCU index {}", i)?; - dump_dwp_sections( - w, - &dwp, - dwo_parent, - dwo_parent_units, - flags, - dwp.cu_index.sections(i)?, - )?; - } - } - - if dwp.tu_index.unit_count() != 0 { - writeln!( - w, - "\n.debug_tu_index: version = {}, sections = {}, units = {}, slots = {}", - dwp.tu_index.version(), - dwp.tu_index.section_count(), - dwp.tu_index.unit_count(), - dwp.tu_index.slot_count(), - )?; - for i in 1..=dwp.tu_index.unit_count() { - writeln!(w, "\nTU index {}", i)?; - dump_dwp_sections( - w, - &dwp, - dwo_parent, - dwo_parent_units, - flags, - dwp.tu_index.sections(i)?, - )?; - } - } - - Ok(()) -} - -fn dump_dwp_sections<R: Reader, W: Write + Send>( - w: &mut W, - dwp: &gimli::DwarfPackage<R>, - dwo_parent: &gimli::Dwarf<R>, - dwo_parent_units: Option<&HashMap<gimli::DwoId, gimli::Unit<R>>>, - flags: &Flags, - sections: gimli::UnitIndexSectionIterator<R>, -) -> Result<()> -where - R::Endian: Send + Sync, -{ - for section in sections.clone() { - writeln!( - w, - " {}: offset = 0x{:x}, size = 0x{:x}", - section.section.dwo_name().unwrap(), - section.offset, - section.size - )?; - } - let dwarf = dwp.sections(sections, dwo_parent)?; - if flags.info { - dump_info(w, &dwarf, dwo_parent_units, flags)?; - dump_types(w, &dwarf, dwo_parent_units, flags)?; - } - if flags.line { - dump_line(w, &dwarf)?; - } - Ok(()) -} - -fn dump_info<R: Reader, W: Write + Send>( - w: &mut W, - dwarf: &gimli::Dwarf<R>, - dwo_parent_units: Option<&HashMap<gimli::DwoId, gimli::Unit<R>>>, - flags: &Flags, -) -> Result<()> -where - R::Endian: Send + Sync, -{ - writeln!(w, "\n.debug_info")?; - - let units = match dwarf.units().collect::<Vec<_>>() { - Ok(units) => units, - Err(err) => { - writeln_error( - w, - dwarf, - Error::GimliError(err), - "Failed to read unit headers", - )?; - return Ok(()); - } - }; - let process_unit = |header: UnitHeader<R>, buf: &mut Vec<u8>| -> Result<()> { - dump_unit(buf, header, dwarf, dwo_parent_units, flags)?; - if !flags - .match_units - .as_ref() - .map(|r| r.is_match(&buf)) - .unwrap_or(true) - { - buf.clear(); - } - Ok(()) - }; - // Don't use more than 16 cores even if available. No point in soaking hundreds - // of cores if you happen to have them. - parallel_output(w, 16, units, process_unit) -} - -fn dump_types<R: Reader, W: Write>( - w: &mut W, - dwarf: &gimli::Dwarf<R>, - dwo_parent_units: Option<&HashMap<gimli::DwoId, gimli::Unit<R>>>, - flags: &Flags, -) -> Result<()> { - writeln!(w, "\n.debug_types")?; - - let mut iter = dwarf.type_units(); - while let Some(header) = iter.next()? { - dump_unit(w, header, dwarf, dwo_parent_units, flags)?; - } - Ok(()) -} - -fn dump_unit<R: Reader, W: Write>( - w: &mut W, - header: UnitHeader<R>, - dwarf: &gimli::Dwarf<R>, - dwo_parent_units: Option<&HashMap<gimli::DwoId, gimli::Unit<R>>>, - flags: &Flags, -) -> Result<()> { - write!(w, "\nUNIT<")?; - match header.offset() { - UnitSectionOffset::DebugInfoOffset(o) => { - write!(w, ".debug_info+0x{:08x}", o.0)?; - } - UnitSectionOffset::DebugTypesOffset(o) => { - write!(w, ".debug_types+0x{:08x}", o.0)?; - } - } - writeln!(w, ">: length = 0x{:x}, format = {:?}, version = {}, address_size = {}, abbrev_offset = 0x{:x}", - header.unit_length(), - header.format(), - header.version(), - header.address_size(), - header.debug_abbrev_offset().0, - )?; - - match header.type_() { - UnitType::Compilation | UnitType::Partial => (), - UnitType::Type { - type_signature, - type_offset, - } - | UnitType::SplitType { - type_signature, - type_offset, - } => { - write!(w, " signature = ")?; - dump_type_signature(w, type_signature)?; - writeln!(w)?; - writeln!(w, " type_offset = 0x{:x}", type_offset.0,)?; - } - UnitType::Skeleton(dwo_id) | UnitType::SplitCompilation(dwo_id) => { - write!(w, " dwo_id = ")?; - writeln!(w, "0x{:016x}", dwo_id.0)?; - } - } - - let mut unit = match dwarf.unit(header) { - Ok(unit) => unit, - Err(err) => { - writeln_error(w, dwarf, err.into(), "Failed to parse unit root entry")?; - return Ok(()); - } - }; - - if let Some(dwo_parent_units) = dwo_parent_units { - if let Some(dwo_id) = unit.dwo_id { - if let Some(parent_unit) = dwo_parent_units.get(&dwo_id) { - unit.copy_relocated_attributes(parent_unit); - } - } - } - - let entries_result = dump_entries(w, unit, dwarf, flags); - if let Err(err) = entries_result { - writeln_error(w, dwarf, err, "Failed to dump entries")?; - } - Ok(()) -} - -fn spaces(buf: &mut String, len: usize) -> &str { - while buf.len() < len { - buf.push(' '); - } - &buf[..len] -} - -// " GOFF=0x{:08x}" adds exactly 16 spaces. -const GOFF_SPACES: usize = 16; - -fn write_offset<R: Reader, W: Write>( - w: &mut W, - unit: &gimli::Unit<R>, - offset: gimli::UnitOffset<R::Offset>, - flags: &Flags, -) -> Result<()> { - write!(w, "<0x{:08x}", offset.0)?; - if flags.goff { - let goff = match offset.to_unit_section_offset(unit) { - UnitSectionOffset::DebugInfoOffset(o) => o.0, - UnitSectionOffset::DebugTypesOffset(o) => o.0, - }; - write!(w, " GOFF=0x{:08x}", goff)?; - } - write!(w, ">")?; - Ok(()) -} - -fn dump_entries<R: Reader, W: Write>( - w: &mut W, - unit: gimli::Unit<R>, - dwarf: &gimli::Dwarf<R>, - flags: &Flags, -) -> Result<()> { - let mut spaces_buf = String::new(); - - let mut entries = unit.entries_raw(None)?; - while !entries.is_empty() { - let offset = entries.next_offset(); - let depth = entries.next_depth(); - let abbrev = entries.read_abbreviation()?; - - let mut indent = if depth >= 0 { - depth as usize * 2 + 2 - } else { - 2 - }; - write!(w, "<{}{}>", if depth < 10 { " " } else { "" }, depth)?; - write_offset(w, &unit, offset, flags)?; - writeln!( - w, - "{}{}", - spaces(&mut spaces_buf, indent), - abbrev.map(|x| x.tag()).unwrap_or(gimli::DW_TAG_null) - )?; - - indent += 18; - if flags.goff { - indent += GOFF_SPACES; - } - - for spec in abbrev.map(|x| x.attributes()).unwrap_or(&[]) { - let attr = entries.read_attribute(*spec)?; - w.write_all(spaces(&mut spaces_buf, indent).as_bytes())?; - if let Some(n) = attr.name().static_string() { - let right_padding = 27 - std::cmp::min(27, n.len()); - write!(w, "{}{} ", n, spaces(&mut spaces_buf, right_padding))?; - } else { - write!(w, "{:27} ", attr.name())?; - } - if flags.raw { - writeln!(w, "{:?}", attr.raw_value())?; - } else { - match dump_attr_value(w, &attr, &unit, dwarf) { - Ok(_) => (), - Err(err) => writeln_error(w, dwarf, err, "Failed to dump attribute value")?, - }; - } - } - } - Ok(()) -} - -fn dump_attr_value<R: Reader, W: Write>( - w: &mut W, - attr: &gimli::Attribute<R>, - unit: &gimli::Unit<R>, - dwarf: &gimli::Dwarf<R>, -) -> Result<()> { - let value = attr.value(); - match value { - gimli::AttributeValue::Addr(address) => { - writeln!(w, "0x{:08x}", address)?; - } - gimli::AttributeValue::Block(data) => { - for byte in data.to_slice()?.iter() { - write!(w, "{:02x}", byte)?; - } - writeln!(w)?; - } - gimli::AttributeValue::Data1(_) - | gimli::AttributeValue::Data2(_) - | gimli::AttributeValue::Data4(_) - | gimli::AttributeValue::Data8(_) => { - if let (Some(udata), Some(sdata)) = (attr.udata_value(), attr.sdata_value()) { - if sdata >= 0 { - writeln!(w, "{}", udata)?; - } else { - writeln!(w, "{} ({})", udata, sdata)?; - } - } else { - writeln!(w, "{:?}", value)?; - } - } - gimli::AttributeValue::Sdata(data) => { - match attr.name() { - gimli::DW_AT_data_member_location => { - writeln!(w, "{}", data)?; - } - _ => { - if data >= 0 { - writeln!(w, "0x{:08x}", data)?; - } else { - writeln!(w, "0x{:08x} ({})", data, data)?; - } - } - }; - } - gimli::AttributeValue::Udata(data) => { - match attr.name() { - gimli::DW_AT_high_pc => { - writeln!(w, "<offset-from-lowpc>{}", data)?; - } - gimli::DW_AT_data_member_location => { - if let Some(sdata) = attr.sdata_value() { - // This is a DW_FORM_data* value. - // libdwarf-dwarfdump displays this as signed too. - if sdata >= 0 { - writeln!(w, "{}", data)?; - } else { - writeln!(w, "{} ({})", data, sdata)?; - } - } else { - writeln!(w, "{}", data)?; - } - } - gimli::DW_AT_lower_bound | gimli::DW_AT_upper_bound => { - writeln!(w, "{}", data)?; - } - _ => { - writeln!(w, "0x{:08x}", data)?; - } - }; - } - gimli::AttributeValue::Exprloc(ref data) => { - if let gimli::AttributeValue::Exprloc(_) = attr.raw_value() { - write!(w, "len 0x{:04x}: ", data.0.len())?; - for byte in data.0.to_slice()?.iter() { - write!(w, "{:02x}", byte)?; - } - write!(w, ": ")?; - } - dump_exprloc(w, unit.encoding(), data)?; - writeln!(w)?; - } - gimli::AttributeValue::Flag(true) => { - writeln!(w, "yes")?; - } - gimli::AttributeValue::Flag(false) => { - writeln!(w, "no")?; - } - gimli::AttributeValue::SecOffset(offset) => { - writeln!(w, "0x{:08x}", offset)?; - } - gimli::AttributeValue::DebugAddrBase(base) => { - writeln!(w, "<.debug_addr+0x{:08x}>", base.0)?; - } - gimli::AttributeValue::DebugAddrIndex(index) => { - write!(w, "(indirect address, index {:#x}): ", index.0)?; - let address = dwarf.address(unit, index)?; - writeln!(w, "0x{:08x}", address)?; - } - gimli::AttributeValue::UnitRef(offset) => { - write!(w, "0x{:08x}", offset.0)?; - match offset.to_unit_section_offset(unit) { - UnitSectionOffset::DebugInfoOffset(goff) => { - write!(w, "<.debug_info+0x{:08x}>", goff.0)?; - } - UnitSectionOffset::DebugTypesOffset(goff) => { - write!(w, "<.debug_types+0x{:08x}>", goff.0)?; - } - } - writeln!(w)?; - } - gimli::AttributeValue::DebugInfoRef(offset) => { - writeln!(w, "<.debug_info+0x{:08x}>", offset.0)?; - } - gimli::AttributeValue::DebugInfoRefSup(offset) => { - writeln!(w, "<.debug_info(sup)+0x{:08x}>", offset.0)?; - } - gimli::AttributeValue::DebugLineRef(offset) => { - writeln!(w, "<.debug_line+0x{:08x}>", offset.0)?; - } - gimli::AttributeValue::LocationListsRef(offset) => { - dump_loc_list(w, offset, unit, dwarf)?; - } - gimli::AttributeValue::DebugLocListsBase(base) => { - writeln!(w, "<.debug_loclists+0x{:08x}>", base.0)?; - } - gimli::AttributeValue::DebugLocListsIndex(index) => { - write!(w, "(indirect location list, index {:#x}): ", index.0)?; - let offset = dwarf.locations_offset(unit, index)?; - dump_loc_list(w, offset, unit, dwarf)?; - } - gimli::AttributeValue::DebugMacinfoRef(offset) => { - writeln!(w, "<.debug_macinfo+0x{:08x}>", offset.0)?; - } - gimli::AttributeValue::DebugMacroRef(offset) => { - writeln!(w, "<.debug_macro+0x{:08x}>", offset.0)?; - } - gimli::AttributeValue::RangeListsRef(offset) => { - let offset = dwarf.ranges_offset_from_raw(unit, offset); - dump_range_list(w, offset, unit, dwarf)?; - } - gimli::AttributeValue::DebugRngListsBase(base) => { - writeln!(w, "<.debug_rnglists+0x{:08x}>", base.0)?; - } - gimli::AttributeValue::DebugRngListsIndex(index) => { - write!(w, "(indirect range list, index {:#x}): ", index.0)?; - let offset = dwarf.ranges_offset(unit, index)?; - dump_range_list(w, offset, unit, dwarf)?; - } - gimli::AttributeValue::DebugTypesRef(signature) => { - dump_type_signature(w, signature)?; - writeln!(w, " <type signature>")?; - } - gimli::AttributeValue::DebugStrRef(offset) => { - if let Ok(s) = dwarf.debug_str.get_str(offset) { - writeln!(w, "{}", s.to_string_lossy()?)?; - } else { - writeln!(w, "<.debug_str+0x{:08x}>", offset.0)?; - } - } - gimli::AttributeValue::DebugStrRefSup(offset) => { - if let Some(s) = dwarf - .sup() - .and_then(|sup| sup.debug_str.get_str(offset).ok()) - { - writeln!(w, "{}", s.to_string_lossy()?)?; - } else { - writeln!(w, "<.debug_str(sup)+0x{:08x}>", offset.0)?; - } - } - gimli::AttributeValue::DebugStrOffsetsBase(base) => { - writeln!(w, "<.debug_str_offsets+0x{:08x}>", base.0)?; - } - gimli::AttributeValue::DebugStrOffsetsIndex(index) => { - write!(w, "(indirect string, index {:#x}): ", index.0)?; - let offset = dwarf.debug_str_offsets.get_str_offset( - unit.encoding().format, - unit.str_offsets_base, - index, - )?; - if let Ok(s) = dwarf.debug_str.get_str(offset) { - writeln!(w, "{}", s.to_string_lossy()?)?; - } else { - writeln!(w, "<.debug_str+0x{:08x}>", offset.0)?; - } - } - gimli::AttributeValue::DebugLineStrRef(offset) => { - if let Ok(s) = dwarf.debug_line_str.get_str(offset) { - writeln!(w, "{}", s.to_string_lossy()?)?; - } else { - writeln!(w, "<.debug_line_str=0x{:08x}>", offset.0)?; - } - } - gimli::AttributeValue::String(s) => { - writeln!(w, "{}", s.to_string_lossy()?)?; - } - gimli::AttributeValue::Encoding(value) => { - writeln!(w, "{}", value)?; - } - gimli::AttributeValue::DecimalSign(value) => { - writeln!(w, "{}", value)?; - } - gimli::AttributeValue::Endianity(value) => { - writeln!(w, "{}", value)?; - } - gimli::AttributeValue::Accessibility(value) => { - writeln!(w, "{}", value)?; - } - gimli::AttributeValue::Visibility(value) => { - writeln!(w, "{}", value)?; - } - gimli::AttributeValue::Virtuality(value) => { - writeln!(w, "{}", value)?; - } - gimli::AttributeValue::Language(value) => { - writeln!(w, "{}", value)?; - } - gimli::AttributeValue::AddressClass(value) => { - writeln!(w, "{}", value)?; - } - gimli::AttributeValue::IdentifierCase(value) => { - writeln!(w, "{}", value)?; - } - gimli::AttributeValue::CallingConvention(value) => { - writeln!(w, "{}", value)?; - } - gimli::AttributeValue::Inline(value) => { - writeln!(w, "{}", value)?; - } - gimli::AttributeValue::Ordering(value) => { - writeln!(w, "{}", value)?; - } - gimli::AttributeValue::FileIndex(value) => { - write!(w, "0x{:08x}", value)?; - dump_file_index(w, value, unit, dwarf)?; - writeln!(w)?; - } - gimli::AttributeValue::DwoId(value) => { - writeln!(w, "0x{:016x}", value.0)?; - } - } - - Ok(()) -} - -fn dump_type_signature<W: Write>(w: &mut W, signature: gimli::DebugTypeSignature) -> Result<()> { - write!(w, "0x{:016x}", signature.0)?; - Ok(()) -} - -fn dump_file_index<R: Reader, W: Write>( - w: &mut W, - file_index: u64, - unit: &gimli::Unit<R>, - dwarf: &gimli::Dwarf<R>, -) -> Result<()> { - if file_index == 0 && unit.header.version() <= 4 { - return Ok(()); - } - let header = match unit.line_program { - Some(ref program) => program.header(), - None => return Ok(()), - }; - let file = match header.file(file_index) { - Some(file) => file, - None => { - writeln!(w, "Unable to get header for file {}", file_index)?; - return Ok(()); - } - }; - write!(w, " ")?; - if let Some(directory) = file.directory(header) { - let directory = dwarf.attr_string(unit, directory)?; - let directory = directory.to_string_lossy()?; - if file.directory_index() != 0 && !directory.starts_with('/') { - if let Some(ref comp_dir) = unit.comp_dir { - write!(w, "{}/", comp_dir.to_string_lossy()?,)?; - } - } - write!(w, "{}/", directory)?; - } - write!( - w, - "{}", - dwarf - .attr_string(unit, file.path_name())? - .to_string_lossy()? - )?; - Ok(()) -} - -fn dump_exprloc<R: Reader, W: Write>( - w: &mut W, - encoding: gimli::Encoding, - data: &gimli::Expression<R>, -) -> Result<()> { - let mut pc = data.0.clone(); - let mut space = false; - while pc.len() != 0 { - let pc_clone = pc.clone(); - match gimli::Operation::parse(&mut pc, encoding) { - Ok(op) => { - if space { - write!(w, " ")?; - } else { - space = true; - } - dump_op(w, encoding, pc_clone, op)?; - } - Err(gimli::Error::InvalidExpression(op)) => { - writeln!(w, "WARNING: unsupported operation 0x{:02x}", op.0)?; - return Ok(()); - } - Err(gimli::Error::UnsupportedRegister(register)) => { - writeln!(w, "WARNING: unsupported register {}", register)?; - return Ok(()); - } - Err(gimli::Error::UnexpectedEof(_)) => { - writeln!(w, "WARNING: truncated or malformed expression")?; - return Ok(()); - } - Err(e) => { - writeln!(w, "WARNING: unexpected operation parse error: {}", e)?; - return Ok(()); - } - } - } - Ok(()) -} - -fn dump_op<R: Reader, W: Write>( - w: &mut W, - encoding: gimli::Encoding, - mut pc: R, - op: gimli::Operation<R>, -) -> Result<()> { - let dwop = gimli::DwOp(pc.read_u8()?); - write!(w, "{}", dwop)?; - match op { - gimli::Operation::Deref { - base_type, size, .. - } => { - if dwop == gimli::DW_OP_deref_size || dwop == gimli::DW_OP_xderef_size { - write!(w, " {}", size)?; - } - if base_type != UnitOffset(0) { - write!(w, " type 0x{:08x}", base_type.0)?; - } - } - gimli::Operation::Pick { index } => { - if dwop == gimli::DW_OP_pick { - write!(w, " {}", index)?; - } - } - gimli::Operation::PlusConstant { value } => { - write!(w, " {}", value as i64)?; - } - gimli::Operation::Bra { target } => { - write!(w, " {}", target)?; - } - gimli::Operation::Skip { target } => { - write!(w, " {}", target)?; - } - gimli::Operation::SignedConstant { value } => match dwop { - gimli::DW_OP_const1s - | gimli::DW_OP_const2s - | gimli::DW_OP_const4s - | gimli::DW_OP_const8s - | gimli::DW_OP_consts => { - write!(w, " {}", value)?; - } - _ => {} - }, - gimli::Operation::UnsignedConstant { value } => match dwop { - gimli::DW_OP_const1u - | gimli::DW_OP_const2u - | gimli::DW_OP_const4u - | gimli::DW_OP_const8u - | gimli::DW_OP_constu => { - write!(w, " {}", value)?; - } - _ => { - // These have the value encoded in the operation, eg DW_OP_lit0. - } - }, - gimli::Operation::Register { register } => { - if dwop == gimli::DW_OP_regx { - write!(w, " {}", register.0)?; - } - } - gimli::Operation::RegisterOffset { - register, - offset, - base_type, - } => { - if dwop >= gimli::DW_OP_breg0 && dwop <= gimli::DW_OP_breg31 { - write!(w, "{:+}", offset)?; - } else { - write!(w, " {}", register.0)?; - if offset != 0 { - write!(w, "{:+}", offset)?; - } - if base_type != UnitOffset(0) { - write!(w, " type 0x{:08x}", base_type.0)?; - } - } - } - gimli::Operation::FrameOffset { offset } => { - write!(w, " {}", offset)?; - } - gimli::Operation::Call { offset } => match offset { - gimli::DieReference::UnitRef(gimli::UnitOffset(offset)) => { - write!(w, " 0x{:08x}", offset)?; - } - gimli::DieReference::DebugInfoRef(gimli::DebugInfoOffset(offset)) => { - write!(w, " 0x{:08x}", offset)?; - } - }, - gimli::Operation::Piece { - size_in_bits, - bit_offset: None, - } => { - write!(w, " {}", size_in_bits / 8)?; - } - gimli::Operation::Piece { - size_in_bits, - bit_offset: Some(bit_offset), - } => { - write!(w, " 0x{:08x} offset 0x{:08x}", size_in_bits, bit_offset)?; - } - gimli::Operation::ImplicitValue { data } => { - let data = data.to_slice()?; - write!(w, " 0x{:08x} contents 0x", data.len())?; - for byte in data.iter() { - write!(w, "{:02x}", byte)?; - } - } - gimli::Operation::ImplicitPointer { value, byte_offset } => { - write!(w, " 0x{:08x} {}", value.0, byte_offset)?; - } - gimli::Operation::EntryValue { expression } => { - write!(w, "(")?; - dump_exprloc(w, encoding, &gimli::Expression(expression))?; - write!(w, ")")?; - } - gimli::Operation::ParameterRef { offset } => { - write!(w, " 0x{:08x}", offset.0)?; - } - gimli::Operation::Address { address } => { - write!(w, " 0x{:08x}", address)?; - } - gimli::Operation::AddressIndex { index } => { - write!(w, " 0x{:08x}", index.0)?; - } - gimli::Operation::ConstantIndex { index } => { - write!(w, " 0x{:08x}", index.0)?; - } - gimli::Operation::TypedLiteral { base_type, value } => { - write!(w, " type 0x{:08x} contents 0x", base_type.0)?; - for byte in value.to_slice()?.iter() { - write!(w, "{:02x}", byte)?; - } - } - gimli::Operation::Convert { base_type } => { - write!(w, " type 0x{:08x}", base_type.0)?; - } - gimli::Operation::Reinterpret { base_type } => { - write!(w, " type 0x{:08x}", base_type.0)?; - } - gimli::Operation::WasmLocal { index } - | gimli::Operation::WasmGlobal { index } - | gimli::Operation::WasmStack { index } => { - let wasmop = pc.read_u8()?; - write!(w, " 0x{:x} 0x{:x}", wasmop, index)?; - } - gimli::Operation::Drop - | gimli::Operation::Swap - | gimli::Operation::Rot - | gimli::Operation::Abs - | gimli::Operation::And - | gimli::Operation::Div - | gimli::Operation::Minus - | gimli::Operation::Mod - | gimli::Operation::Mul - | gimli::Operation::Neg - | gimli::Operation::Not - | gimli::Operation::Or - | gimli::Operation::Plus - | gimli::Operation::Shl - | gimli::Operation::Shr - | gimli::Operation::Shra - | gimli::Operation::Xor - | gimli::Operation::Eq - | gimli::Operation::Ge - | gimli::Operation::Gt - | gimli::Operation::Le - | gimli::Operation::Lt - | gimli::Operation::Ne - | gimli::Operation::Nop - | gimli::Operation::PushObjectAddress - | gimli::Operation::TLS - | gimli::Operation::CallFrameCFA - | gimli::Operation::StackValue => {} - }; - Ok(()) -} - -fn dump_range<W: Write>(w: &mut W, range: Option<gimli::Range>) -> Result<()> { - if let Some(range) = range { - write!(w, " [0x{:08x}, 0x{:08x}]", range.begin, range.end)?; - } else { - write!(w, " [ignored]")?; - } - Ok(()) -} - -fn dump_loc_list<R: Reader, W: Write>( - w: &mut W, - offset: gimli::LocationListsOffset<R::Offset>, - unit: &gimli::Unit<R>, - dwarf: &gimli::Dwarf<R>, -) -> Result<()> { - let mut locations = dwarf.locations(unit, offset)?; - writeln!( - w, - "<loclist at {}+0x{:08x}>", - if unit.encoding().version < 5 { - ".debug_loc" - } else { - ".debug_loclists" - }, - offset.0, - )?; - let mut i = 0; - while let Some(raw) = locations.next_raw()? { - write!(w, "\t\t\t[{:2}]", i)?; - i += 1; - let range = locations - .convert_raw(raw.clone())? - .map(|location| location.range); - match raw { - gimli::RawLocListEntry::BaseAddress { addr } => { - writeln!(w, "<base-address 0x{:08x}>", addr)?; - } - gimli::RawLocListEntry::BaseAddressx { addr } => { - let addr_val = dwarf.address(unit, addr)?; - writeln!(w, "<base-addressx [{}]0x{:08x}>", addr.0, addr_val)?; - } - gimli::RawLocListEntry::StartxEndx { - begin, - end, - ref data, - } => { - let begin_val = dwarf.address(unit, begin)?; - let end_val = dwarf.address(unit, end)?; - write!( - w, - "<startx-endx [{}]0x{:08x}, [{}]0x{:08x}>", - begin.0, begin_val, end.0, end_val, - )?; - dump_range(w, range)?; - dump_exprloc(w, unit.encoding(), data)?; - writeln!(w)?; - } - gimli::RawLocListEntry::StartxLength { - begin, - length, - ref data, - } => { - let begin_val = dwarf.address(unit, begin)?; - write!( - w, - "<startx-length [{}]0x{:08x}, 0x{:08x}>", - begin.0, begin_val, length, - )?; - dump_range(w, range)?; - dump_exprloc(w, unit.encoding(), data)?; - writeln!(w)?; - } - gimli::RawLocListEntry::AddressOrOffsetPair { - begin, - end, - ref data, - } - | gimli::RawLocListEntry::OffsetPair { - begin, - end, - ref data, - } => { - write!(w, "<offset-pair 0x{:08x}, 0x{:08x}>", begin, end)?; - dump_range(w, range)?; - dump_exprloc(w, unit.encoding(), data)?; - writeln!(w)?; - } - gimli::RawLocListEntry::DefaultLocation { ref data } => { - write!(w, "<default location>")?; - dump_exprloc(w, unit.encoding(), data)?; - writeln!(w)?; - } - gimli::RawLocListEntry::StartEnd { - begin, - end, - ref data, - } => { - write!(w, "<start-end 0x{:08x}, 0x{:08x}>", begin, end)?; - dump_range(w, range)?; - dump_exprloc(w, unit.encoding(), data)?; - writeln!(w)?; - } - gimli::RawLocListEntry::StartLength { - begin, - length, - ref data, - } => { - write!(w, "<start-length 0x{:08x}, 0x{:08x}>", begin, length)?; - dump_range(w, range)?; - dump_exprloc(w, unit.encoding(), data)?; - writeln!(w)?; - } - }; - } - Ok(()) -} - -fn dump_range_list<R: Reader, W: Write>( - w: &mut W, - offset: gimli::RangeListsOffset<R::Offset>, - unit: &gimli::Unit<R>, - dwarf: &gimli::Dwarf<R>, -) -> Result<()> { - let mut ranges = dwarf.ranges(unit, offset)?; - writeln!( - w, - "<rnglist at {}+0x{:08x}>", - if unit.encoding().version < 5 { - ".debug_ranges" - } else { - ".debug_rnglists" - }, - offset.0, - )?; - let mut i = 0; - while let Some(raw) = ranges.next_raw()? { - write!(w, "\t\t\t[{:2}] ", i)?; - i += 1; - let range = ranges.convert_raw(raw.clone())?; - match raw { - gimli::RawRngListEntry::BaseAddress { addr } => { - writeln!(w, "<new base address 0x{:08x}>", addr)?; - } - gimli::RawRngListEntry::BaseAddressx { addr } => { - let addr_val = dwarf.address(unit, addr)?; - writeln!(w, "<new base addressx [{}]0x{:08x}>", addr.0, addr_val)?; - } - gimli::RawRngListEntry::StartxEndx { begin, end } => { - let begin_val = dwarf.address(unit, begin)?; - let end_val = dwarf.address(unit, end)?; - write!( - w, - "<startx-endx [{}]0x{:08x}, [{}]0x{:08x}>", - begin.0, begin_val, end.0, end_val, - )?; - dump_range(w, range)?; - writeln!(w)?; - } - gimli::RawRngListEntry::StartxLength { begin, length } => { - let begin_val = dwarf.address(unit, begin)?; - write!( - w, - "<startx-length [{}]0x{:08x}, 0x{:08x}>", - begin.0, begin_val, length, - )?; - dump_range(w, range)?; - writeln!(w)?; - } - gimli::RawRngListEntry::AddressOrOffsetPair { begin, end } - | gimli::RawRngListEntry::OffsetPair { begin, end } => { - write!(w, "<offset-pair 0x{:08x}, 0x{:08x}>", begin, end)?; - dump_range(w, range)?; - writeln!(w)?; - } - gimli::RawRngListEntry::StartEnd { begin, end } => { - write!(w, "<start-end 0x{:08x}, 0x{:08x}>", begin, end)?; - dump_range(w, range)?; - writeln!(w)?; - } - gimli::RawRngListEntry::StartLength { begin, length } => { - write!(w, "<start-length 0x{:08x}, 0x{:08x}>", begin, length)?; - dump_range(w, range)?; - writeln!(w)?; - } - }; - } - Ok(()) -} - -fn dump_line<R: Reader, W: Write>(w: &mut W, dwarf: &gimli::Dwarf<R>) -> Result<()> { - let mut iter = dwarf.units(); - while let Some(header) = iter.next()? { - writeln!( - w, - "\n.debug_line: line number info for unit at .debug_info offset 0x{:08x}", - header.offset().as_debug_info_offset().unwrap().0 - )?; - let unit = match dwarf.unit(header) { - Ok(unit) => unit, - Err(err) => { - writeln_error( - w, - dwarf, - err.into(), - "Failed to parse unit root entry for dump_line", - )?; - continue; - } - }; - match dump_line_program(w, &unit, dwarf) { - Ok(_) => (), - Err(Error::IoError) => return Err(Error::IoError), - Err(err) => writeln_error(w, dwarf, err, "Failed to dump line program")?, - } - } - Ok(()) -} - -fn dump_line_program<R: Reader, W: Write>( - w: &mut W, - unit: &gimli::Unit<R>, - dwarf: &gimli::Dwarf<R>, -) -> Result<()> { - if let Some(program) = unit.line_program.clone() { - { - let header = program.header(); - writeln!(w)?; - writeln!( - w, - "Offset: 0x{:x}", - header.offset().0 - )?; - writeln!( - w, - "Length: {}", - header.unit_length() - )?; - writeln!( - w, - "DWARF version: {}", - header.version() - )?; - writeln!( - w, - "Address size: {}", - header.address_size() - )?; - writeln!( - w, - "Prologue length: {}", - header.header_length() - )?; - writeln!( - w, - "Minimum instruction length: {}", - header.minimum_instruction_length() - )?; - writeln!( - w, - "Maximum operations per instruction: {}", - header.maximum_operations_per_instruction() - )?; - writeln!( - w, - "Default is_stmt: {}", - header.default_is_stmt() - )?; - writeln!( - w, - "Line base: {}", - header.line_base() - )?; - writeln!( - w, - "Line range: {}", - header.line_range() - )?; - writeln!( - w, - "Opcode base: {}", - header.opcode_base() - )?; - - writeln!(w)?; - writeln!(w, "Opcodes:")?; - for (i, length) in header - .standard_opcode_lengths() - .to_slice()? - .iter() - .enumerate() - { - writeln!(w, " Opcode {} has {} args", i + 1, length)?; - } - - let base = if header.version() >= 5 { 0 } else { 1 }; - writeln!(w)?; - writeln!(w, "The Directory Table:")?; - for (i, dir) in header.include_directories().iter().enumerate() { - writeln!( - w, - " {} {}", - base + i, - dwarf.attr_string(unit, dir.clone())?.to_string_lossy()? - )?; - } - - writeln!(w)?; - writeln!(w, "The File Name Table")?; - write!(w, " Entry\tDir\tTime\tSize")?; - if header.file_has_md5() { - write!(w, "\tMD5\t\t\t\t")?; - } - writeln!(w, "\tName")?; - for (i, file) in header.file_names().iter().enumerate() { - write!( - w, - " {}\t{}\t{}\t{}", - base + i, - file.directory_index(), - file.timestamp(), - file.size(), - )?; - if header.file_has_md5() { - let md5 = file.md5(); - write!(w, "\t")?; - for i in 0..16 { - write!(w, "{:02X}", md5[i])?; - } - } - writeln!( - w, - "\t{}", - dwarf - .attr_string(unit, file.path_name())? - .to_string_lossy()? - )?; - } - - writeln!(w)?; - writeln!(w, "Line Number Instructions:")?; - let mut instructions = header.instructions(); - while let Some(instruction) = instructions.next_instruction(header)? { - writeln!(w, " {}", instruction)?; - } - - writeln!(w)?; - writeln!(w, "Line Number Rows:")?; - writeln!(w, "<pc> [lno,col]")?; - } - let mut rows = program.rows(); - let mut file_index = std::u64::MAX; - while let Some((header, row)) = rows.next_row()? { - let line = match row.line() { - Some(line) => line.get(), - None => 0, - }; - let column = match row.column() { - gimli::ColumnType::Column(column) => column.get(), - gimli::ColumnType::LeftEdge => 0, - }; - write!(w, "0x{:08x} [{:4},{:2}]", row.address(), line, column)?; - if row.is_stmt() { - write!(w, " NS")?; - } - if row.basic_block() { - write!(w, " BB")?; - } - if row.end_sequence() { - write!(w, " ET")?; - } - if row.prologue_end() { - write!(w, " PE")?; - } - if row.epilogue_begin() { - write!(w, " EB")?; - } - if row.isa() != 0 { - write!(w, " IS={}", row.isa())?; - } - if row.discriminator() != 0 { - write!(w, " DI={}", row.discriminator())?; - } - if file_index != row.file_index() { - file_index = row.file_index(); - if let Some(file) = row.file(header) { - if let Some(directory) = file.directory(header) { - write!( - w, - " uri: \"{}/{}\"", - dwarf.attr_string(unit, directory)?.to_string_lossy()?, - dwarf - .attr_string(unit, file.path_name())? - .to_string_lossy()? - )?; - } else { - write!( - w, - " uri: \"{}\"", - dwarf - .attr_string(unit, file.path_name())? - .to_string_lossy()? - )?; - } - } - } - writeln!(w)?; - } - } - Ok(()) -} - -fn dump_pubnames<R: Reader, W: Write>( - w: &mut W, - debug_pubnames: &gimli::DebugPubNames<R>, - debug_info: &gimli::DebugInfo<R>, -) -> Result<()> { - writeln!(w, "\n.debug_pubnames")?; - - let mut cu_offset; - let mut cu_die_offset = gimli::DebugInfoOffset(0); - let mut prev_cu_offset = None; - let mut pubnames = debug_pubnames.items(); - while let Some(pubname) = pubnames.next()? { - cu_offset = pubname.unit_header_offset(); - if Some(cu_offset) != prev_cu_offset { - let cu = debug_info.header_from_offset(cu_offset)?; - cu_die_offset = gimli::DebugInfoOffset(cu_offset.0 + cu.header_size()); - prev_cu_offset = Some(cu_offset); - } - let die_in_cu = pubname.die_offset(); - let die_in_sect = cu_offset.0 + die_in_cu.0; - writeln!(w, - "global die-in-sect 0x{:08x}, cu-in-sect 0x{:08x}, die-in-cu 0x{:08x}, cu-header-in-sect 0x{:08x} '{}'", - die_in_sect, - cu_die_offset.0, - die_in_cu.0, - cu_offset.0, - pubname.name().to_string_lossy()? - )?; - } - Ok(()) -} - -fn dump_pubtypes<R: Reader, W: Write>( - w: &mut W, - debug_pubtypes: &gimli::DebugPubTypes<R>, - debug_info: &gimli::DebugInfo<R>, -) -> Result<()> { - writeln!(w, "\n.debug_pubtypes")?; - - let mut cu_offset; - let mut cu_die_offset = gimli::DebugInfoOffset(0); - let mut prev_cu_offset = None; - let mut pubtypes = debug_pubtypes.items(); - while let Some(pubtype) = pubtypes.next()? { - cu_offset = pubtype.unit_header_offset(); - if Some(cu_offset) != prev_cu_offset { - let cu = debug_info.header_from_offset(cu_offset)?; - cu_die_offset = gimli::DebugInfoOffset(cu_offset.0 + cu.header_size()); - prev_cu_offset = Some(cu_offset); - } - let die_in_cu = pubtype.die_offset(); - let die_in_sect = cu_offset.0 + die_in_cu.0; - writeln!(w, - "pubtype die-in-sect 0x{:08x}, cu-in-sect 0x{:08x}, die-in-cu 0x{:08x}, cu-header-in-sect 0x{:08x} '{}'", - die_in_sect, - cu_die_offset.0, - die_in_cu.0, - cu_offset.0, - pubtype.name().to_string_lossy()? - )?; - } - Ok(()) -} - -fn dump_aranges<R: Reader, W: Write>( - w: &mut W, - debug_aranges: &gimli::DebugAranges<R>, -) -> Result<()> { - writeln!(w, "\n.debug_aranges")?; - - let mut headers = debug_aranges.headers(); - while let Some(header) = headers.next()? { - writeln!( - w, - "Address Range Header: length = 0x{:08x}, version = 0x{:04x}, cu_offset = 0x{:08x}, addr_size = 0x{:02x}, seg_size = 0x{:02x}", - header.length(), - header.encoding().version, - header.debug_info_offset().0, - header.encoding().address_size, - header.segment_size(), - )?; - let mut aranges = header.entries(); - while let Some(arange) = aranges.next()? { - let range = arange.range(); - if let Some(segment) = arange.segment() { - writeln!( - w, - "[0x{:016x}, 0x{:016x}) segment 0x{:x}", - range.begin, range.end, segment - )?; - } else { - writeln!(w, "[0x{:016x}, 0x{:016x})", range.begin, range.end)?; - } - } - } - Ok(()) -} diff --git a/vendor/gimli/examples/simple.rs b/vendor/gimli/examples/simple.rs deleted file mode 100644 index 7c958d45c..000000000 --- a/vendor/gimli/examples/simple.rs +++ /dev/null @@ -1,67 +0,0 @@ -//! A simple example of parsing `.debug_info`. - -use object::{Object, ObjectSection}; -use std::{borrow, env, fs}; - -fn main() { - for path in env::args().skip(1) { - let file = fs::File::open(&path).unwrap(); - let mmap = unsafe { memmap2::Mmap::map(&file).unwrap() }; - let object = object::File::parse(&*mmap).unwrap(); - let endian = if object.is_little_endian() { - gimli::RunTimeEndian::Little - } else { - gimli::RunTimeEndian::Big - }; - dump_file(&object, endian).unwrap(); - } -} - -fn dump_file(object: &object::File, endian: gimli::RunTimeEndian) -> Result<(), gimli::Error> { - // Load a section and return as `Cow<[u8]>`. - let load_section = |id: gimli::SectionId| -> Result<borrow::Cow<[u8]>, gimli::Error> { - match object.section_by_name(id.name()) { - Some(ref section) => Ok(section - .uncompressed_data() - .unwrap_or(borrow::Cow::Borrowed(&[][..]))), - None => Ok(borrow::Cow::Borrowed(&[][..])), - } - }; - - // Load all of the sections. - let dwarf_cow = gimli::Dwarf::load(&load_section)?; - - // Borrow a `Cow<[u8]>` to create an `EndianSlice`. - let borrow_section: &dyn for<'a> Fn( - &'a borrow::Cow<[u8]>, - ) -> gimli::EndianSlice<'a, gimli::RunTimeEndian> = - &|section| gimli::EndianSlice::new(&*section, endian); - - // Create `EndianSlice`s for all of the sections. - let dwarf = dwarf_cow.borrow(&borrow_section); - - // Iterate over the compilation units. - let mut iter = dwarf.units(); - while let Some(header) = iter.next()? { - println!( - "Unit at <.debug_info+0x{:x}>", - header.offset().as_debug_info_offset().unwrap().0 - ); - let unit = dwarf.unit(header)?; - - // Iterate over the Debugging Information Entries (DIEs) in the unit. - let mut depth = 0; - let mut entries = unit.entries(); - while let Some((delta_depth, entry)) = entries.next_dfs()? { - depth += delta_depth; - println!("<{}><{:x}> {}", depth, entry.offset().0, entry.tag()); - - // Iterate over the attributes in the DIE. - let mut attrs = entry.attrs(); - while let Some(attr) = attrs.next()? { - println!(" {}: {:?}", attr.name(), attr.value()); - } - } - } - Ok(()) -} diff --git a/vendor/gimli/examples/simple_line.rs b/vendor/gimli/examples/simple_line.rs deleted file mode 100644 index 87b224cda..000000000 --- a/vendor/gimli/examples/simple_line.rs +++ /dev/null @@ -1,106 +0,0 @@ -//! A simple example of parsing `.debug_line`. - -use object::{Object, ObjectSection}; -use std::{borrow, env, fs, path}; - -fn main() { - for path in env::args().skip(1) { - let file = fs::File::open(&path).unwrap(); - let mmap = unsafe { memmap2::Mmap::map(&file).unwrap() }; - let object = object::File::parse(&*mmap).unwrap(); - let endian = if object.is_little_endian() { - gimli::RunTimeEndian::Little - } else { - gimli::RunTimeEndian::Big - }; - dump_file(&object, endian).unwrap(); - } -} - -fn dump_file(object: &object::File, endian: gimli::RunTimeEndian) -> Result<(), gimli::Error> { - // Load a section and return as `Cow<[u8]>`. - let load_section = |id: gimli::SectionId| -> Result<borrow::Cow<[u8]>, gimli::Error> { - match object.section_by_name(id.name()) { - Some(ref section) => Ok(section - .uncompressed_data() - .unwrap_or(borrow::Cow::Borrowed(&[][..]))), - None => Ok(borrow::Cow::Borrowed(&[][..])), - } - }; - - // Load all of the sections. - let dwarf_cow = gimli::Dwarf::load(&load_section)?; - - // Borrow a `Cow<[u8]>` to create an `EndianSlice`. - let borrow_section: &dyn for<'a> Fn( - &'a borrow::Cow<[u8]>, - ) -> gimli::EndianSlice<'a, gimli::RunTimeEndian> = - &|section| gimli::EndianSlice::new(&*section, endian); - - // Create `EndianSlice`s for all of the sections. - let dwarf = dwarf_cow.borrow(&borrow_section); - - // Iterate over the compilation units. - let mut iter = dwarf.units(); - while let Some(header) = iter.next()? { - println!( - "Line number info for unit at <.debug_info+0x{:x}>", - header.offset().as_debug_info_offset().unwrap().0 - ); - let unit = dwarf.unit(header)?; - - // Get the line program for the compilation unit. - if let Some(program) = unit.line_program.clone() { - let comp_dir = if let Some(ref dir) = unit.comp_dir { - path::PathBuf::from(dir.to_string_lossy().into_owned()) - } else { - path::PathBuf::new() - }; - - // Iterate over the line program rows. - let mut rows = program.rows(); - while let Some((header, row)) = rows.next_row()? { - if row.end_sequence() { - // End of sequence indicates a possible gap in addresses. - println!("{:x} end-sequence", row.address()); - } else { - // Determine the path. Real applications should cache this for performance. - let mut path = path::PathBuf::new(); - if let Some(file) = row.file(header) { - path = comp_dir.clone(); - - // The directory index 0 is defined to correspond to the compilation unit directory. - if file.directory_index() != 0 { - if let Some(dir) = file.directory(header) { - path.push( - dwarf.attr_string(&unit, dir)?.to_string_lossy().as_ref(), - ); - } - } - - path.push( - dwarf - .attr_string(&unit, file.path_name())? - .to_string_lossy() - .as_ref(), - ); - } - - // Determine line/column. DWARF line/column is never 0, so we use that - // but other applications may want to display this differently. - let line = match row.line() { - Some(line) => line.get(), - None => 0, - }; - let column = match row.column() { - gimli::ColumnType::LeftEdge => 0, - gimli::ColumnType::Column(column) => column.get(), - }; - - println!("{:x} {}:{}:{}", row.address(), path.display(), line, column); - } - } - } - } - Ok(()) -} |