Skip to main content

twenty_first/math/
b_field_element.rs

1use std::convert::TryFrom;
2use std::fmt;
3use std::fmt::Formatter;
4use std::hash::Hash;
5use std::iter::Sum;
6use std::num::TryFromIntError;
7use std::ops::Add;
8use std::ops::AddAssign;
9use std::ops::Div;
10use std::ops::Mul;
11use std::ops::MulAssign;
12use std::ops::Neg;
13use std::ops::Sub;
14use std::ops::SubAssign;
15use std::str::FromStr;
16
17use arbitrary::Arbitrary;
18use arbitrary::Unstructured;
19use get_size2::GetSize;
20use num_traits::ConstOne;
21use num_traits::ConstZero;
22use num_traits::One;
23use num_traits::Zero;
24use phf::phf_map;
25use rand::Rng;
26use rand::RngExt;
27use rand::distr::Distribution;
28use rand::distr::StandardUniform;
29use serde::Deserialize;
30use serde::Deserializer;
31use serde::Serialize;
32use serde::Serializer;
33
34use super::traits::Inverse;
35use super::traits::PrimitiveRootOfUnity;
36use super::x_field_element::XFieldElement;
37use crate::error::ParseBFieldElementError;
38use crate::math::traits::CyclicGroupGenerator;
39use crate::math::traits::FiniteField;
40use crate::math::traits::ModPowU32;
41use crate::math::traits::ModPowU64;
42
43const PRIMITIVE_ROOTS: phf::Map<u64, u64> = phf_map! {
44    0u64 => 1,
45    1u64 => 1,
46    2u64 => 18446744069414584320,
47    4u64 => 281474976710656,
48    8u64 => 18446744069397807105,
49    16u64 => 17293822564807737345,
50    32u64 => 70368744161280,
51    64u64 => 549755813888,
52    128u64 => 17870292113338400769,
53    256u64 => 13797081185216407910,
54    512u64 => 1803076106186727246,
55    1024u64 => 11353340290879379826,
56    2048u64 => 455906449640507599,
57    4096u64 => 17492915097719143606,
58    8192u64 => 1532612707718625687,
59    16384u64 => 16207902636198568418,
60    32768u64 => 17776499369601055404,
61    65536u64 => 6115771955107415310,
62    131072u64 => 12380578893860276750,
63    262144u64 => 9306717745644682924,
64    524288u64 => 18146160046829613826,
65    1048576u64 => 3511170319078647661,
66    2097152u64 => 17654865857378133588,
67    4194304u64 => 5416168637041100469,
68    8388608u64 => 16905767614792059275,
69    16777216u64 => 9713644485405565297,
70    33554432u64 => 5456943929260765144,
71    67108864u64 => 17096174751763063430,
72    134217728u64 => 1213594585890690845,
73    268435456u64 => 6414415596519834757,
74    536870912u64 => 16116352524544190054,
75    1073741824u64 => 9123114210336311365,
76    2147483648u64 => 4614640910117430873,
77    4294967296u64 => 1753635133440165772,
78};
79
80/// Base field element ∈ ℤ_{2^64 - 2^32 + 1}.
81///
82/// In Montgomery representation. This implementation follows <https://eprint.iacr.org/2022/274.pdf>
83/// and <https://github.com/novifinancial/winterfell/pull/101/files>.
84#[derive(Copy, Clone, Default, Hash, PartialEq, Eq, GetSize)]
85#[repr(transparent)]
86pub struct BFieldElement(u64);
87
88/// Simplifies constructing [base field element][BFieldElement]s.
89///
90/// The type [`BFieldElement`] must be in scope for this macro to work.
91/// See [`BFieldElement::from`] for supported types.
92///
93/// # Examples
94///
95/// ```
96/// # use twenty_first::prelude::*;
97/// let a = bfe!(42);
98/// let b = bfe!(-12); // correctly translates to `BFieldElement::P - 12`
99/// let c = bfe!(42 - 12);
100/// assert_eq!(a + b, c);
101///```
102#[macro_export]
103macro_rules! bfe {
104    ($value:expr) => {
105        BFieldElement::from($value)
106    };
107}
108
109/// Simplifies constructing vectors of [base field element][BFieldElement]s.
110///
111/// The type [`BFieldElement`] must be in scope for this macro to work. See also [`bfe!`].
112///
113/// # Examples
114///
115/// ```
116/// # use twenty_first::prelude::*;
117/// let a = bfe_vec![1, 2, 3];
118/// let b = vec![bfe!(1), bfe!(2), bfe!(3)];
119/// assert_eq!(a, b);
120/// ```
121///
122/// ```
123/// # use twenty_first::prelude::*;
124/// let a = bfe_vec![42; 15];
125/// let b = vec![bfe!(42); 15];
126/// assert_eq!(a, b);
127/// ```
128///
129#[macro_export]
130macro_rules! bfe_vec {
131    ($b:expr; $n:expr) => {
132        vec![BFieldElement::from($b); $n]
133    };
134    ($($b:expr),* $(,)?) => {
135        vec![$(BFieldElement::from($b)),*]
136    };
137}
138
139/// Simplifies constructing arrays of [base field element][BFieldElement]s.
140///
141/// The type [`BFieldElement`] must be in scope for this macro to work. See also [`bfe!`].
142///
143/// # Examples
144///
145/// ```
146/// # use twenty_first::prelude::*;
147/// let a = bfe_array![1, 2, 3];
148/// let b = [bfe!(1), bfe!(2), bfe!(3)];
149/// assert_eq!(a, b);
150/// ```
151///
152/// ```
153/// # use twenty_first::prelude::*;
154/// let a = bfe_array![42; 15];
155/// let b = [bfe!(42); 15];
156/// assert_eq!(a, b);
157/// ```
158#[macro_export]
159macro_rules! bfe_array {
160    ($b:expr; $n:expr) => {
161        [BFieldElement::from($b); $n]
162    };
163    ($($b:expr),* $(,)?) => {
164        [$(BFieldElement::from($b)),*]
165    };
166}
167
168impl fmt::Debug for BFieldElement {
169    fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
170        f.debug_tuple("BFieldElement").field(&self.value()).finish()
171    }
172}
173
174impl fmt::LowerHex for BFieldElement {
175    fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
176        fmt::LowerHex::fmt(&self.value(), f)
177    }
178}
179
180impl fmt::UpperHex for BFieldElement {
181    fn fmt(&self, f: &mut Formatter<'_>) -> fmt::Result {
182        fmt::UpperHex::fmt(&self.value(), f)
183    }
184}
185
186impl<'a> Arbitrary<'a> for BFieldElement {
187    fn arbitrary(u: &mut Unstructured<'a>) -> arbitrary::Result<Self> {
188        u.arbitrary().map(BFieldElement::new)
189    }
190
191    fn size_hint(depth: usize) -> (usize, Option<usize>) {
192        u64::size_hint(depth)
193    }
194}
195
196impl Serialize for BFieldElement {
197    fn serialize<S>(&self, serializer: S) -> Result<S::Ok, S::Error>
198    where
199        S: Serializer,
200    {
201        self.value().serialize(serializer)
202    }
203}
204
205impl<'de> Deserialize<'de> for BFieldElement {
206    fn deserialize<D>(deserializer: D) -> Result<Self, D::Error>
207    where
208        D: Deserializer<'de>,
209    {
210        Ok(Self::new(u64::deserialize(deserializer)?))
211    }
212}
213
214impl Sum for BFieldElement {
215    fn sum<I: Iterator<Item = Self>>(iter: I) -> Self {
216        iter.reduce(|a, b| a + b)
217            .unwrap_or_else(BFieldElement::zero)
218    }
219}
220
221impl BFieldElement {
222    pub const BYTES: usize = 8;
223
224    /// The base field's prime, _i.e._, 2^64 - 2^32 + 1.
225    pub const P: u64 = 0xffff_ffff_0000_0001;
226    pub const MAX: u64 = Self::P - 1;
227
228    /// 2^128 mod P; this is used for conversion of elements into Montgomery representation.
229    const R2: u64 = 0xffff_fffe_0000_0001;
230
231    /// -2^-1
232    pub const MINUS_TWO_INVERSE: Self = Self::new(0x7fff_ffff_8000_0000);
233
234    #[inline]
235    pub const fn new(value: u64) -> Self {
236        Self(Self::montyred((value as u128) * (Self::R2 as u128)))
237    }
238
239    /// Construct a new base field element iff the given value is
240    /// [canonical][Self::is_canonical], an error otherwise.
241    fn try_new(v: u64) -> Result<Self, ParseBFieldElementError> {
242        Self::is_canonical(v)
243            .then(|| Self::new(v))
244            .ok_or(ParseBFieldElementError::NotCanonical(i128::from(v)))
245    }
246
247    #[inline]
248    pub const fn value(&self) -> u64 {
249        self.canonical_representation()
250    }
251
252    #[must_use]
253    #[inline]
254    pub fn inverse(&self) -> Self {
255        #[inline(always)]
256        const fn exp(base: BFieldElement, exponent: u64) -> BFieldElement {
257            let mut res = base;
258            let mut i = 0;
259            while i < exponent {
260                res = BFieldElement(BFieldElement::montyred(res.0 as u128 * res.0 as u128));
261                i += 1;
262            }
263            res
264        }
265
266        let x = *self;
267        assert_ne!(
268            x,
269            Self::ZERO,
270            "Attempted to find the multiplicative inverse of zero."
271        );
272
273        let bin_2_ones = x.square() * x;
274        let bin_3_ones = bin_2_ones.square() * x;
275        let bin_6_ones = exp(bin_3_ones, 3) * bin_3_ones;
276        let bin_12_ones = exp(bin_6_ones, 6) * bin_6_ones;
277        let bin_24_ones = exp(bin_12_ones, 12) * bin_12_ones;
278        let bin_30_ones = exp(bin_24_ones, 6) * bin_6_ones;
279        let bin_31_ones = bin_30_ones.square() * x;
280        let bin_31_ones_1_zero = bin_31_ones.square();
281        let bin_32_ones = bin_31_ones.square() * x;
282
283        exp(bin_31_ones_1_zero, 32) * bin_32_ones
284    }
285
286    /// Square the base M times and multiply the result by the tail value
287    #[inline]
288    pub const fn power_accumulator<const N: usize, const M: usize>(
289        base: [Self; N],
290        tail: [Self; N],
291    ) -> [Self; N] {
292        let mut result = base;
293        let mut i = 0;
294        while i < M {
295            let mut j = 0;
296            while j < N {
297                result[j] = Self(Self::montyred(result[j].0 as u128 * result[j].0 as u128));
298                j += 1;
299            }
300            i += 1;
301        }
302
303        let mut j = 0;
304        while j < N {
305            result[j] = Self(Self::montyred(result[j].0 as u128 * tail[j].0 as u128));
306            j += 1;
307        }
308        result
309    }
310
311    /// A generator for the entire base field.
312    pub const fn generator() -> Self {
313        BFieldElement::new(7)
314    }
315
316    /// Turn this base field element into the corresponding extension field
317    /// element.
318    #[inline]
319    pub const fn lift(&self) -> XFieldElement {
320        XFieldElement::new_const(*self)
321    }
322
323    /// Add [`BFieldElement::ONE`] to `self`.
324    pub fn increment(&mut self) {
325        *self += Self::ONE;
326    }
327
328    /// Subtract [`BFieldElement::ONE`] from `self`.
329    pub fn decrement(&mut self) {
330        *self -= Self::ONE;
331    }
332
333    #[inline]
334    const fn canonical_representation(&self) -> u64 {
335        Self::montyred(self.0 as u128)
336    }
337
338    #[must_use]
339    #[inline]
340    pub const fn mod_pow(&self, exp: u64) -> Self {
341        let mut acc = BFieldElement::ONE;
342        let bit_length = u64::BITS - exp.leading_zeros();
343        let mut i = 0;
344        while i < bit_length {
345            acc = Self(Self::montyred(acc.0 as u128 * acc.0 as u128));
346            if exp & (1 << (bit_length - 1 - i)) != 0 {
347                acc = Self(Self::montyred(acc.0 as u128 * self.0 as u128));
348            }
349            i += 1;
350        }
351
352        acc
353    }
354
355    /// Montgomery reduction
356    #[inline(always)]
357    pub const fn montyred(x: u128) -> u64 {
358        // See reference above for a description of the following implementation.
359        let xl = x as u64;
360        let xh = (x >> 64) as u64;
361        let (a, e) = xl.overflowing_add(xl << 32);
362
363        let b = a.wrapping_sub(a >> 32).wrapping_sub(e as u64);
364
365        let (r, c) = xh.overflowing_sub(b);
366
367        // See https://github.com/Neptune-Crypto/twenty-first/pull/70 for various ways
368        // of expressing this.
369        r.wrapping_sub((1 + !Self::P) * c as u64)
370    }
371
372    /// Return the raw bytes or 8-bit chunks of the Montgomery
373    /// representation, in little-endian byte order
374    pub const fn raw_bytes(&self) -> [u8; 8] {
375        self.0.to_le_bytes()
376    }
377
378    /// Take a slice of 8 bytes and interpret it as an integer in
379    /// little-endian byte order, and cast it to a BFieldElement
380    /// in Montgomery representation
381    pub const fn from_raw_bytes(bytes: &[u8; 8]) -> Self {
382        Self(u64::from_le_bytes(*bytes))
383    }
384
385    /// Return the raw 16-bit chunks of the Montgomery
386    /// representation, in little-endian chunk order
387    pub const fn raw_u16s(&self) -> [u16; 4] {
388        [
389            (self.0 & 0xffff) as u16,
390            ((self.0 >> 16) & 0xffff) as u16,
391            ((self.0 >> 32) & 0xffff) as u16,
392            ((self.0 >> 48) & 0xffff) as u16,
393        ]
394    }
395
396    /// Take a slice of 4 16-bit chunks and interpret it as an integer in
397    /// little-endian chunk order, and cast it to a BFieldElement
398    /// in Montgomery representation
399    pub const fn from_raw_u16s(chunks: &[u16; 4]) -> Self {
400        Self(
401            ((chunks[3] as u64) << 48)
402                | ((chunks[2] as u64) << 32)
403                | ((chunks[1] as u64) << 16)
404                | (chunks[0] as u64),
405        )
406    }
407
408    #[inline]
409    pub fn raw_u128(&self) -> u128 {
410        self.0.into()
411    }
412
413    #[inline]
414    pub const fn from_raw_u64(e: u64) -> BFieldElement {
415        BFieldElement(e)
416    }
417
418    #[inline]
419    pub const fn raw_u64(&self) -> u64 {
420        self.0
421    }
422
423    #[inline]
424    pub const fn is_canonical(x: u64) -> bool {
425        x < Self::P
426    }
427}
428
429impl fmt::Display for BFieldElement {
430    fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
431        let canonical_value = Self::canonical_representation(self);
432        let cutoff = 256;
433        if canonical_value >= Self::P - cutoff {
434            write!(f, "-{}", Self::P - canonical_value)
435        } else if canonical_value <= cutoff {
436            write!(f, "{canonical_value}")
437        } else {
438            write!(f, "{canonical_value:>020}")
439        }
440    }
441}
442
443impl FromStr for BFieldElement {
444    type Err = ParseBFieldElementError;
445
446    fn from_str(s: &str) -> Result<Self, Self::Err> {
447        let parsed = s.parse::<i128>().map_err(Self::Err::ParseIntError)?;
448
449        let p = i128::from(Self::P);
450        let normalized = match parsed {
451            n if n <= -p => return Err(Self::Err::NotCanonical(parsed)),
452            n if n < 0 => n + p,
453            n => n,
454        };
455
456        let bfe_value = u64::try_from(normalized).map_err(|_| Self::Err::NotCanonical(parsed))?;
457        Self::try_new(bfe_value)
458    }
459}
460
461impl From<usize> for BFieldElement {
462    fn from(value: usize) -> Self {
463        // targets with wider target pointers don't exist at the time of writing
464        #[cfg(any(
465            target_pointer_width = "16",
466            target_pointer_width = "32",
467            target_pointer_width = "64",
468        ))]
469        Self::new(value as u64)
470    }
471}
472
473impl From<u128> for BFieldElement {
474    fn from(value: u128) -> Self {
475        fn mod_reduce(x: u128) -> u64 {
476            const LOWER_MASK: u64 = 0xFFFF_FFFF;
477
478            let x_lo = x as u64;
479            let x_hi = (x >> 64) as u64;
480            let x_hi_lo = (x_hi as u32) as u64;
481            let x_hi_hi = x_hi >> 32;
482
483            // x_lo - x_hi_hi; potential underflow because `x_hi_hi` may be greater than `x_lo`
484            let (tmp0, is_underflow) = x_lo.overflowing_sub(x_hi_hi);
485            let tmp1 = tmp0.wrapping_sub(LOWER_MASK * (is_underflow as u64));
486
487            // guaranteed not to underflow
488            let tmp2 = (x_hi_lo << 32) - x_hi_lo;
489
490            // adding tmp values gives final result;
491            // potential overflow because each of the `tmp`s may be up to 64 bits
492            let (result, is_overflow) = tmp1.overflowing_add(tmp2);
493            result.wrapping_add(LOWER_MASK * (is_overflow as u64))
494        }
495
496        Self::new(mod_reduce(value))
497    }
498}
499
500macro_rules! impl_from_small_unsigned_int_for_bfe {
501    ($($t:ident),+ $(,)?) => {$(
502        impl From<$t> for BFieldElement {
503            fn from(value: $t) -> Self {
504                Self::new(u64::from(value))
505            }
506        }
507    )+};
508}
509
510impl_from_small_unsigned_int_for_bfe!(u8, u16, u32, u64);
511
512impl From<isize> for BFieldElement {
513    fn from(value: isize) -> Self {
514        // targets with wider target pointers don't exist at the time of writing
515        #[cfg(target_pointer_width = "16")]
516        {
517            (value as i16).into()
518        }
519        #[cfg(target_pointer_width = "32")]
520        {
521            (value as i32).into()
522        }
523        #[cfg(target_pointer_width = "64")]
524        {
525            (value as i64).into()
526        }
527    }
528}
529
530impl From<i64> for BFieldElement {
531    fn from(value: i64) -> Self {
532        match i128::from(value) {
533            0.. => value as u128,
534            _ => (value as u128) - BFieldElement::R2 as u128,
535        }
536        .into()
537    }
538}
539
540macro_rules! impl_from_small_signed_int_for_bfe {
541    ($($t:ident),+ $(,)?) => {$(
542        impl From<$t> for BFieldElement {
543            fn from(value: $t) -> Self {
544                i64::from(value).into()
545            }
546        }
547    )+};
548}
549
550impl_from_small_signed_int_for_bfe!(i8, i16, i32);
551
552macro_rules! impl_try_from_bfe_for_int {
553    ($($t:ident),+ $(,)?) => {$(
554        impl TryFrom<BFieldElement> for $t {
555            type Error = TryFromIntError;
556
557            fn try_from(value: BFieldElement) -> Result<Self, Self::Error> {
558                $t::try_from(value.canonical_representation())
559            }
560        }
561
562        impl TryFrom<&BFieldElement> for $t {
563            type Error = TryFromIntError;
564
565            fn try_from(value: &BFieldElement) -> Result<Self, Self::Error> {
566                $t::try_from(value.canonical_representation())
567            }
568        }
569    )+};
570}
571
572impl_try_from_bfe_for_int!(u8, i8, u16, i16, u32, i32, usize, isize);
573
574macro_rules! impl_from_bfe_for_int {
575    ($($t:ident),+ $(,)?) => {$(
576        impl From<BFieldElement> for $t {
577            fn from(elem: BFieldElement) -> Self {
578                Self::from(elem.canonical_representation())
579            }
580        }
581
582        impl From<&BFieldElement> for $t {
583            fn from(elem: &BFieldElement) -> Self {
584                Self::from(elem.canonical_representation())
585            }
586        }
587    )+};
588}
589
590impl_from_bfe_for_int!(u64, u128, i128);
591
592impl From<BFieldElement> for i64 {
593    fn from(elem: BFieldElement) -> Self {
594        bfe_to_i64(&elem)
595    }
596}
597
598impl From<&BFieldElement> for i64 {
599    fn from(elem: &BFieldElement) -> Self {
600        bfe_to_i64(elem)
601    }
602}
603
604const fn bfe_to_i64(bfe: &BFieldElement) -> i64 {
605    let v = bfe.canonical_representation();
606    if v <= i64::MAX as u64 {
607        v as i64
608    } else {
609        (v as i128 - BFieldElement::P as i128) as i64
610    }
611}
612
613/// Convert a B-field element to a byte array.
614/// The client uses this for its database.
615impl From<BFieldElement> for [u8; BFieldElement::BYTES] {
616    fn from(bfe: BFieldElement) -> Self {
617        // It's crucial to map this to the canonical representation before converting.
618        // Otherwise, the representation is degenerate.
619        bfe.canonical_representation().to_le_bytes()
620    }
621}
622
623impl TryFrom<[u8; BFieldElement::BYTES]> for BFieldElement {
624    type Error = ParseBFieldElementError;
625
626    fn try_from(array: [u8; BFieldElement::BYTES]) -> Result<Self, Self::Error> {
627        Self::try_new(u64::from_le_bytes(array))
628    }
629}
630
631impl TryFrom<&[u8]> for BFieldElement {
632    type Error = ParseBFieldElementError;
633
634    fn try_from(bytes: &[u8]) -> Result<Self, Self::Error> {
635        <[u8; BFieldElement::BYTES]>::try_from(bytes)
636            .map_err(|_| Self::Error::InvalidNumBytes(bytes.len()))?
637            .try_into()
638    }
639}
640
641impl Inverse for BFieldElement {
642    #[inline]
643    fn inverse(&self) -> Self {
644        self.inverse()
645    }
646}
647
648impl ModPowU32 for BFieldElement {
649    #[inline]
650    fn mod_pow_u32(&self, exp: u32) -> Self {
651        self.mod_pow(exp as u64)
652    }
653}
654
655impl CyclicGroupGenerator for BFieldElement {
656    fn get_cyclic_group_elements(&self, max: Option<usize>) -> Vec<Self> {
657        let mut val = *self;
658        let mut ret: Vec<Self> = vec![Self::one()];
659
660        loop {
661            ret.push(val);
662            val *= *self;
663            if val.is_one() || max.is_some() && ret.len() >= max.unwrap() {
664                break;
665            }
666        }
667        ret
668    }
669}
670
671impl Distribution<BFieldElement> for StandardUniform {
672    fn sample<R: Rng + ?Sized>(&self, rng: &mut R) -> BFieldElement {
673        BFieldElement::new(rng.random_range(0..=BFieldElement::MAX))
674    }
675}
676
677impl FiniteField for BFieldElement {}
678
679impl Zero for BFieldElement {
680    #[inline]
681    fn zero() -> Self {
682        Self::ZERO
683    }
684
685    #[inline]
686    fn is_zero(&self) -> bool {
687        self == &Self::ZERO
688    }
689}
690
691impl ConstZero for BFieldElement {
692    const ZERO: Self = Self::new(0);
693}
694
695impl One for BFieldElement {
696    #[inline]
697    fn one() -> Self {
698        Self::ONE
699    }
700
701    #[inline]
702    fn is_one(&self) -> bool {
703        self == &Self::ONE
704    }
705}
706
707impl ConstOne for BFieldElement {
708    const ONE: Self = Self::new(1);
709}
710
711impl Add for BFieldElement {
712    type Output = Self;
713
714    #[expect(clippy::suspicious_arithmetic_impl)]
715    #[inline(always)]
716    fn add(self, rhs: Self) -> Self {
717        // Compute a + b = a - (p - b).
718        let (x1, c1) = self.0.overflowing_sub(Self::P - rhs.0);
719
720        // The following if/else is equivalent to the commented-out code below but
721        // the if/else was found to be faster.
722        // let adj = 0u32.wrapping_sub(c1 as u32);
723        // Self(x1.wrapping_sub(adj as u64))
724        // See
725        // https://github.com/Neptune-Crypto/twenty-first/pull/70
726        if c1 {
727            Self(x1.wrapping_add(Self::P))
728        } else {
729            Self(x1)
730        }
731    }
732}
733
734impl AddAssign for BFieldElement {
735    #[inline(always)]
736    fn add_assign(&mut self, rhs: Self) {
737        *self = *self + rhs
738    }
739}
740
741impl SubAssign for BFieldElement {
742    #[inline]
743    fn sub_assign(&mut self, rhs: Self) {
744        *self = *self - rhs
745    }
746}
747
748impl MulAssign for BFieldElement {
749    #[inline]
750    fn mul_assign(&mut self, rhs: Self) {
751        *self = *self * rhs;
752    }
753}
754
755impl Mul for BFieldElement {
756    type Output = Self;
757
758    #[inline]
759    fn mul(self, rhs: Self) -> Self {
760        Self(Self::montyred((self.0 as u128) * (rhs.0 as u128)))
761    }
762}
763
764impl Neg for BFieldElement {
765    type Output = Self;
766
767    #[inline]
768    fn neg(self) -> Self {
769        Self::zero() - self
770    }
771}
772
773impl Sub for BFieldElement {
774    type Output = Self;
775
776    #[inline]
777    fn sub(self, rhs: Self) -> Self {
778        let (x1, c1) = self.0.overflowing_sub(rhs.0);
779
780        // The following code is equivalent to the commented-out code below
781        // but they were determined to have near-equiavalent running times. Maybe because
782        // subtraction is not used very often.
783        // See: https://github.com/Neptune-Crypto/twenty-first/pull/70
784        // 1st alternative:
785        // if c1 {
786        //     Self(x1.wrapping_add(Self::P))
787        // } else {
788        //     Self(x1)
789        // }
790        // 2nd alternative:
791        // let adj = 0u32.wrapping_sub(c1 as u32);
792        // Self(x1.wrapping_sub(adj as u64))
793        Self(x1.wrapping_sub((1 + !Self::P) * c1 as u64))
794    }
795}
796
797impl Div for BFieldElement {
798    type Output = Self;
799
800    #[expect(clippy::suspicious_arithmetic_impl)]
801    fn div(self, other: Self) -> Self {
802        other.inverse() * self
803    }
804}
805
806// TODO: We probably wanna make use of Rust's Pow, but for now we copy from ...big:
807impl ModPowU64 for BFieldElement {
808    #[inline]
809    fn mod_pow_u64(&self, pow: u64) -> Self {
810        self.mod_pow(pow)
811    }
812}
813
814impl PrimitiveRootOfUnity for BFieldElement {
815    fn primitive_root_of_unity(n: u64) -> Option<BFieldElement> {
816        PRIMITIVE_ROOTS.get(&n).map(|&r| BFieldElement::new(r))
817    }
818}
819
820#[cfg(test)]
821#[cfg_attr(coverage_nightly, coverage(off))]
822mod tests {
823    use std::collections::hash_map::DefaultHasher;
824    use std::hash::Hasher;
825
826    use itertools::izip;
827    use proptest::prelude::*;
828    use proptest_arbitrary_adapter::arb;
829    use rand::random;
830
831    use crate::math::b_field_element::*;
832    use crate::math::other::random_elements;
833    use crate::math::polynomial::Polynomial;
834    use crate::tests::proptest;
835    use crate::tests::test;
836
837    impl proptest::arbitrary::Arbitrary for BFieldElement {
838        type Parameters = ();
839
840        fn arbitrary_with(_: Self::Parameters) -> Self::Strategy {
841            arb().boxed()
842        }
843
844        type Strategy = BoxedStrategy<Self>;
845    }
846
847    #[macro_rules_attr::apply(proptest)]
848    fn get_size(bfe: BFieldElement) {
849        prop_assert_eq!(8, bfe.get_size());
850    }
851
852    #[macro_rules_attr::apply(proptest)]
853    fn serialization_and_deserialization_to_and_from_json_is_identity(bfe: BFieldElement) {
854        let serialized = serde_json::to_string(&bfe).unwrap();
855        let deserialized: BFieldElement = serde_json::from_str(&serialized).unwrap();
856        prop_assert_eq!(bfe, deserialized);
857    }
858
859    #[macro_rules_attr::apply(proptest)]
860    fn deserializing_u64_is_like_calling_new(#[strategy(0..=BFieldElement::MAX)] value: u64) {
861        let bfe = BFieldElement::new(value);
862        let deserialized: BFieldElement = serde_json::from_str(&value.to_string()).unwrap();
863        prop_assert_eq!(bfe, deserialized);
864    }
865
866    #[macro_rules_attr::apply(test)]
867    fn parsing_interval_is_open_minus_p_to_p() {
868        let p = i128::from(BFieldElement::P);
869        let display_then_parse = |v: i128| BFieldElement::from_str(&v.to_string());
870
871        assert!(display_then_parse(-p).is_err());
872        assert!(display_then_parse(-p + 1).is_ok());
873        assert!(display_then_parse(p - 1).is_ok());
874        assert!(display_then_parse(p).is_err());
875    }
876
877    #[macro_rules_attr::apply(proptest)]
878    fn parsing_string_representing_canonical_negative_integer_gives_correct_bfield_element(
879        #[strategy(0..=BFieldElement::MAX)] v: u64,
880    ) {
881        let bfe = BFieldElement::from_str(&(-i128::from(v)).to_string())?;
882        prop_assert_eq!(BFieldElement::P - v, bfe.value());
883    }
884
885    #[macro_rules_attr::apply(proptest)]
886    fn parsing_string_representing_canonical_positive_integer_gives_correct_bfield_element(
887        #[strategy(0..=BFieldElement::MAX)] v: u64,
888    ) {
889        let bfe = BFieldElement::from_str(&v.to_string())?;
890        prop_assert_eq!(v, bfe.value());
891    }
892
893    #[macro_rules_attr::apply(proptest)]
894    fn parsing_string_representing_too_big_positive_integer_as_bfield_element_gives_error(
895        #[strategy(i128::from(BFieldElement::P)..)] v: i128,
896    ) {
897        let err = BFieldElement::from_str(&v.to_string()).unwrap_err();
898        prop_assert_eq!(ParseBFieldElementError::NotCanonical(v), err);
899    }
900
901    #[macro_rules_attr::apply(proptest)]
902    fn parsing_string_representing_too_small_negative_integer_as_bfield_element_gives_error(
903        #[strategy(..=i128::from(BFieldElement::P))] v: i128,
904    ) {
905        let err = BFieldElement::from_str(&v.to_string()).unwrap_err();
906        prop_assert_eq!(ParseBFieldElementError::NotCanonical(v), err);
907    }
908
909    #[macro_rules_attr::apply(proptest)]
910    fn zero_is_neutral_element_for_addition(bfe: BFieldElement) {
911        let zero = BFieldElement::ZERO;
912        prop_assert_eq!(bfe + zero, bfe);
913    }
914
915    #[macro_rules_attr::apply(proptest)]
916    fn one_is_neutral_element_for_multiplication(bfe: BFieldElement) {
917        let one = BFieldElement::ONE;
918        prop_assert_eq!(bfe * one, bfe);
919    }
920
921    #[macro_rules_attr::apply(proptest)]
922    fn addition_is_commutative(element_0: BFieldElement, element_1: BFieldElement) {
923        prop_assert_eq!(element_0 + element_1, element_1 + element_0);
924    }
925
926    #[macro_rules_attr::apply(proptest)]
927    fn multiplication_is_commutative(element_0: BFieldElement, element_1: BFieldElement) {
928        prop_assert_eq!(element_0 * element_1, element_1 * element_0);
929    }
930
931    #[macro_rules_attr::apply(proptest)]
932
933    fn addition_is_associative(
934        element_0: BFieldElement,
935        element_1: BFieldElement,
936        element_2: BFieldElement,
937    ) {
938        prop_assert_eq!(
939            (element_0 + element_1) + element_2,
940            element_0 + (element_1 + element_2)
941        );
942    }
943
944    #[macro_rules_attr::apply(proptest)]
945    fn multiplication_is_associative(
946        element_0: BFieldElement,
947        element_1: BFieldElement,
948        element_2: BFieldElement,
949    ) {
950        prop_assert_eq!(
951            (element_0 * element_1) * element_2,
952            element_0 * (element_1 * element_2)
953        );
954    }
955
956    #[macro_rules_attr::apply(proptest)]
957    fn multiplication_distributes_over_addition(
958        element_0: BFieldElement,
959        element_1: BFieldElement,
960        element_2: BFieldElement,
961    ) {
962        prop_assert_eq!(
963            element_0 * (element_1 + element_2),
964            element_0 * element_1 + element_0 * element_2
965        );
966    }
967
968    #[macro_rules_attr::apply(proptest)]
969    fn multiplication_with_inverse_gives_identity(#[filter(!#bfe.is_zero())] bfe: BFieldElement) {
970        prop_assert!((bfe.inverse() * bfe).is_one());
971    }
972
973    #[macro_rules_attr::apply(proptest)]
974    fn division_by_self_gives_identity(#[filter(!#bfe.is_zero())] bfe: BFieldElement) {
975        prop_assert!((bfe / bfe).is_one());
976    }
977
978    #[macro_rules_attr::apply(proptest)]
979    fn values_larger_than_modulus_are_handled_correctly(
980        #[strategy(BFieldElement::P..)] large_value: u64,
981    ) {
982        let bfe = BFieldElement::new(large_value);
983        let expected_value = large_value - BFieldElement::P;
984        prop_assert_eq!(expected_value, bfe.value());
985    }
986
987    #[macro_rules_attr::apply(test)]
988    fn display_test() {
989        let seven = BFieldElement::new(7);
990        assert_eq!("7", format!("{seven}"));
991        assert_eq!("7", format!("{seven:x}"));
992        assert_eq!("7", format!("{seven:X}"));
993        assert_eq!("0x7", format!("{seven:#x}"));
994        assert_eq!("0x7", format!("{seven:#X}"));
995        assert_eq!("BFieldElement(7)", format!("{seven:?}"));
996
997        let forty_two = BFieldElement::new(42);
998        assert_eq!("42", format!("{forty_two}"));
999        assert_eq!("2a", format!("{forty_two:x}"));
1000        assert_eq!("2A", format!("{forty_two:X}"));
1001        assert_eq!("0x2a", format!("{forty_two:#x}"));
1002        assert_eq!("0x2A", format!("{forty_two:#X}"));
1003        assert_eq!("BFieldElement(42)", format!("{forty_two:?}"));
1004
1005        let minus_one = BFieldElement::new(BFieldElement::P - 1);
1006        assert_eq!("-1", format!("{minus_one}"));
1007        assert_eq!("ffffffff00000000", format!("{minus_one:x}"));
1008        assert_eq!("FFFFFFFF00000000", format!("{minus_one:X}"));
1009        assert_eq!("0xffffffff00000000", format!("{minus_one:#x}"));
1010        assert_eq!("0xFFFFFFFF00000000", format!("{minus_one:#X}"));
1011        assert_eq!(
1012            "BFieldElement(18446744069414584320)",
1013            format!("{minus_one:?}")
1014        );
1015
1016        let minus_fifteen = BFieldElement::new(BFieldElement::P - 15);
1017        assert_eq!("-15", format!("{minus_fifteen}"));
1018        assert_eq!("fffffffefffffff2", format!("{minus_fifteen:x}"));
1019        assert_eq!("FFFFFFFEFFFFFFF2", format!("{minus_fifteen:X}"));
1020        assert_eq!("0xfffffffefffffff2", format!("{minus_fifteen:#x}"));
1021        assert_eq!("0xFFFFFFFEFFFFFFF2", format!("{minus_fifteen:#X}"));
1022        assert_eq!(
1023            "BFieldElement(18446744069414584306)",
1024            format!("{minus_fifteen:?}")
1025        );
1026    }
1027
1028    #[macro_rules_attr::apply(test)]
1029    fn display_and_from_str_are_reciprocal_unit_test() {
1030        for bfe in bfe_array![
1031            -1000, -500, -200, -100, -10, -1, 0, 1, 10, 100, 200, 500, 1000
1032        ] {
1033            let bfe_again = bfe.to_string().parse().unwrap();
1034            assert_eq!(bfe, bfe_again);
1035        }
1036    }
1037
1038    #[macro_rules_attr::apply(proptest)]
1039    fn display_and_from_str_are_reciprocal_prop_test(bfe: BFieldElement) {
1040        let bfe_again = bfe.to_string().parse()?;
1041        prop_assert_eq!(bfe, bfe_again);
1042    }
1043
1044    #[macro_rules_attr::apply(test)]
1045    fn zero_is_zero() {
1046        let zero = BFieldElement::zero();
1047        assert!(zero.is_zero());
1048        assert_eq!(zero, BFieldElement::ZERO);
1049    }
1050
1051    #[macro_rules_attr::apply(proptest)]
1052    fn not_zero_is_nonzero(bfe: BFieldElement) {
1053        if bfe.value() == 0 {
1054            return Ok(());
1055        }
1056        prop_assert!(!bfe.is_zero());
1057    }
1058
1059    #[macro_rules_attr::apply(test)]
1060    fn one_is_one() {
1061        let one = BFieldElement::one();
1062        assert!(one.is_one());
1063        assert_eq!(one, BFieldElement::ONE);
1064    }
1065
1066    #[macro_rules_attr::apply(proptest)]
1067    fn not_one_is_not_one(bfe: BFieldElement) {
1068        if bfe.value() == 1 {
1069            return Ok(());
1070        }
1071        prop_assert!(!bfe.is_one());
1072    }
1073
1074    #[macro_rules_attr::apply(test)]
1075    fn one_unequal_zero() {
1076        let one = BFieldElement::ONE;
1077        let zero = BFieldElement::ZERO;
1078        assert_ne!(one, zero);
1079    }
1080
1081    #[macro_rules_attr::apply(proptest)]
1082    fn byte_array_of_small_field_elements_is_zero_at_high_indices(value: u8) {
1083        let bfe = BFieldElement::new(value as u64);
1084        let byte_array: [u8; 8] = bfe.into();
1085
1086        prop_assert_eq!(value, byte_array[0]);
1087        (1..8).for_each(|i| {
1088            assert_eq!(0, byte_array[i]);
1089        });
1090    }
1091
1092    #[macro_rules_attr::apply(proptest)]
1093    fn byte_array_conversion(bfe: BFieldElement) {
1094        let array: [u8; 8] = bfe.into();
1095        let bfe_recalculated: BFieldElement = array.try_into()?;
1096        prop_assert_eq!(bfe, bfe_recalculated);
1097    }
1098
1099    #[macro_rules_attr::apply(proptest)]
1100    fn byte_array_outside_range_is_not_accepted(#[strategy(BFieldElement::P..)] value: u64) {
1101        let byte_array = value.to_le_bytes();
1102        prop_assert!(BFieldElement::try_from(byte_array).is_err());
1103    }
1104
1105    #[macro_rules_attr::apply(proptest)]
1106    fn value_is_preserved(#[strategy(0..BFieldElement::P)] value: u64) {
1107        prop_assert_eq!(value, BFieldElement::new(value).value());
1108    }
1109
1110    #[macro_rules_attr::apply(test)]
1111    fn supposed_generator_is_generator() {
1112        let generator = BFieldElement::generator();
1113        let largest_meaningful_power = BFieldElement::P - 1;
1114        let generator_pow_p = generator.mod_pow(largest_meaningful_power);
1115        let generator_pow_p_half = generator.mod_pow(largest_meaningful_power / 2);
1116
1117        assert_eq!(BFieldElement::ONE, generator_pow_p);
1118        assert_ne!(BFieldElement::ONE, generator_pow_p_half);
1119    }
1120
1121    #[macro_rules_attr::apply(proptest)]
1122    fn lift_then_unlift_preserves_element(bfe: BFieldElement) {
1123        prop_assert_eq!(Some(bfe), bfe.lift().unlift());
1124    }
1125
1126    #[macro_rules_attr::apply(proptest)]
1127    fn increment(mut bfe: BFieldElement) {
1128        let old_value = bfe.value();
1129        bfe.increment();
1130        let expected_value = (old_value + 1) % BFieldElement::P;
1131        prop_assert_eq!(expected_value, bfe.value());
1132    }
1133
1134    #[macro_rules_attr::apply(test)]
1135    fn incrementing_max_value_wraps_around() {
1136        let mut bfe = BFieldElement::new(BFieldElement::MAX);
1137        bfe.increment();
1138        assert_eq!(0, bfe.value());
1139    }
1140
1141    #[macro_rules_attr::apply(proptest)]
1142    fn decrement(mut bfe: BFieldElement) {
1143        let old_value = bfe.value();
1144        bfe.decrement();
1145        let expected_value = old_value.checked_sub(1).unwrap_or(BFieldElement::P - 1);
1146        prop_assert_eq!(expected_value, bfe.value());
1147    }
1148
1149    #[macro_rules_attr::apply(test)]
1150    fn decrementing_min_value_wraps_around() {
1151        let mut bfe = BFieldElement::ZERO;
1152        bfe.decrement();
1153        assert_eq!(BFieldElement::MAX, bfe.value());
1154    }
1155
1156    #[macro_rules_attr::apply(test)]
1157    fn empty_batch_inversion() {
1158        let empty_inv = BFieldElement::batch_inversion(vec![]);
1159        assert!(empty_inv.is_empty());
1160    }
1161
1162    #[macro_rules_attr::apply(proptest)]
1163    fn batch_inversion(bfes: Vec<BFieldElement>) {
1164        let bfes_inv = BFieldElement::batch_inversion(bfes.clone());
1165        prop_assert_eq!(bfes.len(), bfes_inv.len());
1166        for (bfe, bfe_inv) in izip!(bfes, bfes_inv) {
1167            prop_assert_eq!(BFieldElement::ONE, bfe * bfe_inv);
1168        }
1169    }
1170
1171    #[macro_rules_attr::apply(test)]
1172    fn power_accumulator_simple_test() {
1173        let input_a = [
1174            BFieldElement::new(10),
1175            BFieldElement::new(100),
1176            BFieldElement::new(1000),
1177            BFieldElement::new(1),
1178        ];
1179        let input_b = [
1180            BFieldElement::new(5),
1181            BFieldElement::new(6),
1182            BFieldElement::new(7),
1183            BFieldElement::new(8),
1184        ];
1185        let powers: [BFieldElement; 4] = BFieldElement::power_accumulator::<4, 2>(input_a, input_b);
1186        assert_eq!(BFieldElement::new(50000), powers[0]);
1187        assert_eq!(BFieldElement::new(600000000), powers[1]);
1188        assert_eq!(BFieldElement::new(7000000000000), powers[2]);
1189        assert_eq!(BFieldElement::new(8), powers[3]);
1190    }
1191
1192    #[macro_rules_attr::apply(test)]
1193    fn mul_div_plus_minus_neg_property_based_test() {
1194        let elements: Vec<BFieldElement> = random_elements(300);
1195        let power_input_b: [BFieldElement; 6] = random();
1196        for i in 1..elements.len() {
1197            let a = elements[i - 1];
1198            let b = elements[i];
1199
1200            let ab = a * b;
1201            let a_o_b = a / b;
1202            let b_o_a = b / a;
1203            assert_eq!(a, ab / b);
1204            assert_eq!(b, ab / a);
1205            assert_eq!(a, a_o_b * b);
1206            assert_eq!(b, b_o_a * a);
1207            assert!((a_o_b * b_o_a).is_one());
1208            assert_eq!(a * a, a.square());
1209
1210            assert_eq!(a - b + b, a);
1211            assert_eq!(b - a + a, b);
1212            assert!((a - a).is_zero());
1213            assert!((b - b).is_zero());
1214
1215            // Test the add/sub/mul assign operators
1216            let mut a_minus_b = a;
1217            a_minus_b -= b;
1218            assert_eq!(a - b, a_minus_b);
1219
1220            let mut a_plus_b = a;
1221            a_plus_b += b;
1222            assert_eq!(a + b, a_plus_b);
1223
1224            let mut a_mul_b = a;
1225            a_mul_b *= b;
1226            assert_eq!(a * b, a_mul_b);
1227            assert_eq!(b * a, a_mul_b);
1228
1229            // Test negation
1230            assert!((-a + a).is_zero());
1231            assert!((-b + b).is_zero());
1232            assert!((-ab + ab).is_zero());
1233            assert!((-a_o_b + a_o_b).is_zero());
1234            assert!((-b_o_a + b_o_a).is_zero());
1235            assert!((-a_minus_b + a_minus_b).is_zero());
1236            assert!((-a_plus_b + a_plus_b).is_zero());
1237            assert!((-a_mul_b + a_mul_b).is_zero());
1238
1239            // Test power_accumulator
1240            let power_input_a = [a, b, ab, a_o_b, b_o_a, a_minus_b];
1241            let powers = BFieldElement::power_accumulator::<6, 4>(power_input_a, power_input_b);
1242            for ((result_element, input_a), input_b) in powers
1243                .iter()
1244                .zip(power_input_a.iter())
1245                .zip(power_input_b.iter())
1246            {
1247                assert_eq!(input_a.mod_pow(16) * *input_b, *result_element);
1248            }
1249        }
1250    }
1251
1252    #[macro_rules_attr::apply(test)]
1253    fn mul_div_pbt() {
1254        // Verify that the mul result is sane
1255        let rands: Vec<BFieldElement> = random_elements(100);
1256        for i in 1..rands.len() {
1257            let prod_mul = rands[i - 1] * rands[i];
1258            let mut prod_mul_assign = rands[i - 1];
1259            prod_mul_assign *= rands[i];
1260            assert_eq!(
1261                prod_mul, prod_mul_assign,
1262                "mul and mul_assign must be the same for B field elements"
1263            );
1264            assert_eq!(prod_mul / rands[i - 1], rands[i]);
1265            assert_eq!(prod_mul / rands[i], rands[i - 1]);
1266        }
1267    }
1268
1269    #[macro_rules_attr::apply(test)]
1270    fn add_sub_wrap_around_test() {
1271        // Ensure that something that exceeds P but is smaller than $2^64$
1272        // is still the correct field element. The property-based test is unlikely
1273        // to hit such an element as the chances of doing so are about 2^(-32) for
1274        // random uniform numbers. So we test this in a separate test
1275        let element = BFieldElement::new(4);
1276        let sum = BFieldElement::new(BFieldElement::MAX) + element;
1277        assert_eq!(BFieldElement::new(3), sum);
1278        let diff = sum - element;
1279        assert_eq!(BFieldElement::new(BFieldElement::MAX), diff);
1280    }
1281
1282    #[macro_rules_attr::apply(test)]
1283    fn neg_test() {
1284        assert_eq!(-BFieldElement::ZERO, BFieldElement::ZERO);
1285        assert_eq!(
1286            (-BFieldElement::ONE).canonical_representation(),
1287            BFieldElement::MAX
1288        );
1289        let max = BFieldElement::new(BFieldElement::MAX);
1290        let max_plus_one = max + BFieldElement::ONE;
1291        let max_plus_two = max_plus_one + BFieldElement::ONE;
1292        assert_eq!(BFieldElement::ZERO, -max_plus_one);
1293        assert_eq!(max, -max_plus_two);
1294    }
1295
1296    #[macro_rules_attr::apply(test)]
1297    fn equality_and_hash_test() {
1298        assert_eq!(BFieldElement::ZERO, BFieldElement::ZERO);
1299        assert_eq!(BFieldElement::ONE, BFieldElement::ONE);
1300        assert_ne!(BFieldElement::ONE, BFieldElement::ZERO);
1301        assert_eq!(BFieldElement::new(42), BFieldElement::new(42));
1302        assert_ne!(BFieldElement::new(42), BFieldElement::new(43));
1303
1304        assert_eq!(
1305            BFieldElement::new(102),
1306            BFieldElement::new(BFieldElement::MAX) + BFieldElement::new(103)
1307        );
1308        assert_ne!(
1309            BFieldElement::new(103),
1310            BFieldElement::new(BFieldElement::MAX) + BFieldElement::new(103)
1311        );
1312
1313        // Verify that hashing works for canonical representations
1314        let mut hasher_a = DefaultHasher::new();
1315        let mut hasher_b = DefaultHasher::new();
1316
1317        std::hash::Hash::hash(&BFieldElement::new(42), &mut hasher_a);
1318        std::hash::Hash::hash(&BFieldElement::new(42), &mut hasher_b);
1319        assert_eq!(hasher_a.finish(), hasher_b.finish());
1320
1321        // Verify that hashing works for non-canonical representations
1322        hasher_a = DefaultHasher::new();
1323        hasher_b = DefaultHasher::new();
1324        let non_canonical = BFieldElement::new(BFieldElement::MAX) + BFieldElement::new(103);
1325        std::hash::Hash::hash(&(non_canonical), &mut hasher_a);
1326        std::hash::Hash::hash(&BFieldElement::new(102), &mut hasher_b);
1327        assert_eq!(hasher_a.finish(), hasher_b.finish());
1328    }
1329
1330    #[macro_rules_attr::apply(test)]
1331    fn create_polynomial_test() {
1332        let a = Polynomial::from([1, 3, 7]);
1333        let b = Polynomial::from([2, 5, -1]);
1334        let expected = Polynomial::<BFieldElement>::from([3, 8, 6]);
1335
1336        assert_eq!(expected, a + b);
1337    }
1338
1339    #[macro_rules_attr::apply(test)]
1340    fn mod_pow_test_powers_of_two() {
1341        let two = BFieldElement::new(2);
1342        // 2^63 < 2^64, so no wrap-around of B-field element
1343        for i in 0..64 {
1344            assert_eq!(BFieldElement::new(1 << i), two.mod_pow(i));
1345        }
1346    }
1347
1348    #[macro_rules_attr::apply(test)]
1349    fn mod_pow_test_powers_of_three() {
1350        let three = BFieldElement::new(3);
1351        // 3^40 < 2^64, so no wrap-around of B-field element
1352        for i in 0..41 {
1353            assert_eq!(BFieldElement::new(3u64.pow(i as u32)), three.mod_pow(i));
1354        }
1355    }
1356
1357    #[macro_rules_attr::apply(test)]
1358    fn mod_pow_test() {
1359        // These values were found by finding primitive roots of unity and verifying
1360        // that they are group generators of the right order
1361        assert!(BFieldElement::new(281474976710656).mod_pow(4).is_one());
1362        assert_eq!(
1363            BFieldElement::new(281474976710656),
1364            BFieldElement::new(281474976710656).mod_pow(5)
1365        );
1366        assert!(BFieldElement::new(18446744069414584320).mod_pow(2).is_one());
1367        assert!(BFieldElement::new(18446744069397807105).mod_pow(8).is_one());
1368        assert!(BFieldElement::new(2625919085333925275).mod_pow(10).is_one());
1369        assert!(BFieldElement::new(281474976645120).mod_pow(12).is_one());
1370        assert!(BFieldElement::new(0).mod_pow(0).is_one());
1371    }
1372
1373    #[macro_rules_attr::apply(test)]
1374    fn get_primitive_root_of_unity_test() {
1375        for i in 1..33 {
1376            let power = 1 << i;
1377            let root_result = BFieldElement::primitive_root_of_unity(power);
1378            match root_result {
1379                Some(root) => println!("{power} => {root},"),
1380                None => println!("Found no primitive root of unity for n = {power}"),
1381            };
1382            let root = root_result.unwrap();
1383            assert!(root.mod_pow(power).is_one());
1384            assert!(!root.mod_pow(power / 2).is_one());
1385        }
1386    }
1387
1388    #[macro_rules_attr::apply(test)]
1389    #[should_panic(expected = "Attempted to find the multiplicative inverse of zero.")]
1390    fn multiplicative_inverse_of_zero() {
1391        let zero = BFieldElement::ZERO;
1392        let _ = zero.inverse();
1393    }
1394
1395    #[macro_rules_attr::apply(test)]
1396    fn u32_conversion() {
1397        let val = BFieldElement::new(u32::MAX as u64);
1398        let as_u32: u32 = val.try_into().unwrap();
1399        assert_eq!(u32::MAX, as_u32);
1400
1401        for i in 1..100 {
1402            let invalid_val_0 = BFieldElement::new((u32::MAX as u64) + i);
1403            let converted_0 = TryInto::<u32>::try_into(invalid_val_0);
1404            assert!(converted_0.is_err());
1405        }
1406    }
1407
1408    #[macro_rules_attr::apply(test)]
1409    fn inverse_or_zero_bfe() {
1410        let zero = BFieldElement::ZERO;
1411        let one = BFieldElement::ONE;
1412        assert_eq!(zero, zero.inverse_or_zero());
1413
1414        let mut rng = rand::rng();
1415        let elem: BFieldElement = rng.random();
1416        if elem.is_zero() {
1417            assert_eq!(zero, elem.inverse_or_zero())
1418        } else {
1419            assert_eq!(one, elem * elem.inverse_or_zero());
1420        }
1421    }
1422
1423    #[macro_rules_attr::apply(test)]
1424    fn test_random_squares() {
1425        let mut rng = rand::rng();
1426        let p = BFieldElement::P;
1427        for _ in 0..100 {
1428            let a = rng.random_range(0..p);
1429            let asq = (((a as u128) * (a as u128)) % (p as u128)) as u64;
1430            let b = BFieldElement::new(a);
1431            let bsq = BFieldElement::new(asq);
1432            assert_eq!(bsq, b * b);
1433            assert_eq!(bsq.value(), (b * b).value());
1434            assert_eq!(b.value(), a);
1435            assert_eq!(bsq.value(), asq);
1436        }
1437        let one = BFieldElement::new(1);
1438        assert_eq!(one, one * one);
1439    }
1440
1441    #[macro_rules_attr::apply(test)]
1442    fn equals() {
1443        let a = BFieldElement::ONE;
1444        let b = bfe!(BFieldElement::MAX) * bfe!(BFieldElement::MAX);
1445
1446        // elements are equal
1447        assert_eq!(a, b);
1448        assert_eq!(a.value(), b.value());
1449    }
1450
1451    #[macro_rules_attr::apply(test)]
1452    fn test_random_raw() {
1453        let mut rng = rand::rng();
1454        for _ in 0..100 {
1455            let e: BFieldElement = rng.random();
1456            let bytes = e.raw_bytes();
1457            let c = BFieldElement::from_raw_bytes(&bytes);
1458            assert_eq!(e, c);
1459
1460            let mut f = 0u64;
1461            for (i, b) in bytes.iter().enumerate() {
1462                f += (*b as u64) << (8 * i);
1463            }
1464            assert_eq!(e, BFieldElement(f));
1465
1466            let chunks = e.raw_u16s();
1467            let g = BFieldElement::from_raw_u16s(&chunks);
1468            assert_eq!(e, g);
1469
1470            let mut h = 0u64;
1471            for (i, ch) in chunks.iter().enumerate() {
1472                h += (*ch as u64) << (16 * i);
1473            }
1474            assert_eq!(e, BFieldElement(h));
1475        }
1476    }
1477
1478    #[macro_rules_attr::apply(test)]
1479    fn test_fixed_inverse() {
1480        // (8561862112314395584, 17307602810081694772)
1481        let a = BFieldElement::new(8561862112314395584);
1482        let a_inv = a.inverse();
1483        let a_inv_or_0 = a.inverse_or_zero();
1484        let expected = BFieldElement::new(17307602810081694772);
1485        assert_eq!(a_inv, a_inv_or_0);
1486        assert_eq!(a_inv, expected);
1487    }
1488
1489    #[macro_rules_attr::apply(test)]
1490    fn test_fixed_modpow() {
1491        let exponent = 16608971246357572739u64;
1492        let base = BFieldElement::new(7808276826625786800);
1493        let expected = BFieldElement::new(2288673415394035783);
1494        assert_eq!(base.mod_pow_u64(exponent), expected);
1495    }
1496
1497    #[macro_rules_attr::apply(test)]
1498    fn test_fixed_mul() {
1499        {
1500            let a = BFieldElement::new(2779336007265862836);
1501            let b = BFieldElement::new(8146517303801474933);
1502            let c = a * b;
1503            let expected = BFieldElement::new(1857758653037316764);
1504            assert_eq!(c, expected);
1505        }
1506
1507        {
1508            let a = BFieldElement::new(9223372036854775808);
1509            let b = BFieldElement::new(9223372036854775808);
1510            let c = a * b;
1511            let expected = BFieldElement::new(18446744068340842497);
1512            assert_eq!(c, expected);
1513        }
1514    }
1515
1516    #[macro_rules_attr::apply(proptest)]
1517    fn conversion_from_i32_to_bfe_is_correct(v: i32) {
1518        let bfe = BFieldElement::from(v);
1519        match v {
1520            0.. => prop_assert_eq!(u64::try_from(v)?, bfe.value()),
1521            _ => prop_assert_eq!(u64::try_from(-v)?, BFieldElement::P - bfe.value()),
1522        }
1523    }
1524
1525    #[macro_rules_attr::apply(proptest)]
1526    fn conversion_from_isize_to_bfe_is_correct(v: isize) {
1527        let bfe = BFieldElement::from(v);
1528        match v {
1529            0.. => prop_assert_eq!(u64::try_from(v)?, bfe.value()),
1530            _ => prop_assert_eq!(u64::try_from(-v)?, BFieldElement::P - bfe.value()),
1531        }
1532    }
1533
1534    #[macro_rules_attr::apply(test)]
1535    fn bfield_element_can_be_converted_to_and_from_many_types() {
1536        let _ = BFieldElement::from(0_u8);
1537        let _ = BFieldElement::from(0_u16);
1538        let _ = BFieldElement::from(0_u32);
1539        let _ = BFieldElement::from(0_u64);
1540        let _ = BFieldElement::from(0_u128);
1541        let _ = BFieldElement::from(0_usize);
1542
1543        let max = bfe!(BFieldElement::MAX);
1544        assert_eq!(max, BFieldElement::from(-1_i8));
1545        assert_eq!(max, BFieldElement::from(-1_i16));
1546        assert_eq!(max, BFieldElement::from(-1_i32));
1547        assert_eq!(max, BFieldElement::from(-1_i64));
1548        assert_eq!(max, BFieldElement::from(-1_isize));
1549
1550        assert!(u8::try_from(BFieldElement::ZERO).is_ok());
1551        assert!(i8::try_from(BFieldElement::ZERO).is_ok());
1552        assert!(u16::try_from(BFieldElement::ZERO).is_ok());
1553        assert!(i16::try_from(BFieldElement::ZERO).is_ok());
1554        assert!(u32::try_from(BFieldElement::ZERO).is_ok());
1555        assert!(i32::try_from(BFieldElement::ZERO).is_ok());
1556        assert!(usize::try_from(BFieldElement::ZERO).is_ok());
1557        assert!(isize::try_from(BFieldElement::ZERO).is_ok());
1558
1559        let _ = u64::from(max);
1560        let _ = i64::from(max);
1561        let _ = u128::from(max);
1562        let _ = i128::from(max);
1563    }
1564
1565    #[macro_rules_attr::apply(test)]
1566    fn bfield_conversion_works_for_types_min_and_max() {
1567        let _ = BFieldElement::from(u8::MIN);
1568        let _ = BFieldElement::from(u8::MAX);
1569        let _ = BFieldElement::from(u16::MIN);
1570        let _ = BFieldElement::from(u16::MAX);
1571        let _ = BFieldElement::from(u32::MIN);
1572        let _ = BFieldElement::from(u32::MAX);
1573        let _ = BFieldElement::from(u64::MIN);
1574        let _ = BFieldElement::from(u64::MAX);
1575        let _ = BFieldElement::from(u128::MIN);
1576        let _ = BFieldElement::from(u128::MAX);
1577        let _ = BFieldElement::from(usize::MIN);
1578        let _ = BFieldElement::from(usize::MAX);
1579        let _ = BFieldElement::from(i8::MIN);
1580        let _ = BFieldElement::from(i8::MAX);
1581        let _ = BFieldElement::from(i16::MIN);
1582        let _ = BFieldElement::from(i16::MAX);
1583        let _ = BFieldElement::from(i32::MIN);
1584        let _ = BFieldElement::from(i32::MAX);
1585        let _ = BFieldElement::from(i64::MIN);
1586        let _ = BFieldElement::from(i64::MAX);
1587        let _ = BFieldElement::from(isize::MIN);
1588        let _ = BFieldElement::from(isize::MAX);
1589    }
1590
1591    #[macro_rules_attr::apply(proptest)]
1592    fn naive_and_actual_conversion_from_u128_agree(v: u128) {
1593        fn naive_conversion(x: u128) -> BFieldElement {
1594            let p = BFieldElement::P as u128;
1595            let value = (x % p) as u64;
1596            BFieldElement::new(value)
1597        }
1598
1599        prop_assert_eq!(naive_conversion(v), BFieldElement::from(v));
1600    }
1601
1602    #[macro_rules_attr::apply(proptest)]
1603    fn naive_and_actual_conversion_from_i64_agree(v: i64) {
1604        fn naive_conversion(x: i64) -> BFieldElement {
1605            let p = BFieldElement::P as i128;
1606            let value = i128::from(x).rem_euclid(p) as u64;
1607            BFieldElement::new(value)
1608        }
1609
1610        prop_assert_eq!(naive_conversion(v), BFieldElement::from(v));
1611    }
1612
1613    #[macro_rules_attr::apply(test)]
1614    fn bfe_macro_can_be_used() {
1615        let b = bfe!(42);
1616        let _ = bfe!(42u32);
1617        let _ = bfe!(-1);
1618        let _ = bfe!(b);
1619        let _ = bfe!(b.0);
1620        let _ = bfe!(42_usize);
1621        let _ = bfe!(-2_isize);
1622
1623        let c: Vec<BFieldElement> = bfe_vec![1, 2, 3];
1624        let d: [BFieldElement; 3] = bfe_array![1, 2, 3];
1625        assert_eq!(c, d);
1626    }
1627
1628    #[macro_rules_attr::apply(proptest)]
1629    fn bfe_macro_produces_same_result_as_calling_new(value: u64) {
1630        prop_assert_eq!(BFieldElement::new(value), bfe!(value));
1631    }
1632
1633    #[macro_rules_attr::apply(test)]
1634    fn const_minus_two_inverse_is_really_minus_two_inverse() {
1635        assert_eq!(bfe!(-2).inverse(), BFieldElement::MINUS_TWO_INVERSE);
1636    }
1637}