Skip to content

Commit 501befb

Browse files
committed
Auto merge of rust-lang#115974 - m-ou-se:panicinfo-and-panicinfo, r=<try>
Split core's PanicInfo and std's PanicInfo This experimental PR is based on some thoughts I was having here: rust-lang#115561 (comment) --- `PanicInfo` is used in two ways: 1. As argument to the `#[panic_handler]` in `no_std` context. 2. As argument to the [panic hook](https://doc.rust-lang.org/stable/std/panic/fn.set_hook.html) in `std` context. In situation 1, the `PanicInfo` always has a *message* (of type `fmt::Arguments`), but never a *payload* (of type `&dyn Any`). In situation 2, the `PanicInfo` always has a *payload* (which is often a `String`), but not always a *message*. Having these as the same type is annoying. It means we can't add `.message()` to the first one without also finding a way to properly support it on the second one. (Which is what rust-lang#66745 is blocked on.) It also means that, because the implementation is in `core`, the implementation cannot make use of the `String` type (which doesn't exist in `core`): https://github.com/rust-lang/rust/blob/0692db1a9082380e027f354912229dfd6af37e78/library/core/src/panic/panic_info.rs#L171-L172 This also means that we cannot easily add a useful method like `PanicInfo::payload_as_str() -> Option<&str>` that works for both `&'static str` and `String` payloads. I don't see any good reasons for these to be the same type, other than historical reasons. This PR is an experiment to see what happens if we make 1 and 2 separate types. To try to avoid breaking existing code, they are still named `core::panic::PanicInfo` and `std::panic::PanicInfo`, in the hope that people write the former when used in `#[panic_handler]` (since then they'd be using `no_std`) and the latter when used in a panic hook (since then they're definitely using `std`). Let's see how much explodes. :)
2 parents 0692db1 + 6707f29 commit 501befb

File tree

4 files changed

+177
-96
lines changed

4 files changed

+177
-96
lines changed

library/core/src/panic/panic_info.rs

+7-72
Original file line numberDiff line numberDiff line change
@@ -1,31 +1,14 @@
1-
use crate::any::Any;
21
use crate::fmt;
32
use crate::panic::Location;
43

54
/// A struct providing information about a panic.
65
///
7-
/// `PanicInfo` structure is passed to a panic hook set by the [`set_hook`]
8-
/// function.
9-
///
10-
/// [`set_hook`]: ../../std/panic/fn.set_hook.html
11-
///
12-
/// # Examples
13-
///
14-
/// ```should_panic
15-
/// use std::panic;
16-
///
17-
/// panic::set_hook(Box::new(|panic_info| {
18-
/// println!("panic occurred: {panic_info}");
19-
/// }));
20-
///
21-
/// panic!("critical system failure");
22-
/// ```
6+
/// A `PanicInfo` structure is passed to the panic handler defined by `#[panic_handler]`.
237
#[lang = "panic_info"]
248
#[stable(feature = "panic_hooks", since = "1.10.0")]
259
#[derive(Debug)]
2610
pub struct PanicInfo<'a> {
27-
payload: &'a (dyn Any + Send),
28-
message: Option<&'a fmt::Arguments<'a>>,
11+
message: fmt::Arguments<'a>,
2912
location: &'a Location<'a>,
3013
can_unwind: bool,
3114
force_no_backtrace: bool,
@@ -40,59 +23,20 @@ impl<'a> PanicInfo<'a> {
4023
#[doc(hidden)]
4124
#[inline]
4225
pub fn internal_constructor(
43-
message: Option<&'a fmt::Arguments<'a>>,
26+
message: fmt::Arguments<'a>,
4427
location: &'a Location<'a>,
4528
can_unwind: bool,
4629
force_no_backtrace: bool,
4730
) -> Self {
48-
struct NoPayload;
49-
PanicInfo { location, message, payload: &NoPayload, can_unwind, force_no_backtrace }
50-
}
51-
52-
#[unstable(
53-
feature = "panic_internals",
54-
reason = "internal details of the implementation of the `panic!` and related macros",
55-
issue = "none"
56-
)]
57-
#[doc(hidden)]
58-
#[inline]
59-
pub fn set_payload(&mut self, info: &'a (dyn Any + Send)) {
60-
self.payload = info;
61-
}
62-
63-
/// Returns the payload associated with the panic.
64-
///
65-
/// This will commonly, but not always, be a `&'static str` or [`String`].
66-
///
67-
/// [`String`]: ../../std/string/struct.String.html
68-
///
69-
/// # Examples
70-
///
71-
/// ```should_panic
72-
/// use std::panic;
73-
///
74-
/// panic::set_hook(Box::new(|panic_info| {
75-
/// if let Some(s) = panic_info.payload().downcast_ref::<&str>() {
76-
/// println!("panic occurred: {s:?}");
77-
/// } else {
78-
/// println!("panic occurred");
79-
/// }
80-
/// }));
81-
///
82-
/// panic!("Normal panic");
83-
/// ```
84-
#[must_use]
85-
#[stable(feature = "panic_hooks", since = "1.10.0")]
86-
pub fn payload(&self) -> &(dyn Any + Send) {
87-
self.payload
31+
PanicInfo { location, message, can_unwind, force_no_backtrace }
8832
}
8933

