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#[derive(Copy, Clone, Default, Hash, PartialEq, Eq, GetSize)]
85#[repr(transparent)]
86pub struct BFieldElement(u64);
87
88#[macro_export]
103macro_rules! bfe {
104 ($value:expr) => {
105 BFieldElement::from($value)
106 };
107}
108
109#[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#[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 pub const P: u64 = 0xffff_ffff_0000_0001;
226 pub const MAX: u64 = Self::P - 1;
227
228 const R2: u64 = 0xffff_fffe_0000_0001;
230
231 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 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 #[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 pub const fn generator() -> Self {
313 BFieldElement::new(7)
314 }
315
316 #[inline]
319 pub const fn lift(&self) -> XFieldElement {
320 XFieldElement::new_const(*self)
321 }
322
323 pub fn increment(&mut self) {
325 *self += Self::ONE;
326 }
327
328 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 #[inline(always)]
357 pub const fn montyred(x: u128) -> u64 {
358 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 r.wrapping_sub((1 + !Self::P) * c as u64)
370 }
371
372 pub const fn raw_bytes(&self) -> [u8; 8] {
375 self.0.to_le_bytes()
376 }
377
378 pub const fn from_raw_bytes(bytes: &[u8; 8]) -> Self {
382 Self(u64::from_le_bytes(*bytes))
383 }
384
385 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 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 #[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 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 let tmp2 = (x_hi_lo << 32) - x_hi_lo;
489
490 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 #[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
613impl From<BFieldElement> for [u8; BFieldElement::BYTES] {
616 fn from(bfe: BFieldElement) -> Self {
617 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 let (x1, c1) = self.0.overflowing_sub(Self::P - rhs.0);
719
720 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 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
806impl 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 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 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 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 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 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 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 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 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 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 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 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 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}