use crate::store::{InstanceId, StoreOpaque};
use crate::trampoline::create_handle;
use crate::{GlobalType, Mutability, Val};
use anyhow::Result;
use wasmtime_environ::{
AnyfuncIndex, EntityIndex, Global, GlobalInit, Module, ModuleType, SignatureIndex,
};
use wasmtime_runtime::VMFunctionImport;
pub fn create_global(store: &mut StoreOpaque, gt: &GlobalType, val: Val) -> Result<InstanceId> {
let mut module = Module::new();
let mut func_imports = Vec::new();
let mut externref_init = None;
let mut one_signature = None;
let global = Global {
wasm_ty: gt.content().to_wasm_type(),
mutability: match gt.mutability() {
Mutability::Const => false,
Mutability::Var => true,
},
initializer: match val {
Val::I32(i) => GlobalInit::I32Const(i),
Val::I64(i) => GlobalInit::I64Const(i),
Val::F32(f) => GlobalInit::F32Const(f),
Val::F64(f) => GlobalInit::F64Const(f),
Val::V128(i) => GlobalInit::V128Const(i.into()),
Val::ExternRef(None) | Val::FuncRef(None) => GlobalInit::RefNullConst,
Val::ExternRef(Some(x)) => {
externref_init = Some(x);
GlobalInit::RefNullConst
}
Val::FuncRef(Some(f)) => {
let f = f.caller_checked_anyfunc(store);
let f = unsafe { f.as_ref() };
let sig_id = SignatureIndex::from_u32(u32::max_value() - 1);
one_signature = Some((sig_id, f.type_index));
module.types.push(ModuleType::Function(sig_id));
let func_index = module.push_escaped_function(sig_id, AnyfuncIndex::from_u32(0));
module.num_imported_funcs = 1;
module.num_escaped_funcs = 1;
module
.initializers
.push(wasmtime_environ::Initializer::Import {
name: "".into(),
field: "".into(),
index: EntityIndex::Function(func_index),
});
func_imports.push(VMFunctionImport {
body: f.func_ptr,
vmctx: f.vmctx,
});
GlobalInit::RefFunc(func_index)
}
},
};
let global_id = module.globals.push(global);
module
.exports
.insert(String::new(), EntityIndex::Global(global_id));
let id = create_handle(module, store, Box::new(()), &func_imports, one_signature)?;
if let Some(x) = externref_init {
let instance = store.instance_mut(id);
let g = instance.get_exported_global(global_id);
unsafe {
*(*g.definition).as_externref_mut() = Some(x.inner);
}
}
Ok(id)
}