use crate::runtime::vm::{GcRootsList, SendSyncPtr};
use crate::{
store::{AutoAssertNoGc, StoreData, StoreOpaque, Stored},
trampoline::generate_global_export,
AnyRef, AsContext, AsContextMut, ExternRef, Func, GlobalType, HeapType, Mutability, Ref,
RootedGcRefImpl, Val, ValType,
};
use anyhow::{bail, Context, Result};
use core::ptr;
use core::ptr::NonNull;
use wasmtime_environ::TypeTrace;
#[derive(Copy, Clone, Debug)]
#[repr(transparent)] pub struct Global(pub(super) Stored<crate::runtime::vm::ExportGlobal>);
impl Global {
pub fn new(mut store: impl AsContextMut, ty: GlobalType, val: Val) -> Result<Global> {
Global::_new(store.as_context_mut().0, ty, val)
}
fn _new(store: &mut StoreOpaque, ty: GlobalType, val: Val) -> Result<Global> {
val.ensure_matches_ty(store, ty.content()).context(
"type mismatch: initial value provided does not match the type of this global",
)?;
unsafe {
let wasmtime_export = generate_global_export(store, ty, val)?;
Ok(Global::from_wasmtime_global(wasmtime_export, store))
}
}
pub fn ty(&self, store: impl AsContext) -> GlobalType {
self._ty(store.as_context().0)
}
pub(crate) fn _ty(&self, store: &StoreOpaque) -> GlobalType {
let ty = &store[self.0].global;
GlobalType::from_wasmtime_global(store.engine(), &ty)
}
pub fn get(&self, mut store: impl AsContextMut) -> Val {
unsafe {
let store = store.as_context_mut();
let mut store = AutoAssertNoGc::new(store.0);
let definition = &*store[self.0].definition;
match self._ty(&store).content() {
ValType::I32 => Val::from(*definition.as_i32()),
ValType::I64 => Val::from(*definition.as_i64()),
ValType::F32 => Val::F32(*definition.as_u32()),
ValType::F64 => Val::F64(*definition.as_u64()),
ValType::V128 => Val::V128((*definition.as_u128()).into()),
ValType::Ref(ref_ty) => {
let reference: Ref = match ref_ty.heap_type() {
HeapType::Func | HeapType::ConcreteFunc(_) => {
Func::_from_raw(&mut store, definition.as_func_ref().cast()).into()
}
HeapType::NoFunc => Ref::Func(None),
HeapType::Extern => Ref::Extern(
definition
.as_gc_ref()
.map(|r| {
let r = store.unwrap_gc_store_mut().clone_gc_ref(r);
ExternRef::from_cloned_gc_ref(&mut store, r)
})
.into(),
),
HeapType::NoExtern => Ref::Extern(None),
HeapType::Any
| HeapType::Eq
| HeapType::I31
| HeapType::Struct
| HeapType::ConcreteStruct(_)
| HeapType::Array
| HeapType::ConcreteArray(_) => definition
.as_gc_ref()
.map(|r| {
let r = store.unwrap_gc_store_mut().clone_gc_ref(r);
AnyRef::from_cloned_gc_ref(&mut store, r)
})
.into(),
HeapType::None => Ref::Any(None),
};
debug_assert!(
ref_ty.is_nullable() || !reference.is_null(),
"if the type is non-nullable, we better have a non-null reference"
);
reference.into()
}
}
}
}
pub fn set(&self, mut store: impl AsContextMut, val: Val) -> Result<()> {
let mut store = AutoAssertNoGc::new(store.as_context_mut().0);
let global_ty = self._ty(&store);
if global_ty.mutability() != Mutability::Var {
bail!("immutable global cannot be set");
}
val.ensure_matches_ty(&store, global_ty.content())
.context("type mismatch: attempt to set global to value of wrong type")?;
unsafe {
let definition = &mut *store[self.0].definition;
match val {
Val::I32(i) => *definition.as_i32_mut() = i,
Val::I64(i) => *definition.as_i64_mut() = i,
Val::F32(f) => *definition.as_u32_mut() = f,
Val::F64(f) => *definition.as_u64_mut() = f,
Val::V128(i) => *definition.as_u128_mut() = i.into(),
Val::FuncRef(f) => {
*definition.as_func_ref_mut() = f.map_or(ptr::null_mut(), |f| {
f.vm_func_ref(&mut store).as_ptr().cast()
});
}
Val::ExternRef(e) => {
let new = match e {
None => None,
Some(e) => Some(e.try_gc_ref(&mut store)?.unchecked_copy()),
};
let new = new.as_ref();
definition.write_gc_ref(store.unwrap_gc_store_mut(), new);
}
Val::AnyRef(a) => {
let new = match a {
None => None,
Some(a) => Some(a.try_gc_ref(&mut store)?.unchecked_copy()),
};
let new = new.as_ref();
definition.write_gc_ref(store.unwrap_gc_store_mut(), new);
}
}
}
Ok(())
}
pub(crate) fn trace_root(&self, store: &mut StoreOpaque, gc_roots_list: &mut GcRootsList) {
if let Some(ref_ty) = self._ty(store).content().as_ref() {
if !ref_ty.is_vmgcref_type_and_points_to_object() {
return;
}
if let Some(gc_ref) = unsafe { (*store[self.0].definition).as_gc_ref() } {
let gc_ref = NonNull::from(gc_ref);
let gc_ref = SendSyncPtr::new(gc_ref);
unsafe {
gc_roots_list.add_root(gc_ref);
}
}
}
}
pub(crate) unsafe fn from_wasmtime_global(
mut wasmtime_export: crate::runtime::vm::ExportGlobal,
store: &mut StoreOpaque,
) -> Global {
wasmtime_export
.global
.wasm_ty
.canonicalize_for_runtime_usage(&mut |module_index| {
crate::runtime::vm::Instance::from_vmctx(wasmtime_export.vmctx, |instance| {
instance.engine_type_index(module_index)
})
});
Global(store.store_data_mut().insert(wasmtime_export))
}
pub(crate) fn wasmtime_ty<'a>(&self, data: &'a StoreData) -> &'a wasmtime_environ::Global {
&data[self.0].global
}
pub(crate) fn vmimport(&self, store: &StoreOpaque) -> crate::runtime::vm::VMGlobalImport {
crate::runtime::vm::VMGlobalImport {
from: store[self.0].definition,
}
}
pub(crate) fn hash_key(&self, store: &StoreOpaque) -> impl core::hash::Hash + Eq {
store[self.0].definition as usize
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::{Instance, Module, Store};
#[test]
fn hash_key_is_stable_across_duplicate_store_data_entries() -> Result<()> {
let mut store = Store::<()>::default();
let module = Module::new(
store.engine(),
r#"
(module
(global (export "g") (mut i32) (i32.const 0))
)
"#,
)?;
let instance = Instance::new(&mut store, &module, &[])?;
let g1 = instance.get_global(&mut store, "g").unwrap();
let g2 = instance.get_global(&mut store, "g").unwrap();
assert_eq!(g1.get(&mut store).unwrap_i32(), 0);
assert_eq!(g2.get(&mut store).unwrap_i32(), 0);
g1.set(&mut store, Val::I32(42))?;
assert_eq!(g1.get(&mut store).unwrap_i32(), 42);
assert_eq!(g2.get(&mut store).unwrap_i32(), 42);
assert!(g1.hash_key(&store.as_context().0) == g2.hash_key(&store.as_context().0));
let instance2 = Instance::new(&mut store, &module, &[])?;
let g3 = instance2.get_global(&mut store, "g").unwrap();
assert!(g1.hash_key(&store.as_context().0) != g3.hash_key(&store.as_context().0));
Ok(())
}
}