| 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 | |