#![no_std]
extern crate alloc;
mod external;
mod generic;
mod internal;
mod round_numbers;
use alloc::vec::Vec;
use core::marker::PhantomData;
pub use external::*;
pub use generic::*;
pub use internal::*;
use p3_field::{Algebra, InjectiveMonomial, PrimeField, PrimeField64};
use p3_symmetric::{CryptographicPermutation, Permutation};
use rand::Rng;
use rand::distr::{Distribution, StandardUniform};
pub use round_numbers::poseidon2_round_numbers_128;
const SUPPORTED_WIDTHS: [usize; 8] = [2, 3, 4, 8, 12, 16, 20, 24];
#[derive(Clone, Debug)]
pub struct Poseidon2<F, ExternalPerm, InternalPerm, const WIDTH: usize, const D: u64> {
external_layer: ExternalPerm,
internal_layer: InternalPerm,
_phantom: PhantomData<F>,
}
impl<F, ExternalPerm, InternalPerm, const WIDTH: usize, const D: u64>
Poseidon2<F, ExternalPerm, InternalPerm, WIDTH, D>
where
F: PrimeField,
ExternalPerm: ExternalLayerConstructor<F, WIDTH>,
InternalPerm: InternalLayerConstructor<F>,
{
pub fn new(
external_constants: ExternalLayerConstants<F, WIDTH>,
internal_constants: Vec<F>,
) -> Self {
assert!(SUPPORTED_WIDTHS.contains(&WIDTH));
let external_layer = ExternalPerm::new_from_constants(external_constants);
let internal_layer = InternalPerm::new_from_constants(internal_constants);
Self {
external_layer,
internal_layer,
_phantom: PhantomData,
}
}
pub fn new_from_rng<R: Rng>(rounds_f: usize, rounds_p: usize, rng: &mut R) -> Self
where
StandardUniform: Distribution<F> + Distribution<[F; WIDTH]>,
{
let external_constants = ExternalLayerConstants::new_from_rng(rounds_f, rng);
let internal_constants = rng.sample_iter(StandardUniform).take(rounds_p).collect();
Self::new(external_constants, internal_constants)
}
}
impl<F, ExternalPerm, InternalPerm, const WIDTH: usize, const D: u64>
Poseidon2<F, ExternalPerm, InternalPerm, WIDTH, D>
where
F: PrimeField64,
ExternalPerm: ExternalLayerConstructor<F, WIDTH>,
InternalPerm: InternalLayerConstructor<F>,
{
pub fn new_from_rng_128<R: Rng>(rng: &mut R) -> Self
where
StandardUniform: Distribution<F> + Distribution<[F; WIDTH]>,
{
let round_numbers = poseidon2_round_numbers_128::<F>(WIDTH, D);
let (rounds_f, rounds_p) = round_numbers.unwrap_or_else(|e| panic!("{e}"));
Self::new_from_rng(rounds_f, rounds_p, rng)
}
}
impl<F, A, ExternalPerm, InternalPerm, const WIDTH: usize, const D: u64> Permutation<[A; WIDTH]>
for Poseidon2<F, ExternalPerm, InternalPerm, WIDTH, D>
where
F: PrimeField + InjectiveMonomial<D>,
A: Algebra<F> + Sync + InjectiveMonomial<D>,
ExternalPerm: ExternalLayer<A, WIDTH, D>,
InternalPerm: InternalLayer<A, WIDTH, D>,
{
fn permute_mut(&self, state: &mut [A; WIDTH]) {
self.external_layer.permute_state_initial(state);
self.internal_layer.permute_state(state);
self.external_layer.permute_state_terminal(state);
}
}
impl<F, A, ExternalPerm, InternalPerm, const WIDTH: usize, const D: u64>
CryptographicPermutation<[A; WIDTH]> for Poseidon2<F, ExternalPerm, InternalPerm, WIDTH, D>
where
F: PrimeField + InjectiveMonomial<D>,
A: Algebra<F> + Sync + InjectiveMonomial<D>,
ExternalPerm: ExternalLayer<A, WIDTH, D>,
InternalPerm: InternalLayer<A, WIDTH, D>,
{
}