use crate::{
core::{TypedVal, UntypedVal},
ExternRef,
Func,
Ref,
ValType,
F32,
F64,
V128,
};
pub trait WithType {
type Output;
fn with_type(self, ty: ValType) -> Self::Output;
}
impl WithType for UntypedVal {
type Output = Val;
fn with_type(self, ty: ValType) -> Self::Output {
match ty {
ValType::I32 => Val::I32(self.into()),
ValType::I64 => Val::I64(self.into()),
ValType::F32 => Val::F32(self.into()),
ValType::F64 => Val::F64(self.into()),
#[cfg(feature = "simd")]
ValType::V128 => Val::V128(self.into()),
ValType::FuncRef => Val::FuncRef(self.into()),
ValType::ExternRef => Val::ExternRef(self.into()),
#[cfg(not(feature = "simd"))]
unsupported => unimplemented!("encountered unsupported `ValType`: {unsupported:?}"),
}
}
}
impl From<Val> for UntypedVal {
fn from(value: Val) -> Self {
match value {
Val::I32(value) => value.into(),
Val::I64(value) => value.into(),
Val::F32(value) => value.into(),
Val::F64(value) => value.into(),
#[cfg(feature = "simd")]
Val::V128(value) => value.into(),
Val::FuncRef(value) => value.into(),
Val::ExternRef(value) => value.into(),
#[cfg(not(feature = "simd"))]
unsupported => unimplemented!("encountered unsupported `Val`: {unsupported:?}"),
}
}
}
#[derive(Clone, Debug)]
pub enum Val {
I32(i32),
I64(i64),
F32(F32),
F64(F64),
V128(V128),
FuncRef(Ref<Func>),
ExternRef(Ref<ExternRef>),
}
impl Val {
#[inline]
pub fn default(value_type: ValType) -> Self {
match value_type {
ValType::I32 => Self::I32(0),
ValType::I64 => Self::I64(0),
ValType::F32 => Self::F32(0f32.into()),
ValType::F64 => Self::F64(0f64.into()),
ValType::V128 => Self::V128(V128::from(0_u128)),
ValType::FuncRef => Self::from(<Ref<Func>>::Null),
ValType::ExternRef => Self::from(<Ref<ExternRef>>::Null),
}
}
#[inline]
pub fn ty(&self) -> ValType {
match *self {
Self::I32(_) => ValType::I32,
Self::I64(_) => ValType::I64,
Self::F32(_) => ValType::F32,
Self::F64(_) => ValType::F64,
Self::V128(_) => ValType::V128,
Self::FuncRef(_) => ValType::FuncRef,
Self::ExternRef(_) => ValType::ExternRef,
}
}
pub fn i32(&self) -> Option<i32> {
match self {
Self::I32(value) => Some(*value),
_ => None,
}
}
pub fn i64(&self) -> Option<i64> {
match self {
Self::I64(value) => Some(*value),
_ => None,
}
}
pub fn f32(&self) -> Option<F32> {
match self {
Self::F32(value) => Some(*value),
_ => None,
}
}
pub fn f64(&self) -> Option<F64> {
match self {
Self::F64(value) => Some(*value),
_ => None,
}
}
pub fn funcref(&self) -> Option<Ref<&Func>> {
match self {
Self::FuncRef(value) => Some(value.as_ref()),
_ => None,
}
}
pub fn externref(&self) -> Option<Ref<&ExternRef>> {
match self {
Self::ExternRef(value) => Some(value.as_ref()),
_ => None,
}
}
}
impl From<i32> for Val {
#[inline]
fn from(val: i32) -> Self {
Self::I32(val)
}
}
impl From<i64> for Val {
#[inline]
fn from(val: i64) -> Self {
Self::I64(val)
}
}
impl From<F32> for Val {
#[inline]
fn from(val: F32) -> Self {
Self::F32(val)
}
}
impl From<F64> for Val {
#[inline]
fn from(val: F64) -> Self {
Self::F64(val)
}
}
impl From<Func> for Val {
#[inline]
fn from(func: Func) -> Self {
Self::FuncRef(Ref::Val(func))
}
}
impl From<ExternRef> for Val {
#[inline]
fn from(externref: ExternRef) -> Self {
Self::ExternRef(Ref::Val(externref))
}
}
impl From<Ref<Func>> for Val {
#[inline]
fn from(funcref: Ref<Func>) -> Self {
Self::FuncRef(funcref)
}
}
impl From<Ref<ExternRef>> for Val {
#[inline]
fn from(externref: Ref<ExternRef>) -> Self {
Self::ExternRef(externref)
}
}
impl From<V128> for Val {
#[inline]
fn from(value: V128) -> Self {
Self::V128(value)
}
}
impl From<TypedVal> for Val {
fn from(value: TypedVal) -> Self {
let untyped = value.untyped();
match value.ty() {
ValType::I32 => Self::I32(untyped.into()),
ValType::I64 => Self::I64(untyped.into()),
ValType::F32 => Self::F32(untyped.into()),
ValType::F64 => Self::F64(untyped.into()),
ValType::V128 => {
#[cfg(feature = "simd")]
{
Self::V128(untyped.into())
}
#[cfg(not(feature = "simd"))]
{
panic!("`simd` crate feature is disabled")
}
}
ValType::FuncRef => Self::FuncRef(untyped.into()),
ValType::ExternRef => Self::ExternRef(untyped.into()),
}
}
}
impl From<Val> for TypedVal {
fn from(value: Val) -> Self {
Self::new(value.ty(), value.into())
}
}