108 lines
2.9 KiB
Rust
Raw Normal View History

//! This module contains types to manipulate and access a Wasm module's exports
//! including memories, tables, globals, and functions.
2019-01-10 22:59:57 -05:00
use crate::{
error,
global::Global,
instance::{Exports, InstanceInner},
memory::Memory,
module::ExportIndex,
module::ModuleInner,
table::Table,
types::FuncSig,
vm,
2019-01-10 22:59:57 -05:00
};
use indexmap::map::Iter as IndexMapIter;
use std::{ptr::NonNull, sync::Arc};
2019-01-10 22:59:57 -05:00
/// A kind of Context.
2019-01-12 16:24:17 -05:00
#[derive(Debug, Copy, Clone)]
2019-01-10 22:59:57 -05:00
pub enum Context {
/// External context include a mutable pointer to `Ctx`.
2019-01-10 22:59:57 -05:00
External(*mut vm::Ctx),
/// External context with an environment include a mutable pointer
/// to `Ctx` and an optional non-null pointer to `FuncEnv`.
ExternalWithEnv(*mut vm::Ctx, Option<NonNull<vm::FuncEnv>>),
/// Internal context.
2019-01-10 22:59:57 -05:00
Internal,
}
// Manually implemented because context contains a raw pointer to Ctx
2019-09-17 10:37:37 -07:00
unsafe impl Send for Context {}
/// Kind of WebAssembly export.
2019-01-10 22:59:57 -05:00
#[derive(Debug, Clone)]
pub enum Export {
/// Function export.
2019-01-10 22:59:57 -05:00
Function {
/// A pointer to a function.
2019-01-12 17:52:14 -05:00
func: FuncPointer,
/// A kind of context.
2019-01-10 22:59:57 -05:00
ctx: Context,
/// The signature of the function.
2019-01-29 10:16:39 -08:00
signature: Arc<FuncSig>,
2019-01-10 22:59:57 -05:00
},
/// Memory export.
Memory(Memory),
/// Table export.
2019-01-29 10:16:39 -08:00
Table(Table),
/// Global export.
2019-01-28 11:55:44 -08:00
Global(Global),
2019-01-10 22:59:57 -05:00
}
2019-01-12 17:52:14 -05:00
/// Const pointer to a `Func`.
2019-01-12 17:52:14 -05:00
#[derive(Debug, Clone)]
pub struct FuncPointer(*const vm::Func);
// Manually implemented because FuncPointer contains a raw pointer to Ctx
2019-09-17 10:37:37 -07:00
unsafe impl Send for FuncPointer {}
2019-01-12 17:52:14 -05:00
impl FuncPointer {
/// This needs to be unsafe because there is
/// no way to check whether the passed function
/// is valid and has the right signature.
pub unsafe fn new(f: *const vm::Func) -> Self {
FuncPointer(f)
}
pub(crate) fn inner(&self) -> *const vm::Func {
self.0
}
}
/// An iterator to an instance's exports.
2019-01-12 17:52:14 -05:00
pub struct ExportIter<'a> {
inner: &'a InstanceInner,
iter: IndexMapIter<'a, String, ExportIndex>,
2019-01-13 16:44:14 -05:00
module: &'a ModuleInner,
2019-01-12 17:52:14 -05:00
}
impl<'a> ExportIter<'a> {
pub(crate) fn new(module: &'a ModuleInner, inner: &'a InstanceInner) -> Self {
2019-01-12 17:52:14 -05:00
Self {
2019-01-13 16:44:14 -05:00
inner,
iter: module.info.exports.iter(),
2019-01-19 01:03:07 -06:00
module,
2019-01-12 17:52:14 -05:00
}
}
}
impl<'a> Iterator for ExportIter<'a> {
type Item = (String, Export);
fn next(&mut self) -> Option<(String, Export)> {
let (name, export_index) = self.iter.next()?;
Some((
name.clone(),
2019-01-13 16:44:14 -05:00
self.inner.get_export_from_index(&self.module, export_index),
2019-01-12 17:52:14 -05:00
))
}
2019-01-12 17:53:17 -05:00
}
/// This trait is used to mark types as gettable from an [`Instance`].
pub trait Exportable<'a>: Sized {
/// Implementation of how to get the export corresponding to the implementing type
/// from an [`Instance`] by name.
fn get_self(exports: &'a Exports, name: &str) -> error::ResolveResult<Self>;
}