use crate::error::{Error, Result};
use crate::hash::HashFunction;
use dcrypt_common::security::{SecretBuffer, SecureZeroingType};
use subtle::ConstantTimeEq;
use zeroize::{Zeroize, ZeroizeOnDrop};
const MAX_BLOCK: usize = 144;
#[derive(Clone, Zeroize, ZeroizeOnDrop)]
pub struct Hmac<H: HashFunction + Clone> {
#[zeroize(skip)] hash: H,
ipad: SecretBuffer<MAX_BLOCK>,
opad: SecretBuffer<MAX_BLOCK>,
block_size: usize,
is_finalized: bool,
}
impl<H> Hmac<H>
where
H: HashFunction + Clone,
H::Output: AsRef<[u8]> + Clone,
{
const IPAD_BYTE: u8 = 0x36;
const OPAD_BYTE: u8 = 0x5c;
pub fn new(key: &[u8]) -> Result<Self> {
let bs = H::block_size();
debug_assert!(bs <= MAX_BLOCK);
let mut hk = H::new();
hk.update(key)?;
let hashed = hk.finalize()?;
let mut k_prime = [0u8; MAX_BLOCK];
let long = (key.len() > bs) as u8; let mask = long.wrapping_neg(); #[allow(clippy::needless_range_loop)] for i in 0..bs {
let k = *key.get(i).unwrap_or(&0);
let hk = hashed.as_ref().get(i).copied().unwrap_or(0);
k_prime[i] = (hk & mask) | (k & !mask);
}
let mut ipad_bytes = [0u8; MAX_BLOCK];
let mut opad_bytes = [0u8; MAX_BLOCK];
#[allow(clippy::needless_range_loop)] for i in 0..bs {
ipad_bytes[i] = k_prime[i] ^ Self::IPAD_BYTE;
opad_bytes[i] = k_prime[i] ^ Self::OPAD_BYTE;
}
for b in k_prime.iter_mut().take(bs) {
*b = 0;
}
let mut hash = H::new();
hash.update(&ipad_bytes[..bs])?;
Ok(Self {
hash,
ipad: SecretBuffer::new(ipad_bytes),
opad: SecretBuffer::new(opad_bytes),
block_size: bs,
is_finalized: false,
})
}
pub fn update(&mut self, data: &[u8]) -> Result<()> {
if self.is_finalized {
let mut dummy = H::new();
dummy.update(data)?;
let _ = dummy.finalize();
return Err(Error::param(
"hmac_state",
"Cannot update after finalization",
));
}
self.hash.update(data).map(|_| ())
}
pub fn finalize(&mut self) -> Result<Vec<u8>> {
if self.is_finalized {
let inner_dummy = [0u8; 64]; let mut outer = H::new();
outer.update(&self.opad.as_ref()[..self.block_size])?;
outer.update(&inner_dummy[..H::output_size()])?;
let _ = outer.finalize();
return Err(Error::param("hmac_state", "HMAC already finalized"));
}
self.is_finalized = true;
let inner_hash = self.hash.finalize()?;
let mut outer = H::new();
outer.update(&self.opad.as_ref()[..self.block_size])?;
outer.update(inner_hash.as_ref())?;
outer.finalize().map(|out| out.as_ref().to_vec())
}
pub fn mac(key: &[u8], data: &[u8]) -> Result<Vec<u8>> {
let mut h = Self::new(key)?;
h.update(data)?;
h.finalize()
}
pub fn verify(key: &[u8], data: &[u8], tag: &[u8]) -> Result<bool> {
let expected = Self::mac(key, data)?;
let mut diff = 0u8;
#[allow(clippy::needless_range_loop)] for i in 0..H::output_size() {
let a = expected.get(i).copied().unwrap_or(0);
let b = tag.get(i).copied().unwrap_or(0);
diff |= a ^ b;
}
diff |= (tag.len() ^ H::output_size()) as u8;
Ok(diff.ct_eq(&0u8).unwrap_u8() == 1)
}
}
impl<H> SecureZeroingType for Hmac<H>
where
H: HashFunction + Default + Clone,
{
fn zeroed() -> Self {
Self {
hash: H::default(),
ipad: SecretBuffer::zeroed(),
opad: SecretBuffer::zeroed(),
block_size: 0,
is_finalized: false,
}
}
fn secure_clone(&self) -> Self {
Self {
hash: self.hash.clone(),
ipad: self.ipad.secure_clone(),
opad: self.opad.secure_clone(),
block_size: self.block_size,
is_finalized: self.is_finalized,
}
}
}
#[cfg(test)]
mod tests;