9034
/// If the `panic!` macro from the `core` crate (not from `std`)
9135
/// was used with a formatting string and some additional arguments,
9236
/// returns that message ready to be used for example with [`fmt::write`]
9337
#[must_use]
9438
#[unstable(feature = "panic_info_message", issue = "66745")]
95-
pub fn message(&self) -> Option<&fmt::Arguments<'_>> {
39+
pub fn message(&self) -> fmt::Arguments<'_> {
9640
self.message
9741
}
9842

@@ -161,17 +105,8 @@ impl fmt::Display for PanicInfo<'_> {
161105
fn fmt(&self, formatter: &mut fmt::Formatter<'_>) -> fmt::Result {
162106
formatter.write_str("panicked at ")?;
163107
self.location.fmt(formatter)?;
164-
if let Some(message) = self.message {
165-
formatter.write_str(":\n")?;
166-
formatter.write_fmt(*message)?;
167-
} else if let Some(payload) = self.payload.downcast_ref::<&'static str>() {
168-
formatter.write_str(":\n")?;
169-
formatter.write_str(payload)?;
170-
}
171-
// NOTE: we cannot use downcast_ref::<String>() here
172-
// since String is not available in core!
173-
// The payload is a String when `std::panic!` is called with multiple arguments,
174-
// but in that case the message is also available.
108+
formatter.write_str(":\n")?;
109+
formatter.write_fmt(self.message)?;
175110
Ok(())
176111
}
177112
}

library/core/src/panicking.rs

+2-2
Original file line numberDiff line numberDiff line change
@@ -62,7 +62,7 @@ pub const fn panic_fmt(fmt: fmt::Arguments<'_>) -> ! {
6262
}
6363

