]> git.lizzy.rs Git - rust.git/blob - tests/run-pass/backtrace-api.rs
Auto merge of #1575 - Aaron1011:fix/macro-backtrace, r=oli-obk
[rust.git] / tests / run-pass / backtrace-api.rs
1 // normalize-stderr-test ".*/(rust|checkout)/library/" -> "RUSTLIB/"
2 // normalize-stderr-test "RUSTLIB/(.*):\d+:\d+ "-> "RUSTLIB/$1:LL:COL "
3 // normalize-stderr-test "::<.*>" -> ""
4
5 extern "Rust" {
6     fn miri_get_backtrace(flags: u64) -> Box<[*mut ()]>;
7     fn miri_resolve_frame(ptr: *mut (), flags: u64) -> MiriFrame;
8 }
9
10 #[derive(Debug)]
11 #[repr(C)]
12 struct MiriFrame {
13     name: Box<[u8]>,
14     filename: Box<[u8]>,
15     lineno: u32,
16     colno: u32
17 }
18
19 #[inline(never)] fn func_a() -> Box<[*mut ()]> { func_b::<u8>() }
20 #[inline(never)] fn func_b<T>() -> Box<[*mut ()]> { func_c() }
21
22 macro_rules! invoke_func_d {
23     () => { func_d() }
24 }
25
26 #[inline(never)] fn func_c() -> Box<[*mut ()]> { invoke_func_d!() }
27 #[inline(never)] fn func_d() -> Box<[*mut ()]> { unsafe { miri_get_backtrace(0) } }
28
29 fn main() {
30     let mut seen_main = false;
31     let frames = func_a();
32     for frame in frames.into_iter() {
33         let miri_frame = unsafe { miri_resolve_frame(*frame, 0) };
34         let name = String::from_utf8(miri_frame.name.into()).unwrap();
35         let filename = String::from_utf8(miri_frame.filename.into()).unwrap();
36
37         // Print every frame to stderr.
38         let out = format!("{}:{}:{} ({})", filename, miri_frame.lineno, miri_frame.colno, name);
39         eprintln!("{}", out);
40         // Print the 'main' frame (and everything before it) to stdout, skipping
41         // the printing of internal (and possibly fragile) libstd frames.
42         if !seen_main {
43             println!("{}", out);
44             seen_main = name == "main";
45         }
46     }
47 }