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
use core::fmt;

#[derive(Debug, Copy, Clone)]
pub struct ConvertError;
impl fmt::Display for ConvertError {
    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
        write!(f, "conversion error")
    }
}
impl From<ConvertError> for wasmi::Trap {
    fn from(_: ConvertError) -> wasmi::Trap {
        wasmi::TrapKind::UnexpectedSignature.into()
    }
}

/// Trait describing a type which can be used in wasm method signatures.
// XXX: Should this require `wasmi::FromRuntimeValue` and
// `Into<wasmi::RuntimeValue>`?
pub trait WasmPrimitive: Sized {
    const TYPE: wasmi::ValueType;

    fn from_wasm_value(value: wasmi::RuntimeValue) -> Result<Self, ConvertError>;
    fn into_wasm_value(self) -> wasmi::RuntimeValue;
}

macro_rules! impl_wasm_primitive {
    ($($rust:ty = $wasm:ident;)*) => {
        $(
            impl WasmPrimitive for $rust {
                const TYPE: wasmi::ValueType = wasmi::ValueType::$wasm;

                fn from_wasm_value(value: wasmi::RuntimeValue) -> Result<Self, ConvertError> {
                    value.try_into().ok_or(ConvertError)
                }

                fn into_wasm_value(self) -> wasmi::RuntimeValue {
                    self.into()
                }
            }
        )*
    }
}

impl_wasm_primitive! {
    i8 = I32;
    i16 = I32;
    i32 = I32;
    i64 = I64;
    u8 = I32;
    u16 = I32;
    u32 = I32;
    u64 = I64;
    wasmi::nan_preserving_float::F32 = F32;
    wasmi::nan_preserving_float::F64 = F64;
}