1 | //! Panic support in the standard library. |
2 | |
3 | #![stable (feature = "std_panic" , since = "1.9.0" )] |
4 | |
5 | use crate::any::Any; |
6 | use crate::sync::atomic::{Atomic, AtomicU8, Ordering}; |
7 | use crate::sync::{Condvar, Mutex, RwLock}; |
8 | use crate::thread::Result; |
9 | use crate::{collections, fmt, panicking}; |
10 | |
11 | #[stable (feature = "panic_hooks" , since = "1.10.0" )] |
12 | #[deprecated ( |
13 | since = "1.82.0" , |
14 | note = "use `PanicHookInfo` instead" , |
15 | suggestion = "std::panic::PanicHookInfo" |
16 | )] |
17 | /// A struct providing information about a panic. |
18 | /// |
19 | /// `PanicInfo` has been renamed to [`PanicHookInfo`] to avoid confusion with |
20 | /// [`core::panic::PanicInfo`]. |
21 | pub type PanicInfo<'a> = PanicHookInfo<'a>; |
22 | |
23 | /// A struct providing information about a panic. |
24 | /// |
25 | /// `PanicHookInfo` structure is passed to a panic hook set by the [`set_hook`] function. |
26 | /// |
27 | /// # Examples |
28 | /// |
29 | /// ```should_panic |
30 | /// use std::panic; |
31 | /// |
32 | /// panic::set_hook(Box::new(|panic_info| { |
33 | /// println!("panic occurred: {panic_info}" ); |
34 | /// })); |
35 | /// |
36 | /// panic!("critical system failure" ); |
37 | /// ``` |
38 | /// |
39 | /// [`set_hook`]: ../../std/panic/fn.set_hook.html |
40 | #[stable (feature = "panic_hook_info" , since = "1.81.0" )] |
41 | #[derive (Debug)] |
42 | pub struct PanicHookInfo<'a> { |
43 | payload: &'a (dyn Any + Send), |
44 | location: &'a Location<'a>, |
45 | can_unwind: bool, |
46 | force_no_backtrace: bool, |
47 | } |
48 | |
49 | impl<'a> PanicHookInfo<'a> { |
50 | #[inline ] |
51 | pub(crate) fn new( |
52 | location: &'a Location<'a>, |
53 | payload: &'a (dyn Any + Send), |
54 | can_unwind: bool, |
55 | force_no_backtrace: bool, |
56 | ) -> Self { |
57 | PanicHookInfo { payload, location, can_unwind, force_no_backtrace } |
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 | /// A invocation of the `panic!()` macro in Rust 2021 or later will always result in a |
65 | /// panic payload of type `&'static str` or `String`. |
66 | /// |
67 | /// Only an invocation of [`panic_any`] |
68 | /// (or, in Rust 2018 and earlier, `panic!(x)` where `x` is something other than a string) |
69 | /// can result in a panic payload other than a `&'static str` or `String`. |
70 | /// |
71 | /// [`String`]: ../../std/string/struct.String.html |
72 | /// |
73 | /// # Examples |
74 | /// |
75 | /// ```should_panic |
76 | /// use std::panic; |
77 | /// |
78 | /// panic::set_hook(Box::new(|panic_info| { |
79 | /// if let Some(s) = panic_info.payload().downcast_ref::<&str>() { |
80 | /// println!("panic occurred: {s:?}" ); |
81 | /// } else if let Some(s) = panic_info.payload().downcast_ref::<String>() { |
82 | /// println!("panic occurred: {s:?}" ); |
83 | /// } else { |
84 | /// println!("panic occurred" ); |
85 | /// } |
86 | /// })); |
87 | /// |
88 | /// panic!("Normal panic" ); |
89 | /// ``` |
90 | #[must_use ] |
91 | #[inline ] |
92 | #[stable (feature = "panic_hooks" , since = "1.10.0" )] |
93 | pub fn payload(&self) -> &(dyn Any + Send) { |
94 | self.payload |
95 | } |
96 | |
97 | /// Returns the payload associated with the panic, if it is a string. |
98 | /// |
99 | /// This returns the payload if it is of type `&'static str` or `String`. |
100 | /// |
101 | /// A invocation of the `panic!()` macro in Rust 2021 or later will always result in a |
102 | /// panic payload where `payload_as_str` returns `Some`. |
103 | /// |
104 | /// Only an invocation of [`panic_any`] |
105 | /// (or, in Rust 2018 and earlier, `panic!(x)` where `x` is something other than a string) |
106 | /// can result in a panic payload where `payload_as_str` returns `None`. |
107 | /// |
108 | /// # Example |
109 | /// |
110 | /// ```should_panic |
111 | /// #![feature(panic_payload_as_str)] |
112 | /// |
113 | /// std::panic::set_hook(Box::new(|panic_info| { |
114 | /// if let Some(s) = panic_info.payload_as_str() { |
115 | /// println!("panic occurred: {s:?}" ); |
116 | /// } else { |
117 | /// println!("panic occurred" ); |
118 | /// } |
119 | /// })); |
120 | /// |
121 | /// panic!("Normal panic" ); |
122 | /// ``` |
123 | #[must_use ] |
124 | #[inline ] |
125 | #[unstable (feature = "panic_payload_as_str" , issue = "125175" )] |
126 | pub fn payload_as_str(&self) -> Option<&str> { |
127 | if let Some(s) = self.payload.downcast_ref::<&str>() { |
128 | Some(s) |
129 | } else if let Some(s) = self.payload.downcast_ref::<String>() { |
130 | Some(s) |
131 | } else { |
132 | None |
133 | } |
134 | } |
135 | |
136 | /// Returns information about the location from which the panic originated, |
137 | /// if available. |
138 | /// |
139 | /// This method will currently always return [`Some`], but this may change |
140 | /// in future versions. |
141 | /// |
142 | /// # Examples |
143 | /// |
144 | /// ```should_panic |
145 | /// use std::panic; |
146 | /// |
147 | /// panic::set_hook(Box::new(|panic_info| { |
148 | /// if let Some(location) = panic_info.location() { |
149 | /// println!("panic occurred in file '{}' at line {}" , |
150 | /// location.file(), |
151 | /// location.line(), |
152 | /// ); |
153 | /// } else { |
154 | /// println!("panic occurred but can't get location information..." ); |
155 | /// } |
156 | /// })); |
157 | /// |
158 | /// panic!("Normal panic" ); |
159 | /// ``` |
160 | #[must_use ] |
161 | #[inline ] |
162 | #[stable (feature = "panic_hooks" , since = "1.10.0" )] |
163 | pub fn location(&self) -> Option<&Location<'_>> { |
164 | // NOTE: If this is changed to sometimes return None, |
165 | // deal with that case in std::panicking::default_hook and core::panicking::panic_fmt. |
166 | Some(&self.location) |
167 | } |
168 | |
169 | /// Returns whether the panic handler is allowed to unwind the stack from |
170 | /// the point where the panic occurred. |
171 | /// |
172 | /// This is true for most kinds of panics with the exception of panics |
173 | /// caused by trying to unwind out of a `Drop` implementation or a function |
174 | /// whose ABI does not support unwinding. |
175 | /// |
176 | /// It is safe for a panic handler to unwind even when this function returns |
177 | /// false, however this will simply cause the panic handler to be called |
178 | /// again. |
179 | #[must_use ] |
180 | #[inline ] |
181 | #[unstable (feature = "panic_can_unwind" , issue = "92988" )] |
182 | pub fn can_unwind(&self) -> bool { |
183 | self.can_unwind |
184 | } |
185 | |
186 | #[unstable ( |
187 | feature = "panic_internals" , |
188 | reason = "internal details of the implementation of the `panic!` and related macros" , |
189 | issue = "none" |
190 | )] |
191 | #[doc (hidden)] |
192 | #[inline ] |
193 | pub fn force_no_backtrace(&self) -> bool { |
194 | self.force_no_backtrace |
195 | } |
196 | } |
197 | |
198 | #[stable (feature = "panic_hook_display" , since = "1.26.0" )] |
199 | impl fmt::Display for PanicHookInfo<'_> { |
200 | fn fmt(&self, formatter: &mut fmt::Formatter<'_>) -> fmt::Result { |
201 | formatter.write_str(data:"panicked at " )?; |
202 | self.location.fmt(formatter)?; |
203 | if let Some(payload: &str) = self.payload_as_str() { |
204 | formatter.write_str(data:": \n" )?; |
205 | formatter.write_str(data:payload)?; |
206 | } |
207 | Ok(()) |
208 | } |
209 | } |
210 | |
211 | #[doc (hidden)] |
212 | #[unstable (feature = "edition_panic" , issue = "none" , reason = "use panic!() instead" )] |
213 | #[allow_internal_unstable (libstd_sys_internals, const_format_args, panic_internals, rt)] |
214 | #[cfg_attr (not(test), rustc_diagnostic_item = "std_panic_2015_macro" )] |
215 | #[rustc_macro_transparency = "semitransparent" ] |
216 | pub macro panic_2015 { |
217 | () => ({ |
218 | $crate::rt::begin_panic("explicit panic" ) |
219 | }), |
220 | ($msg:expr $(,)?) => ({ |
221 | $crate::rt::begin_panic($msg); |
222 | }), |
223 | // Special-case the single-argument case for const_panic. |
224 | ("{}" , $arg:expr $(,)?) => ({ |
225 | $crate::rt::panic_display(&$arg); |
226 | }), |
227 | ($fmt:expr, $($arg:tt)+) => ({ |
228 | // Semicolon to prevent temporaries inside the formatting machinery from |
229 | // being considered alive in the caller after the panic_fmt call. |
230 | $crate::rt::panic_fmt($crate::const_format_args!($fmt, $($arg)+)); |
231 | }), |
232 | } |
233 | |
234 | #[stable (feature = "panic_hooks" , since = "1.10.0" )] |
235 | pub use core::panic::Location; |
236 | #[doc (hidden)] |
237 | #[unstable (feature = "edition_panic" , issue = "none" , reason = "use panic!() instead" )] |
238 | pub use core::panic::panic_2021; |
239 | #[stable (feature = "catch_unwind" , since = "1.9.0" )] |
240 | pub use core::panic::{AssertUnwindSafe, RefUnwindSafe, UnwindSafe}; |
241 | |
242 | #[unstable (feature = "panic_update_hook" , issue = "92649" )] |
243 | pub use crate::panicking::update_hook; |
244 | #[stable (feature = "panic_hooks" , since = "1.10.0" )] |
245 | pub use crate::panicking::{set_hook, take_hook}; |
246 | |
247 | /// Panics the current thread with the given message as the panic payload. |
248 | /// |
249 | /// The message can be of any (`Any + Send`) type, not just strings. |
250 | /// |
251 | /// The message is wrapped in a `Box<'static + Any + Send>`, which can be |
252 | /// accessed later using [`PanicHookInfo::payload`]. |
253 | /// |
254 | /// See the [`panic!`] macro for more information about panicking. |
255 | #[stable (feature = "panic_any" , since = "1.51.0" )] |
256 | #[inline ] |
257 | #[track_caller ] |
258 | #[cfg_attr (not(test), rustc_diagnostic_item = "panic_any" )] |
259 | pub fn panic_any<M: 'static + Any + Send>(msg: M) -> ! { |
260 | crate::panicking::begin_panic(msg); |
261 | } |
262 | |
263 | #[stable (feature = "catch_unwind" , since = "1.9.0" )] |
264 | impl<T: ?Sized> UnwindSafe for Mutex<T> {} |
265 | #[stable (feature = "catch_unwind" , since = "1.9.0" )] |
266 | impl<T: ?Sized> UnwindSafe for RwLock<T> {} |
267 | #[stable (feature = "catch_unwind" , since = "1.9.0" )] |
268 | impl UnwindSafe for Condvar {} |
269 | |
270 | #[stable (feature = "unwind_safe_lock_refs" , since = "1.12.0" )] |
271 | impl<T: ?Sized> RefUnwindSafe for Mutex<T> {} |
272 | #[stable (feature = "unwind_safe_lock_refs" , since = "1.12.0" )] |
273 | impl<T: ?Sized> RefUnwindSafe for RwLock<T> {} |
274 | #[stable (feature = "unwind_safe_lock_refs" , since = "1.12.0" )] |
275 | impl RefUnwindSafe for Condvar {} |
276 | |
277 | // https://github.com/rust-lang/rust/issues/62301 |
278 | #[stable (feature = "hashbrown" , since = "1.36.0" )] |
279 | impl<K, V, S> UnwindSafe for collections::HashMap<K, V, S> |
280 | where |
281 | K: UnwindSafe, |
282 | V: UnwindSafe, |
283 | S: UnwindSafe, |
284 | { |
285 | } |
286 | |
287 | #[unstable (feature = "abort_unwind" , issue = "130338" )] |
288 | pub use core::panic::abort_unwind; |
289 | |
290 | /// Invokes a closure, capturing the cause of an unwinding panic if one occurs. |
291 | /// |
292 | /// This function will return `Ok` with the closure's result if the closure does |
293 | /// not panic, and will return `Err(cause)` if the closure panics. The `cause` |
294 | /// returned is the object with which panic was originally invoked. |
295 | /// |
296 | /// Rust functions that are expected to be called from foreign code that does |
297 | /// not support unwinding (such as C compiled with `-fno-exceptions`) should be |
298 | /// defined using `extern "C"`, which ensures that if the Rust code panics, it |
299 | /// is automatically caught and the process is aborted. If this is the desired |
300 | /// behavior, it is not necessary to use `catch_unwind` explicitly. This |
301 | /// function should instead be used when more graceful error-handling is needed. |
302 | /// |
303 | /// It is **not** recommended to use this function for a general try/catch |
304 | /// mechanism. The [`Result`] type is more appropriate to use for functions that |
305 | /// can fail on a regular basis. Additionally, this function is not guaranteed |
306 | /// to catch all panics, see the "Notes" section below. |
307 | /// |
308 | /// The closure provided is required to adhere to the [`UnwindSafe`] trait to |
309 | /// ensure that all captured variables are safe to cross this boundary. The |
310 | /// purpose of this bound is to encode the concept of [exception safety][rfc] in |
311 | /// the type system. Most usage of this function should not need to worry about |
312 | /// this bound as programs are naturally unwind safe without `unsafe` code. If |
313 | /// it becomes a problem the [`AssertUnwindSafe`] wrapper struct can be used to |
314 | /// quickly assert that the usage here is indeed unwind safe. |
315 | /// |
316 | /// [rfc]: https://github.com/rust-lang/rfcs/blob/master/text/1236-stabilize-catch-panic.md |
317 | /// |
318 | /// # Notes |
319 | /// |
320 | /// This function **might not catch all Rust panics**. A Rust panic is not |
321 | /// always implemented via unwinding, but can be implemented by aborting the |
322 | /// process as well. This function *only* catches unwinding panics, not those |
323 | /// that abort the process. |
324 | /// |
325 | /// If a custom panic hook has been set, it will be invoked before the panic is |
326 | /// caught, before unwinding. |
327 | /// |
328 | /// Although unwinding into Rust code with a foreign exception (e.g. an |
329 | /// exception thrown from C++ code, or a `panic!` in Rust code compiled or |
330 | /// linked with a different runtime) via an appropriate ABI (e.g. `"C-unwind"`) |
331 | /// is permitted, catching such an exception using this function will have one |
332 | /// of two behaviors, and it is unspecified which will occur: |
333 | /// |
334 | /// * The process aborts, after executing all destructors of `f` and the |
335 | /// functions it called. |
336 | /// * The function returns a `Result::Err` containing an opaque type. |
337 | /// |
338 | /// Finally, be **careful in how you drop the result of this function**. If it |
339 | /// is `Err`, it contains the panic payload, and dropping that may in turn |
340 | /// panic! |
341 | /// |
342 | /// # Examples |
343 | /// |
344 | /// ``` |
345 | /// use std::panic; |
346 | /// |
347 | /// let result = panic::catch_unwind(|| { |
348 | /// println!("hello!" ); |
349 | /// }); |
350 | /// assert!(result.is_ok()); |
351 | /// |
352 | /// let result = panic::catch_unwind(|| { |
353 | /// panic!("oh no!" ); |
354 | /// }); |
355 | /// assert!(result.is_err()); |
356 | /// ``` |
357 | #[stable (feature = "catch_unwind" , since = "1.9.0" )] |
358 | pub fn catch_unwind<F: FnOnce() -> R + UnwindSafe, R>(f: F) -> Result<R> { |
359 | unsafe { panicking::catch_unwind(f) } |
360 | } |
361 | |
362 | /// Triggers a panic without invoking the panic hook. |
363 | /// |
364 | /// This is designed to be used in conjunction with [`catch_unwind`] to, for |
365 | /// example, carry a panic across a layer of C code. |
366 | /// |
367 | /// # Notes |
368 | /// |
369 | /// Note that panics in Rust are not always implemented via unwinding, but they |
370 | /// may be implemented by aborting the process. If this function is called when |
371 | /// panics are implemented this way then this function will abort the process, |
372 | /// not trigger an unwind. |
373 | /// |
374 | /// # Examples |
375 | /// |
376 | /// ```should_panic |
377 | /// use std::panic; |
378 | /// |
379 | /// let result = panic::catch_unwind(|| { |
380 | /// if 1 != 2 { |
381 | /// panic!("oh no!" ); |
382 | /// } |
383 | /// }); |
384 | /// |
385 | /// if let Err(err) = result { |
386 | /// panic::resume_unwind(err); |
387 | /// } |
388 | /// ``` |
389 | #[stable (feature = "resume_unwind" , since = "1.9.0" )] |
390 | pub fn resume_unwind(payload: Box<dyn Any + Send>) -> ! { |
391 | panicking::rust_panic_without_hook(payload) |
392 | } |
393 | |
394 | /// Makes all future panics abort directly without running the panic hook or unwinding. |
395 | /// |
396 | /// There is no way to undo this; the effect lasts until the process exits or |
397 | /// execs (or the equivalent). |
398 | /// |
399 | /// # Use after fork |
400 | /// |
401 | /// This function is particularly useful for calling after `libc::fork`. After `fork`, in a |
402 | /// multithreaded program it is (on many platforms) not safe to call the allocator. It is also |
403 | /// generally highly undesirable for an unwind to unwind past the `fork`, because that results in |
404 | /// the unwind propagating to code that was only ever expecting to run in the parent. |
405 | /// |
406 | /// `panic::always_abort()` helps avoid both of these. It directly avoids any further unwinding, |
407 | /// and if there is a panic, the abort will occur without allocating provided that the arguments to |
408 | /// panic can be formatted without allocating. |
409 | /// |
410 | /// Examples |
411 | /// |
412 | /// ```no_run |
413 | /// #![feature(panic_always_abort)] |
414 | /// use std::panic; |
415 | /// |
416 | /// panic::always_abort(); |
417 | /// |
418 | /// let _ = panic::catch_unwind(|| { |
419 | /// panic!("inside the catch" ); |
420 | /// }); |
421 | /// |
422 | /// // We will have aborted already, due to the panic. |
423 | /// unreachable!(); |
424 | /// ``` |
425 | #[unstable (feature = "panic_always_abort" , issue = "84438" )] |
426 | pub fn always_abort() { |
427 | crate::panicking::panic_count::set_always_abort(); |
428 | } |
429 | |
430 | /// The configuration for whether and how the default panic hook will capture |
431 | /// and display the backtrace. |
432 | #[derive (Debug, Copy, Clone, PartialEq, Eq)] |
433 | #[unstable (feature = "panic_backtrace_config" , issue = "93346" )] |
434 | #[non_exhaustive ] |
435 | pub enum BacktraceStyle { |
436 | /// Prints a terser backtrace which ideally only contains relevant |
437 | /// information. |
438 | Short, |
439 | /// Prints a backtrace with all possible information. |
440 | Full, |
441 | /// Disable collecting and displaying backtraces. |
442 | Off, |
443 | } |
444 | |
445 | impl BacktraceStyle { |
446 | pub(crate) fn full() -> Option<Self> { |
447 | if cfg!(feature = "backtrace" ) { Some(BacktraceStyle::Full) } else { None } |
448 | } |
449 | |
450 | fn as_u8(self) -> u8 { |
451 | match self { |
452 | BacktraceStyle::Short => 1, |
453 | BacktraceStyle::Full => 2, |
454 | BacktraceStyle::Off => 3, |
455 | } |
456 | } |
457 | |
458 | fn from_u8(s: u8) -> Option<Self> { |
459 | match s { |
460 | 1 => Some(BacktraceStyle::Short), |
461 | 2 => Some(BacktraceStyle::Full), |
462 | 3 => Some(BacktraceStyle::Off), |
463 | _ => None, |
464 | } |
465 | } |
466 | } |
467 | |
468 | // Tracks whether we should/can capture a backtrace, and how we should display |
469 | // that backtrace. |
470 | // |
471 | // Internally stores equivalent of an Option<BacktraceStyle>. |
472 | static SHOULD_CAPTURE: Atomic<u8> = AtomicU8::new(0); |
473 | |
474 | /// Configures whether the default panic hook will capture and display a |
475 | /// backtrace. |
476 | /// |
477 | /// The default value for this setting may be set by the `RUST_BACKTRACE` |
478 | /// environment variable; see the details in [`get_backtrace_style`]. |
479 | #[unstable (feature = "panic_backtrace_config" , issue = "93346" )] |
480 | pub fn set_backtrace_style(style: BacktraceStyle) { |
481 | if cfg!(feature = "backtrace" ) { |
482 | // If the `backtrace` feature of this crate is enabled, set the backtrace style. |
483 | SHOULD_CAPTURE.store(style.as_u8(), Ordering::Relaxed); |
484 | } |
485 | } |
486 | |
487 | /// Checks whether the standard library's panic hook will capture and print a |
488 | /// backtrace. |
489 | /// |
490 | /// This function will, if a backtrace style has not been set via |
491 | /// [`set_backtrace_style`], read the environment variable `RUST_BACKTRACE` to |
492 | /// determine a default value for the backtrace formatting: |
493 | /// |
494 | /// The first call to `get_backtrace_style` may read the `RUST_BACKTRACE` |
495 | /// environment variable if `set_backtrace_style` has not been called to |
496 | /// override the default value. After a call to `set_backtrace_style` or |
497 | /// `get_backtrace_style`, any changes to `RUST_BACKTRACE` will have no effect. |
498 | /// |
499 | /// `RUST_BACKTRACE` is read according to these rules: |
500 | /// |
501 | /// * `0` for `BacktraceStyle::Off` |
502 | /// * `full` for `BacktraceStyle::Full` |
503 | /// * `1` for `BacktraceStyle::Short` |
504 | /// * Other values are currently `BacktraceStyle::Short`, but this may change in |
505 | /// the future |
506 | /// |
507 | /// Returns `None` if backtraces aren't currently supported. |
508 | #[unstable (feature = "panic_backtrace_config" , issue = "93346" )] |
509 | pub fn get_backtrace_style() -> Option<BacktraceStyle> { |
510 | if !cfg!(feature = "backtrace" ) { |
511 | // If the `backtrace` feature of this crate isn't enabled quickly return |
512 | // `Unsupported` so this can be constant propagated all over the place |
513 | // to optimize away callers. |
514 | return None; |
515 | } |
516 | |
517 | let current = SHOULD_CAPTURE.load(Ordering::Relaxed); |
518 | if let Some(style) = BacktraceStyle::from_u8(current) { |
519 | return Some(style); |
520 | } |
521 | |
522 | let format = match crate::env::var_os("RUST_BACKTRACE" ) { |
523 | Some(x) if &x == "0" => BacktraceStyle::Off, |
524 | Some(x) if &x == "full" => BacktraceStyle::Full, |
525 | Some(_) => BacktraceStyle::Short, |
526 | None if crate::sys::FULL_BACKTRACE_DEFAULT => BacktraceStyle::Full, |
527 | None => BacktraceStyle::Off, |
528 | }; |
529 | |
530 | match SHOULD_CAPTURE.compare_exchange(0, format.as_u8(), Ordering::Relaxed, Ordering::Relaxed) { |
531 | Ok(_) => Some(format), |
532 | Err(new) => BacktraceStyle::from_u8(new), |
533 | } |
534 | } |
535 | |