use crate::fixed_hash::HashConstructor;
use crate::utils;
use core::cmp;
use proc_macro2::TokenStream;
use quote::quote;
impl HashConstructor {
pub fn define_kernel(&self) {
self.defun_priv_kernel();
self.defun_pub_kernel();
self.deftrait_hash_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 => quote!(
let inner = self.inner();
let val = #that_name::new(inner.clone());
(val, false)
),
cmp::Ordering::Less => {
let this_bytes_size = &self.ts.unit_amount;
quote!(
let mut ret = #that_name::empty();
ret.mut_inner()[..#this_bytes_size].copy_from_slice(&self.inner()[..]);
(ret, false)
)
}
cmp::Ordering::Greater => {
let that_bytes_size = &uc.ts.unit_amount;
quote!(
let mut ret = #that_name::empty();
ret.mut_inner()
.copy_from_slice(&self.inner()[..#that_bytes_size]);
(ret, true)
)
}
};
quote!(
#[cfg(all(feature = #this_feature, feature = #that_feature))]
impl prelude::HashConvert<#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 part = quote!(
#[inline]
pub const fn repeat_byte(byte: u8) -> Self {
Self::new([byte; #unit_amount])
}
#[deprecated(
since = "0.1.5",
note = "Please use the empty function instead"
)]
#[inline]
pub const fn zero() -> Self {
Self::new([0; #unit_amount])
}
#[deprecated(
since = "0.1.5",
note = "Please use the is_empty function instead"
)]
#[inline]
pub fn is_zero(&self) -> bool {
let inner = self.inner();
#({
if inner[#loop_unit_amount] != 0 {
return false;
}
})*
true
}
#[deprecated(
since = "0.1.5",
note = "Please use the is_full function instead"
)]
#[inline]
pub fn is_max(&self) -> bool {
let inner = self.inner();
#({
if inner[#loop_unit_amount] != !0 {
return false;
}
})*
true
}
#[inline]
pub const fn empty() -> Self {
Self::new([0; #unit_amount])
}
#[inline]
pub fn is_empty(&self) -> bool {
let inner = self.inner();
#({
if inner[#loop_unit_amount] != 0 {
return false;
}
})*
true
}
#[inline]
pub const fn full() -> Self {
Self::new([!0; #unit_amount])
}
#[inline]
pub fn is_full(&self) -> bool {
let inner = self.inner();
#({
if inner[#loop_unit_amount] != !0 {
return false;
}
})*
true
}
#[inline]
pub fn covers(&self, hash: &Self) -> bool {
let inner = self.inner();
let rhs = hash.inner();
#({
let idx = #loop_unit_amount;
if inner[idx] & rhs[idx] != rhs[idx] {
return false;
}
})*
true
}
);
self.defun(part);
}
fn deftrait_hash_convert(&self) {
let part = quote!(
pub trait HashConvert<T> {
fn convert_into(&self) -> (T, bool);
}
);
self.prelude(part);
}
}