6464
let pi = PanicInfo::internal_constructor(
65-
Some(&fmt),
65+
fmt,
6666
Location::caller(),
6767
/* can_unwind */ true,
6868
/* force_no_backtrace */ false,
@@ -96,7 +96,7 @@ pub fn panic_nounwind_fmt(fmt: fmt::Arguments<'_>, force_no_backtrace: bool) ->
9696

9797
// PanicInfo with the `can_unwind` flag set to false forces an abort.
9898
let pi = PanicInfo::internal_constructor(
99-
Some(&fmt),
99+
fmt,
100100
Location::caller(),
101101
/* can_unwind */ false,
102102
force_no_backtrace,

library/std/src/panic.rs

+152-1
Original file line numberDiff line numberDiff line change
@@ -4,11 +4,162 @@
44

55
use crate::any::Any;
66
use crate::collections;
7+
use crate::fmt;
78
use crate::panicking;
89
use crate::sync::atomic::{AtomicUsize, Ordering};
910
use crate::sync::{Mutex, RwLock};
1011
use crate::thread::Result;
1112

13+
/// A struct providing information about a panic.
14+
///
15+
/// `PanicInfo` structure is passed to a panic hook set by the [`set_hook`]
16+
/// function.
17+
///
18+
/// [`set_hook`]: ../../std/panic/fn.set_hook.html
19+
///
20+
/// # Examples
21+
///
22+
/// ```should_panic
23+
/// use std::panic;
24+
///
25+
/// panic::set_hook(Box::new(|panic_info| {
26+
/// println!("panic occurred: {panic_info}");
27+
/// }));
28+
///
29+
/// panic!("critical system failure");
30+
/// ```
31+
#[stable(feature = "panic_hooks", since = "1.10.0")]
32+
#[derive(Debug)]
33+
pub struct PanicInfo<'a> {
34+
payload: &'a (dyn Any + Send),
35+
location: &'a Location<'a>,
36+
can_unwind: bool,
37+
force_no_backtrace: bool,
38+
}
39+
40+
impl<'a> PanicInfo<'a> {
41+
#[unstable(feature = "panic_internals", issue = "none")]
42+
#[doc(hidden)]
43+
#[inline]
44+
pub fn internal_constructor(
45+
location: &'a Location<'a>,
46+
can_unwind: bool,
47+
force_no_backtrace: bool,
48+
) -> Self {
49+
struct NoPayload;
50+
PanicInfo { payload: &NoPayload, location, can_unwind, force_no_backtrace }
51+
}
52+
53+
#[unstable(feature = "panic_internals", issue = "none")]
54+
#[doc(hidden)]
55+
#[inline]
56+
pub fn set_payload(&mut self, info: &'a (dyn Any + Send)) {
57+
self.payload = info;
58+
}
59+
60+
/// Returns the payload associated with the panic.
61+
///
62+
/// This will commonly, but not always, be a `&'static str` or [`String`].
63+
///
64+
/// [`String`]: ../../std/string/struct.String.html
65+
///
66+
/// # Examples
67+
///
68+
/// ```should_panic
69+
/// use std::panic;
70+
///
71+
/// panic::set_hook(Box::new(|panic_info| {
72+
/// if let Some(s) = panic_info.payload().downcast_ref::<&str>() {
73+
/// println!("panic occurred: {s:?}");
74+
/// } else {
75+
/// println!("panic occurred");
76+
/// }
77+
/// }));
78+
///
79+
/// panic!("Normal panic");
80+
/// ```
81+
#[must_use]
82+
#[stable(feature = "panic_hooks", since = "1.10.0")]
83+
pub fn payload(&self) -> &(dyn Any + Send) {
84+
self.payload
85+
}
86+
87+
/// Returns information about the location from which the panic originated,
88+
/// if available.
89+
///
90+
/// This method will currently always return [`Some`], but this may change
91+
/// in future versions.
92+
///
93+
/// # Examples
94+
///
95+
/// ```should_panic
96+
/// use std::panic;
97+
///
98+
/// panic::set_hook(Box::new(|panic_info| {
99+
/// if let Some(location) = panic_info.location() {
100+
/// println!("panic occurred in file '{}' at line {}",
101+
/// location.file(),
102+
/// location.line(),
103+
/// );
104+
/// } else {
105+
/// println!("panic occurred but can't get location information...");
106+
/// }
107+
/// }));
108+
///
109+
/// panic!("Normal panic");
110+
/// ```
111+
#[must_use]
112+
#[stable(feature = "panic_hooks", since = "1.10.0")]
113+
pub fn location(&self) -> Option<&Location<'_>> {
114+
// NOTE: If this is changed to sometimes return None,
115+
// deal with that case in std::panicking::default_hook and core::panicking::panic_fmt.
116+
Some(&self.location)
117+
}
118+
119+
/// Returns whether the panic handler is allowed to unwind the stack from
120+
/// the point where the panic occurred.
121+
///
122+
/// This is true for most kinds of panics with the exception of panics
123+
/// caused by trying to unwind out of a `Drop` implementation or a function
124+
/// whose ABI does not support unwinding.
125+
///
126+
/// It is safe for a panic handler to unwind even when this function returns
127+
/// false, however this will simply cause the panic handler to be called
128+
/// again.
129+
#[must_use]
130+
#[unstable(feature = "panic_can_unwind", issue = "92988")]
131+
pub fn can_unwind(&self) -> bool {
132+
self.can_unwind
133+
}
134+
135+
#[unstable(
136+
feature = "panic_internals",
137+
reason = "internal details of the implementation of the `panic!` and related macros",
138+
issue = "none"
139+
)]
140+
#[doc(hidden)]
141+
#[inline]
142+
pub fn force_no_backtrace(&self) -> bool {
143+
self.force_no_backtrace
144+
}
145+
}
146+
147+
#[stable(feature = "panic_hook_display", since = "1.26.0")]
148+
impl fmt::Display for PanicInfo<'_> {
149+
fn fmt(&self, formatter: &mut fmt::Formatter<'_>) -> fmt::Result {
150+
formatter.write_str("panicked at ")?;
151+
self.location.fmt(formatter)?;
152+
if let Some(payload) = self.payload.downcast_ref::<&'static str>() {
153+
formatter.write_str(":\n")?;
154+
formatter.write_str(payload)?;
155+
} else if let Some(payload) = self.payload.downcast_ref::<String>() {
156+
formatter.write_str(":\n")?;
157+
formatter.write_str(payload)?;
158+
}
159+
Ok(())
160+
}
161+
}
162+
12163
#[doc(hidden)]
13164
#[unstable(feature = "edition_panic", issue = "none", reason = "use panic!() instead")]
14165
#[allow_internal_unstable(libstd_sys_internals, const_format_args, core_panic, rt)]
@@ -43,7 +194,7 @@ pub use crate::panicking::{set_hook, take_hook};
43194
pub use crate::panicking::update_hook;
44195

