2019-02-14 09:58:33 -08:00
|
|
|
use crate::relocation::{TrapCode, TrapData};
|
2019-04-22 15:36:47 -07:00
|
|
|
use crate::signal::{CallProtError, HandlerData};
|
2019-02-14 09:58:33 -08:00
|
|
|
use crate::trampoline::Trampoline;
|
|
|
|
use std::cell::Cell;
|
|
|
|
use std::ffi::c_void;
|
2019-04-09 16:46:01 -07:00
|
|
|
use std::ptr::{self, NonNull};
|
2019-03-15 14:10:17 -07:00
|
|
|
use wasmer_runtime_core::error::{RuntimeError, RuntimeResult};
|
2019-04-22 15:06:40 -07:00
|
|
|
use wasmer_runtime_core::typed_func::WasmTrapInfo;
|
2019-02-14 09:58:33 -08:00
|
|
|
use wasmer_runtime_core::vm::Ctx;
|
|
|
|
use wasmer_runtime_core::vm::Func;
|
|
|
|
use wasmer_win_exception_handler::CallProtectedData;
|
|
|
|
pub use wasmer_win_exception_handler::_call_protected;
|
|
|
|
use winapi::shared::minwindef::DWORD;
|
|
|
|
use winapi::um::minwinbase::{
|
|
|
|
EXCEPTION_ACCESS_VIOLATION, EXCEPTION_FLT_DENORMAL_OPERAND, EXCEPTION_FLT_DIVIDE_BY_ZERO,
|
|
|
|
EXCEPTION_FLT_INEXACT_RESULT, EXCEPTION_FLT_INVALID_OPERATION, EXCEPTION_FLT_OVERFLOW,
|
|
|
|
EXCEPTION_FLT_STACK_CHECK, EXCEPTION_FLT_UNDERFLOW, EXCEPTION_ILLEGAL_INSTRUCTION,
|
|
|
|
EXCEPTION_INT_DIVIDE_BY_ZERO, EXCEPTION_INT_OVERFLOW, EXCEPTION_STACK_OVERFLOW,
|
|
|
|
};
|
2019-02-07 17:08:42 -08:00
|
|
|
|
2019-02-14 09:58:33 -08:00
|
|
|
thread_local! {
|
|
|
|
pub static CURRENT_EXECUTABLE_BUFFER: Cell<*const c_void> = Cell::new(ptr::null());
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn call_protected(
|
|
|
|
handler_data: &HandlerData,
|
|
|
|
trampoline: Trampoline,
|
|
|
|
ctx: *mut Ctx,
|
2019-04-09 16:46:01 -07:00
|
|
|
func: NonNull<Func>,
|
2019-02-14 09:58:33 -08:00
|
|
|
param_vec: *const u64,
|
|
|
|
return_vec: *mut u64,
|
2019-04-22 15:36:47 -07:00
|
|
|
) -> Result<(), CallProtError> {
|
2019-02-14 09:58:33 -08:00
|
|
|
// TODO: trap early
|
|
|
|
// user code error
|
|
|
|
// if let Some(msg) = super::TRAP_EARLY_DATA.with(|cell| cell.replace(None)) {
|
|
|
|
// return Err(RuntimeError::User { msg });
|
|
|
|
// }
|
|
|
|
|
|
|
|
let result = _call_protected(trampoline, ctx, func, param_vec, return_vec);
|
|
|
|
|
|
|
|
if let Ok(_) = result {
|
|
|
|
return Ok(());
|
|
|
|
}
|
|
|
|
|
|
|
|
let CallProtectedData {
|
|
|
|
code: signum,
|
2019-03-15 14:10:17 -07:00
|
|
|
exception_address,
|
|
|
|
instruction_pointer,
|
2019-02-14 09:58:33 -08:00
|
|
|
} = result.unwrap_err();
|
|
|
|
|
|
|
|
if let Some(TrapData {
|
|
|
|
trapcode,
|
|
|
|
srcloc: _,
|
|
|
|
}) = handler_data.lookup(instruction_pointer as _)
|
|
|
|
{
|
2019-04-22 15:36:47 -07:00
|
|
|
Err(CallProtError::Trap(match signum as DWORD {
|
2019-04-22 15:06:40 -07:00
|
|
|
EXCEPTION_ACCESS_VIOLATION => WasmTrapInfo::MemoryOutOfBounds,
|
2019-02-14 09:58:33 -08:00
|
|
|
EXCEPTION_ILLEGAL_INSTRUCTION => match trapcode {
|
2019-04-22 15:06:40 -07:00
|
|
|
TrapCode::BadSignature => WasmTrapInfo::IncorrectCallIndirectSignature,
|
|
|
|
TrapCode::IndirectCallToNull => WasmTrapInfo::CallIndirectOOB,
|
|
|
|
TrapCode::HeapOutOfBounds => WasmTrapInfo::MemoryOutOfBounds,
|
|
|
|
TrapCode::TableOutOfBounds => WasmTrapInfo::CallIndirectOOB,
|
|
|
|
TrapCode::UnreachableCodeReached => WasmTrapInfo::Unreachable,
|
|
|
|
_ => WasmTrapInfo::Unknown,
|
2019-03-04 12:57:26 -08:00
|
|
|
},
|
2019-04-22 15:06:40 -07:00
|
|
|
EXCEPTION_STACK_OVERFLOW => WasmTrapInfo::Unknown,
|
|
|
|
EXCEPTION_INT_DIVIDE_BY_ZERO | EXCEPTION_INT_OVERFLOW => {
|
|
|
|
WasmTrapInfo::IllegalArithmetic
|
|
|
|
}
|
|
|
|
_ => WasmTrapInfo::Unknown,
|
|
|
|
}))
|
2019-02-14 09:58:33 -08:00
|
|
|
} else {
|
|
|
|
let signal = match signum as DWORD {
|
|
|
|
EXCEPTION_FLT_DENORMAL_OPERAND
|
|
|
|
| EXCEPTION_FLT_DIVIDE_BY_ZERO
|
|
|
|
| EXCEPTION_FLT_INEXACT_RESULT
|
|
|
|
| EXCEPTION_FLT_INVALID_OPERATION
|
|
|
|
| EXCEPTION_FLT_OVERFLOW
|
|
|
|
| EXCEPTION_FLT_STACK_CHECK
|
|
|
|
| EXCEPTION_FLT_UNDERFLOW => "floating-point exception",
|
|
|
|
EXCEPTION_ILLEGAL_INSTRUCTION => "illegal instruction",
|
|
|
|
EXCEPTION_ACCESS_VIOLATION => "segmentation violation",
|
|
|
|
_ => "unkown trapped signal",
|
|
|
|
};
|
|
|
|
|
2019-04-22 15:06:40 -07:00
|
|
|
let s = format!("unknown trap at {} - {}", exception_address, signal);
|
|
|
|
|
2019-04-22 15:36:47 -07:00
|
|
|
Err(CallProtError::Error(Box::new(s)))
|
2019-02-14 09:58:33 -08:00
|
|
|
}
|
2019-02-07 17:08:42 -08:00
|
|
|
}
|
2019-02-08 13:51:29 -08:00
|
|
|
|
|
|
|
pub unsafe fn trigger_trap() -> ! {
|
2019-02-14 09:58:33 -08:00
|
|
|
// TODO
|
|
|
|
unimplemented!();
|
2019-02-08 13:51:29 -08:00
|
|
|
}
|