summaryrefslogtreecommitdiffstats
path: root/vendor/addr2line/examples
diff options
context:
space:
mode:
Diffstat (limited to 'vendor/addr2line/examples')
-rw-r--r--vendor/addr2line/examples/addr2line.rs128
1 files changed, 78 insertions, 50 deletions
diff --git a/vendor/addr2line/examples/addr2line.rs b/vendor/addr2line/examples/addr2line.rs
index fa4d8e457..97fc80222 100644
--- a/vendor/addr2line/examples/addr2line.rs
+++ b/vendor/addr2line/examples/addr2line.rs
@@ -1,17 +1,9 @@
-extern crate addr2line;
-extern crate clap;
-extern crate fallible_iterator;
-extern crate gimli;
-extern crate memmap2;
-extern crate object;
-extern crate typed_arena;
-
use std::borrow::Cow;
use std::fs::File;
use std::io::{BufRead, Lines, StdinLock, Write};
-use std::path::Path;
+use std::path::{Path, PathBuf};
-use clap::{Arg, Command, Values};
+use clap::{Arg, ArgAction, Command};
use fallible_iterator::FallibleIterator;
use object::{Object, ObjectSection, SymbolMap, SymbolMapName};
use typed_arena::Arena;
@@ -27,7 +19,7 @@ fn parse_uint_from_hex_string(string: &str) -> Option<u64> {
}
enum Addrs<'a> {
- Args(Values<'a>),
+ Args(clap::parser::ValuesRef<'a, String>),
Stdin(Lines<StdinLock<'a>>),
}
@@ -45,8 +37,8 @@ impl<'a> Iterator for Addrs<'a> {
}
}
-fn print_loc(loc: Option<&Location>, basenames: bool, llvm: bool) {
- if let Some(ref loc) = loc {
+fn print_loc(loc: Option<&Location<'_>>, basenames: bool, llvm: bool) {
+ if let Some(loc) = loc {
if let Some(ref file) = loc.file.as_ref() {
let path = if basenames {
Path::new(Path::new(file).file_name().unwrap())
@@ -102,21 +94,35 @@ fn load_file_section<'input, 'arena, Endian: gimli::Endianity>(
}
fn find_name_from_symbols<'a>(
- symbols: &'a SymbolMap<SymbolMapName>,
+ symbols: &'a SymbolMap<SymbolMapName<'_>>,
probe: u64,
) -> Option<&'a str> {
symbols.get(probe).map(|x| x.name())
}
+struct Options<'a> {
+ do_functions: bool,
+ do_inlines: bool,
+ pretty: bool,
+ print_addrs: bool,
+ basenames: bool,
+ demangle: bool,
+ llvm: bool,
+ exe: &'a PathBuf,
+ sup: Option<&'a PathBuf>,
+}
+
fn main() {
let matches = Command::new("addr2line")
- .version("0.1")
+ .version(env!("CARGO_PKG_VERSION"))
.about("A fast addr2line Rust port")
+ .max_term_width(100)
.args(&[
Arg::new("exe")
.short('e')
.long("exe")
.value_name("filename")
+ .value_parser(clap::value_parser!(PathBuf))
.help(
"Specify the name of the executable for which addresses should be translated.",
)
@@ -124,52 +130,66 @@ fn main() {
Arg::new("sup")
.long("sup")
.value_name("filename")
+ .value_parser(clap::value_parser!(PathBuf))
.help("Path to supplementary object file."),
Arg::new("functions")
.short('f')
.long("functions")
+ .action(ArgAction::SetTrue)
.help("Display function names as well as file and line number information."),
- Arg::new("pretty").short('p').long("pretty-print").help(
+ Arg::new("pretty").short('p').long("pretty-print")
+ .action(ArgAction::SetTrue)
+ .help(
"Make the output more human friendly: each location are printed on one line.",
),
- Arg::new("inlines").short('i').long("inlines").help(
+ Arg::new("inlines").short('i').long("inlines")
+ .action(ArgAction::SetTrue)
+ .help(
"If the address belongs to a function that was inlined, the source information for \
all enclosing scopes back to the first non-inlined function will also be printed.",
),
- Arg::new("addresses").short('a').long("addresses").help(
+ Arg::new("addresses").short('a').long("addresses")
+ .action(ArgAction::SetTrue)
+ .help(
"Display the address before the function name, file and line number information.",
),
Arg::new("basenames")
.short('s')
.long("basenames")
+ .action(ArgAction::SetTrue)
.help("Display only the base of each file name."),
- Arg::new("demangle").short('C').long("demangle").help(
+ Arg::new("demangle").short('C').long("demangle")
+ .action(ArgAction::SetTrue)
+ .help(
"Demangle function names. \
Specifying a specific demangling style (like GNU addr2line) is not supported. \
(TODO)"
),
Arg::new("llvm")
.long("llvm")
+ .action(ArgAction::SetTrue)
.help("Display output in the same format as llvm-symbolizer."),
Arg::new("addrs")
- .takes_value(true)
- .multiple_occurrences(true)
+ .action(ArgAction::Append)
.help("Addresses to use instead of reading from stdin."),
])
.get_matches();
let arena_data = Arena::new();
- let do_functions = matches.is_present("functions");
- let do_inlines = matches.is_present("inlines");
- let pretty = matches.is_present("pretty");
- let print_addrs = matches.is_present("addresses");
- let basenames = matches.is_present("basenames");
- let demangle = matches.is_present("demangle");
- let llvm = matches.is_present("llvm");
- let path = matches.value_of("exe").unwrap();
+ let opts = Options {
+ do_functions: matches.get_flag("functions"),
+ do_inlines: matches.get_flag("inlines"),
+ pretty: matches.get_flag("pretty"),
+ print_addrs: matches.get_flag("addresses"),
+ basenames: matches.get_flag("basenames"),
+ demangle: matches.get_flag("demangle"),
+ llvm: matches.get_flag("llvm"),
+ exe: matches.get_one::<PathBuf>("exe").unwrap(),
+ sup: matches.get_one::<PathBuf>("sup"),
+ };
- let file = File::open(path).unwrap();
+ let file = File::open(opts.exe).unwrap();
let map = unsafe { memmap2::Mmap::map(&file).unwrap() };
let object = &object::File::parse(&*map).unwrap();
@@ -184,7 +204,7 @@ fn main() {
};
let sup_map;
- let sup_object = if let Some(sup_path) = matches.value_of("sup") {
+ let sup_object = if let Some(sup_path) = opts.sup {
let sup_file = File::open(sup_path).unwrap();
sup_map = unsafe { memmap2::Mmap::map(&sup_file).unwrap() };
Some(object::File::parse(&*sup_map).unwrap())
@@ -201,87 +221,95 @@ fn main() {
dwarf.load_sup(&mut load_sup_section).unwrap();
}
+ let mut split_dwarf_loader = addr2line::builtin_split_dwarf_loader::SplitDwarfLoader::new(
+ |data, endian| {
+ gimli::EndianSlice::new(arena_data.alloc(Cow::Owned(data.into_owned())), endian)
+ },
+ Some(opts.exe.clone()),
+ );
let ctx = Context::from_dwarf(dwarf).unwrap();
let stdin = std::io::stdin();
let addrs = matches
- .values_of("addrs")
+ .get_many::<String>("addrs")
.map(Addrs::Args)
.unwrap_or_else(|| Addrs::Stdin(stdin.lock().lines()));
for probe in addrs {
- if print_addrs {
+ if opts.print_addrs {
let addr = probe.unwrap_or(0);
- if llvm {
+ if opts.llvm {
print!("0x{:x}", addr);
} else {
print!("0x{:016x}", addr);
}
- if pretty {
+ if opts.pretty {
print!(": ");
} else {
println!();
}
}
- if do_functions || do_inlines {
+ if opts.do_functions || opts.do_inlines {
let mut printed_anything = false;
if let Some(probe) = probe {
- let mut frames = ctx.find_frames(probe).unwrap().enumerate();
+ let frames = ctx.find_frames(probe);
+ let frames = split_dwarf_loader.run(frames).unwrap();
+ let mut frames = frames.enumerate();
while let Some((i, frame)) = frames.next().unwrap() {
- if pretty && i != 0 {
+ if opts.pretty && i != 0 {
print!(" (inlined by) ");
}
- if do_functions {
+ if opts.do_functions {
if let Some(func) = frame.function {
print_function(
func.raw_name().ok().as_ref().map(AsRef::as_ref),
func.language,
- demangle,
+ opts.demangle,
);
} else {
let name = find_name_from_symbols(&symbols, probe);
- print_function(name, None, demangle);
+ print_function(name, None, opts.demangle);
}
- if pretty {
+ if opts.pretty {
print!(" at ");
} else {
println!();
}
}
- print_loc(frame.location.as_ref(), basenames, llvm);
+ print_loc(frame.location.as_ref(), opts.basenames, opts.llvm);
printed_anything = true;
- if !do_inlines {
+ if !opts.do_inlines {
break;
}
}
}
if !printed_anything {
- if do_functions {
+ if opts.do_functions {
let name = probe.and_then(|probe| find_name_from_symbols(&symbols, probe));
- print_function(name, None, demangle);
+ print_function(name, None, opts.demangle);
- if pretty {
+ if opts.pretty {
print!(" at ");
} else {
println!();
}
}
- print_loc(None, basenames, llvm);
+ print_loc(None, opts.basenames, opts.llvm);
}
} else {
let loc = probe.and_then(|probe| ctx.find_location(probe).unwrap());
- print_loc(loc.as_ref(), basenames, llvm);
+ print_loc(loc.as_ref(), opts.basenames, opts.llvm);
}
- if llvm {
+ if opts.llvm {
println!();
}
std::io::stdout().flush().unwrap();