-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathbigd.h
359 lines (269 loc) · 11.1 KB
/
bigd.h
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
/* $Id: bigd.h $ */
/******************** SHORT COPYRIGHT NOTICE**************************
This source code is part of the BigDigits multiple-precision
arithmetic library Version 2.2 originally written by David Ireland,
copyright (c) 2001-8 D.I. Management Services Pty Limited, all rights
reserved. It is provided "as is" with no warranties. You may use
this software under the terms of the full copyright notice
"bigdigitsCopyright.txt" that should have been included with this
library or can be obtained from <www.di-mgt.com.au/bigdigits.html>.
This notice must always be retained in any copy.
******************* END OF COPYRIGHT NOTICE***************************/
/*
Last updated:
$Date: 2008-07-31 12:54:00 $
$Revision: 2.2.0 $
$Author: dai $
*/
/*
Interface to the BIGD library (BigDigits "bd" functions)
Multiple-precision arithmetic for non-negative numbers
using memory management and opaque pointers.
*/
#ifndef BIGD_H_
#define BIGD_H_ 1
#include "bigdtypes.h"
/**** USER CONFIGURABLE SECTION ****/
/* [v2.1] Changed to use exact width integer types.
[v2.2] Moved macros for exact-width types to "bigdtypes.h"
*/
/* DIGIT_T type is not exposed by this library so we expose
a synonym `bdigit_t' for a single digit */
typedef uint32_t bdigit_t;
/**** END OF USER CONFIGURABLE SECTION ****/
#define OCTETS_PER_DIGIT (sizeof(bdigit_t))
/* Options for bdPrint */
#define BD_PRINT_NL 0x1 /* append a newline after printing */
#define BD_PRINT_TRIM 0x2 /* trim leading zero digits */
#ifdef NO_ALLOCS
#error NO_ALLOCS is not permitted with bd functions
#endif
/*
This interface uses opaque pointers of type BIGD using
the conventions in "C Interfaces and Implementions" by
David R. Hanson, Addison-Wesley, 1996, pp21-4.
Thanks to Ian Tree for the C++ fudge.
*/
#define T BIGD
#ifdef __cplusplus
typedef struct T2 *T;
#else
typedef struct T *T;
#endif
#ifdef __cplusplus
extern "C" {
#endif
/* PROGRAMMER'S NOTES
Where the function computes a new BIGD value,
the result is returned in the first argument.
Some functions do not allow variables to overlap (noted by w#v).
Functions of type int generally return 0 to denote success
but some return True/False (1/0) or borrow (+1) or error (-1).
Functions of type size_t (an unsigned int) return a length.
Memory is allocated if the function might need it.
It is only released when freed with bdFree(), e.g.:
BIGD b;
b = bdNew();
...
bdFree(&b);
*/
/******************************/
/* CONSTRUCTOR AND DESTRUCTOR */
/******************************/
BIGD bdNew(void);
/* Return handle to new BIGD object */
void bdFree(BIGD *bd);
/* Zeroise and free BIGD memory
- NB pass a pointer to the handle */
/*************************/
/* ARITHMETIC OPERATIONS */
/*************************/
int bdAdd(BIGD w, BIGD u, BIGD v);
/* Compute w = u + v, w#v */
int bdSubtract(BIGD w, BIGD u, BIGD v);
/* Compute w = u - v, return borrow, w#v */
int bdMultiply(BIGD w, BIGD u, BIGD v);
/* Compute w = u * v, w#u */
int bdDivide(BIGD q, BIGD r, BIGD u, BIGD v);
/* Computes quotient q = u / v and remainder r = u mod v
Trashes q and r first. q#(u,v), r#(u,v) */
int bdModulo(BIGD r, BIGD u, BIGD v);
/* Computes r = u mod v, r#u */
int bdSquare(BIGD w, BIGD x);
/* Computes w = x^2, w#x */
int bdSqrt(BIGD s, BIGD x);
/* Computes integer square root s = floor(sqrt(x)) */
int bdIncrement(BIGD a);
/* Sets a = a + 1, returns carry */
int bdDecrement(BIGD a);
/* Sets a = a - 1, returns borrow */
/* 'Safe' versions with no restrictions on overlap */
/* [v2.2] Changed name from ~Ex to ~_s */
int bdAdd_s(BIGD w, BIGD u, BIGD v);
int bdSubtract_s(BIGD w, BIGD u, BIGD v);
int bdMultiply_s(BIGD w, BIGD u, BIGD v);
int bdDivide_s(BIGD q, BIGD r, BIGD u, BIGD v);
int bdModulo_s(BIGD r, BIGD u, BIGD v);
int bdSquare_s(BIGD s, BIGD x);
/* Keep the faith with the deprecated ~Ex version names */
#define bdAddEx(w, u, v) bdAdd_s((w), (u), (v))
#define bdSubtractEx(w, u, v) bdSubtract_s((w), (u), (v))
#define bdMultiplyEx(w, u, v) bdMultiply_s((w), (u), (v))
#define bdDivideEx(q, r, u, v) bdDivide_s((q), (r), (u), (v))
#define bdModuloEx(r, u, v) bdModulo_s((r), (u), (v))
#define bdSquareEx(s, x) bdSquare_s((s), (x))
/*************************/
/* COMPARISON OPERATIONS */
/*************************/
int bdIsEqual(BIGD a, BIGD b);
/* Returns true if a == b, else false */
int bdCompare(BIGD a, BIGD b);
/* Returns sign of (a-b) */
int bdIsZero(BIGD a);
/* Returns true if a == 0, else false */
int bdIsEven(BIGD a);
int bdIsOdd(BIGD a);
/* Return TRUE or FALSE */
/*************************/
/* ASSIGNMENT OPERATIONS */
/*************************/
int bdSetEqual(BIGD a, BIGD b);
/* Sets a = b */
int bdSetZero(BIGD a);
/* Sets a = 0 */
/****************************/
/* NUMBER THEORY OPERATIONS */
/****************************/
int bdModExp(BIGD y, BIGD x, BIGD e, BIGD m);
/* Compute y = x^e mod m */
int bdModMult(BIGD a, BIGD x, BIGD y, BIGD m);
/* Compute a = (x * y) mod m */
int bdModInv(BIGD x, BIGD a, BIGD m);
/* Compute x = a^-1 mod m */
int bdGcd(BIGD g, BIGD x, BIGD y);
/* Compute g = gcd(x, y) */
int bdJacobi(BIGD a, BIGD n);
/* Returns Jacobi(a, n) = {0, +1, -1} */
int bdIsPrime(BIGD b, size_t ntests);
/* Returns true if passes ntests x Miller-Rabin tests with trial division */
int bdRabinMiller(BIGD b, size_t ntests);
/* Returns true if passes ntests x Miller-Rabin tests without trial division, b > 2 */
/**********************************************/
/* FUNCTIONS THAT OPERATE WITH A SINGLE DIGIT */
/**********************************************/
int bdSetShort(BIGD b, bdigit_t d);
/* Converts single digit d into a BIGD b */
int bdShortAdd(BIGD w, BIGD u, bdigit_t d);
/* Compute w = u + d */
int bdShortSub(BIGD w, BIGD u, bdigit_t d);
/* Compute w = u - d, return borrow */
int bdShortMult(BIGD w, BIGD u, bdigit_t d);
/* Compute w = u * d */
int bdShortDiv(BIGD q, BIGD r, BIGD u, bdigit_t d);
/* Computes quotient q = u / d and remainder r = u mod d */
bdigit_t bdShortMod(BIGD r, BIGD u, bdigit_t d);
/* Computes r = u mod d. Returns r. */
int bdShortCmp(BIGD a, bdigit_t d);
/* Returns sign of (a-d) */
/***********************/
/* BIT-WISE OPERATIONS */
/***********************/
/* [v2.1.0] Added ModPowerOf2, Xor, Or and AndBits functions */
/* [v2.2.0] Added NotBits, GetBit, SetBit functions */
size_t bdBitLength(BIGD bd);
/* Returns number of significant bits in bd */
int bdSetBit(BIGD a, size_t n, int value);
/* Set bit n (0..nbits-1) with value 1 or 0 */
int bdGetBit(BIGD a, size_t ibit);
/* Returns value 1 or 0 of bit n (0..nbits-1) */
/* [v2.1.0] Removed restriction on shift size */
void bdShiftLeft(BIGD a, BIGD b, size_t n);
/* Computes a = b << n */
void bdShiftRight(BIGD a, BIGD b, size_t n);
/* Computes a = b >> n */
void bdModPowerOf2(BIGD a, size_t L);
/* Computes a = a mod 2^L */
void bdXorBits(BIGD a, BIGD b, BIGD c);
/* Computes bitwise operation a = b XOR c */
void bdOrBits(BIGD a, BIGD b, BIGD c);
/* Computes bitwise operation a = b OR c */
void bdAndBits(BIGD a, BIGD b, BIGD c);
/* Computes bitwise operation a = b AND c */
void bdNotBits(BIGD a, BIGD b);
/* Computes bitwise a = NOT b */
/*******************/
/* MISC OPERATIONS */
/*******************/
size_t bdSizeof(BIGD b);
/* Returns number of significant digits in b */
void bdPrint(BIGD bd, size_t flags);
/* Print bd to stdout (see flags above) */
/***************/
/* CONVERSIONS */
/***************/
/* [v2.1] Tightened up specs here */
/* All bdConv functions return 0 on error */
/* `Octet' = an 8-bit byte */
size_t bdConvFromOctets(BIGD b, const unsigned char *octets, size_t nbytes);
/* Converts nbytes octets into big digit b, resizing b if necessary.
Returns actual number of digits set, which may be larger than mpSizeof(b). */
size_t bdConvToOctets(BIGD b, unsigned char *octets, size_t nbytes);
/* Convert big digit b into array of octets, in big-endian order,
padding on the left with zeroes to nbytes, or truncating as necessary.
It returns the exact number of significant bytes <nsig> in the result.
If octets is NULL or nbytes == 0 then just return required <nsig>;
if nbytes== nsig, octets <- nnnnnn;
if nbytes > nsig, octets <- 000nnnnnn;
if nbytes < nsig, octets <- nnn. */
/* For Hex and Decimal conversions:
The parameter `smax' in the bdConvFrom functions
is the size INCLUDING the terminating zero (as in sizeof(s))
but the bdConvTo functions return the number of characters required
EXCLUDING the terminating zero.
The bdConvToHex and bdConvToDecimal functions return the
total length of the string they tried to create even if
the output string was truncated at a shorter length,
so remember to check!.
*/
size_t bdConvFromHex(BIGD b, const char *s);
/* Convert string s of hex chars into big digit b.
Returns number of significant digits actually set, which could be 0. */
size_t bdConvToHex(BIGD b, char *s, size_t smax);
/* Convert big digit b into string s of hex characters
where s has size smax including the terminating zero.
Returns number of chars required for s excluding leading zeroes.
If s is NULL or smax == 0 then just return required size. */
size_t bdConvFromDecimal(BIGD b, const char *s);
/* Convert string s of decimal chars into big digit b.
Returns number of significant digits actually set, which could be 0. */
size_t bdConvToDecimal(BIGD b, char *s, size_t smax);
/* Convert big digit b into string s of decimal characters
where s has size smax including the terminating zero.
Returns number of chars required for s excluding leading zeroes.
If s is NULL or smax == 0 then just return required size. */
/****************************/
/* RANDOM NUMBER OPERATIONS */
/****************************/
/* [Version 2.1: bdRandDigit and bdRandomBits moved to bigdRand.h] */
size_t bdSetRandTest(BIGD a, size_t ndigits);
/* Fill a with [1,ndigits] pseudo-random digits. Returns # digits set.
Randomly clears a random number of bits in the most significant digit.
For testing only. Not crypto secure. Not thread safe. */
/* TYPEDEF for user-defined random byte generator function */
typedef int (* BD_RANDFUNC)(unsigned char *buf, size_t nbytes, const unsigned char *seed, size_t seedlen);
int bdRandomSeeded(BIGD a, size_t nbits, const unsigned char *seed,
size_t seedlen, BD_RANDFUNC RandFunc);
/* Generate a random number nbits long using RandFunc */
int bdGeneratePrime(BIGD b, size_t nbits, size_t ntests, const unsigned char *seed,
size_t seedlen, BD_RANDFUNC RandFunc);
/* Generate a prime number nbits long; carry out ntests x primality tests */
/****************/
/* VERSION INFO */
/****************/
int bdVersion(void);
/* Returns version number = major*1000+minor*100+release*10+uses_asm(0|1)+uses_64(0|2)+uses_noalloc(0|5) */
#undef T /* (for opaque BIGD pointer) */
#ifdef __cplusplus
}
#endif
#endif /* BIGD_H_ */