Skip to content

Commit 474c342

Browse files
author
Aaron Power
committed
no_std support
1 parent a73a623 commit 474c342

17 files changed

+453
-219
lines changed

Cargo.toml

+28-2
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,9 @@ documentation = "https://docs.rs/backtrace"
1111
description = """
1212
A library to acquire a stack trace (backtrace) at runtime in a Rust program.
1313
"""
14+
autoexamples = true
15+
autotests = true
16+
1417
[dependencies]
1518
cfg-if = "0.1"
1619
rustc-demangle = "0.1.4"
@@ -47,7 +50,10 @@ backtrace-sys = { path = "backtrace-sys", version = "0.1.17", optional = true }
4750
# Note that not all features are available on all platforms, so even though a
4851
# feature is enabled some other feature may be used instead.
4952
[features]
50-
default = ["libunwind", "libbacktrace", "coresymbolication", "dladdr", "dbghelp"]
53+
default = ["std", "libunwind", "libbacktrace", "coresymbolication", "dladdr", "dbghelp"]
54+
55+
# Include std support.
56+
std = []
5157

5258
#=======================================
5359
# Methods of acquiring a backtrace
@@ -86,7 +92,7 @@ kernel32 = []
8692
# the moment, this is only possible when targetting Linux, since macOS
8793
# splits DWARF out into a separate object file. Enabling this feature
8894
# means one less C dependency.
89-
libbacktrace = ["backtrace-sys"]
95+
libbacktrace = ["backtrace-sys", "std"]
9096
dladdr = []
9197
coresymbolication = []
9298
gimli-symbolize = ["addr2line", "findshlibs", "gimli", "memmap", "object" ]
@@ -97,3 +103,23 @@ gimli-symbolize = ["addr2line", "findshlibs", "gimli", "memmap", "object" ]
97103
# Various features used for enabling rustc-serialize or syntex codegen.
98104
serialize-rustc = ["rustc-serialize"]
99105
serialize-serde = ["serde", "serde_derive"]
106+
107+
[[example]]
108+
name = "backtrace"
109+
required-features = ["std"]
110+
111+
[[example]]
112+
name = "raw"
113+
required-features = ["std"]
114+
115+
[[test]]
116+
name = "skip_inner_frames"
117+
required-features = ["std"]
118+
119+
[[test]]
120+
name = "long_fn_name"
121+
required-features = ["std"]
122+
123+
[[test]]
124+
name = "smoke"
125+
required-features = ["std"]

backtrace-sys/src/lib.rs

+1-2
Original file line numberDiff line numberDiff line change
@@ -2,8 +2,7 @@
22

33
extern crate libc;
44

5-
use libc::uintptr_t;
6-
use std::os::raw::{c_void, c_char, c_int};
5+
use libc::{c_void, c_char, c_int, uintptr_t};
76

