ark_ff/biginteger/
arithmetic.rs1use 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#[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#[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#[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#[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#[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#[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#[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#[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
153pub 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
207pub 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}