use core::{cmp::Ordering, convert::Infallible, fmt::Debug};
use super::{
env::internal::{Env as _, EnvBase as _, I256Small, I256Val, U256Small, U256Val},
ConversionError, Env, RawVal, TryFromVal, TryIntoVal,
};
#[cfg(not(target_family = "wasm"))]
use crate::env::internal::xdr::ScVal;
use crate::{env::MaybeEnv, unwrap::UnwrapInfallible};
macro_rules! impl_num_wrapping_val_type {
($wrapper:ident, $val:ty, $small:ty) => {
#[derive(Clone)]
pub struct $wrapper {
env: MaybeEnv,
val: $val,
}
impl Debug for $wrapper {
fn fmt(&self, f: &mut core::fmt::Formatter<'_>) -> core::fmt::Result {
write!(f, "{:?}", self.val.as_raw())
}
}
impl Eq for $wrapper {}
impl PartialEq for $wrapper {
fn eq(&self, other: &Self) -> bool {
self.partial_cmp(other) == Some(Ordering::Equal)
}
}
impl PartialOrd for $wrapper {
fn partial_cmp(&self, other: &Self) -> Option<Ordering> {
Some(Ord::cmp(self, other))
}
}
impl Ord for $wrapper {
fn cmp(&self, other: &Self) -> Ordering {
let self_raw = self.val.to_raw();
let other_raw = other.val.to_raw();
match (<$small>::try_from(self_raw), <$small>::try_from(other_raw)) {
(Ok(self_num), Ok(other_num)) => self_num.cmp(&other_num),
_ => {
let env: Option<Env> =
match (self.env.clone().try_into(), other.env.clone().try_into()) {
(Err(_), Err(_)) => None,
(Err(_), Ok(e)) => Some(e),
(Ok(e), Err(_)) => Some(e),
(Ok(e1), Ok(e2)) => {
e1.check_same_env(&e2);
Some(e1)
}
};
if let Some(env) = env {
let v = env.obj_cmp(self_raw, other_raw).unwrap_infallible();
v.cmp(&0)
} else {
panic!("$wrapper object is missing the env reference");
}
}
}
}
}
impl TryFromVal<Env, $val> for $wrapper {
type Error = Infallible;
fn try_from_val(env: &Env, val: &$val) -> Result<Self, Self::Error> {
Ok(unsafe { $wrapper::unchecked_new(env.clone(), *val) })
}
}
impl TryFromVal<Env, RawVal> for $wrapper {
type Error = ConversionError;
fn try_from_val(env: &Env, val: &RawVal) -> Result<Self, Self::Error> {
Ok(<$val>::try_from_val(env, val)?
.try_into_val(env)
.unwrap_infallible())
}
}
impl TryFromVal<Env, $wrapper> for RawVal {
type Error = ConversionError;
fn try_from_val(_env: &Env, v: &$wrapper) -> Result<Self, Self::Error> {
Ok(v.to_raw())
}
}
impl TryFromVal<Env, &$wrapper> for RawVal {
type Error = ConversionError;
fn try_from_val(_env: &Env, v: &&$wrapper) -> Result<Self, Self::Error> {
Ok(v.to_raw())
}
}
#[cfg(not(target_family = "wasm"))]
impl TryFrom<&$wrapper> for ScVal {
type Error = ConversionError;
fn try_from(v: &$wrapper) -> Result<Self, Self::Error> {
if let Ok(ss) = <$small>::try_from(v.val) {
ScVal::try_from(ss)
} else {
let e: Env = v.env.clone().try_into()?;
ScVal::try_from_val(&e, &v.to_raw())
}
}
}
#[cfg(not(target_family = "wasm"))]
impl TryFrom<$wrapper> for ScVal {
type Error = ConversionError;
fn try_from(v: $wrapper) -> Result<Self, Self::Error> {
(&v).try_into()
}
}
#[cfg(not(target_family = "wasm"))]
impl TryFromVal<Env, ScVal> for $wrapper {
type Error = ConversionError;
fn try_from_val(env: &Env, val: &ScVal) -> Result<Self, Self::Error> {
Ok(<$val>::try_from_val(env, &RawVal::try_from_val(env, val)?)?
.try_into_val(env)
.unwrap_infallible())
}
}
impl $wrapper {
#[inline(always)]
pub(crate) unsafe fn unchecked_new(env: Env, val: $val) -> Self {
Self {
env: env.into(),
val,
}
}
pub fn as_raw(&self) -> &RawVal {
self.val.as_raw()
}
pub fn to_raw(&self) -> RawVal {
self.val.to_raw()
}
pub fn to_val(&self) -> $val {
self.val
}
}
};
}
impl_num_wrapping_val_type!(U256, U256Val, U256Small);
impl_num_wrapping_val_type!(I256, I256Val, I256Small);