87
pub type backtrace_syminfo_callback =
98
extern fn(data: *mut c_void,

src/backtrace/dbghelp.rs

+34-39
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@
1111
#![allow(bad_style)]
1212

1313
use std::mem;
14+
use std::prelude::v1::*;
1415
use winapi::ctypes::*;
1516
use winapi::shared::minwindef::*;
1617
use winapi::um::processthreadsapi;
@@ -33,49 +34,43 @@ impl Frame {
3334
}
3435

3536
#[inline(always)]
36-
pub fn trace(cb: &mut FnMut(&super::Frame) -> bool) {
37-
// According to windows documentation, all dbghelp functions are
38-
// single-threaded.
39-
let _g = ::lock::lock();
37+
pub unsafe fn trace(cb: &mut FnMut(&super::Frame) -> bool) {
38+
// Allocate necessary structures for doing the stack walk
39+
let process = processthreadsapi::GetCurrentProcess();
40+
let thread = processthreadsapi::GetCurrentThread();
4041

41-
unsafe {
42-
// Allocate necessary structures for doing the stack walk
43-
let process = processthreadsapi::GetCurrentProcess();
44-
let thread = processthreadsapi::GetCurrentThread();
42+
// The CONTEXT structure needs to be aligned on a 16-byte boundary for
43+
// 64-bit Windows, but currently we don't have a way to express that in
44+
// Rust. Allocations are generally aligned to 16-bytes, though, so we
45+
// box this up.
46+
let mut context = Box::new(mem::zeroed::<CONTEXT>());
47+
winnt::RtlCaptureContext(&mut *context);
48+
let mut frame = super::Frame {
49+
inner: Frame { inner: mem::zeroed() },
50+
};
51+
let image = init_frame(&mut frame.inner.inner, &context);
4552

46-
// The CONTEXT structure needs to be aligned on a 16-byte boundary for
47-
// 64-bit Windows, but currently we don't have a way to express that in
48-
// Rust. Allocations are generally aligned to 16-bytes, though, so we
49-
// box this up.
50-
let mut context = Box::new(mem::zeroed::<CONTEXT>());
51-
winnt::RtlCaptureContext(&mut *context);
52-
let mut frame = super::Frame {
53-
inner: Frame { inner: mem::zeroed() },
54-
};
55-
let image = init_frame(&mut frame.inner.inner, &context);
53+
// Initialize this process's symbols
54+
let _c = ::dbghelp_init();
5655

57-
// Initialize this process's symbols
58-
let _c = ::dbghelp_init();
56+
// And now that we're done with all the setup, do the stack walking!
57+
while dbghelp::StackWalk64(image as DWORD,
58+
process,
59+
thread,
60+
&mut frame.inner.inner,
61+
&mut *context as *mut _ as *mut _,
62+
None,
63+
Some(dbghelp::SymFunctionTableAccess64),
64+
Some(dbghelp::SymGetModuleBase64),
65+
None) == TRUE {
66+
if frame.inner.inner.AddrPC.Offset == frame.inner.inner.AddrReturn.Offset ||
67+
frame.inner.inner.AddrPC.Offset == 0 ||
68+
frame.inner.inner.AddrReturn.Offset == 0 {
69+
break
70+
}
5971

60-
// And now that we're done with all the setup, do the stack walking!
61-
while dbghelp::StackWalk64(image as DWORD,
62-
process,
63-
thread,
64-
&mut frame.inner.inner,
65-
&mut *context as *mut _ as *mut _,
66-
None,
67-
Some(dbghelp::SymFunctionTableAccess64),
68-
Some(dbghelp::SymGetModuleBase64),
69-
None) == TRUE {
70-
if frame.inner.inner.AddrPC.Offset == frame.inner.inner.AddrReturn.Offset ||
71-
frame.inner.inner.AddrPC.Offset == 0 ||
72-
frame.inner.inner.AddrReturn.Offset == 0 {
73-
break
74-
}
75-
76-
if !cb(&frame) {
77-
break
78-
}
72+
if !cb(&frame) {
73+
break
7974
}
8075
}
8176
}

src/backtrace/libunwind.rs

+4-6
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@
88
// option. This file may not be copied, modified, or distributed
99
// except according to those terms.
1010

11-
use std::os::raw::c_void;
11+
use types::c_void;
1212

1313
pub struct Frame {
1414
ctx: *mut uw::_Unwind_Context,
@@ -48,10 +48,8 @@ impl Frame {
4848
}
4949

5050
#[inline(always)]
51-
pub fn trace(mut cb: &mut FnMut(&super::Frame) -> bool) {
52-
unsafe {
53-
uw::_Unwind_Backtrace(trace_fn, &mut cb as *mut _ as *mut _);
54-
}
51+
pub unsafe fn trace(mut cb: &mut FnMut(&super::Frame) -> bool) {
52+
uw::_Unwind_Backtrace(trace_fn, &mut cb as *mut _ as *mut _);
5553

5654
extern fn trace_fn(ctx: *mut uw::_Unwind_Context,
5755
arg: *mut c_void) -> uw::_Unwind_Reason_Code {
@@ -84,7 +82,7 @@ mod uw {
8482
pub use self::_Unwind_Reason_Code::*;
8583

8684
use libc;
87-
use std::os::raw::{c_int, c_void};
85+
use types::{c_int, c_void};
8886

8987
#[repr(C)]
9088
pub enum _Unwind_Reason_Code {

src/backtrace/mod.rs

+12-3
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,5 @@
1-
use std::fmt;
2-
3-
use std::os::raw::c_void;
1+
use core::fmt;
2+
use types::c_void;
43

54
/// Inspects the current call-stack, passing all active frames into the closure
65
/// provided to calculate a stack trace.
@@ -37,10 +36,20 @@ use std::os::raw::c_void;
3736
/// }
3837
/// ```
3938
#[inline(never)]
39+
#[cfg(feature = "std")]
4040
pub fn trace<F: FnMut(&Frame) -> bool>(mut cb: F) {
41+
let _guard = ::lock::lock();
42+
unsafe { trace_imp(&mut cb) }
43+
}
44+
45+
/// Without std this function now does not have synchronization guarentees.
46+
/// Please refer to std documentation for examples and explaination.
47+
#[inline(never)]
48+
pub unsafe fn trace_unsynchronized<F: FnMut(&Frame) -> bool>(mut cb: F) {
4149
trace_imp(&mut cb)
4250
}
4351

52+
4453
/// A trait representing one frame of a backtrace, yielded to the `trace`
4554
/// function of this crate.
4655
///

src/backtrace/noop.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
use std::os::raw::c_void;
1+
use types::c_void;
22

33
#[inline(always)]
44
pub fn trace(_cb: &mut FnMut(&super::Frame) -> bool) {}

src/backtrace/unix_backtrace.rs

+6-7
Original file line numberDiff line numberDiff line change
@@ -8,8 +8,8 @@
88
// option. This file may not be copied, modified, or distributed
99
// except according to those terms.
1010

11-
use std::mem;
12-
use std::os::raw::{c_void, c_int};
11+
use core::mem;
12+
use types::{c_void, c_int};
1313

1414
pub struct Frame {
1515
addr: *mut c_void,
@@ -25,15 +25,14 @@ extern {
2525
}
2626

2727
#[inline(always)]
28-
pub fn trace(cb: &mut FnMut(&super::Frame) -> bool) {
28+
pub unsafe fn trace(cb: &mut FnMut(&super::Frame) -> bool) {
2929
const SIZE: usize = 100;
3030

3131
let mut buf: [*mut c_void; SIZE];
3232
let cnt;
33-
unsafe {
34-
buf = mem::zeroed();
35-
cnt = backtrace(buf.as_mut_ptr(), SIZE as c_int);
36-
}
33+
34+
buf = mem::zeroed();
35+
cnt = backtrace(buf.as_mut_ptr(), SIZE as c_int);
3736

3837
for addr in buf[..cnt as usize].iter() {
3938
let cx = super::Frame {

src/capture.rs

+3-2
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,10 @@
1+
use std::prelude::v1::*;
12
use std::fmt;
23
use std::mem;
3-
use std::os::raw::c_void;
44
use std::path::{Path, PathBuf};
55

66
use {trace, resolve, SymbolName};
7+
use types::c_void;
78

89
/// Representation of an owned and self-contained backtrace.
910
///
@@ -144,7 +145,7 @@ impl Backtrace {
144145
symbols.push(BacktraceSymbol {
145146
name: symbol.name().map(|m| m.as_bytes().to_vec()),
146147
addr: symbol.addr().map(|a| a as usize),
147-
filename: symbol.filename().map(|m| m.to_path_buf()),
148+
filename: symbol.filename().map(|m| m.to_owned()),
148149
lineno: symbol.lineno(),
149150
});
150151
});

src/lib.rs

+23-5
Original file line numberDiff line numberDiff line change
@@ -46,6 +46,8 @@
4646
//! extern crate backtrace;
4747
//!
4848
//! fn main() {
49+
//! # // Unsafe here so test passes on no_std.
50+
//! # #[cfg(feature = "std")] {
4951
//! backtrace::trace(|frame| {
5052
//! let ip = frame.ip();
5153
//! let symbol_address = frame.symbol_address();
@@ -63,10 +65,15 @@
6365
//! true // keep going to the next frame
6466
//! });
6567
//! }
68+
//! # }
6669
//! ```
6770
6871
#![doc(html_root_url = "https://docs.rs/backtrace")]
6972
#![deny(missing_docs)]
73+
#![no_std]
74+
75+
#[cfg(feature = "std")]
76+
#[macro_use] extern crate std;
7077

7178
#[cfg(unix)]
7279
extern crate libc;
@@ -98,18 +105,27 @@ cfg_if! {
98105
}
99106

100107
#[allow(dead_code)] // not used everywhere
101-
#[cfg(unix)]
108+
#[cfg(all(unix, feature = "std"))]
102109
#[macro_use]
103110
mod dylib;
104111

105-
pub use backtrace::{trace, Frame};
112+
pub use backtrace::{trace_unsynchronized, Frame};
106113
mod backtrace;
107114

108-
pub use symbolize::{resolve, Symbol, SymbolName};
115+
pub use symbolize::{resolve_unsynchronized, Symbol, SymbolName};
109116
mod symbolize;
110117

111-
pub use capture::{Backtrace, BacktraceFrame, BacktraceSymbol};
112-
mod capture;
118+
pub use types::BytesOrWideString;
119+
mod types;
120+
121+
cfg_if! {
122+
if #[cfg(feature = "std")] {
123+
pub use backtrace::trace;
124+
pub use symbolize::resolve;
125+
pub use capture::{Backtrace, BacktraceFrame, BacktraceSymbol};
126+
mod capture;
127+
}
128+
}
113129

114130
#[allow(dead_code)]
115131
struct Bomb {
@@ -126,8 +142,10 @@ impl Drop for Bomb {
126142
}
127143

128144
#[allow(dead_code)]
145+
#[cfg(feature = "std")]
129146
mod lock {
130147
use std::cell::Cell;
148+
use std::boxed::Box;
131149
use std::sync::{Once, Mutex, MutexGuard, ONCE_INIT};
132150

133151
pub struct LockGuard(MutexGuard<'static, ()>);

0 commit comments

Comments
 (0)