use crate::fixed_uint::UintConstructor;
use crate::utils;
use core::cmp;
use proc_macro2::TokenStream;
use quote::quote;
impl UintConstructor {
pub fn define_kernel(&self) {
self.defun_priv_kernel();
self.defun_pub_kernel();
self.deftrait_uint_convert();
}
pub fn convert_into(&self, uc: &Self) -> TokenStream {
let this_name = &self.ts.name;
let this_feature = &self.ts.feature;
let that_name = &uc.ts.name;
let that_feature = &uc.ts.feature;
let stmts = match self.info.bits_size.cmp(&uc.info.bits_size) {
cmp::Ordering::Equal => {
if self.info.unit_bits_size == uc.info.unit_bits_size {
quote!(
let inner = self.inner();
let val = #that_name::new(inner.clone());
(val, false)
)
} else if uc.info.unit_bits_size == 8 {
let that_unit_amount = &uc.ts.unit_amount;
quote!(
let mut inner = [0u8; #that_unit_amount];
self.into_little_endian(&mut inner[..]).unwrap();
let val = #that_name::new(inner);
(val, false)
)
} else if self.info.unit_bits_size == 8 {
quote!(
let inner = self.inner();
let val = #that_name::from_little_endian(&inner[..]).unwrap();
(val, false)
)
} else {
let that_bytes_size = &uc.ts.bytes_size;
quote!(
let mut tmp = [0u8; #that_bytes_size];
self.into_little_endian(&mut tmp[..]).unwrap();
let val = #that_name::from_little_endian(&tmp[..]).unwrap();
(val, false)
)
}
}
cmp::Ordering::Less => {
let this_bytes_size = &self.ts.bytes_size;
quote!(
let mut tmp = [0u8; #this_bytes_size];
self.into_little_endian(&mut tmp[..]).unwrap();
let val = #that_name::from_little_endian(&tmp[..]).unwrap();
(val, false)
)
}
cmp::Ordering::Greater => {
let this_bytes_size = &self.ts.bytes_size;
let that_bytes_size = &uc.ts.bytes_size;
quote!(
let mut tmp = [0u8; #this_bytes_size];
self.into_little_endian(&mut tmp[..]).unwrap();
let val = #that_name::from_little_endian(&tmp[..#that_bytes_size]).unwrap();
(val, true)
)
}
};
quote!(
#[cfg(all(feature = #this_feature, feature = #that_feature))]
impl prelude::UintConvert<#that_name> for #this_name {
#[inline]
fn convert_into(&self) -> (#that_name, bool) {
#stmts
}
}
)
}
fn defun_priv_kernel(&self) {
let name = &self.ts.name;
let inner_type = &self.ts.inner_type;
let part = quote!(
#[inline]
const fn new(data: #inner_type) -> Self {
#name(data)
}
#[inline]
pub(crate) fn inner<'a>(&'a self) -> &'a #inner_type {
&self.0
}
#[inline]
pub(crate) fn mut_inner<'a>(&'a mut self) -> &'a mut #inner_type {
&mut self.0
}
#[inline]
pub(crate) fn into_inner(self) -> #inner_type {
self.0
}
);
self.defun(part);
}
fn defun_pub_kernel(&self) {
let unit_amount = &self.ts.unit_amount;
let loop_unit_amount = &utils::pure_uint_list_to_ts(0..self.info.unit_amount);
let zero_padding = &utils::pure_uint_list_to_ts(
::std::iter::repeat(0).take((self.info.unit_amount - 1) as usize),
);
let one = quote!([1, #(#zero_padding),* ]);
let part = quote!(
#[inline]
pub const fn zero() -> Self {
Self::new([0; #unit_amount])
}
#[inline]
pub const fn one() -> Self {
Self::new( #one )
}
#[inline]
pub fn is_zero(&self) -> bool {
let inner = self.inner();
#({
if inner[#loop_unit_amount] != 0 {
return false;
}
})*
true
}
#[inline]
pub fn is_max(&self) -> bool {
let inner = self.inner();
#({
if inner[#loop_unit_amount] != !0 {
return false;
}
})*
true
}
);
self.defun(part);
}
fn deftrait_uint_convert(&self) {
let part = quote!(
pub trait UintConvert<T> {
fn convert_into(&self) -> (T, bool);
}
);
self.prelude(part);
}
}