wasmtime/runtime/vm/unwind.rs
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59
//! Support for low-level primitives of unwinding the stack.
#[cfg(has_host_compiler_backend)]
use crate::runtime::vm::arch;
/// Implementation necessary to unwind the stack, used by `Backtrace`.
pub unsafe trait Unwind {
/// Returns the offset, from the current frame pointer, of where to get to
/// the previous frame pointer on the stack.
fn next_older_fp_from_fp_offset(&self) -> usize;
/// Load the return address of a frame given the frame pointer for that
/// frame.
unsafe fn get_next_older_pc_from_fp(&self, fp: usize) -> usize;
/// Debug assertion that the frame pointer is aligned.
fn assert_fp_is_aligned(&self, fp: usize);
}
/// A host-backed implementation of unwinding, using the native platform ABI
/// that Cranelift has.
#[cfg(has_host_compiler_backend)]
pub struct UnwindHost;
#[cfg(has_host_compiler_backend)]
unsafe impl Unwind for UnwindHost {
fn next_older_fp_from_fp_offset(&self) -> usize {
arch::NEXT_OLDER_FP_FROM_FP_OFFSET
}
unsafe fn get_next_older_pc_from_fp(&self, fp: usize) -> usize {
arch::get_next_older_pc_from_fp(fp)
}
fn assert_fp_is_aligned(&self, fp: usize) {
arch::assert_fp_is_aligned(fp)
}
}
/// An implementation specifically designed for unwinding Pulley's runtime stack
/// (which might not match the native host).
pub struct UnwindPulley;
unsafe impl Unwind for UnwindPulley {
fn next_older_fp_from_fp_offset(&self) -> usize {
0
}
unsafe fn get_next_older_pc_from_fp(&self, fp: usize) -> usize {
// The calling convention always pushes the return pointer (aka the PC
// of the next older frame) just before this frame.
*(fp as *mut usize).offset(1)
}
fn assert_fp_is_aligned(&self, fp: usize) {
let expected = if cfg!(target_pointer_width = "32") {
8
} else {
16
};
assert_eq!(fp % expected, 0, "stack should always be aligned");
}
}