Files
wasmtime/wasmtime-api/src/values.rs
2019-09-12 17:11:34 -07:00

256 lines
6.3 KiB
Rust

use crate::callable::{WasmtimeFn, WrappedCallable};
use crate::runtime::{SignatureRegistry, Store};
use crate::types::ValType;
use std::any::Any;
use std::cell::RefCell;
use std::fmt;
use std::ptr;
use std::rc::Rc;
use cranelift_codegen::ir;
use wasmtime_jit::RuntimeValue;
#[derive(Clone)]
pub enum AnyRef {
Null,
Rc(Rc<dyn Any>),
Func(FuncRef),
}
impl AnyRef {
pub fn null() -> AnyRef {
AnyRef::Null
}
}
impl fmt::Debug for AnyRef {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
AnyRef::Null => write!(f, "null"),
AnyRef::Rc(_) => write!(f, "anyref"),
AnyRef::Func(func) => func.fmt(f),
}
}
}
#[derive(Clone)]
pub struct FuncRef(pub(crate) Rc<dyn WrappedCallable + 'static>);
impl fmt::Debug for FuncRef {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
write!(f, "funcref")
}
}
impl From<AnyRef> for FuncRef {
fn from(anyref: AnyRef) -> FuncRef {
match anyref {
AnyRef::Func(f) => f,
AnyRef::Rc(_) => unimplemented!("try to unwrap?"),
AnyRef::Null => panic!("null anyref"),
}
}
}
impl Into<AnyRef> for FuncRef {
fn into(self) -> AnyRef {
AnyRef::Func(self)
}
}
#[derive(Debug, Clone)]
pub enum Val {
I32(i32),
I64(i64),
F32(u32),
F64(u64),
AnyRef(AnyRef),
FuncRef(FuncRef),
}
impl Val {
pub fn default() -> Val {
Val::AnyRef(AnyRef::null())
}
pub fn r#type(&self) -> ValType {
match self {
Val::I32(_) => ValType::I32,
Val::I64(_) => ValType::I64,
Val::F32(_) => ValType::F32,
Val::F64(_) => ValType::F64,
Val::AnyRef(_) => ValType::AnyRef,
Val::FuncRef(_) => ValType::FuncRef,
}
}
pub(crate) unsafe fn write_value_to(&self, p: *mut i64) {
match self {
Val::I32(i) => ptr::write(p as *mut i32, *i),
Val::I64(i) => ptr::write(p as *mut i64, *i),
Val::F32(u) => ptr::write(p as *mut u32, *u),
Val::F64(u) => ptr::write(p as *mut u64, *u),
_ => unimplemented!("Val::write_value_to"),
}
}
pub(crate) unsafe fn read_value_from(p: *const i64, ty: ir::Type) -> Val {
match ty {
ir::types::I32 => Val::I32(ptr::read(p as *const i32)),
ir::types::I64 => Val::I64(ptr::read(p as *const i64)),
ir::types::F32 => Val::F32(ptr::read(p as *const u32)),
ir::types::F64 => Val::F64(ptr::read(p as *const u64)),
_ => unimplemented!("Val::read_value_from"),
}
}
pub fn from_f32_bits(v: u32) -> Val {
Val::F32(v)
}
pub fn from_f64_bits(v: u64) -> Val {
Val::F64(v)
}
}
impl From<i32> for Val {
fn from(val: i32) -> Val {
Val::I32(val)
}
}
impl From<i64> for Val {
fn from(val: i64) -> Val {
Val::I64(val)
}
}
impl From<f32> for Val {
fn from(val: f32) -> Val {
Val::F32(val.to_bits())
}
}
impl From<f64> for Val {
fn from(val: f64) -> Val {
Val::F64(val.to_bits())
}
}
impl Into<i32> for Val {
fn into(self) -> i32 {
if let Val::I32(i) = self {
i
} else {
panic!("Invalid conversion of {:?} to i32.", self);
}
}
}
impl Into<i64> for Val {
fn into(self) -> i64 {
if let Val::I64(i) = self {
i
} else {
panic!("Invalid conversion of {:?} to i64.", self);
}
}
}
impl Into<f32> for Val {
fn into(self) -> f32 {
if let Val::F32(i) = self {
RuntimeValue::F32(i).unwrap_f32()
} else {
panic!("Invalid conversion of {:?} to f32.", self);
}
}
}
impl Into<f64> for Val {
fn into(self) -> f64 {
if let Val::F64(i) = self {
RuntimeValue::F64(i).unwrap_f64()
} else {
panic!("Invalid conversion of {:?} to f64.", self);
}
}
}
impl From<AnyRef> for Val {
fn from(val: AnyRef) -> Val {
match val {
AnyRef::Func(f) => Val::FuncRef(f),
_ => Val::AnyRef(val),
}
}
}
impl From<FuncRef> for Val {
fn from(val: FuncRef) -> Val {
Val::FuncRef(val)
}
}
impl Into<AnyRef> for Val {
fn into(self) -> AnyRef {
match self {
Val::AnyRef(r) => r,
Val::FuncRef(f) => AnyRef::Func(f),
_ => panic!("Invalid conversion of {:?} to anyref.", self),
}
}
}
pub(crate) fn into_checked_anyfunc(
val: Val,
store: &Rc<RefCell<Store>>,
) -> wasmtime_runtime::VMCallerCheckedAnyfunc {
match val {
Val::AnyRef(AnyRef::Null) => wasmtime_runtime::VMCallerCheckedAnyfunc {
func_ptr: ptr::null(),
type_index: wasmtime_runtime::VMSharedSignatureIndex::default(),
vmctx: ptr::null_mut(),
},
Val::AnyRef(AnyRef::Func(f)) | Val::FuncRef(f) => {
let (vmctx, func_ptr, signature) = match f.0.wasmtime_export() {
wasmtime_runtime::Export::Function {
vmctx,
address,
signature,
} => (*vmctx, *address, signature),
_ => panic!("expected function export"),
};
let type_index = store.borrow_mut().register_cranelift_signature(signature);
wasmtime_runtime::VMCallerCheckedAnyfunc {
func_ptr,
type_index,
vmctx,
}
}
_ => panic!("val is not funcref"),
}
}
pub(crate) fn from_checked_anyfunc(
item: &wasmtime_runtime::VMCallerCheckedAnyfunc,
store: &Rc<RefCell<Store>>,
) -> Val {
if item.type_index == wasmtime_runtime::VMSharedSignatureIndex::default() {
return Val::AnyRef(AnyRef::Null);
}
let signature = store
.borrow()
.lookup_cranelift_signature(item.type_index)
.expect("signature")
.clone();
let instance_handle = unsafe { wasmtime_runtime::InstanceHandle::from_vmctx(item.vmctx) };
let export = wasmtime_runtime::Export::Function {
address: item.func_ptr,
signature,
vmctx: item.vmctx,
};
let f = WasmtimeFn::new(store.clone(), instance_handle, export);
Val::FuncRef(FuncRef(Rc::new(f)))
}