1use alloc::vec;
2use alloc::vec::Vec;
3use core::iter::Sum;
4use core::mem::MaybeUninit;
5use core::ops::Mul;
6
7use num_bigint::BigUint;
8use p3_maybe_rayon::prelude::*;
9
10use crate::field::Field;
11use crate::{PackedValue, PrimeCharacteristicRing, PrimeField, PrimeField32};
12
13pub fn cyclic_subgroup_known_order<F: Field>(
15 generator: F,
16 order: usize,
17) -> impl Iterator<Item = F> + Clone {
18 generator.powers().take(order)
19}
20
21pub fn cyclic_subgroup_coset_known_order<F: Field>(
23 generator: F,
24 shift: F,
25 order: usize,
26) -> impl Iterator<Item = F> + Clone {
27 generator.shifted_powers(shift).take(order)
28}
29
30pub fn scale_slice_in_place_single_core<F: Field>(slice: &mut [F], s: F) {
35 let (packed, sfx) = F::Packing::pack_slice_with_suffix_mut(slice);
36 let packed_s: F::Packing = s.into();
37 packed.iter_mut().for_each(|x| *x *= packed_s);
38 sfx.iter_mut().for_each(|x| *x *= s);
39}
40
41#[inline]
47pub fn par_scale_slice_in_place<F: Field>(slice: &mut [F], s: F) {
48 let (packed, sfx) = F::Packing::pack_slice_with_suffix_mut(slice);
49 let packed_s: F::Packing = s.into();
50 packed.par_iter_mut().for_each(|x| *x *= packed_s);
51 sfx.iter_mut().for_each(|x| *x *= s);
52}
53
54pub fn add_scaled_slice_in_place<F: Field>(slice: &mut [F], other: &[F], s: F) {
59 debug_assert_eq!(slice.len(), other.len(), "slices must have equal length");
60 let (slice_packed, slice_sfx) = F::Packing::pack_slice_with_suffix_mut(slice);
61 let (other_packed, other_sfx) = F::Packing::pack_slice_with_suffix(other);
62 let packed_s: F::Packing = s.into();
63 slice_packed
64 .iter_mut()
65 .zip(other_packed)
66 .for_each(|(x, y)| *x += *y * packed_s);
67 slice_sfx
68 .iter_mut()
69 .zip(other_sfx)
70 .for_each(|(x, y)| *x += *y * s);
71}
72
73pub fn par_add_scaled_slice_in_place<F: Field>(slice: &mut [F], other: &[F], s: F) {
79 debug_assert_eq!(slice.len(), other.len(), "slices must have equal length");
80 let (slice_packed, slice_sfx) = F::Packing::pack_slice_with_suffix_mut(slice);
81 let (other_packed, other_sfx) = F::Packing::pack_slice_with_suffix(other);
82 let packed_s: F::Packing = s.into();
83 slice_packed
84 .par_iter_mut()
85 .zip(other_packed.par_iter())
86 .for_each(|(x, y)| *x += *y * packed_s);
87 slice_sfx
88 .iter_mut()
89 .zip(other_sfx)
90 .for_each(|(x, y)| *x += *y * s);
91}
92
93#[inline]
96#[must_use]
97pub const fn field_to_array<R: PrimeCharacteristicRing, const D: usize>(x: R) -> [R; D] {
98 let mut arr = [const { MaybeUninit::uninit() }; D];
99 arr[0] = MaybeUninit::new(x);
100 let mut i = 1;
101 while i < D {
102 arr[i] = MaybeUninit::new(R::ZERO);
103 i += 1;
104 }
105 unsafe { core::mem::transmute_copy::<_, [R; D]>(&arr) }
106}
107
108#[inline]
110#[must_use]
111pub const fn halve_u32<const P: u32>(x: u32) -> u32 {
112 let shift = (P + 1) >> 1;
113 let half = x >> 1;
114 if x & 1 == 0 { half } else { half + shift }
115}
116
117#[inline]
119#[must_use]
120pub const fn halve_u64<const P: u64>(x: u64) -> u64 {
121 let shift = (P + 1) >> 1;
122 let half = x >> 1;
123 if x & 1 == 0 { half } else { half + shift }
124}
125
126#[must_use]
138pub fn reduce_32<SF: PrimeField32, TF: PrimeField>(vals: &[SF]) -> TF {
139 reduce_packed(vals, 32)
140}
141
142#[must_use]
147pub fn reduce_packed_shifted<SF: PrimeField32, TF: PrimeField>(vals: &[SF], radix_bits: u32) -> TF {
148 debug_assert!((radix_bits < 64) && ((SF::ORDER_U32 as u64) < (1u64 << radix_bits)));
149 let base = TF::from_int(1u64 << radix_bits);
150 vals.iter().rev().fold(TF::ZERO, |acc, val| {
151 acc * base + TF::from_int(val.as_canonical_u32() as u64 + 1)
152 })
153}
154
155#[inline]
161#[must_use]
162pub const fn absorb_radix_bits<F: PrimeField32>() -> u32 {
163 u32::BITS - (F::ORDER_U32 - 1).leading_zeros()
164}
165
166#[must_use]
171pub fn reduce_packed<SF: PrimeField32, TF: PrimeField>(vals: &[SF], radix_bits: u32) -> TF {
172 debug_assert!((absorb_radix_bits::<SF>() <= radix_bits) && (radix_bits < 64));
173 let base = TF::from_int(1u64 << radix_bits);
174 vals.iter().rev().fold(TF::ZERO, |acc, val| {
175 acc * base + TF::from_int(val.as_canonical_u32())
176 })
177}
178
179#[inline]
183#[must_use]
184pub const fn injective_pack_bits<F: PrimeField32>() -> u32 {
185 (F::ORDER_U32 - 1).ilog2()
186}
187
188#[must_use]
195pub fn max_packed_injective_limbs<F: PrimeField32, PF: PrimeField>(radix_bits: u32) -> usize {
196 max_packed_injective_limbs_with_max_digit::<PF>(radix_bits, F::ORDER_U32 - 1)
197}
198
199fn max_packed_injective_limbs_with_max_digit<PF: PrimeField>(
200 radix_bits: u32,
201 max_digit: u32,
202) -> usize {
203 debug_assert!((0 < radix_bits) && (radix_bits < 64));
204 let max_digit = BigUint::from(max_digit);
205 let base = BigUint::from(1u32) << (radix_bits as usize);
206 let pf_order = PF::order();
207 let mut k = 0usize;
208 let mut max_val = BigUint::ZERO;
209 let mut power = BigUint::from(1u32);
210 loop {
211 let new_max = &max_val + &max_digit * &power;
212 if new_max >= pf_order {
213 break k;
214 }
215 max_val = new_max;
216 power *= &base;
217 k += 1;
218 }
219}
220
221#[must_use]
227pub fn max_shifted_packed_injective_limbs<F: PrimeField32, PF: PrimeField>(
228 radix_bits: u32,
229) -> usize {
230 max_packed_injective_limbs_with_max_digit::<PF>(radix_bits, F::ORDER_U32)
231}
232
233#[must_use]
236pub fn max_absorb_injective_limbs<F: PrimeField32, PF: PrimeField>() -> usize {
237 max_packed_injective_limbs::<F, PF>(absorb_radix_bits::<F>())
238}
239
240#[must_use]
243pub fn max_shifted_absorb_injective_limbs<F: PrimeField32, PF: PrimeField>() -> usize {
244 max_shifted_packed_injective_limbs::<F, PF>(absorb_radix_bits::<F>())
245}
246
247#[must_use]
250pub fn pf_packed_limbs_cover_order<SF: PrimeField>(num_limbs: usize, radix_bits: u32) -> bool {
251 let Some(total_bits) = num_limbs.checked_mul(radix_bits as usize) else {
252 return false;
253 };
254 (BigUint::from(1u32) << total_bits) >= SF::order()
255}
256
257#[must_use]
272pub fn split_pf_to_packed_limbs<SF: PrimeField, TF: PrimeField32>(
273 val: SF,
274 num_limbs: usize,
275 radix_bits: u32,
276) -> Vec<TF> {
277 debug_assert!((0 < radix_bits) && (radix_bits < 64));
278 debug_assert!(
279 radix_bits <= injective_pack_bits::<TF>(),
280 "radix_bits must be ≤ injective_pack_bits::<TF>() for injective limb embedding"
281 );
282
283 let mask_u32: u32 = (1u32 << radix_bits) - 1;
285 let mut rem = val.as_canonical_biguint();
286 let mut out = vec![TF::ZERO; num_limbs];
287
288 for item in out.iter_mut() {
289 let limb = rem.iter_u32_digits().next().unwrap_or(0) & mask_u32;
291 *item = TF::from_int(limb);
292
293 rem >>= radix_bits;
295 }
296
297 out
298}
299
300#[must_use]
317pub fn squeeze_field_order_num_limbs<PF: PrimeField, TF: PrimeField32>() -> usize {
318 let p = BigUint::from(TF::ORDER_U32);
319 let n = PF::order();
320 let mut count = 0usize;
321 let mut power = BigUint::from(1u32);
322 while &power * &p < n {
323 power *= &p;
324 count += 1;
325 }
326 count.saturating_sub(1)
327}
328
329#[must_use]
338pub fn split_pf_to_field_order_limbs<SF: PrimeField, TF: PrimeField32>(
339 val: SF,
340 num_limbs: usize,
341) -> Vec<TF> {
342 let p_u32 = TF::ORDER_U32;
343 let mut rem = val.as_canonical_biguint();
344 let mut out = Vec::with_capacity(num_limbs);
345
346 for _ in 0..num_limbs {
347 let limb = (&rem % p_u32).to_u32_digits().first().copied().unwrap_or(0);
349 out.push(TF::from_int(limb));
350
351 rem /= p_u32;
353 }
354 out
355}
356
357#[must_use]
368pub fn split_32<SF: PrimeField, TF: PrimeField32>(val: SF, n: usize) -> Vec<TF> {
369 let mut result: Vec<TF> = val
370 .as_canonical_biguint()
371 .to_u64_digits()
372 .iter()
373 .take(n)
374 .map(|d| TF::from_u64(*d))
375 .collect();
376
377 result.resize_with(n, || TF::ZERO);
379 result
380}
381
382#[must_use]
384pub fn dot_product<S, LI, RI>(li: LI, ri: RI) -> S
385where
386 LI: Iterator,
387 RI: Iterator,
388 LI::Item: Mul<RI::Item>,
389 S: Sum<<LI::Item as Mul<RI::Item>>::Output>,
390{
391 li.zip(ri).map(|(l, r)| l * r).sum()
392}