use super::ternary_types::{
TernaryTier, TernaryTier1, TernaryTier2, TernaryTier3, TernaryTier4, TernaryTier5, TernaryTier6,
TernaryValue, UniversalTernaryFixed,
};
use crate::fixed_point::{I256, I512, I1024};
use crate::fixed_point::core_types::errors::OverflowDetected;
#[inline]
pub fn negate_ternary_tq8_8(a: i32) -> Result<i32, OverflowDetected> {
match a.checked_neg() {
Some(result) => Ok(result),
None => Err(OverflowDetected::TierOverflow), }
}
#[inline]
pub fn negate_ternary_tq16_16(a: i64) -> Result<i64, OverflowDetected> {
match a.checked_neg() {
Some(result) => Ok(result),
None => Err(OverflowDetected::TierOverflow),
}
}
#[inline]
pub fn negate_ternary_tq32_32(a: i128) -> Result<i128, OverflowDetected> {
match a.checked_neg() {
Some(result) => Ok(result),
None => Err(OverflowDetected::TierOverflow),
}
}
#[inline]
pub fn negate_ternary_tq64_64(a: I256) -> I256 {
a.saturating_neg()
}
#[inline]
pub fn negate_ternary_tq128_128(a: I512) -> Result<I512, OverflowDetected> {
match a.checked_neg() {
Some(result) => Ok(result),
None => Err(OverflowDetected::TierOverflow),
}
}
#[inline]
pub fn negate_ternary_tq256_256(a: I1024) -> I1024 {
-a
}
impl UniversalTernaryFixed {
pub fn negate(&self) -> Result<Self, OverflowDetected> {
match &self.value {
TernaryValue::Tier1(t) => {
match negate_ternary_tq8_8(t.raw()) {
Ok(result) => Ok(Self { value: TernaryValue::Tier1(TernaryTier1::from_raw(result)), current_tier: TernaryTier::Tier1 }),
Err(_) => {
let promoted = self.promote_to_tier2()?;
promoted.negate()
}
}
}
TernaryValue::Tier2(t) => {
match negate_ternary_tq16_16(t.raw()) {
Ok(result) => Ok(Self { value: TernaryValue::Tier2(TernaryTier2::from_raw(result)), current_tier: TernaryTier::Tier2 }),
Err(_) => {
let promoted = self.promote_to_tier3()?;
promoted.negate()
}
}
}
TernaryValue::Tier3(t) => {
match negate_ternary_tq32_32(t.raw()) {
Ok(result) => Ok(Self { value: TernaryValue::Tier3(TernaryTier3::from_raw(result)), current_tier: TernaryTier::Tier3 }),
Err(_) => {
let promoted = self.promote_to_tier4();
promoted.negate()
}
}
}
TernaryValue::Tier4(t) => {
let result = negate_ternary_tq64_64(t.raw().clone());
Ok(Self { value: TernaryValue::Tier4(TernaryTier4::from_raw(result)), current_tier: TernaryTier::Tier4 })
}
TernaryValue::Tier5(t) => {
match negate_ternary_tq128_128(t.raw().clone()) {
Ok(result) => Ok(Self { value: TernaryValue::Tier5(TernaryTier5::from_raw(result)), current_tier: TernaryTier::Tier5 }),
Err(_) => {
let promoted = self.promote_to_tier6();
promoted.negate()
}
}
}
TernaryValue::Tier6(t) => {
let result = negate_ternary_tq256_256(t.raw().clone());
Ok(Self { value: TernaryValue::Tier6(TernaryTier6::from_raw(result)), current_tier: TernaryTier::Tier6 })
}
}
}
}