use digest::consts::{U16, U24, U32, U48, U64};
macro_rules! impl_blake2_mac {
(
$(#[$attr:meta])*
$name:ident,
wc_ty = $wc_ty:path,
digest_size = $digest_size:literal,
out = $out_size:ty,
key = $key_size:ty
) => {
$(#[$attr])*
pub struct $name {
blake2: $wc_ty,
}
$(#[$attr])*
impl digest::MacMarker for $name {}
$(#[$attr])*
impl digest::OutputSizeUser for $name {
type OutputSize = $out_size;
}
$(#[$attr])*
impl digest::common::KeySizeUser for $name {
type KeySize = $key_size;
}
$(#[$attr])*
impl digest::KeyInit for $name {
fn new(key: &digest::Key<Self>) -> Self {
Self {
blake2: <$wc_ty>::new_with_key($digest_size, key.as_slice())
.expect("wolfCrypt BLAKE2 init failed"),
}
}
fn new_from_slice(key: &[u8]) -> Result<Self, digest::InvalidLength> {
if key.len() > <Self as digest::common::KeySizeUser>::key_size() {
return Err(digest::InvalidLength);
}
Ok(Self {
blake2: <$wc_ty>::new_with_key($digest_size, key)
.map_err(|_| digest::InvalidLength)?,
})
}
}
$(#[$attr])*
impl digest::Update for $name {
fn update(&mut self, data: &[u8]) {
<$wc_ty>::update(&mut self.blake2, data)
.expect("wolfCrypt BLAKE2 update failed");
}
}
$(#[$attr])*
impl digest::FixedOutput for $name {
fn finalize_into(mut self, out: &mut digest::Output<Self>) {
<$wc_ty>::finalize(&mut self.blake2, out.as_mut_slice())
.expect("wolfCrypt BLAKE2 finalize failed");
}
}
};
}
impl_blake2_mac! {
#[cfg(blake2b)]
Blake2bMac256,
wc_ty = crate::blake2::BLAKE2b,
digest_size = 32,
out = U32,
key = U64
}
impl_blake2_mac! {
#[cfg(blake2b)]
Blake2bMac384,
wc_ty = crate::blake2::BLAKE2b,
digest_size = 48,
out = U48,
key = U64
}
impl_blake2_mac! {
#[cfg(blake2b)]
Blake2bMac512,
wc_ty = crate::blake2::BLAKE2b,
digest_size = 64,
out = U64,
key = U64
}
impl_blake2_mac! {
#[cfg(blake2s)]
Blake2sMac128,
wc_ty = crate::blake2::BLAKE2s,
digest_size = 16,
out = U16,
key = U32
}
impl_blake2_mac! {
#[cfg(blake2s)]
Blake2sMac192,
wc_ty = crate::blake2::BLAKE2s,
digest_size = 24,
out = U24,
key = U32
}
impl_blake2_mac! {
#[cfg(blake2s)]
Blake2sMac256,
wc_ty = crate::blake2::BLAKE2s,
digest_size = 32,
out = U32,
key = U32
}