Skip to content

Commit

Permalink
digest internals: Provide an infallible I-U-F API internally.
Browse files Browse the repository at this point in the history
Return an error instead of panic when too much input is
processed by a `BlockContext`, or when the precondition of
`BlockContext::finish` is violated. Jump through some hoops
to get the compiler to prefer the non-error path.

This is a step towards providing an non-panicking variant of
the API.
  • Loading branch information
briansmith committed Dec 15, 2024
1 parent 39a5e7b commit b4d440d
Show file tree
Hide file tree
Showing 2 changed files with 64 additions and 16 deletions.
70 changes: 56 additions & 14 deletions src/digest.rs
Original file line number Diff line number Diff line change
Expand Up @@ -24,7 +24,7 @@ use self::{
};
use crate::{
bits::{BitLength, FromByteLen as _},
cpu, debug,
cpu, debug, error,
polyfill::{self, slice, sliceutil},
};
use core::num::Wrapping;
Expand Down Expand Up @@ -74,12 +74,20 @@ impl BlockContext {
// `num_pending < self.algorithm.block_len`.
//
// `block` may be arbitrarily overwritten.
pub(crate) fn finish(
#[inline(never)]
pub(crate) fn try_finish(
mut self,
block: &mut [u8; MAX_BLOCK_LEN],
num_pending: usize,
cpu_features: cpu::Features,
) -> Digest {
) -> Result<Digest, FinishError> {
let completed_bytes = self
.completed_bytes
.checked_add(polyfill::u64_from_usize(num_pending))
.ok_or_else(|| FinishError::too_much_input(self.completed_bytes))?;
let copmleted_bits = BitLength::from_byte_len(completed_bytes)
.map_err(|_: error::Unspecified| FinishError::too_much_input(self.completed_bytes))?;

let block_len = self.algorithm.block_len();
let block = &mut block[..block_len];

Expand All @@ -89,7 +97,11 @@ impl BlockContext {
padding
}
// Precondition violated.
Some([]) | None => unreachable!(),
unreachable => {
return Err(FinishError::pending_not_a_partial_block(
unreachable.as_deref(),
));
}
};

let padding = match padding
Expand All @@ -107,23 +119,17 @@ impl BlockContext {
}
};

let completed_bytes = self
.completed_bytes
.checked_add(polyfill::u64_from_usize(num_pending))
.unwrap();
let copmleted_bits = BitLength::from_byte_len(completed_bytes).unwrap();

let (to_zero, len) = padding.split_at_mut(padding.len() - 8);
to_zero.fill(0);
len.copy_from_slice(&copmleted_bits.to_be_bytes());

let (completed_bytes, leftover) = self.block_data_order(block, cpu_features);
debug_assert_eq!((completed_bytes, leftover.len()), (block_len, 0));

Digest {
Ok(Digest {
algorithm: self.algorithm,
value: (self.algorithm.format_output)(self.state),
}
})
}

#[must_use]
Expand All @@ -136,6 +142,37 @@ impl BlockContext {
}
}

pub(crate) enum FinishError {
#[allow(dead_code)]
TooMuchInput(u64),
#[allow(dead_code)]
PendingNotAPartialBlock(usize),
}

impl FinishError {
#[cold]
#[inline(never)]
fn too_much_input(completed_bytes: u64) -> Self {
Self::TooMuchInput(completed_bytes)
}

// unreachable
#[cold]
#[inline(never)]
fn pending_not_a_partial_block(padding: Option<&[u8]>) -> Self {
match padding {
None => Self::PendingNotAPartialBlock(0),
Some(padding) => Self::PendingNotAPartialBlock(padding.len()),
}
}
}

impl From<FinishError> for error::Unspecified {
fn from(_: FinishError) -> Self {
Self
}
}

/// A context for multi-step (Init-Update-Finish) digest calculations.
///
/// # Examples
Expand Down Expand Up @@ -227,10 +264,15 @@ impl Context {
///
/// `finish` consumes the context so it cannot be (mis-)used after `finish`
/// has been called.
pub fn finish(mut self) -> Digest {
pub fn finish(self) -> Digest {
self.try_finish().unwrap()
}

fn try_finish(mut self) -> Result<Digest, error::Unspecified> {
let cpu_features = cpu::features();
self.block
.finish(&mut self.pending, self.num_pending, cpu_features)
.try_finish(&mut self.pending, self.num_pending, cpu_features)
.map_err(error::Unspecified::from)
}

/// The algorithm that this context is using.
Expand Down
10 changes: 8 additions & 2 deletions src/hmac.rs
Original file line number Diff line number Diff line change
Expand Up @@ -312,13 +312,19 @@ impl Context {
/// the return value of `sign` to a tag. Use `verify` for verification
/// instead.
pub fn sign(self) -> Tag {
let cpu_features = cpu::features();
self.try_sign().unwrap()
}

fn try_sign(self) -> Result<Tag, error::Unspecified> {
let cpu_features = cpu::features();
let algorithm = self.inner.algorithm();
let buffer = &mut [0u8; digest::MAX_BLOCK_LEN];
let num_pending = algorithm.output_len();
buffer[..num_pending].copy_from_slice(self.inner.finish().as_ref());
Tag(self.outer.finish(buffer, num_pending, cpu_features))
self.outer
.try_finish(buffer, num_pending, cpu_features)
.map(Tag)
.map_err(error::Unspecified::from)
}
}

Expand Down

0 comments on commit b4d440d

Please sign in to comment.