use super::CtrFlavor;
use cipher::generic_array::{
typenum::{operator_aliases::PartialQuot, type_operators::PartialDiv, Unsigned, U4},
ArrayLength, GenericArray,
};
use core::convert::TryInto;
type ChunkSize = U4;
type Chunks<B> = PartialQuot<B, ChunkSize>;
const CS: usize = ChunkSize::USIZE;
#[derive(Default, Copy, Clone)]
#[repr(transparent)]
pub struct Ctr32BE(u32);
impl<B> CtrFlavor<B> for Ctr32BE
where
Self: Default + Clone,
B: ArrayLength<u8> + PartialDiv<ChunkSize>,
Chunks<B>: ArrayLength<u32>,
{
type Nonce = GenericArray<u32, Chunks<B>>;
type Backend = u32;
#[inline]
fn generate_block(&self, nonce: &Self::Nonce) -> GenericArray<u8, B> {
let mut block = GenericArray::<u8, B>::default();
for i in 0..Chunks::<B>::USIZE {
let t = if i == Chunks::<B>::USIZE - 1 {
self.0.wrapping_add(nonce[i]).to_be_bytes()
} else {
nonce[i].to_ne_bytes()
};
block[CS * i..][..CS].copy_from_slice(&t);
}
block
}
#[inline]
fn load(block: &GenericArray<u8, B>) -> Self::Nonce {
let mut res = Self::Nonce::default();
for i in 0..Chunks::<B>::USIZE {
let chunk = block[CS * i..][..CS].try_into().unwrap();
res[i] = if i == Chunks::<B>::USIZE - 1 {
u32::from_be_bytes(chunk)
} else {
u32::from_ne_bytes(chunk)
}
}
res
}
#[inline]
fn checked_add(&self, rhs: usize) -> Option<Self> {
rhs.try_into()
.ok()
.and_then(|rhs| self.0.checked_add(rhs))
.map(Self)
}
#[inline]
fn increment(&mut self) {
self.0 = self.0.wrapping_add(1);
}
#[inline]
fn to_backend(&self) -> Self::Backend {
self.0
}
#[inline]
fn from_backend(v: Self::Backend) -> Self {
Self(v)
}
}
#[derive(Default, Clone)]
#[repr(transparent)]
pub struct Ctr32LE(u32);
impl<B> CtrFlavor<B> for Ctr32LE
where
Self: Default + Clone,
B: ArrayLength<u8> + PartialDiv<ChunkSize>,
Chunks<B>: ArrayLength<u32>,
{
type Nonce = GenericArray<u32, Chunks<B>>;
type Backend = u32;
#[inline]
fn generate_block(&self, nonce: &Self::Nonce) -> GenericArray<u8, B> {
let mut block = GenericArray::<u8, B>::default();
for i in 0..Chunks::<B>::USIZE {
let t = if i == 0 {
self.0.wrapping_add(nonce[i]).to_le_bytes()
} else {
nonce[i].to_ne_bytes()
};
block[CS * i..][..CS].copy_from_slice(&t);
}
block
}
#[inline]
fn load(block: &GenericArray<u8, B>) -> Self::Nonce {
let mut res = Self::Nonce::default();
for i in 0..Chunks::<B>::USIZE {
let chunk = block[CS * i..][..CS].try_into().unwrap();
res[i] = if i == 0 {
u32::from_le_bytes(chunk)
} else {
u32::from_ne_bytes(chunk)
}
}
res
}
#[inline]
fn checked_add(&self, rhs: usize) -> Option<Self> {
rhs.try_into()
.ok()
.and_then(|rhs| self.0.checked_add(rhs))
.map(Self)
}
#[inline]
fn increment(&mut self) {
self.0 = self.0.wrapping_add(1);
}
#[inline]
fn to_backend(&self) -> Self::Backend {
self.0
}
#[inline]
fn from_backend(v: Self::Backend) -> Self {
Self(v)
}
}