mirror of
https://github.com/italicsjenga/agb.git
synced 2025-01-09 08:31:33 +11:00
Print the entire backtrace
This commit is contained in:
parent
8453b46eab
commit
3f374f3e9c
|
@ -1,6 +1,6 @@
|
||||||
use std::{fs, path::PathBuf, str::FromStr};
|
use std::{fs, path::PathBuf};
|
||||||
|
|
||||||
use addr2line::object;
|
use addr2line::{gimli, object};
|
||||||
use clap::Parser;
|
use clap::Parser;
|
||||||
|
|
||||||
#[derive(Parser, Debug)]
|
#[derive(Parser, Debug)]
|
||||||
|
@ -35,9 +35,20 @@ fn main() -> anyhow::Result<()> {
|
||||||
|
|
||||||
let ctx = addr2line::Context::new(&object)?;
|
let ctx = addr2line::Context::new(&object)?;
|
||||||
|
|
||||||
let mut frames = ctx
|
for address in cli.dump.split('-') {
|
||||||
.find_frames(parse_address(&cli.dump)?)
|
let mut address = u64::from_str_radix(address, 16)?;
|
||||||
.skip_all_loads()?;
|
if address <= 0xFFFF {
|
||||||
|
address += 0x0800_0000;
|
||||||
|
}
|
||||||
|
|
||||||
|
print_address(&ctx, address)?;
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
fn print_address(ctx: &addr2line::Context<impl gimli::Reader>, address: u64) -> anyhow::Result<()> {
|
||||||
|
let mut frames = ctx.find_frames(address).skip_all_loads()?;
|
||||||
|
|
||||||
while let Some(frame) = frames.next()? {
|
while let Some(frame) = frames.next()? {
|
||||||
let function_name = if let Some(func) = frame.function {
|
let function_name = if let Some(func) = frame.function {
|
||||||
|
@ -54,19 +65,8 @@ fn main() -> anyhow::Result<()> {
|
||||||
})
|
})
|
||||||
.unwrap_or_default();
|
.unwrap_or_default();
|
||||||
|
|
||||||
println!(
|
println!("{function_name} ({}:{})", location.filename, location.line);
|
||||||
"{}:{} ({})",
|
|
||||||
location.filename, location.line, function_name
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
}
|
}
|
||||||
|
|
||||||
fn parse_address(input: &str) -> Result<u64, <u64 as FromStr>::Err> {
|
|
||||||
if let Some(input) = input.strip_prefix("0x") {
|
|
||||||
u64::from_str_radix(input, 16)
|
|
||||||
} else {
|
|
||||||
input.parse()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
|
@ -9,8 +9,8 @@ struct Context {
|
||||||
registers: [u32; 11],
|
registers: [u32; 11],
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct Frame {
|
pub struct Frames {
|
||||||
pub address: u32,
|
frames: Vec<u32>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[allow(unused)]
|
#[allow(unused)]
|
||||||
|
@ -37,7 +37,7 @@ impl Index<Register> for Context {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[inline(never)]
|
#[inline(never)]
|
||||||
pub(crate) fn unwind_exception() -> Vec<Frame> {
|
pub(crate) fn unwind_exception() -> Frames {
|
||||||
let mut context = Context::default();
|
let mut context = Context::default();
|
||||||
|
|
||||||
unsafe {
|
unsafe {
|
||||||
|
@ -77,10 +77,33 @@ pub(crate) fn unwind_exception() -> Vec<Frame> {
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
frames.push(Frame { address: lr });
|
frames.push(lr);
|
||||||
|
|
||||||
frame_pointer = sp;
|
frame_pointer = sp;
|
||||||
}
|
}
|
||||||
|
|
||||||
frames
|
Frames { frames }
|
||||||
|
}
|
||||||
|
|
||||||
|
impl core::fmt::Display for Frames {
|
||||||
|
fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
|
||||||
|
let mut is_first = true;
|
||||||
|
|
||||||
|
for frame in &self.frames {
|
||||||
|
if !is_first {
|
||||||
|
write!(f, "-")?;
|
||||||
|
}
|
||||||
|
|
||||||
|
if frame & 0xFFFF_0000 == 0x0800_0000 {
|
||||||
|
let frame = frame & 0xFFFF;
|
||||||
|
write!(f, "{frame:x}")?;
|
||||||
|
} else {
|
||||||
|
write!(f, "{frame:x}")?;
|
||||||
|
}
|
||||||
|
|
||||||
|
is_first = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -323,12 +323,10 @@ pub mod test_runner {
|
||||||
if let Some(mut mgba) = mgba::Mgba::new() {
|
if let Some(mut mgba) = mgba::Mgba::new() {
|
||||||
let _ = mgba.print(format_args!("[failed]"), mgba::DebugLevel::Error);
|
let _ = mgba.print(format_args!("[failed]"), mgba::DebugLevel::Error);
|
||||||
|
|
||||||
for frame in frames {
|
let _ = mgba.print(
|
||||||
let _ = mgba.print(
|
format_args!("debug data: {frames}"),
|
||||||
format_args!("{:#08x}", frame.address),
|
mgba::DebugLevel::Error,
|
||||||
mgba::DebugLevel::Error,
|
);
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
let _ = mgba.print(format_args!("Error: {info}"), mgba::DebugLevel::Fatal);
|
let _ = mgba.print(format_args!("Error: {info}"), mgba::DebugLevel::Fatal);
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue