ark_ff/biginteger/
arithmetic.rs

1use ark_std::{vec, vec::*};
2
3macro_rules! adc {
4    ($a:expr, $b:expr, &mut $carry:expr$(,)?) => {{
5        let tmp = ($a as u128) + ($b as u128) + ($carry as u128);
6        $carry = (tmp >> 64) as u64;
7        tmp as u64
8    }};
9}
10
11/// Sets a = a + b + carry, and returns the new carry.
12#[inline(always)]
13#[allow(unused_mut)]
14#[doc(hidden)]
15pub fn adc(a: &mut u64, b: u64, carry: u64) -> u64 {
16    let tmp = *a as u128 + b as u128 + carry as u128;
17    *a = tmp as u64;
18    (tmp >> 64) as u64
19}
20
21/// Sets a = a + b + carry, and returns the new carry.
22#[inline(always)]
23#[allow(unused_mut)]
24#[doc(hidden)]
25pub fn adc_for_add_with_carry(a: &mut u64, b: u64, carry: u8) -> u8 {
26    #[cfg(all(target_arch = "x86_64", feature = "asm"))]
27    #[allow(unsafe_code)]
28    unsafe {
29        use core::arch::x86_64::_addcarry_u64;
30        _addcarry_u64(carry, *a, b, a)
31    }
32    #[cfg(not(all(target_arch = "x86_64", feature = "asm")))]
33    {
34        let tmp = *a as u128 + b as u128 + carry as u128;
35        *a = tmp as u64;
36        (tmp >> 64) as u8
37    }
38}
39
40/// Calculate a + b + carry, returning the sum
41#[inline(always)]
42#[doc(hidden)]
43pub fn adc_no_carry(a: u64, b: u64, carry: &mut u64) -> u64 {
44    let tmp = a as u128 + b as u128 + *carry as u128;
45    tmp as u64
46}
47
48#[macro_export]
49macro_rules! sbb {
50    ($a:expr, $b:expr, &mut $borrow:expr$(,)?) => {{
51        let tmp = (1u128 << 64) + ($a as u128) - ($b as u128) - ($borrow as u128);
52        $borrow = if tmp >> 64 == 0 { 1 } else { 0 };
53        tmp as u64
54    }};
55}
56
57/// Sets a = a - b - borrow, and returns the borrow.
58#[inline(always)]
59#[allow(unused_mut)]
60pub(crate) fn sbb(a: &mut u64, b: u64, borrow: u64) -> u64 {
61    let tmp = (1u128 << 64) + (*a as u128) - (b as u128) - (borrow as u128);
62    *a = tmp as u64;
63    (tmp >> 64 == 0) as u64
64}
65
66/// Sets a = a - b - borrow, and returns the borrow.
67#[inline(always)]
68#[allow(unused_mut)]
69#[doc(hidden)]
70pub fn sbb_for_sub_with_borrow(a: &mut u64, b: u64, borrow: u8) -> u8 {
71    #[cfg(all(target_arch = "x86_64", feature = "asm"))]
72    #[allow(unsafe_code)]
73    unsafe {
74        use core::arch::x86_64::_subborrow_u64;
75        _subborrow_u64(borrow, *a, b, a)
76    }
77    #[cfg(not(all(target_arch = "x86_64", feature = "asm")))]
78    {
79        let tmp = (1u128 << 64) + (*a as u128) - (b as u128) - (borrow as u128);
80        *a = tmp as u64;
81        u8::from(tmp >> 64 == 0)
82    }
83}
84
85#[inline(always)]
86#[doc(hidden)]
87pub const fn widening_mul(a: u64, b: u64) -> u128 {
88    #[cfg(not(target_family = "wasm"))]
89    {
90        a as u128 * b as u128
91    }
92    #[cfg(target_family = "wasm")]
93    {
94        let a_lo = a as u32 as u64;
95        let a_hi = a >> 32;
96        let b_lo = b as u32 as u64;
97        let b_hi = b >> 32;
98
99        let lolo = (a_lo * b_lo) as u128;
100        let lohi = ((a_lo * b_hi) as u128) << 32;
101        let hilo = ((a_hi * b_lo) as u128) << 32;
102        let hihi = ((a_hi * b_hi) as u128) << 64;
103        (lolo | hihi) + (lohi + hilo)
104    }
105}
106
107/// Calculate a + b * c, returning the lower 64 bits of the result and setting
108/// `carry` to the upper 64 bits.
109#[inline(always)]
110#[doc(hidden)]
111pub fn mac(a: u64, b: u64, c: u64, carry: &mut u64) -> u64 {
112    let tmp = (a as u128) + widening_mul(b, c);
113    *carry = (tmp >> 64) as u64;
114    tmp as u64
115}
116
117/// Calculate a + b * c, discarding the lower 64 bits of the result and setting
118/// `carry` to the upper 64 bits.
119#[inline(always)]
120#[doc(hidden)]
121pub fn mac_discard(a: u64, b: u64, c: u64, carry: &mut u64) {
122    let tmp = (a as u128) + widening_mul(b, c);
123    *carry = (tmp >> 64) as u64;
124}
125
126macro_rules! mac_with_carry {
127    ($a:expr, $b:expr, $c:expr, &mut $carry:expr$(,)?) => {{
128        let tmp =
129            ($a as u128) + $crate::biginteger::arithmetic::widening_mul($b, $c) + ($carry as u128);
130        $carry = (tmp >> 64) as u64;
131        tmp as u64
132    }};
133}
134
135macro_rules! mac {
136    ($a:expr, $b:expr, $c:expr, &mut $carry:expr$(,)?) => {{
137        let tmp = ($a as u128) + $crate::biginteger::arithmetic::widening_mul($b, $c);
138        $carry = (tmp >> 64) as u64;
139        tmp as u64
140    }};
141}
142
143/// Calculate a + (b * c) + carry, returning the least significant digit
144/// and setting carry to the most significant digit.
145#[inline(always)]
146#[doc(hidden)]
147pub fn mac_with_carry(a: u64, b: u64, c: u64, carry: &mut u64) -> u64 {
148    let tmp = (a as u128) + widening_mul(b, c) + (*carry as u128);
149    *carry = (tmp >> 64) as u64;
150    tmp as u64
151}
152
153/// Compute the NAF (non-adjacent form) of num
154pub fn find_naf(num: &[u64]) -> Vec<i8> {
155    let is_zero = |num: &[u64]| num.iter().all(|x| *x == 0u64);
156    let is_odd = |num: &[u64]| num[0] & 1 == 1;
157    let sub_noborrow = |num: &mut [u64], z: u64| {
158        let mut other = vec![0u64; num.len()];
159        other[0] = z;
160        let mut borrow = 0;
161
162        for (a, b) in num.iter_mut().zip(other) {
163            borrow = sbb(a, b, borrow);
164        }
165    };
166    let add_nocarry = |num: &mut [u64], z: u64| {
167        let mut other = vec![0u64; num.len()];
168        other[0] = z;
169        let mut carry = 0;
170
171        for (a, b) in num.iter_mut().zip(other) {
172            carry = adc(a, b, carry);
173        }
174    };
175    let div2 = |num: &mut [u64]| {
176        let mut t = 0;
177        for i in num.iter_mut().rev() {
178            let t2 = *i << 63;
179            *i >>= 1;
180            *i |= t;
181            t = t2;
182        }
183    };
184
185    let mut num = num.to_vec();
186    let mut res = vec![];
187
188    while !is_zero(&num) {
189        let z: i8;
190        if is_odd(&num) {
191            z = 2 - (num[0] % 4) as i8;
192            if z >= 0 {
193                sub_noborrow(&mut num, z as u64)
194            } else {
195                add_nocarry(&mut num, (-z) as u64)
196            }
197        } else {
198            z = 0;
199        }
200        res.push(z);
201        div2(&mut num);
202    }
203
204    res
205}
206
207/// We define relaxed NAF as a variant of NAF with a very small tweak.
208///
209/// Note that the cost of scalar multiplication grows with the length of the sequence (for doubling)
210/// plus the Hamming weight of the sequence (for addition, or subtraction).
211///
212/// NAF is optimizing for the Hamming weight only and therefore can be suboptimal.
213/// For example, NAF may generate a sequence (in little-endian) of the form ...0 -1 0 1.
214///
215/// This can be rewritten as ...0 1 1 to avoid one doubling, at the cost that we are making an
216/// exception of non-adjacence for the most significant bit.
217///
218/// Since this representation is no longer a strict NAF, we call it "relaxed NAF".
219pub fn find_relaxed_naf(num: &[u64]) -> Vec<i8> {
220    let mut res = find_naf(num);
221
222    let len = res.len();
223    if res[len - 2] == 0 && res[len - 3] == -1 {
224        res[len - 3] = 1;
225        res[len - 2] = 1;
226        res.resize(len - 1, 0);
227    }
228
229    res
230}
231
232#[test]
233fn test_find_relaxed_naf_usefulness() {
234    let vec = find_naf(&[12u64]);
235    assert_eq!(vec.len(), 5);
236
237    let vec = find_relaxed_naf(&[12u64]);
238    assert_eq!(vec.len(), 4);
239}
240
241#[test]
242fn test_find_relaxed_naf_correctness() {
243    use ark_std::{One, UniformRand, Zero};
244    use num_bigint::BigInt;
245
246    let mut rng = ark_std::test_rng();
247
248    for _ in 0..10 {
249        let num = [
250            u64::rand(&mut rng),
251            u64::rand(&mut rng),
252            u64::rand(&mut rng),
253            u64::rand(&mut rng),
254        ];
255        let relaxed_naf = find_relaxed_naf(&num);
256
257        let test = {
258            let mut sum = BigInt::zero();
259            let mut cur = BigInt::one();
260            for v in relaxed_naf {
261                sum += cur.clone() * v;
262                cur *= 2;
263            }
264            sum
265        };
266
267        let test_expected = {
268            let mut sum = BigInt::zero();
269            let mut cur = BigInt::one();
270            for v in num.iter() {
271                sum += cur.clone() * v;
272                cur <<= 64;
273            }
274            sum
275        };
276
277        assert_eq!(test, test_expected);
278    }
279}