wasmer/lib/emscripten/src/jmp.rs

43 lines
1.6 KiB
Rust
Raw Normal View History

use super::env::get_emscripten_data;
2018-12-19 01:23:45 +00:00
use libc::{c_int, c_void};
use std::cell::UnsafeCell;
2019-01-23 07:27:13 +00:00
use wasmer_runtime_core::vm::Ctx;
2018-12-19 01:21:12 +00:00
/// setjmp
pub extern "C" fn __setjmp(env_addr: u32, ctx: &mut Ctx) -> c_int {
2018-12-19 01:21:12 +00:00
debug!("emscripten::__setjmp (setjmp)");
unsafe {
// Rather than using the env as the holder of the jump buffer pointer,
// we use the environment address to store the index relative to jumps
// so the address of the jump it's outside the wasm memory itself.
let jump_index = ctx.memory(0)[env_addr as usize] as *mut i8;
// We create the jump buffer outside of the wasm memory
let jump_buf: UnsafeCell<[c_int; 27]> = UnsafeCell::new([0; 27]);
let mut jumps = &mut get_emscripten_data(ctx).jumps;
let result = setjmp(jump_buf.get() as _);
// We set the jump index to be the last value of jumps
*jump_index = jumps.len() as _;
// We hold the reference of the jump buffer
jumps.push(jump_buf);
result
}
2018-12-19 01:21:12 +00:00
}
/// longjmp
pub extern "C" fn __longjmp(env_addr: u32, val: c_int, ctx: &mut Ctx) -> ! {
debug!("emscripten::__longjmp (longmp)");
unsafe {
// We retrieve the jump index from the env address
let jump_index = ctx.memory(0)[env_addr as usize] as *mut i8;
let mut jumps = &mut get_emscripten_data(ctx).jumps;
// We get the real jump buffer from the jumps vector, using the retrieved index
let mut jump_buf = &jumps[*jump_index as usize];
longjmp(jump_buf.get() as _, val)
};
2018-12-19 01:21:12 +00:00
}
extern "C" {
fn setjmp(env: *mut c_void) -> c_int;
fn longjmp(env: *mut c_void, val: c_int) -> !;
}