binius_core/fiat_shamir/
hasher_challenger.rsuse std::{cmp::min, mem};
use bytes::{buf::UninitSlice, Buf, BufMut};
use digest::{
core_api::{Block, BlockSizeUser},
Digest, FixedOutputReset, Output,
};
use super::Challenger;
#[derive(Debug, Default)]
pub struct HasherSampler<H: Digest> {
index: usize,
buffer: Output<H>,
hasher: H,
}
#[derive(Debug, Default)]
pub struct HasherObserver<H: Digest + BlockSizeUser> {
index: usize,
buffer: Block<H>,
hasher: H,
}
#[derive(Debug)]
pub enum HasherChallenger<H: Digest + BlockSizeUser> {
Observer(HasherObserver<H>),
Sampler(HasherSampler<H>),
}
impl<H> Default for HasherChallenger<H>
where
H: Digest + BlockSizeUser,
{
fn default() -> Self {
Self::Observer(HasherObserver {
hasher: H::new(),
index: 0,
buffer: Block::<H>::default(),
})
}
}
impl<H: Digest + BlockSizeUser + FixedOutputReset + Default> Challenger for HasherChallenger<H> {
fn observer(&mut self) -> &mut impl BufMut {
match self {
Self::Observer(observer) => observer,
Self::Sampler(sampler) => {
*self = Self::Observer(mem::take(sampler).into_observer());
match self {
Self::Observer(observer) => observer,
_ => unreachable!(),
}
}
}
}
fn sampler(&mut self) -> &mut impl Buf {
match self {
Self::Sampler(sampler) => sampler,
Self::Observer(observer) => {
*self = Self::Sampler(mem::take(observer).into_sampler());
match self {
Self::Sampler(sampler) => sampler,
_ => unreachable!(),
}
}
}
}
}
impl<H> HasherSampler<H>
where
H: Digest + Default + BlockSizeUser,
{
fn into_observer(mut self) -> HasherObserver<H> {
Digest::update(&mut self.hasher, self.index.to_le_bytes());
HasherObserver {
hasher: self.hasher,
index: 0,
buffer: Block::<H>::default(),
}
}
}
impl<H> HasherSampler<H>
where
H: Digest + FixedOutputReset,
{
fn fill_buffer(&mut self) {
let digest = self.hasher.finalize_reset();
Digest::update(&mut self.hasher, &digest);
self.buffer = digest;
self.index = 0
}
}
impl<H> HasherObserver<H>
where
H: Digest + BlockSizeUser + Default,
{
fn into_sampler(mut self) -> HasherSampler<H> {
self.flush();
HasherSampler {
hasher: self.hasher,
index: <H as Digest>::output_size(),
buffer: Output::<H>::default(),
}
}
}
impl<H> HasherObserver<H>
where
H: Digest + BlockSizeUser,
{
fn flush(&mut self) {
self.hasher.update(&self.buffer[..self.index]);
self.index = 0
}
}
impl<H> Buf for HasherSampler<H>
where
H: Digest + FixedOutputReset + Default,
{
fn remaining(&self) -> usize {
usize::MAX
}
fn chunk(&self) -> &[u8] {
&self.buffer[self.index..]
}
fn advance(&mut self, mut cnt: usize) {
if self.index == <H as Digest>::output_size() {
self.fill_buffer();
}
while cnt > 0 {
let remaining = min(<H as Digest>::output_size() - self.index, cnt);
if remaining == 0 {
self.fill_buffer();
continue;
}
cnt -= remaining;
self.index += remaining;
}
}
}
unsafe impl<H> BufMut for HasherObserver<H>
where
H: Digest + BlockSizeUser,
{
fn remaining_mut(&self) -> usize {
usize::MAX
}
unsafe fn advance_mut(&mut self, mut cnt: usize) {
while cnt > 0 {
let remaining = min(<H as BlockSizeUser>::block_size() - self.index, cnt);
cnt -= remaining;
self.index += remaining;
if self.index == <H as BlockSizeUser>::block_size() {
self.flush();
}
}
}
fn chunk_mut(&mut self) -> &mut UninitSlice {
let buffer = &mut self.buffer[self.index..];
buffer.into()
}
}
#[cfg(test)]
mod tests {
use groestl_crypto::Groestl256;
use rand::{thread_rng, RngCore};
use super::*;
#[test]
fn test_starting_sampler() {
let mut hasher = Groestl256::default();
let mut challenger = HasherChallenger::<Groestl256>::default();
let mut out = [0u8; 8];
challenger.sampler().copy_to_slice(&mut out);
let first_hash_out = hasher.finalize_reset();
hasher.update(first_hash_out);
assert_eq!(first_hash_out[0..8], out);
let mut out = [0u8; 24];
challenger.sampler().copy_to_slice(&mut out);
assert_eq!(first_hash_out[8..], out);
challenger.observer().put_slice(&[0x48, 0x55]);
hasher.update([32, 0, 0, 0, 0, 0, 0, 0]);
hasher.update([0x48, 0x55]);
let mut out_after_observe = [0u8; 2];
challenger.sampler().copy_to_slice(&mut out_after_observe);
let second_hash_out = hasher.finalize_reset();
assert_eq!(out_after_observe, second_hash_out[..2]);
}
#[test]
fn test_starting_observer() {
let mut hasher = Groestl256::default();
let mut challenger = HasherChallenger::<Groestl256>::default();
let mut observable = [0u8; 1019];
thread_rng().fill_bytes(&mut observable);
challenger.observer().put_slice(&observable[..39]);
challenger.observer().put_slice(&observable[39..300]);
challenger.observer().put_slice(&observable[300..987]);
challenger.observer().put_slice(&observable[987..]);
hasher.update(observable);
let mut out = [0u8; 7];
challenger.sampler().copy_to_slice(&mut out);
let first_hash_out = hasher.finalize_reset();
hasher.update(first_hash_out);
assert_eq!(first_hash_out[..7], out);
thread_rng().fill_bytes(&mut observable);
challenger.observer().put_slice(&observable[..128]);
hasher.update([7, 0, 0, 0, 0, 0, 0, 0]);
hasher.update(&observable[..128]);
let mut out = [0u8; 32];
challenger.sampler().copy_to_slice(&mut out);
let second_hasher_out = hasher.finalize_reset();
hasher.update(second_hasher_out);
assert_eq!(second_hasher_out[..], out);
challenger.observer().put_slice(&observable[128..]);
hasher.update([32, 0, 0, 0, 0, 0, 0, 0]);
hasher.update(&observable[128..]);
let mut out_again = [0u8; 7];
challenger.sampler().copy_to_slice(&mut out_again);
let final_hasher_out = hasher.finalize_reset();
assert_eq!(final_hasher_out[..7], out_again);
}
}