1 //! Support for capturing a stack backtrace of an OS thread
3 //! This module contains the support necessary to capture a stack backtrace of a
4 //! running OS thread from the OS thread itself. The `Backtrace` type supports
5 //! capturing a stack trace via the `Backtrace::capture` and
6 //! `Backtrace::force_capture` functions.
8 //! A backtrace is typically quite handy to attach to errors (e.g. types
9 //! implementing `std::error::Error`) to get a causal chain of where an error
12 //! > **Note**: this module is unstable and is designed in [RFC 2504], and you
13 //! > can learn more about its status in the [tracking issue].
15 //! [RFC 2504]: https://github.com/rust-lang/rfcs/blob/master/text/2504-fix-error.md
16 //! [tracking issue]: https://github.com/rust-lang/rust/issues/53487
20 //! Backtraces are attempted to be as accurate as possible, but no guarantees
21 //! are provided about the exact accuracy of a backtrace. Instruction pointers,
22 //! symbol names, filenames, line numbers, etc, may all be incorrect when
23 //! reported. Accuracy is attempted on a best-effort basis, however, and bugs
24 //! are always welcome to indicate areas of improvement!
26 //! For most platforms a backtrace with a filename/line number requires that
27 //! programs be compiled with debug information. Without debug information
28 //! filenames/line numbers will not be reported.
30 //! ## Platform support
32 //! Not all platforms that libstd compiles for support capturing backtraces.
33 //! Some platforms simply do nothing when capturing a backtrace. To check
34 //! whether the platform supports capturing backtraces you can consult the
35 //! `BacktraceStatus` enum as a result of `Backtrace::status`.
37 //! Like above with accuracy platform support is done on a best effort basis.
38 //! Sometimes libraries may not be available at runtime or something may go
39 //! wrong which would cause a backtrace to not be captured. Please feel free to
40 //! report issues with platforms where a backtrace cannot be captured though!
42 //! ## Environment Variables
44 //! The `Backtrace::capture` function may not actually capture a backtrace by
45 //! default. Its behavior is governed by two environment variables:
47 //! * `RUST_LIB_BACKTRACE` - if this is set to `0` then `Backtrace::capture`
48 //! will never capture a backtrace. Any other value this is set to will enable
49 //! `Backtrace::capture`.
51 //! * `RUST_BACKTRACE` - if `RUST_LIB_BACKTRACE` is not set, then this variable
52 //! is consulted with the same rules of `RUST_LIB_BACKTRACE`.
54 //! * If neither of the above env vars are set, then `Backtrace::capture` will
57 //! Capturing a backtrace can be a quite expensive runtime operation, so the
58 //! environment variables allow either forcibly disabling this runtime
59 //! performance hit or allow selectively enabling it in some programs.
61 //! Note that the `Backtrace::force_capture` function can be used to ignore
62 //! these environment variables. Also note that the state of environment
63 //! variables is cached once the first backtrace is created, so altering
64 //! `RUST_LIB_BACKTRACE` or `RUST_BACKTRACE` at runtime may not actually change
65 //! how backtraces are captured.
67 #![unstable(feature = "backtrace", issue = "53487")]
69 // NB: A note on resolution of a backtrace:
71 // Backtraces primarily happen in two steps, one is where we actually capture
72 // the stack backtrace, giving us a list of instruction pointers corresponding
73 // to stack frames. Next we take these instruction pointers and, one-by-one,
74 // turn them into a human readable name (like `main`).
76 // The first phase can be somewhat expensive (walking the stack), especially
77 // on MSVC where debug information is consulted to return inline frames each as
78 // their own frame. The second phase, however, is almost always extremely
79 // expensive (on the order of milliseconds sometimes) when it's consulting debug
82 // We attempt to amortize this cost as much as possible by delaying resolution
83 // of an address to a human readable name for as long as possible. When
84 // `Backtrace::create` is called to capture a backtrace it doesn't actually
85 // perform any symbol resolution, but rather we lazily resolve symbols only just
86 // before they're needed for printing. This way we can make capturing a
87 // backtrace and throwing it away much cheaper, but actually printing a
88 // backtrace is still basically the same cost.
90 // This strategy comes at the cost of some synchronization required inside of a
91 // `Backtrace`, but that's a relatively small price to pay relative to capturing
92 // a backtrace or actually symbolizing it.
96 use crate::sync::atomic::{AtomicUsize, Ordering::SeqCst};
97 use crate::sync::Mutex;
98 use crate::sys_common::backtrace::{lock, output_filename};
100 use backtrace::BytesOrWideString;
101 use backtrace_rs as backtrace;
103 /// A captured OS thread stack backtrace.
105 /// This type represents a stack backtrace for an OS thread captured at a
106 /// previous point in time. In some instances the `Backtrace` type may
107 /// internally be empty due to configuration. For more information see
108 /// `Backtrace::capture`.
109 pub struct Backtrace {
113 /// The current status of a backtrace, indicating whether it was captured or
114 /// whether it is empty for some other reason.
116 #[derive(Debug, PartialEq, Eq)]
117 pub enum BacktraceStatus {
118 /// Capturing a backtrace is not supported, likely because it's not
119 /// implemented for the current platform.
121 /// Capturing a backtrace has been disabled through either the
122 /// `RUST_LIB_BACKTRACE` or `RUST_BACKTRACE` environment variables.
124 /// A backtrace has been captured and the `Backtrace` should print
125 /// reasonable information when rendered.
132 Captured(Mutex<Capture>),
138 frames: Vec<BacktraceFrame>,
141 fn _assert_send_sync() {
142 fn _assert<T: Send + Sync>() {}
143 _assert::<Backtrace>();
146 struct BacktraceFrame {
147 frame: backtrace::Frame,
148 symbols: Vec<BacktraceSymbol>,
151 struct BacktraceSymbol {
152 name: Option<Vec<u8>>,
153 filename: Option<BytesOrWide>,
162 impl fmt::Debug for Backtrace {
163 fn fmt(&self, fmt: &mut fmt::Formatter<'_>) -> fmt::Result {
164 let mut capture = match &self.inner {
165 Inner::Unsupported => return fmt.write_str("unsupported backtrace"),
166 Inner::Disabled => return fmt.write_str("disabled backtrace"),
167 Inner::Captured(c) => c.lock().unwrap(),
171 let frames = if fmt.alternate() {
174 &capture.frames[capture.actual_start..]
177 write!(fmt, "Backtrace ")?;
179 let mut dbg = fmt.debug_list();
181 for frame in frames {
182 if frame.frame.ip().is_null() {
186 dbg.entries(&frame.symbols);
193 impl fmt::Debug for BacktraceSymbol {
194 fn fmt(&self, fmt: &mut fmt::Formatter<'_>) -> fmt::Result {
197 if let Some(fn_name) = self.name.as_ref().map(|b| backtrace::SymbolName::new(b)) {
198 write!(fmt, "fn: ")?;
199 fmt::Display::fmt(&fn_name, fmt)?;
201 write!(fmt, "fn: <unknown>")?;
204 if let Some(fname) = self.filename.as_ref() {
205 write!(fmt, ", file: {:?}", fname)?;
208 if let Some(line) = self.lineno.as_ref() {
209 write!(fmt, ", line: {:?}", line)?;
216 impl fmt::Debug for BytesOrWide {
217 fn fmt(&self, fmt: &mut fmt::Formatter<'_>) -> fmt::Result {
221 BytesOrWide::Bytes(w) => BytesOrWideString::Bytes(w),
222 BytesOrWide::Wide(w) => BytesOrWideString::Wide(w),
224 if fmt.alternate() { backtrace::PrintFmt::Full } else { backtrace::PrintFmt::Short },
225 crate::env::current_dir().as_ref().ok(),
231 /// Returns whether backtrace captures are enabled through environment
233 fn enabled() -> bool {
234 // Cache the result of reading the environment variables to make
235 // backtrace captures speedy, because otherwise reading environment
236 // variables every time can be somewhat slow.
237 static ENABLED: AtomicUsize = AtomicUsize::new(0);
238 match ENABLED.load(SeqCst) {
243 let enabled = match env::var("RUST_LIB_BACKTRACE") {
245 Err(_) => match env::var("RUST_BACKTRACE") {
250 ENABLED.store(enabled as usize + 1, SeqCst);
254 /// Capture a stack backtrace of the current thread.
256 /// This function will capture a stack backtrace of the current OS thread of
257 /// execution, returning a `Backtrace` type which can be later used to print
258 /// the entire stack trace or render it to a string.
260 /// This function will be a noop if the `RUST_BACKTRACE` or
261 /// `RUST_LIB_BACKTRACE` backtrace variables are both not set. If either
262 /// environment variable is set and enabled then this function will actually
263 /// capture a backtrace. Capturing a backtrace can be both memory intensive
264 /// and slow, so these environment variables allow liberally using
265 /// `Backtrace::capture` and only incurring a slowdown when the environment
266 /// variables are set.
268 /// To forcibly capture a backtrace regardless of environment variables, use
269 /// the `Backtrace::force_capture` function.
270 #[inline(never)] // want to make sure there's a frame here to remove
271 pub fn capture() -> Backtrace {
272 if !Backtrace::enabled() {
273 return Backtrace { inner: Inner::Disabled };
275 Backtrace::create(Backtrace::capture as usize)
278 /// Forcibly captures a full backtrace, regardless of environment variable
281 /// This function behaves the same as `capture` except that it ignores the
282 /// values of the `RUST_BACKTRACE` and `RUST_LIB_BACKTRACE` environment
283 /// variables, always capturing a backtrace.
285 /// Note that capturing a backtrace can be an expensive operation on some
286 /// platforms, so this should be used with caution in performance-sensitive
288 #[inline(never)] // want to make sure there's a frame here to remove
289 pub fn force_capture() -> Backtrace {
290 Backtrace::create(Backtrace::force_capture as usize)
293 // Capture a backtrace which start just before the function addressed by
295 fn create(ip: usize) -> Backtrace {
297 let mut frames = Vec::new();
298 let mut actual_start = None;
300 backtrace::trace_unsynchronized(|frame| {
301 frames.push(BacktraceFrame { frame: frame.clone(), symbols: Vec::new() });
302 if frame.symbol_address() as usize == ip && actual_start.is_none() {
303 actual_start = Some(frames.len());
309 // If no frames came out assume that this is an unsupported platform
310 // since `backtrace` doesn't provide a way of learning this right now,
311 // and this should be a good enough approximation.
312 let inner = if frames.len() == 0 {
315 Inner::Captured(Mutex::new(Capture {
316 actual_start: actual_start.unwrap_or(0),
325 /// Returns the status of this backtrace, indicating whether this backtrace
326 /// request was unsupported, disabled, or a stack trace was actually
328 pub fn status(&self) -> BacktraceStatus {
330 Inner::Unsupported => BacktraceStatus::Unsupported,
331 Inner::Disabled => BacktraceStatus::Disabled,
332 Inner::Captured(_) => BacktraceStatus::Captured,
337 impl fmt::Display for Backtrace {
338 fn fmt(&self, fmt: &mut fmt::Formatter<'_>) -> fmt::Result {
339 let mut capture = match &self.inner {
340 Inner::Unsupported => return fmt.write_str("unsupported backtrace"),
341 Inner::Disabled => return fmt.write_str("disabled backtrace"),
342 Inner::Captured(c) => c.lock().unwrap(),
346 let full = fmt.alternate();
347 let (frames, style) = if full {
348 (&capture.frames[..], backtrace::PrintFmt::Full)
350 (&capture.frames[capture.actual_start..], backtrace::PrintFmt::Short)
353 // When printing paths we try to strip the cwd if it exists, otherwise
354 // we just print the path as-is. Note that we also only do this for the
355 // short format, because if it's full we presumably want to print
357 let cwd = crate::env::current_dir();
358 let mut print_path = move |fmt: &mut fmt::Formatter<'_>, path: BytesOrWideString<'_>| {
359 output_filename(fmt, path, style, cwd.as_ref().ok())
362 let mut f = backtrace::BacktraceFmt::new(fmt, style, &mut print_path);
364 for frame in frames {
365 let mut f = f.frame();
366 if frame.symbols.is_empty() {
367 f.print_raw(frame.frame.ip(), None, None, None)?;
369 for symbol in frame.symbols.iter() {
372 symbol.name.as_ref().map(|b| backtrace::SymbolName::new(b)),
373 symbol.filename.as_ref().map(|b| match b {
374 BytesOrWide::Bytes(w) => BytesOrWideString::Bytes(w),
375 BytesOrWide::Wide(w) => BytesOrWideString::Wide(w),
388 fn resolve(&mut self) {
389 // If we're already resolved, nothing to do!
393 self.resolved = true;
395 // Use the global backtrace lock to synchronize this as it's a
396 // requirement of the `backtrace` crate, and then actually resolve
399 for frame in self.frames.iter_mut() {
400 let symbols = &mut frame.symbols;
402 backtrace::resolve_frame_unsynchronized(&frame.frame, |symbol| {
403 symbols.push(BacktraceSymbol {
404 name: symbol.name().map(|m| m.as_bytes().to_vec()),
405 filename: symbol.filename_raw().map(|b| match b {
406 BytesOrWideString::Bytes(b) => BytesOrWide::Bytes(b.to_owned()),
407 BytesOrWideString::Wide(b) => BytesOrWide::Wide(b.to_owned()),
409 lineno: symbol.lineno(),
422 fn debug_backtrace_fmt() {
423 let bt = Backtrace::capture();
424 eprintln!("uncaptured: {:?}", bt);
425 let bt = Backtrace::force_capture();
426 eprintln!("captured: {:?}", bt);
427 eprintln!("display print: {}", bt);
428 eprintln!("display print alt: {:#}", bt);
429 eprintln!("resolved: {:?}", bt);
430 eprintln!("resolved alt: {:#?}", bt);