use crate::modulus::barrett::Barrett; use crate::modulus::ONCE; use crate::poly::Poly; use crate::ring::Ring; use std::cmp::min; impl Ring { pub fn pack( &self, polys: &mut Vec>>, log_gap: usize, ) { let log_n: usize = self.log_n(); let log_nth_root: usize = log_n + 1; let nth_root: usize = 1 << log_nth_root; let log_start: usize = log_n - log_gap; let mut log_end: usize = log_n; let mut indices: Vec = Vec::::new(); // Retrives non-empty indexes polys.iter().enumerate().for_each(|(i, poly)| { if !poly.is_none() { indices.push(i); } }); let gap: usize = max_gap(&indices); if !ZEROGARBAGE { if gap > 0 { log_end -= gap.trailing_zeros() as usize; } } assert!( log_start < log_end, "invalid input polys: gap between non None value is smaller than 2^log_gap" ); let n_inv: Barrett = self .modulus .barrett .prepare(self.modulus.inv(1 << (log_end - log_start))); indices.iter().for_each(|i| { if let Some(poly) = polys[*i].as_mut() { if !NTT { self.ntt_inplace::(poly); } self.a_mul_b_scalar_barrett_into_a::(&n_inv, poly); } }); let x_pow2: Vec> = self.gen_x_pow_2::(log_n); let mut tmpa: Poly = self.new_poly(); let mut tmpb: Poly = self.new_poly(); for i in log_start..log_end { let t: usize = 1 << (log_n - 1 - i); let (polys_lo, polys_hi) = polys.split_at_mut(t); for j in 0..t { if let Some(poly_hi) = polys_hi[j].as_mut() { self.a_mul_b_montgomery_into_a::(&x_pow2[log_n - i - 1], poly_hi); if let Some(poly_lo) = polys_lo[j].as_mut() { self.a_sub_b_into_c::<1, ONCE>(poly_lo, poly_hi, &mut tmpa); self.a_add_b_into_b::(poly_hi, poly_lo); } } if let Some(poly_lo) = polys_lo[j].as_mut() { let gal_el: usize = self.galois_element((1 << i) >> 1, i == 0, log_nth_root); if !polys_hi[j].is_none() { self.a_apply_automorphism_add_b_into_b::(&tmpa, gal_el, 2 << self.log_n(), poly_lo); } else { self.a_apply_automorphism_into_b::(poly_lo, gal_el, nth_root, &mut tmpa); self.a_add_b_into_b::(&tmpa, poly_lo); } } else if let Some(poly_hi) = polys_hi[j].as_mut() { let gal_el: usize = self.galois_element((1 << i) >> 1, i == 0, log_nth_root); self.a_apply_automorphism_into_b::(poly_hi, gal_el, nth_root, &mut tmpa); self.a_sub_b_into_a::<1, ONCE>(&tmpa, poly_hi); std::mem::swap(&mut polys_lo[j], &mut polys_hi[j]); } } polys.truncate(t); } if !NTT { if let Some(poly) = polys[0].as_mut() { self.intt_inplace::(poly); } } } } // Returns the largest gap between two values in an ordered array of distinct values. // Panics if the array is not ordered or values are not distincts. fn max_gap(vec: &[usize]) -> usize { let mut gap: usize = usize::MAX; for i in 1..vec.len() { let (l, r) = (vec[i - 1], vec[i]); assert!(r > l, "invalid input vec: not sorted or collision between indices"); gap = min(gap, r - l); if gap == 1 { break; } } gap }