Skip to content

Commit 5792840

Browse files
committed
Prevent __rust_begin_short_backtrace frames from being tail-call optimised away
1 parent 4d43423 commit 5792840

File tree

6 files changed

+74
-12
lines changed

6 files changed

+74
-12
lines changed

library/std/src/panicking.rs

+7-2
Original file line numberDiff line numberDiff line change
@@ -434,7 +434,9 @@ pub fn begin_panic_handler(info: &PanicInfo<'_>) -> ! {
434434

435435
let loc = info.location().unwrap(); // The current implementation always returns Some
436436
let msg = info.message().unwrap(); // The current implementation always returns Some
437-
rust_panic_with_hook(&mut PanicPayload::new(msg), info.message(), loc);
437+
crate::sys_common::backtrace::__rust_end_short_backtrace(move || {
438+
rust_panic_with_hook(&mut PanicPayload::new(msg), info.message(), loc);
439+
})
438440
}
439441

440442
/// This is the entry point of panicking for the non-format-string variants of
@@ -453,7 +455,10 @@ pub fn begin_panic<M: Any + Send>(msg: M) -> ! {
453455
intrinsics::abort()
454456
}
455457

456-
rust_panic_with_hook(&mut PanicPayload::new(msg), None, Location::caller());
458+
let loc = Location::caller();
459+
return crate::sys_common::backtrace::__rust_end_short_backtrace(move || {
460+
rust_panic_with_hook(&mut PanicPayload::new(msg), None, loc)
461+
});
457462

458463
struct PanicPayload<A> {
459464
inner: Option<A>,

library/std/src/rt.rs

+6-4
Original file line numberDiff line numberDiff line change
@@ -48,9 +48,7 @@ fn lang_start_internal(
4848
sys::args::init(argc, argv);
4949

5050
// Let's run some code!
51-
let exit_code = panic::catch_unwind(|| {
52-
sys_common::backtrace::__rust_begin_short_backtrace(move || main())
53-
});
51+
let exit_code = panic::catch_unwind(main);
5452

5553
sys_common::cleanup();
5654
exit_code.unwrap_or(101) as isize
@@ -64,5 +62,9 @@ fn lang_start<T: crate::process::Termination + 'static>(
6462
argc: isize,
6563
argv: *const *const u8,
6664
) -> isize {
67-
lang_start_internal(&move || main().report(), argc, argv)
65+
lang_start_internal(
66+
&move || crate::sys_common::backtrace::__rust_begin_short_backtrace(main).report(),
67+
argc,
68+
argv,
69+
)
6870
}

library/std/src/sys_common/backtrace.rs

+34-5
Original file line numberDiff line numberDiff line change
@@ -74,6 +74,8 @@ unsafe fn _print_fmt(fmt: &mut fmt::Formatter<'_>, print_fmt: PrintFmt) -> fmt::
7474
bt_fmt.add_context()?;
7575
let mut idx = 0;
7676
let mut res = Ok(());
77+
// Start immediately if we're not using a short backtrace.
78+
let mut start = print_fmt != PrintFmt::Short;
7779
backtrace_rs::trace_unsynchronized(|frame| {
7880
if print_fmt == PrintFmt::Short && idx > MAX_NB_FRAMES {
7981
return false;
@@ -89,16 +91,24 @@ unsafe fn _print_fmt(fmt: &mut fmt::Formatter<'_>, print_fmt: PrintFmt) -> fmt::
8991
stop = true;
9092
return;
9193
}
94+
if sym.contains("__rust_end_short_backtrace") {
95+
start = true;
96+
return;
97+
}
9298
}
9399
}
94100

95-
res = bt_fmt.frame().symbol(frame, symbol);
101+
if start {
102+
res = bt_fmt.frame().symbol(frame, symbol);
103+
}
96104
});
97105
if stop {
98106
return false;
99107
}
100108
if !hit {
101-
res = bt_fmt.frame().print_raw(frame.ip(), None, None, None);
109+
if start {
110+
res = bt_fmt.frame().print_raw(frame.ip(), None, None, None);
111+
}
102112
}
103113

104114
idx += 1;
@@ -123,10 +133,29 @@ unsafe fn _print_fmt(fmt: &mut fmt::Formatter<'_>, print_fmt: PrintFmt) -> fmt::
123133
pub fn __rust_begin_short_backtrace<F, T>(f: F) -> T
124134
where
125135
F: FnOnce() -> T,
126-
F: Send,
127-
T: Send,
128136
{
129-
f()
137+
let result = f();
138+
139+
// prevent this frame from being tail-call optimised away
140+
crate::hint::black_box(());
141+
142+
result
143+
}
144+
145+
/// Fixed frame used to clean the backtrace with `RUST_BACKTRACE=1`. Note that
146+
/// this is only inline(never) when backtraces in libstd are enabled, otherwise
147+
/// it's fine to optimize away.
148+
#[cfg_attr(feature = "backtrace", inline(never))]
149+
pub fn __rust_end_short_backtrace<F, T>(f: F) -> T
150+
where
151+
F: FnOnce() -> T,
152+
{
153+
let result = f();
154+
155+
// prevent this frame from being tail-call optimised away
156+
crate::hint::black_box(());
157+
158+
result
130159
}
131160

132161
pub enum RustBacktrace {

library/test/src/lib.rs

+4-1
Original file line numberDiff line numberDiff line change
@@ -514,7 +514,10 @@ pub fn run_test(
514514
/// Fixed frame used to clean the backtrace with `RUST_BACKTRACE=1`.
515515
#[inline(never)]
516516
fn __rust_begin_short_backtrace<F: FnOnce()>(f: F) {
517-
f()
517+
f();
518+
519+
// prevent this frame from being tail-call optimised away
520+
black_box(());
518521
}
519522

520523
fn run_test_in_process(
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
// Regression test for #47429: short backtraces were not terminating correctly
2+
3+
// compile-flags: -O
4+
// run-fail
5+
// check-run-results
6+
// exec-env:RUST_BACKTRACE=1
7+
8+
// ignore-msvc see #62897 and `backtrace-debuginfo.rs` test
9+
// ignore-android FIXME #17520
10+
// ignore-cloudabi spawning processes is not supported
11+
// ignore-openbsd no support for libbacktrace without filename
12+
// ignore-wasm no panic or subprocess support
13+
// ignore-emscripten no panic or subprocess support
14+
// ignore-sgx no subprocess support
15+
16+
fn main() {
17+
panic!()
18+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
thread 'main' panicked at 'explicit panic', $DIR/issue-47429-short-backtraces.rs:17:5
2+
stack backtrace:
3+
0: std::panicking::begin_panic
4+
1: issue_47429_short_backtraces::main
5+
note: Some details are omitted, run with `RUST_BACKTRACE=full` for a verbose backtrace.

0 commit comments

Comments
 (0)