use crate::{nonnative_params::*, AlgebraicSponge, DuplexSpongeMode};
use snarkvm_fields::{FieldParameters, PoseidonParameters, PrimeField, ToConstraintField};
use snarkvm_utilities::{BigInteger, FromBits, ToBits};
use smallvec::SmallVec;
use std::{
iter::Peekable,
ops::{Index, IndexMut},
sync::Arc,
};
#[derive(Copy, Clone, Debug)]
pub struct State<F: PrimeField, const RATE: usize, const CAPACITY: usize> {
capacity_state: [F; CAPACITY],
rate_state: [F; RATE],
}
impl<F: PrimeField, const RATE: usize, const CAPACITY: usize> Default for State<F, RATE, CAPACITY> {
fn default() -> Self {
Self { capacity_state: [F::zero(); CAPACITY], rate_state: [F::zero(); RATE] }
}
}
impl<F: PrimeField, const RATE: usize, const CAPACITY: usize> State<F, RATE, CAPACITY> {
pub fn iter(&self) -> impl Iterator<Item = &F> + Clone {
self.capacity_state.iter().chain(self.rate_state.iter())
}
pub fn iter_mut(&mut self) -> impl Iterator<Item = &mut F> {
self.capacity_state.iter_mut().chain(self.rate_state.iter_mut())
}
}
impl<F: PrimeField, const RATE: usize, const CAPACITY: usize> Index<usize> for State<F, RATE, CAPACITY> {
type Output = F;
fn index(&self, index: usize) -> &Self::Output {
assert!(index < RATE + CAPACITY, "Index out of bounds: index is {} but length is {}", index, RATE + CAPACITY);
if index < CAPACITY { &self.capacity_state[index] } else { &self.rate_state[index - CAPACITY] }
}
}
impl<F: PrimeField, const RATE: usize, const CAPACITY: usize> IndexMut<usize> for State<F, RATE, CAPACITY> {
fn index_mut(&mut self, index: usize) -> &mut Self::Output {
assert!(index < RATE + CAPACITY, "Index out of bounds: index is {} but length is {}", index, RATE + CAPACITY);
if index < CAPACITY { &mut self.capacity_state[index] } else { &mut self.rate_state[index - CAPACITY] }
}
}
#[derive(Clone, Debug, PartialEq, Eq)]
pub struct Poseidon<F: PrimeField, const RATE: usize> {
parameters: Arc<PoseidonParameters<F, RATE, 1>>,
}
impl<F: PrimeField, const RATE: usize> Poseidon<F, RATE> {
pub fn setup() -> Self {
Self { parameters: Arc::new(F::default_poseidon_parameters::<RATE>().unwrap()) }
}
pub fn evaluate(&self, input: &[F]) -> F {
self.evaluate_many(input, 1)[0]
}
pub fn evaluate_many(&self, input: &[F], num_outputs: usize) -> Vec<F> {
let mut sponge = PoseidonSponge::<F, RATE, 1>::new_with_parameters(&self.parameters);
sponge.absorb_native_field_elements(input);
sponge.squeeze_native_field_elements(num_outputs).to_vec()
}
pub fn evaluate_with_len(&self, input: &[F]) -> F {
self.evaluate(&[vec![F::from(input.len() as u128)], input.to_vec()].concat())
}
pub fn parameters(&self) -> &Arc<PoseidonParameters<F, RATE, 1>> {
&self.parameters
}
}
#[derive(Clone, Debug)]
pub struct PoseidonSponge<F: PrimeField, const RATE: usize, const CAPACITY: usize> {
parameters: Arc<PoseidonParameters<F, RATE, CAPACITY>>,
state: State<F, RATE, CAPACITY>,
pub mode: DuplexSpongeMode,
adjustment_factor_lookup_table: Arc<[F]>,
}
impl<F: PrimeField, const RATE: usize> AlgebraicSponge<F, RATE> for PoseidonSponge<F, RATE, 1> {
type Parameters = Arc<PoseidonParameters<F, RATE, 1>>;
fn sample_parameters() -> Self::Parameters {
Arc::new(F::default_poseidon_parameters::<RATE>().unwrap())
}
fn new_with_parameters(parameters: &Self::Parameters) -> Self {
Self {
parameters: parameters.clone(),
state: State::default(),
mode: DuplexSpongeMode::Absorbing { next_absorb_index: 0 },
adjustment_factor_lookup_table: {
let capacity = F::size_in_bits() - 1;
let mut table = Vec::<F>::with_capacity(capacity);
let mut cur = F::one();
for _ in 0..capacity {
table.push(cur);
cur.double_in_place();
}
table.into()
},
}
}
fn absorb_native_field_elements<T: ToConstraintField<F>>(&mut self, elements: &[T]) {
let input = elements.iter().flat_map(|e| e.to_field_elements().unwrap()).collect::<Vec<_>>();
if !input.is_empty() {
match self.mode {
DuplexSpongeMode::Absorbing { mut next_absorb_index } => {
if next_absorb_index == RATE {
self.permute();
next_absorb_index = 0;
}
self.absorb_internal(next_absorb_index, &input);
}
DuplexSpongeMode::Squeezing { next_squeeze_index: _ } => {
self.permute();
self.absorb_internal(0, &input);
}
}
}
}
fn absorb_nonnative_field_elements<Target: PrimeField>(&mut self, elements: impl IntoIterator<Item = Target>) {
Self::push_elements_to_sponge(self, elements, OptimizationType::Weight);
}
fn squeeze_nonnative_field_elements<Target: PrimeField>(&mut self, num: usize) -> SmallVec<[Target; 10]> {
self.get_fe(num, false)
}
fn squeeze_native_field_elements(&mut self, num_elements: usize) -> SmallVec<[F; 10]> {
if num_elements == 0 {
return SmallVec::<[F; 10]>::new();
}
let mut output = if num_elements <= 10 {
smallvec::smallvec_inline![F::zero(); 10]
} else {
smallvec::smallvec![F::zero(); num_elements]
};
match self.mode {
DuplexSpongeMode::Absorbing { next_absorb_index: _ } => {
self.permute();
self.squeeze_internal(0, &mut output[..num_elements]);
}
DuplexSpongeMode::Squeezing { mut next_squeeze_index } => {
if next_squeeze_index == RATE {
self.permute();
next_squeeze_index = 0;
}
self.squeeze_internal(next_squeeze_index, &mut output[..num_elements]);
}
}
output.truncate(num_elements);
output
}
fn squeeze_short_nonnative_field_elements<Target: PrimeField>(&mut self, num: usize) -> SmallVec<[Target; 10]> {
self.get_fe(num, true)
}
}
impl<F: PrimeField, const RATE: usize> PoseidonSponge<F, RATE, 1> {
#[inline]
fn apply_ark(&mut self, round_number: usize) {
for (state_elem, ark_elem) in self.state.iter_mut().zip(&self.parameters.ark[round_number]) {
*state_elem += ark_elem;
}
}
#[inline]
fn apply_s_box(&mut self, is_full_round: bool) {
if is_full_round {
for elem in self.state.iter_mut() {
*elem = elem.pow([self.parameters.alpha]);
}
} else {
self.state[0] = self.state[0].pow([self.parameters.alpha]);
}
}
#[inline]
fn apply_mds(&mut self) {
let mut new_state = State::default();
new_state.iter_mut().zip(&self.parameters.mds).for_each(|(new_elem, mds_row)| {
*new_elem = F::sum_of_products(self.state.iter(), mds_row.iter());
});
self.state = new_state;
}
#[inline]
fn permute(&mut self) {
let partial_rounds = self.parameters.partial_rounds;
let full_rounds = self.parameters.full_rounds;
let full_rounds_over_2 = full_rounds / 2;
let partial_round_range = full_rounds_over_2..(full_rounds_over_2 + partial_rounds);
for i in 0..(partial_rounds + full_rounds) {
let is_full_round = !partial_round_range.contains(&i);
self.apply_ark(i);
self.apply_s_box(is_full_round);
self.apply_mds();
}
}
#[inline]
fn absorb_internal(&mut self, mut rate_start: usize, input: &[F]) {
if !input.is_empty() {
let first_chunk_size = std::cmp::min(RATE - rate_start, input.len());
let num_elements_remaining = input.len() - first_chunk_size;
let (first_chunk, rest_chunk) = input.split_at(first_chunk_size);
let rest_chunks = rest_chunk.chunks(RATE);
let total_num_chunks = 1 + (num_elements_remaining / RATE) +
usize::from((num_elements_remaining % RATE) != 0);
for (i, chunk) in std::iter::once(first_chunk).chain(rest_chunks).enumerate() {
for (element, state_elem) in chunk.iter().zip(&mut self.state.rate_state[rate_start..]) {
*state_elem += element;
}
if i == total_num_chunks - 1 {
self.mode = DuplexSpongeMode::Absorbing { next_absorb_index: rate_start + chunk.len() };
return;
} else {
self.permute();
}
rate_start = 0;
}
}
}
#[inline]
fn squeeze_internal(&mut self, mut rate_start: usize, output: &mut [F]) {
let output_size = output.len();
if output_size != 0 {
let first_chunk_size = std::cmp::min(RATE - rate_start, output.len());
let num_output_remaining = output.len() - first_chunk_size;
let (first_chunk, rest_chunk) = output.split_at_mut(first_chunk_size);
assert_eq!(rest_chunk.len(), num_output_remaining);
let rest_chunks = rest_chunk.chunks_mut(RATE);
let total_num_chunks = 1 + (num_output_remaining / RATE) +
usize::from((num_output_remaining % RATE) != 0);
for (i, chunk) in std::iter::once(first_chunk).chain(rest_chunks).enumerate() {
let range = rate_start..(rate_start + chunk.len());
debug_assert_eq!(
chunk.len(),
self.state.rate_state[range.clone()].len(),
"failed with squeeze {output_size} at rate {RATE} and rate_start {rate_start}"
);
chunk.copy_from_slice(&self.state.rate_state[range]);
if i == total_num_chunks - 1 {
self.mode = DuplexSpongeMode::Squeezing { next_squeeze_index: (rate_start + chunk.len()) };
return;
} else {
self.permute();
}
rate_start = 0;
}
}
}
pub fn compress_elements<TargetField: PrimeField, I: Iterator<Item = (F, F)>>(
&self,
mut src_limbs: Peekable<I>,
ty: OptimizationType,
) -> Vec<F> {
let capacity = F::size_in_bits() - 1;
let mut dest_limbs = Vec::<F>::new();
let params = get_params(TargetField::size_in_bits(), F::size_in_bits(), ty);
let mut num_bits = Vec::new();
while let Some(first) = src_limbs.next() {
let second = src_limbs.peek();
let first_max_bits_per_limb = params.bits_per_limb + crate::overhead!(first.1 + F::one(), &mut num_bits);
let second_max_bits_per_limb = if let Some(second) = second {
params.bits_per_limb + crate::overhead!(second.1 + F::one(), &mut num_bits)
} else {
0
};
if let Some(second) = second {
if first_max_bits_per_limb + second_max_bits_per_limb <= capacity {
let adjustment_factor = &self.adjustment_factor_lookup_table[second_max_bits_per_limb];
dest_limbs.push(first.0 * adjustment_factor + second.0);
src_limbs.next();
} else {
dest_limbs.push(first.0);
}
} else {
dest_limbs.push(first.0);
}
}
dest_limbs
}
pub fn get_limbs_representations<TargetField: PrimeField>(
elem: &TargetField,
optimization_type: OptimizationType,
) -> SmallVec<[F; 10]> {
Self::get_limbs_representations_from_big_integer::<TargetField>(&elem.to_bigint(), optimization_type)
}
pub fn get_limbs_representations_from_big_integer<TargetField: PrimeField>(
elem: &<TargetField as PrimeField>::BigInteger,
optimization_type: OptimizationType,
) -> SmallVec<[F; 10]> {
let params = get_params(TargetField::size_in_bits(), F::size_in_bits(), optimization_type);
let mut cur_bits = Vec::new();
let mut limbs: SmallVec<[F; 10]> = SmallVec::new();
let mut cur = *elem;
for _ in 0..params.num_limbs {
cur.write_bits_be(&mut cur_bits); let cur_mod_r =
<F as PrimeField>::BigInteger::from_bits_be(&cur_bits[cur_bits.len() - params.bits_per_limb..])
.unwrap(); limbs.push(F::from_bigint(cur_mod_r).unwrap());
cur.divn(params.bits_per_limb as u32);
cur_bits.clear();
}
limbs.reverse();
limbs
}
pub fn push_elements_to_sponge<TargetField: PrimeField>(
&mut self,
src: impl IntoIterator<Item = TargetField>,
ty: OptimizationType,
) {
let src_limbs = src
.into_iter()
.flat_map(|elem| {
let limbs = Self::get_limbs_representations(&elem, ty);
limbs.into_iter().map(|limb| (limb, F::one()))
})
.peekable();
let dest_limbs = self.compress_elements::<TargetField, _>(src_limbs, ty);
self.absorb_native_field_elements(&dest_limbs);
}
pub fn get_bits(&mut self, num_bits: usize) -> Vec<bool> {
let bits_per_element = F::size_in_bits() - 1;
let num_elements = (num_bits + bits_per_element - 1) / bits_per_element;
let src_elements = self.squeeze_native_field_elements(num_elements);
let mut dest_bits = Vec::<bool>::with_capacity(num_elements * bits_per_element);
let skip = (F::Parameters::REPR_SHAVE_BITS + 1) as usize;
for elem in src_elements.iter() {
let elem_bits = elem.to_bigint().to_bits_be();
dest_bits.extend_from_slice(&elem_bits[skip..]);
}
dest_bits.truncate(num_bits);
dest_bits
}
pub fn get_fe<TargetField: PrimeField>(
&mut self,
num_elements: usize,
outputs_short_elements: bool,
) -> SmallVec<[TargetField; 10]> {
let num_bits_per_nonnative = if outputs_short_elements {
168
} else {
TargetField::size_in_bits() - 1 };
let bits = self.get_bits(num_bits_per_nonnative * num_elements);
let mut lookup_table = Vec::<TargetField>::with_capacity(num_bits_per_nonnative);
let mut cur = TargetField::one();
for _ in 0..num_bits_per_nonnative {
lookup_table.push(cur);
cur.double_in_place();
}
let dest_elements = bits
.chunks_exact(num_bits_per_nonnative)
.map(|per_nonnative_bits| {
let mut res = TargetField::zero();
for (i, bit) in per_nonnative_bits.iter().rev().enumerate() {
if *bit {
res += &lookup_table[i];
}
}
res
})
.collect::<SmallVec<_>>();
debug_assert_eq!(dest_elements.len(), num_elements);
dest_elements
}
}