1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93
//! Generic Artifact abstraction for Wasmer Engines.
use crate::Features;
use enumset::EnumSet;
use std::any::Any;
use std::sync::Arc;
use wasmer_types::entity::PrimaryMap;
use wasmer_types::SerializeError;
use wasmer_types::{
CpuFeature, DataInitializerLike, MemoryIndex, MemoryStyle, ModuleInfo, TableIndex, TableStyle,
};
/// An `Artifact` is the product that the `Engine`
/// implementation produce and use.
///
/// The `Artifact` contains the compiled data for a given
/// module as well as extra information needed to run the
/// module at runtime, such as [`ModuleInfo`] and [`Features`].
pub trait ArtifactCreate<'a>: Send + Sync + Upcastable {
/// Type of `OwnedDataInitializer` returned by the `data_initializers` method
type OwnedDataInitializer: DataInitializerLike<'a> + 'a;
/// Type of iterator returned by the `data_initializers` method
type OwnedDataInitializerIterator: Iterator<Item = Self::OwnedDataInitializer>;
/// Create a `ModuleInfo` for instantiation
fn create_module_info(&'a self) -> Arc<ModuleInfo>;
/// Sets the `ModuleInfo` name
fn set_module_info_name(&'a mut self, name: String) -> bool;
/// Returns the `ModuleInfo` for instantiation
fn module_info(&'a self) -> &ModuleInfo;
/// Returns the features for this Artifact
fn features(&'a self) -> &Features;
/// Returns the CPU features for this Artifact
fn cpu_features(&'a self) -> EnumSet<CpuFeature>;
/// Returns the memory styles associated with this `Artifact`.
fn memory_styles(&'a self) -> &PrimaryMap<MemoryIndex, MemoryStyle>;
/// Returns the table plans associated with this `Artifact`.
fn table_styles(&'a self) -> &PrimaryMap<TableIndex, TableStyle>;
/// Returns data initializers to pass to `VMInstance::initialize`
fn data_initializers(&'a self) -> Self::OwnedDataInitializerIterator;
/// Serializes an artifact into bytes
fn serialize(&'a self) -> Result<Vec<u8>, SerializeError>;
}
// Implementation of `Upcastable` taken from https://users.rust-lang.org/t/why-does-downcasting-not-work-for-subtraits/33286/7 .
/// Trait needed to get downcasting of `Engine`s to work.
pub trait Upcastable {
/// upcast ref
fn upcast_any_ref(&'_ self) -> &'_ dyn Any;
/// upcast mut ref
fn upcast_any_mut(&'_ mut self) -> &'_ mut dyn Any;
/// upcast boxed dyn
fn upcast_any_box(self: Box<Self>) -> Box<dyn Any>;
}
impl<T: Any + Send + Sync + 'static> Upcastable for T {
#[inline]
fn upcast_any_ref(&'_ self) -> &'_ dyn Any {
self
}
#[inline]
fn upcast_any_mut(&'_ mut self) -> &'_ mut dyn Any {
self
}
#[inline]
fn upcast_any_box(self: Box<Self>) -> Box<dyn Any> {
self
}
}
impl<'a, O, I>
dyn ArtifactCreate<'a, OwnedDataInitializer = O, OwnedDataInitializerIterator = I> + 'static
{
/// Try to downcast the artifact into a given type.
#[inline]
pub fn downcast_ref<T: 'static>(&'a self) -> Option<&'a T> {
self.upcast_any_ref().downcast_ref::<T>()
}
/// Try to downcast the artifact into a given type mutably.
#[inline]
pub fn downcast_mut<T: 'static>(&'a mut self) -> Option<&'a mut T> {
self.upcast_any_mut().downcast_mut::<T>()
}
}