Skip to content
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.

Commit 8ef7580

Browse files
author
Aaron Power
committedOct 5, 2018
no_std support
1 parent a73a623 commit 8ef7580

15 files changed

+136
-45
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/libunwind.rs

+2-2
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,
@@ -84,7 +84,7 @@ mod uw {
8484
pub use self::_Unwind_Reason_Code::*;
8585

8686
use libc;
87-
use std::os::raw::{c_int, c_void};
87+
use types::{c_int, c_void};
8888

8989
#[repr(C)]
9090
pub enum _Unwind_Reason_Code {

‎src/backtrace/mod.rs

+3-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+
#[cfg(feature = "std")] use std::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.
@@ -77,6 +76,7 @@ impl Frame {
7776
}
7877
}
7978

79+
#[cfg(feature = "std")]
8080
impl fmt::Debug for Frame {
8181
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
8282
f.debug_struct("Frame")

‎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

+8-2
Original file line numberDiff line numberDiff line change
@@ -8,8 +8,14 @@
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+
cfg_if! {
12+
if #[cfg(feature = "std")] {
13+
use std::mem;
14+
} else {
15+
use core::mem;
16+
}
17+
}
18+
use types::{c_void, c_int};
1319

1420
pub struct Frame {
1521
addr: *mut c_void,

‎src/capture.rs

+22-5
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,14 @@
1-
use std::fmt;
2-
use std::mem;
3-
use std::os::raw::c_void;
4-
use std::path::{Path, PathBuf};
1+
cfg_if! {
2+
if #[cfg(feature = "std")] {
3+
use std::fmt;
4+
use std::mem;
5+
use std::path::{Path, PathBuf};
56

6-
use {trace, resolve, SymbolName};
7+
use {trace, resolve, SymbolName};
8+
}
9+
}
10+
11+
use types::c_void;
712

813
/// Representation of an owned and self-contained backtrace.
914
///
@@ -12,6 +17,7 @@ use {trace, resolve, SymbolName};
1217
#[derive(Clone)]
1318
#[cfg_attr(feature = "serialize-rustc", derive(RustcDecodable, RustcEncodable))]
1419
#[cfg_attr(feature = "serialize-serde", derive(Deserialize, Serialize))]
20+
#[cfg(feature = "std")]
1521
pub struct Backtrace {
1622
// Frames here are listed from top-to-bottom of the stack
1723
frames: Vec<BacktraceFrame>,
@@ -30,6 +36,7 @@ pub struct Backtrace {
3036
pub struct BacktraceFrame {
3137
ip: usize,
3238
symbol_address: usize,
39+
#[cfg(feature = "std")]
3340
symbols: Option<Vec<BacktraceSymbol>>,
3441
}
3542

@@ -41,12 +48,15 @@ pub struct BacktraceFrame {
4148
#[cfg_attr(feature = "serialize-rustc", derive(RustcDecodable, RustcEncodable))]
4249
#[cfg_attr(feature = "serialize-serde", derive(Deserialize, Serialize))]
4350
pub struct BacktraceSymbol {
51+
#[cfg(feature = "std")]
4452
name: Option<Vec<u8>>,
4553
addr: Option<usize>,
54+
#[cfg(feature = "std")]
4655
filename: Option<PathBuf>,
4756
lineno: Option<u32>,
4857
}
4958

59+
#[cfg(feature = "std")]
5060
impl Backtrace {
5161
/// Captures a backtrace at the callsite of this function, returning an
5262
/// owned representation.
@@ -153,6 +163,7 @@ impl Backtrace {
153163
}
154164
}
155165

166+
#[cfg(feature = "std")]
156167
impl From<Vec<BacktraceFrame>> for Backtrace {
157168
fn from(frames: Vec<BacktraceFrame>) -> Self {
158169
Backtrace {
@@ -162,6 +173,7 @@ impl From<Vec<BacktraceFrame>> for Backtrace {
162173
}
163174
}
164175

176+
#[cfg(feature = "std")]
165177
impl Into<Vec<BacktraceFrame>> for Backtrace {
166178
fn into(self) -> Vec<BacktraceFrame> {
167179
self.frames
@@ -188,13 +200,15 @@ impl BacktraceFrame {
188200
///
189201
/// Note that if this frame came from an unresolved backtrace then this will
190202
/// return an empty list.
203+
#[cfg(feature = "std")]
191204
pub fn symbols(&self) -> &[BacktraceSymbol] {
192205
self.symbols.as_ref().map(|s| &s[..]).unwrap_or(&[])
193206
}
194207
}
195208

196209
impl BacktraceSymbol {
197210
/// Same as `Symbol::name`
211+
#[cfg(feature = "std")]
198212
pub fn name(&self) -> Option<SymbolName> {
199213
self.name.as_ref().map(|s| SymbolName::new(s))
200214
}
@@ -205,6 +219,7 @@ impl BacktraceSymbol {
205219
}
206220

207221
/// Same as `Symbol::filename`
222+
#[cfg(feature = "std")]
208223
pub fn filename(&self) -> Option<&Path> {
209224
self.filename.as_ref().map(|p| &**p)
210225
}
@@ -215,6 +230,7 @@ impl BacktraceSymbol {
215230
}
216231
}
217232

233+
#[cfg(feature = "std")]
218234
impl fmt::Debug for Backtrace {
219235
fn fmt(&self, fmt: &mut fmt::Formatter) -> fmt::Result {
220236
let hex_width = mem::size_of::<usize>() * 2 + 2;
@@ -263,6 +279,7 @@ impl fmt::Debug for Backtrace {
263279
}
264280
}
265281

282+
#[cfg(feature = "std")]
266283
impl Default for Backtrace {
267284
fn default() -> Backtrace {
268285
Backtrace::new()

‎src/lib.rs

+9-2
Original file line numberDiff line numberDiff line change
@@ -46,6 +46,7 @@
4646
//! extern crate backtrace;
4747
//!
4848
//! fn main() {
49+
//! # #[cfg(feature = "std")]
4950
//! backtrace::trace(|frame| {
5051
//! let ip = frame.ip();
5152
//! let symbol_address = frame.symbol_address();
@@ -67,6 +68,7 @@
6768
6869
#![doc(html_root_url = "https://docs.rs/backtrace")]
6970
#![deny(missing_docs)]
71+
#![cfg_attr(not(feature = "std"), no_std)]
7072

7173
#[cfg(unix)]
7274
extern crate libc;
@@ -98,7 +100,7 @@ cfg_if! {
98100
}
99101

100102
#[allow(dead_code)] // not used everywhere
101-
#[cfg(unix)]
103+
#[cfg(all(unix, feature = "std"))]
102104
#[macro_use]
103105
mod dylib;
104106

@@ -108,9 +110,13 @@ mod backtrace;
108110
pub use symbolize::{resolve, Symbol, SymbolName};
109111
mod symbolize;
110112

111-
pub use capture::{Backtrace, BacktraceFrame, BacktraceSymbol};
113+
#[cfg(feature = "std")]
114+
pub use capture::Backtrace;
115+
pub use capture::{BacktraceFrame, BacktraceSymbol};
112116
mod capture;
113117

118+
mod types;
119+
114120
#[allow(dead_code)]
115121
struct Bomb {
116122
enabled: bool,
@@ -126,6 +132,7 @@ impl Drop for Bomb {
126132
}
127133

128134
#[allow(dead_code)]
135+
#[cfg(feature = "std")]
129136
mod lock {
130137
use std::cell::Cell;
131138
use std::sync::{Once, Mutex, MutexGuard, ONCE_INIT};

‎src/symbolize/coresymbolication.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,6 @@
1212

1313
use std::ffi::{CStr, OsStr};
1414
use std::mem;
15-
use std::os::raw::{c_void, c_char, c_int};
1615
use std::os::unix::prelude::*;
1716
use std::path::Path;
1817
use std::ptr;
@@ -23,6 +22,7 @@ use libc::{self, Dl_info};
2322
use SymbolName;
2423
use dylib::Dylib;
2524
use dylib::Symbol as DylibSymbol;
25+
use types::{c_void, c_char, c_int};
2626

2727
#[repr(C)]
2828
#[derive(Copy, Clone, PartialEq)]

‎src/symbolize/dladdr.rs

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

11-
use std::ffi::CStr;
12-
use std::mem;
13-
use std::os::raw::c_void;
14-
use std::path::Path;
11+
use core::mem;
1512

13+
#[cfg(feature = "std")] use std::ffi::CStr;
14+
#[cfg(feature = "std")] use std::path::Path;
15+
16+
use types::c_void;
1617
use libc::{self, Dl_info};
1718

18-
use SymbolName;
19+
#[cfg(feature = "std")] use SymbolName;
1920

2021
pub struct Symbol {
2122
inner: Dl_info,
2223
}
2324

2425
impl Symbol {
26+
#[cfg(feature = "std")]
2527
pub fn name(&self) -> Option<SymbolName> {
2628
if self.inner.dli_sname.is_null() {
2729
None
@@ -36,6 +38,7 @@ impl Symbol {
3638
Some(self.inner.dli_saddr as *mut _)
3739
}
3840

41+
#[cfg(feature = "std")]
3942
pub fn filename(&self) -> Option<&Path> {
4043
None
4144
}

‎src/symbolize/gimli.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@ use std::cell::RefCell;
77
use std::env;
88
use std::fs::File;
99
use std::mem;
10-
use std::os::raw::c_void;
10+
use libc::c_void;
1111
use std::path::{Path, PathBuf};
1212
use std::u32;
1313

‎src/symbolize/libbacktrace.rs

+4-4
Original file line numberDiff line numberDiff line change
@@ -12,14 +12,14 @@
1212

1313
extern crate backtrace_sys as bt;
1414

15-
use libc::uintptr_t;
1615
use std::ffi::{CStr, OsStr};
17-
use std::os::raw::{c_void, c_char, c_int};
1816
use std::os::unix::prelude::*;
1917
use std::path::Path;
2018
use std::ptr;
2119
use std::sync::{ONCE_INIT, Once};
2220

21+
use libc::{c_char, c_int, c_void, uintptr_t};
22+
2323
use SymbolName;
2424

2525
pub enum Symbol {
@@ -169,11 +169,11 @@ pub fn resolve(symaddr: *mut c_void, mut cb: &mut FnMut(&super::Symbol)) {
169169
}
170170

171171
let ret = bt::backtrace_pcinfo(state, symaddr as uintptr_t,
172-
pcinfo_cb, error_cb,
172+
pcinfo_cb as *mut _, error_cb as *mut _,
173173
&mut cb as *mut _ as *mut _);
174174
if ret != 0 {
175175
bt::backtrace_syminfo(state, symaddr as uintptr_t,
176-
syminfo_cb, error_cb,
176+
syminfo_cb as *mut _, error_cb as *mut _,
177177
&mut cb as *mut _ as *mut _);
178178
}
179179
}

‎src/symbolize/mod.rs

+35-13
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,27 @@
1-
use std::fmt;
2-
#[cfg(not(feature = "cpp_demangle"))]
3-
use std::marker::PhantomData;
4-
use std::os::raw::c_void;
5-
use std::path::Path;
6-
use std::str;
1+
cfg_if! {
2+
if #[cfg(feature = "std")] {
3+
use std::fmt;
4+
use std::path::Path;
5+
}
6+
}
7+
8+
cfg_if! {
9+
if #[cfg(all(feature = "std", not(feature = "cpp_demangle")))] {
10+
use std::marker::PhantomData;
11+
} else if #[cfg(not(feature = "cpp_demangle"))] {
12+
use core::marker::PhantomData;
13+
}
14+
}
15+
16+
cfg_if! {
17+
if #[cfg(feature = "std")] {
18+
use std::str;
19+
} else {
20+
use core::str;
21+
}
22+
}
23+
24+
use types::c_void;
725
use rustc_demangle::{try_demangle, Demangle};
826

927
/// Resolve an address to a symbol, passing the symbol to the specified
@@ -37,7 +55,7 @@ use rustc_demangle::{try_demangle, Demangle};
3755
/// }
3856
/// ```
3957
pub fn resolve<F: FnMut(&Symbol)>(addr: *mut c_void, mut cb: F) {
40-
resolve_imp(addr, &mut cb)
58+
resolve_imp(addr as *mut _, &mut cb)
4159
}
4260

4361
/// A trait representing the resolution of a symbol in a file.
@@ -63,13 +81,14 @@ impl Symbol {
6381
/// * The raw `str` value of the symbol can be accessed (if it's valid
6482
/// utf-8).
6583
/// * The raw bytes for the symbol name can be accessed.
84+
#[cfg(not(all(not(feature = "std"), feature = "dladdr")))]
6685
pub fn name(&self) -> Option<SymbolName> {
6786
self.inner.name()
6887
}
6988

7089
/// Returns the starting address of this function.
7190
pub fn addr(&self) -> Option<*mut c_void> {
72-
self.inner.addr()
91+
self.inner.addr().map(|p| p as *mut _)
7392
}
7493

7594
/// Returns the file name where this function was defined.
@@ -78,6 +97,7 @@ impl Symbol {
7897
/// unix platforms other than OSX) and when a binary is compiled with
7998
/// debuginfo. If neither of these conditions is met then this will likely
8099
/// return `None`.
100+
#[cfg(feature = "std")]
81101
pub fn filename(&self) -> Option<&Path> {
82102
self.inner.filename()
83103
}
@@ -91,6 +111,7 @@ impl Symbol {
91111
}
92112
}
93113

114+
#[cfg(feature = "std")]
94115
impl fmt::Debug for Symbol {
95116
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
96117
let mut d = f.debug_struct("Symbol");
@@ -189,7 +210,7 @@ impl<'a> SymbolName<'a> {
189210
}
190211

191212
cfg_if! {
192-
if #[cfg(feature = "cpp_demangle")] {
213+
if #[cfg(all(feature = "cpp_demangle", feature = "std"))] {
193214
impl<'a> fmt::Display for SymbolName<'a> {
194215
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
195216
if let Some(ref s) = self.demangled {
@@ -201,7 +222,7 @@ cfg_if! {
201222
}
202223
}
203224
}
204-
} else {
225+
} else if #[cfg(feature = "std")] {
205226
impl<'a> fmt::Display for SymbolName<'a> {
206227
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
207228
if let Some(ref s) = self.demangled {
@@ -215,7 +236,7 @@ cfg_if! {
215236
}
216237

217238
cfg_if! {
218-
if #[cfg(feature = "cpp_demangle")] {
239+
if #[cfg(all(feature = "cpp_demangle", feature = "std"))] {
219240
impl<'a> fmt::Debug for SymbolName<'a> {
220241
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
221242
use std::fmt::Write;
@@ -237,7 +258,7 @@ cfg_if! {
237258
String::from_utf8_lossy(self.bytes).fmt(f)
238259
}
239260
}
240-
} else {
261+
} else if #[cfg(feature = "std")] {
241262
impl<'a> fmt::Debug for SymbolName<'a> {
242263
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
243264
if let Some(ref s) = self.demangled {
@@ -255,7 +276,8 @@ cfg_if! {
255276
mod dbghelp;
256277
use self::dbghelp::resolve as resolve_imp;
257278
use self::dbghelp::Symbol as SymbolImp;
258-
} else if #[cfg(all(feature = "gimli-symbolize",
279+
} else if #[cfg(all(feature = "std",
280+
feature = "gimli-symbolize",
259281
unix,
260282
target_os = "linux"))] {
261283
mod gimli;

‎src/symbolize/noop.rs

+3-2
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
1-
use std::path::Path;
2-
use std::os::raw::c_void;
1+
#[cfg(feature = "std")] use std::path::Path;
32

3+
use types::c_void;
44
use SymbolName;
55

66
pub fn resolve(_addr: *mut c_void, _cb: &mut FnMut(&super::Symbol)) {
@@ -17,6 +17,7 @@ impl Symbol {
1717
None
1818
}
1919

20+
#[cfg(feature = "std")]
2021
pub fn filename(&self) -> Option<&Path> {
2122
None
2223
}

‎src/types.rs

+10
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
cfg_if! {
2+
if #[cfg(feature = "std")] {
3+
pub use std::os::raw::*;
4+
} else if #[cfg(all(windows, feature = "winapi"))] {
5+
pub use winapi::ctypes::*;
6+
} else {
7+
pub use libc::*;
8+
}
9+
}
10+

0 commit comments

Comments
 (0)
Please sign in to comment.