45196
#[stable(feature = "panic_hooks", since = "1.10.0")]
46-
pub use core::panic::{Location, PanicInfo};
197+
pub use core::panic::Location;
47198

48199
#[stable(feature = "catch_unwind", since = "1.9.0")]
49200
pub use core::panic::{AssertUnwindSafe, RefUnwindSafe, UnwindSafe};

library/std/src/panicking.rs

+16-21
Original file line numberDiff line numberDiff line change
@@ -9,8 +9,8 @@
99
1010
#![deny(unsafe_op_in_unsafe_fn)]
1111

12-
use crate::panic::BacktraceStyle;
13-
use core::panic::{BoxMeUp, Location, PanicInfo};
12+
use crate::panic::{BacktraceStyle, PanicInfo};
13+
use core::panic::{BoxMeUp, Location};
1414

1515
use crate::any::Any;
1616
use crate::fmt;
@@ -564,7 +564,7 @@ pub fn panicking() -> bool {
564564
/// Entry point of panics from the core crate (`panic_impl` lang item).
565565
#[cfg(not(test))]
566566
#[panic_handler]
567-
pub fn begin_panic_handler(info: &PanicInfo<'_>) -> ! {
567+
pub fn begin_panic_handler(info: &core::panic::PanicInfo<'_>) -> ! {
568568
struct PanicPayload<'a> {
569569
inner: &'a fmt::Arguments<'a>,
570570
string: Option<String>,
@@ -615,22 +615,20 @@ pub fn begin_panic_handler(info: &PanicInfo<'_>) -> ! {
615615
}
616616

617617
let loc = info.location().unwrap(); // The current implementation always returns Some
618-
let msg = info.message().unwrap(); // The current implementation always returns Some
618+
let msg = info.message();
619619
crate::sys_common::backtrace::__rust_end_short_backtrace(move || {
620-
// FIXME: can we just pass `info` along rather than taking it apart here, only to have
621-
// `rust_panic_with_hook` construct a new `PanicInfo`?
622-
if let Some(msg) = msg.as_str() {
620+
if let Some(s) = msg.as_str() {
623621
rust_panic_with_hook(
624-
&mut StrPanicPayload(msg),
625-
info.message(),
622+
&mut StrPanicPayload(s),
623+
Some(msg),
626624
loc,
627625
info.can_unwind(),
628626
info.force_no_backtrace(),
629627
);
630628
} else {
631629
rust_panic_with_hook(
632-
&mut PanicPayload::new(msg),
633-
info.message(),
630+
&mut PanicPayload::new(&msg),
631+
Some(msg),
634632
loc,
635633
info.can_unwind(),
636634
info.force_no_backtrace(),
@@ -707,7 +705,7 @@ pub const fn begin_panic<M: Any + Send>(msg: M) -> ! {
707705
/// abort or unwind.
708706
fn rust_panic_with_hook(
709707
payload: &mut dyn BoxMeUp,
710-
message: Option<&fmt::Arguments<'_>>,
708+
message: Option<fmt::Arguments<'_>>,
711709
location: &Location<'_>,
712710
can_unwind: bool,
713711
force_no_backtrace: bool,
@@ -725,20 +723,17 @@ fn rust_panic_with_hook(
725723
panic_count::MustAbort::AlwaysAbort => {
726724
// Unfortunately, this does not print a backtrace, because creating
727725
// a `Backtrace` will allocate, which we must to avoid here.
728-
let panicinfo = PanicInfo::internal_constructor(
729-
message,
730-
location,
731-
can_unwind,
732-
force_no_backtrace,
733-
);
734-
rtprintpanic!("{panicinfo}\npanicked after panic::always_abort(), aborting.\n");
726+
if let Some(message) = message {
727+
rtprintpanic!("aborting due to panic at {location}:\n{message}\n");
728+
} else {
729+
rtprintpanic!("aborting due to panic at {location}\n");
730+
}
735731
}
736732
}
737733
crate::sys::abort_internal();
738734
}
739735

740-
let mut info =
741-
PanicInfo::internal_constructor(message, location, can_unwind, force_no_backtrace);
736+
let mut info = PanicInfo::internal_constructor(location, can_unwind, force_no_backtrace);
742737
let hook = HOOK.read().unwrap_or_else(PoisonError::into_inner);
743738
match *hook {
744739
// Some platforms (like wasm) know that printing to stderr won't ever actually

0 commit comments

Comments
 (0)