use once_cell::sync::Lazy;
use p3_goldilocks::Goldilocks;
use p3_symmetric::Permutation;
use super::{
AlgebraicSponge, CAPACITY_RANGE, DIGEST_RANGE, Felt, RATE_RANGE, RATE0_RANGE, RATE1_RANGE,
Range, STATE_WIDTH, Word,
};
use crate::{
ZERO,
hash::algebraic_sponge::poseidon2::constants::{
ARK_EXT_INITIAL, ARK_EXT_TERMINAL, ARK_INT, MAT_DIAG,
},
};
mod constants;
use constants::{NUM_EXTERNAL_ROUNDS_HALF, NUM_INTERNAL_ROUNDS};
#[cfg(test)]
mod test;
static P3_POSEIDON2: Lazy<p3_goldilocks::Poseidon2Goldilocks<12>> =
Lazy::new(p3_goldilocks::default_goldilocks_poseidon2_12);
#[inline(always)]
fn p3_permute(state: &mut [Felt; STATE_WIDTH]) {
let gl_state =
unsafe { &mut *(state as *mut [Felt; STATE_WIDTH] as *mut [Goldilocks; STATE_WIDTH]) };
P3_POSEIDON2.permute_mut(gl_state);
}
#[allow(rustdoc::private_intra_doc_links)]
#[derive(Debug, Copy, Clone, Eq, PartialEq)]
pub struct Poseidon2();
impl AlgebraicSponge for Poseidon2 {
fn apply_permutation(state: &mut [Felt; STATE_WIDTH]) {
p3_permute(state);
}
}
impl Poseidon2 {
pub const COLLISION_RESISTANCE: u32 = 128;
pub const NUM_EXTERNAL_ROUNDS_HALF: usize = NUM_EXTERNAL_ROUNDS_HALF;
pub const NUM_INTERNAL_ROUNDS: usize = NUM_INTERNAL_ROUNDS;
pub const STATE_WIDTH: usize = STATE_WIDTH;
pub const RATE_RANGE: Range<usize> = RATE_RANGE;
pub const RATE0_RANGE: Range<usize> = RATE0_RANGE;
pub const RATE1_RANGE: Range<usize> = RATE1_RANGE;
pub const CAPACITY_RANGE: Range<usize> = CAPACITY_RANGE;
pub const DIGEST_RANGE: Range<usize> = DIGEST_RANGE;
pub const MAT_DIAG: [Felt; STATE_WIDTH] = MAT_DIAG;
pub const ARK_EXT_INITIAL: [[Felt; STATE_WIDTH]; NUM_EXTERNAL_ROUNDS_HALF] = ARK_EXT_INITIAL;
pub const ARK_EXT_TERMINAL: [[Felt; STATE_WIDTH]; NUM_EXTERNAL_ROUNDS_HALF] = ARK_EXT_TERMINAL;
pub const ARK_INT: [Felt; NUM_INTERNAL_ROUNDS] = ARK_INT;
#[inline(always)]
pub fn hash(bytes: &[u8]) -> Word {
<Self as AlgebraicSponge>::hash(bytes)
}
#[inline(always)]
pub fn apply_permutation(state: &mut [Felt; STATE_WIDTH]) {
<Self as AlgebraicSponge>::apply_permutation(state);
}
#[inline(always)]
pub fn hash_elements<E: BasedVectorSpace<Felt>>(elements: &[E]) -> Word {
<Self as AlgebraicSponge>::hash_elements(elements)
}
#[inline(always)]
pub fn merge(values: &[Word; 2]) -> Word {
<Self as AlgebraicSponge>::merge(values)
}
#[inline(always)]
pub fn merge_many(values: &[Word]) -> Word {
<Self as AlgebraicSponge>::merge_many(values)
}
#[inline(always)]
pub fn merge_in_domain(values: &[Word; 2], domain: Felt) -> Word {
<Self as AlgebraicSponge>::merge_in_domain(values, domain)
}
#[inline(always)]
pub fn hash_elements_in_domain<E: BasedVectorSpace<Felt>>(
elements: &[E],
domain: Felt,
) -> Word {
<Self as AlgebraicSponge>::hash_elements_in_domain(elements, domain)
}
#[inline(always)]
pub fn apply_matmul_external(state: &mut [Felt; STATE_WIDTH]) {
Self::matmul_m4(state);
let number_blocks = STATE_WIDTH / 4;
let mut stored = [ZERO; 4];
for j in 0..number_blocks {
let base = j * 4;
for l in 0..4 {
stored[l] += state[base + l];
}
}
for (i, val) in state.iter_mut().enumerate() {
*val += stored[i % 4];
}
}
#[inline(always)]
fn matmul_m4(state: &mut [Felt; STATE_WIDTH]) {
const N_CHUNKS: usize = STATE_WIDTH / 4;
for i in 0..N_CHUNKS {
let base = i * 4;
let x = &mut state[base..base + 4];
let t01 = x[0] + x[1];
let t23 = x[2] + x[3];
let t0123 = t01 + t23;
let t01123 = t0123 + x[1];
let t01233 = t0123 + x[3];
x[3] = t01233 + x[0].double(); x[1] = t01123 + x[2].double(); x[0] = t01123 + t01; x[2] = t01233 + t23; }
}
#[inline(always)]
pub fn matmul_internal(state: &mut [Felt; STATE_WIDTH], mat_diag: [Felt; 12]) {
let mut sum = ZERO;
for s in state.iter().take(STATE_WIDTH) {
sum += *s
}
for i in 0..state.len() {
state[i] = state[i] * mat_diag[i] + sum;
}
}
#[inline(always)]
pub fn add_rc(state: &mut [Felt; STATE_WIDTH], ark: &[Felt; 12]) {
state.iter_mut().zip(ark).for_each(|(s, &k)| *s += k);
}
#[inline(always)]
pub fn apply_sbox(state: &mut [Felt; STATE_WIDTH]) {
state[0] = state[0].exp_const_u64::<7>();
state[1] = state[1].exp_const_u64::<7>();
state[2] = state[2].exp_const_u64::<7>();
state[3] = state[3].exp_const_u64::<7>();
state[4] = state[4].exp_const_u64::<7>();
state[5] = state[5].exp_const_u64::<7>();
state[6] = state[6].exp_const_u64::<7>();
state[7] = state[7].exp_const_u64::<7>();
state[8] = state[8].exp_const_u64::<7>();
state[9] = state[9].exp_const_u64::<7>();
state[10] = state[10].exp_const_u64::<7>();
state[11] = state[11].exp_const_u64::<7>();
}
}
use p3_challenger::DuplexChallenger;
use p3_symmetric::{CryptographicPermutation, PaddingFreeSponge, TruncatedPermutation};
use crate::field::BasedVectorSpace;
#[derive(Debug, Copy, Clone, Eq, PartialEq)]
pub struct Poseidon2Permutation256;
impl Poseidon2Permutation256 {
pub const NUM_EXTERNAL_ROUNDS_HALF: usize = Poseidon2::NUM_EXTERNAL_ROUNDS_HALF;
pub const NUM_INTERNAL_ROUNDS: usize = Poseidon2::NUM_INTERNAL_ROUNDS;
pub const STATE_WIDTH: usize = STATE_WIDTH;
pub const RATE_RANGE: Range<usize> = Poseidon2::RATE_RANGE;
pub const CAPACITY_RANGE: Range<usize> = Poseidon2::CAPACITY_RANGE;
pub const DIGEST_RANGE: Range<usize> = Poseidon2::DIGEST_RANGE;
#[inline(always)]
pub fn apply_permutation(state: &mut [Felt; STATE_WIDTH]) {
Poseidon2::apply_permutation(state);
}
}
impl Permutation<[Felt; STATE_WIDTH]> for Poseidon2Permutation256 {
fn permute_mut(&self, state: &mut [Felt; STATE_WIDTH]) {
p3_permute(state);
}
}
impl CryptographicPermutation<[Felt; STATE_WIDTH]> for Poseidon2Permutation256 {}
pub type Poseidon2Hasher = PaddingFreeSponge<Poseidon2Permutation256, 12, 8, 4>;
pub type Poseidon2Compression = TruncatedPermutation<Poseidon2Permutation256, 2, 4, 12>;
pub type Poseidon2Challenger<F> = DuplexChallenger<F, Poseidon2Permutation256, 12, 8>;