mirror of
https://github.com/radareorg/radare2.git
synced 2024-11-27 15:10:53 +00:00
691 lines
15 KiB
C
691 lines
15 KiB
C
/* radare2 - LGPL - Copyright 2010 - FXTi */
|
|
/* Based on https://github.com/kokke/tiny-bignum-c */
|
|
|
|
#include <r_util.h>
|
|
|
|
#if !HAVE_LIB_GMP && !HAVE_LIB_SSL
|
|
/* Functions for shifting number in-place. */
|
|
static void _lshift_one_bit(RNumBig *a);
|
|
static void _rshift_one_bit(RNumBig *a);
|
|
static void _lshift_word(RNumBig *a, int nwords);
|
|
static void _rshift_word(RNumBig *a, int nwords);
|
|
static void _r_big_zero_out(RNumBig *n);
|
|
|
|
R_API RNumBig *r_big_new(void) {
|
|
RNumBig *n = R_NEW (RNumBig);
|
|
if (n) {
|
|
_r_big_zero_out (n);
|
|
}
|
|
return n;
|
|
}
|
|
|
|
R_API void r_big_free(RNumBig *b) {
|
|
free (b);
|
|
}
|
|
|
|
R_API void r_big_init(RNumBig *b) {
|
|
_r_big_zero_out (b);
|
|
}
|
|
|
|
R_API void r_big_fini(RNumBig *b) {
|
|
_r_big_zero_out (b);
|
|
}
|
|
|
|
R_API void r_big_from_int(RNumBig *b, st64 n) {
|
|
R_RETURN_IF_FAIL (b);
|
|
|
|
_r_big_zero_out (b);
|
|
b->sign = (n < 0)? -1: 1;
|
|
R_BIG_DTYPE_TMP v = n * b->sign;
|
|
|
|
/* Endianness issue if machine is not little-endian? */
|
|
#ifdef R_BIG_WORD_SIZE
|
|
#if (R_BIG_WORD_SIZE == 1)
|
|
b->array[0] = (v & 0x000000ff);
|
|
b->array[1] = (v & 0x0000ff00) >> 8;
|
|
b->array[2] = (v & 0x00ff0000) >> 16;
|
|
b->array[3] = (v & 0xff000000) >> 24;
|
|
#elif (R_BIG_WORD_SIZE == 2)
|
|
b->array[0] = (v & 0x0000ffff);
|
|
b->array[1] = (v & 0xffff0000) >> 16;
|
|
#elif (R_BIG_WORD_SIZE == 4)
|
|
b->array[0] = v;
|
|
R_BIG_DTYPE_TMP num_32 = 32;
|
|
R_BIG_DTYPE_TMP tmp = v >> num_32;
|
|
b->array[1] = tmp;
|
|
#endif
|
|
#endif
|
|
}
|
|
|
|
static void r_big_from_unsigned(RNumBig *b, ut64 v) {
|
|
R_RETURN_IF_FAIL (b);
|
|
|
|
_r_big_zero_out (b);
|
|
|
|
/* Endianness issue if machine is not little-endian? */
|
|
#ifdef R_BIG_WORD_SIZE
|
|
#if (R_BIG_WORD_SIZE == 1)
|
|
b->array[0] = (v & 0x000000ff);
|
|
b->array[1] = (v & 0x0000ff00) >> 8;
|
|
b->array[2] = (v & 0x00ff0000) >> 16;
|
|
b->array[3] = (v & 0xff000000) >> 24;
|
|
#elif (R_BIG_WORD_SIZE == 2)
|
|
b->array[0] = (v & 0x0000ffff);
|
|
b->array[1] = (v & 0xffff0000) >> 16;
|
|
#elif (R_BIG_WORD_SIZE == 4)
|
|
b->array[0] = v;
|
|
R_BIG_DTYPE_TMP num_32 = 32;
|
|
R_BIG_DTYPE_TMP tmp = v >> num_32;
|
|
b->array[1] = tmp;
|
|
#endif
|
|
#endif
|
|
}
|
|
|
|
R_API st64 r_big_to_int(RNumBig *b) {
|
|
R_RETURN_VAL_IF_FAIL (b, 0);
|
|
|
|
R_BIG_DTYPE_TMP ret = 0;
|
|
|
|
/* Endianness issue if machine is not little-endian? */
|
|
#if (R_BIG_WORD_SIZE == 1)
|
|
ret += b->array[0];
|
|
ret += b->array[1] << 8;
|
|
ret += b->array[2] << 16;
|
|
ret += b->array[3] << 24;
|
|
#elif (R_BIG_WORD_SIZE == 2)
|
|
ret += b->array[0];
|
|
ret += b->array[1] << 16;
|
|
#elif (R_BIG_WORD_SIZE == 4)
|
|
ret += b->array[1];
|
|
ret <<= 32;
|
|
ret += b->array[0];
|
|
#endif
|
|
if (b->sign < 0) {
|
|
return -(st64)ret;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
R_API void r_big_from_hexstr(RNumBig *n, const char *str) {
|
|
R_RETURN_IF_FAIL (n);
|
|
R_RETURN_IF_FAIL (str);
|
|
int nbytes = strlen (str);
|
|
|
|
_r_big_zero_out (n);
|
|
|
|
if (str[0] == '-') {
|
|
n->sign = -1;
|
|
str += 1;
|
|
nbytes -= 1;
|
|
}
|
|
|
|
if (str[0] == '0' && str[1] == 'x') {
|
|
str += 2;
|
|
nbytes -= 2;
|
|
}
|
|
R_RETURN_IF_FAIL (nbytes > 0);
|
|
|
|
R_BIG_DTYPE tmp;
|
|
int i = nbytes - (2 * R_BIG_WORD_SIZE); /* index into string */
|
|
int j = 0; /* index into array */
|
|
|
|
while (i >= 0) {
|
|
tmp = 0;
|
|
sscanf (&str[i], R_BIG_SSCANF_FORMAT_STR, &tmp);
|
|
n->array[j] = tmp;
|
|
i -= (2 * R_BIG_WORD_SIZE); /* step R_BIG_WORD_SIZE hex-byte(s) back in the string. */
|
|
j += 1; /* step one element forward in the array. */
|
|
}
|
|
|
|
if (-2 * R_BIG_WORD_SIZE < i) {
|
|
char buffer[2 * R_BIG_WORD_SIZE];
|
|
memset (buffer, 0, sizeof (buffer));
|
|
i += 2 * R_BIG_WORD_SIZE - 1;
|
|
for (; i >= 0; i--) {
|
|
buffer[i] = str[i];
|
|
}
|
|
tmp = 0;
|
|
sscanf (buffer, R_BIG_SSCANF_FORMAT_STR, &tmp);
|
|
n->array[j] = tmp;
|
|
}
|
|
}
|
|
|
|
R_API char *r_big_to_hexstr(RNumBig *b) {
|
|
R_RETURN_VAL_IF_FAIL (b, NULL);
|
|
|
|
int j = R_BIG_ARRAY_SIZE - 1; /* index into array - reading "MSB" first -> big-endian */
|
|
size_t i = 0; /* index into string representation. */
|
|
size_t k = 0; /* Leading zero's amount */
|
|
size_t z, last_z = 2 * R_BIG_WORD_SIZE;
|
|
|
|
for (; b->array[j] == 0 && j >= 0; j--) {
|
|
}
|
|
if (j == -1) {
|
|
return "0x0";
|
|
}
|
|
|
|
size_t size = 3 + 2 * R_BIG_WORD_SIZE * (j + 1) + ((b->sign > 0)? 0: 1);
|
|
char *ret_str = calloc (size, sizeof (char));
|
|
if (!ret_str) {
|
|
return NULL;
|
|
}
|
|
|
|
if (b->sign < 0) {
|
|
ret_str[i++] = '-';
|
|
}
|
|
ret_str[i++] = '0';
|
|
ret_str[i++] = 'x';
|
|
|
|
r_snprintf (ret_str + i, R_BIG_FORMAT_STR_LEN, R_BIG_SPRINTF_FORMAT_STR, b->array[j--]);
|
|
for (; ret_str[i + k] == '0' && k < 2 * R_BIG_WORD_SIZE; k++) {
|
|
}
|
|
for (z = k; ret_str[i + z] && z < last_z; z++) {
|
|
ret_str[i + z - k] = ret_str[i + z];
|
|
}
|
|
i += z - k;
|
|
ret_str[i] = '\x00'; // Truncate string for case(j < 0)
|
|
|
|
for (; j >= 0; j--) {
|
|
r_snprintf (ret_str + i, R_BIG_FORMAT_STR_LEN, R_BIG_SPRINTF_FORMAT_STR, b->array[j]);
|
|
i += 2 * R_BIG_WORD_SIZE;
|
|
}
|
|
|
|
return ret_str;
|
|
}
|
|
|
|
R_API void r_big_assign(RNumBig *dst, RNumBig *src) {
|
|
R_RETURN_IF_FAIL (dst);
|
|
R_RETURN_IF_FAIL (src);
|
|
|
|
memcpy (dst, src, sizeof (RNumBig));
|
|
}
|
|
|
|
static void r_big_add_inner(RNumBig *c, RNumBig *a, RNumBig *b) {
|
|
R_BIG_DTYPE_TMP tmp;
|
|
int carry = 0;
|
|
int i;
|
|
for (i = 0; i < R_BIG_ARRAY_SIZE; i++) {
|
|
tmp = (R_BIG_DTYPE_TMP)a->array[i] + b->array[i] + carry;
|
|
carry = (tmp > R_BIG_MAX_VAL);
|
|
c->array[i] = (tmp & R_BIG_MAX_VAL);
|
|
}
|
|
}
|
|
|
|
static void r_big_sub_inner(RNumBig *c, RNumBig *a, RNumBig *b) {
|
|
R_BIG_DTYPE_TMP res;
|
|
RNumBig *tmp;
|
|
R_BIG_DTYPE_TMP tmp1;
|
|
R_BIG_DTYPE_TMP tmp2;
|
|
int borrow = 0;
|
|
int sign = r_big_cmp (a, b);
|
|
c->sign = (sign >= 0? 1: -1);
|
|
if (sign < 0) {
|
|
tmp = a;
|
|
a = b;
|
|
b = tmp;
|
|
}
|
|
int i;
|
|
for (i = 0; i < R_BIG_ARRAY_SIZE; i++) {
|
|
tmp1 = (R_BIG_DTYPE_TMP)a->array[i] + (R_BIG_MAX_VAL + 1); /* + number_base */
|
|
tmp2 = (R_BIG_DTYPE_TMP)b->array[i] + borrow;
|
|
|
|
res = (tmp1 - tmp2);
|
|
c->array[i] = (R_BIG_DTYPE) (res & R_BIG_MAX_VAL); /* "modulo number_base" == "% (number_base - 1)" if nu mber_base is 2^N */
|
|
borrow = (res <= R_BIG_MAX_VAL);
|
|
}
|
|
}
|
|
|
|
R_API void r_big_add(RNumBig *c, RNumBig *a, RNumBig *b) {
|
|
R_RETURN_IF_FAIL (a);
|
|
R_RETURN_IF_FAIL (b);
|
|
R_RETURN_IF_FAIL (c);
|
|
|
|
if (a->sign >= 0 && b->sign >= 0) {
|
|
r_big_add_inner (c, a, b);
|
|
c->sign = 1;
|
|
return;
|
|
}
|
|
if (a->sign >= 0 && b->sign < 0) {
|
|
r_big_sub_inner (c, a, b);
|
|
return;
|
|
}
|
|
if (a->sign < 0 && b->sign >= 0) {
|
|
r_big_sub_inner (c, b, a);
|
|
return;
|
|
}
|
|
if (a->sign < 0 && b->sign < 0) {
|
|
r_big_add_inner (c, a, b);
|
|
c->sign = -1;
|
|
return;
|
|
}
|
|
}
|
|
|
|
R_API void r_big_sub(RNumBig *c, RNumBig *a, RNumBig *b) {
|
|
R_RETURN_IF_FAIL (a);
|
|
R_RETURN_IF_FAIL (b);
|
|
R_RETURN_IF_FAIL (c);
|
|
|
|
if (a->sign >= 0 && b->sign >= 0) {
|
|
r_big_sub_inner (c, a, b);
|
|
return;
|
|
}
|
|
if (a->sign >= 0 && b->sign < 0) {
|
|
r_big_add_inner (c, a, b);
|
|
c->sign = 1;
|
|
return;
|
|
}
|
|
if (a->sign < 0 && b->sign >= 0) {
|
|
r_big_add_inner (c, a, b);
|
|
c->sign = -1;
|
|
return;
|
|
}
|
|
if (a->sign < 0 && b->sign < 0) {
|
|
r_big_sub_inner (c, b, a);
|
|
return;
|
|
}
|
|
}
|
|
|
|
R_API void r_big_mul(RNumBig *c, RNumBig *a, RNumBig *b) {
|
|
R_RETURN_IF_FAIL (a);
|
|
R_RETURN_IF_FAIL (b);
|
|
R_RETURN_IF_FAIL (c);
|
|
|
|
RNumBig *row = r_big_new ();
|
|
RNumBig *tmp = r_big_new ();
|
|
RNumBig *res = r_big_new ();
|
|
int i, j;
|
|
|
|
for (i = 0; i < R_BIG_ARRAY_SIZE; i++) {
|
|
_r_big_zero_out (row);
|
|
|
|
for (j = 0; j < R_BIG_ARRAY_SIZE; j++) {
|
|
if (i + j < R_BIG_ARRAY_SIZE) {
|
|
_r_big_zero_out (tmp);
|
|
R_BIG_DTYPE_TMP intermediate = ((R_BIG_DTYPE_TMP)a->array[i] * (R_BIG_DTYPE_TMP)b->array[j]);
|
|
r_big_from_unsigned (tmp, intermediate);
|
|
_lshift_word (tmp, i + j);
|
|
r_big_add (row, row, tmp);
|
|
}
|
|
}
|
|
r_big_add (res, row, res);
|
|
}
|
|
|
|
res->sign = a->sign * b->sign;
|
|
if (r_big_is_zero (res)) {
|
|
res->sign = 1; // For -1 * 0 case
|
|
}
|
|
r_big_assign (c, res);
|
|
|
|
r_big_free (row);
|
|
r_big_free (tmp);
|
|
r_big_free (res);
|
|
}
|
|
|
|
R_API void r_big_div(RNumBig *c, RNumBig *a, RNumBig *b) {
|
|
R_RETURN_IF_FAIL (a);
|
|
R_RETURN_IF_FAIL (b);
|
|
R_RETURN_IF_FAIL (c);
|
|
R_RETURN_IF_FAIL (!r_big_is_zero (b));
|
|
|
|
RNumBig *current = r_big_new ();
|
|
RNumBig *denom = r_big_new ();
|
|
;
|
|
RNumBig *tmp = r_big_new ();
|
|
int sign = a->sign * b->sign;
|
|
|
|
r_big_from_int (current, 1); // int current = 1;
|
|
r_big_assign (denom, b); // denom = b
|
|
denom->sign = 1;
|
|
r_big_assign (tmp, denom); // tmp = denom = b
|
|
_lshift_one_bit (tmp); // tmp <= 1
|
|
|
|
while (r_big_cmp (tmp, a) != 1) { // while (tmp <= a)
|
|
if ((denom->array[R_BIG_ARRAY_SIZE - 1] >> (R_BIG_WORD_SIZE * 8 - 1)) == 1) {
|
|
break; // Reach the max value
|
|
}
|
|
_lshift_one_bit (tmp); // tmp <= 1
|
|
_lshift_one_bit (denom); // denom <= 1
|
|
_lshift_one_bit (current); // current <= 1
|
|
}
|
|
|
|
r_big_assign (tmp, a); // tmp = a
|
|
tmp->sign = 1;
|
|
_r_big_zero_out (c); // int answer = 0;
|
|
|
|
while (!r_big_is_zero (current)) // while (current != 0)
|
|
{
|
|
if (r_big_cmp (tmp, denom) != -1) // if (dividend >= denom)
|
|
{
|
|
r_big_sub (tmp, tmp, denom); // dividend -= denom;
|
|
r_big_or (c, current, c); // answer |= current;
|
|
}
|
|
_rshift_one_bit (current); // current >>= 1;
|
|
_rshift_one_bit (denom); // denom >>= 1;
|
|
} // return answer;
|
|
|
|
c->sign = sign;
|
|
if (r_big_is_zero (c)) {
|
|
c->sign = 1; // For -1 * 0 case
|
|
}
|
|
r_big_free (current);
|
|
r_big_free (denom);
|
|
r_big_free (tmp);
|
|
}
|
|
|
|
R_API void r_big_mod(RNumBig *c, RNumBig *a, RNumBig *b) {
|
|
/*
|
|
Take divmod and throw away div part
|
|
*/
|
|
R_RETURN_IF_FAIL (a);
|
|
R_RETURN_IF_FAIL (b);
|
|
R_RETURN_IF_FAIL (c);
|
|
R_RETURN_IF_FAIL (!r_big_is_zero (b));
|
|
|
|
RNumBig *tmp = r_big_new ();
|
|
|
|
r_big_divmod (tmp, c, a, b);
|
|
|
|
r_big_free (tmp);
|
|
}
|
|
|
|
R_API void r_big_divmod(RNumBig *c, RNumBig *d, RNumBig *a, RNumBig *b) {
|
|
/*
|
|
Puts a%b in d
|
|
and a/b in c
|
|
|
|
mod(a,b) = a - ((a / b) * b)
|
|
|
|
example:
|
|
mod(8, 3) = 8 - ((8 / 3) * 3) = 2
|
|
*/
|
|
R_RETURN_IF_FAIL (a);
|
|
R_RETURN_IF_FAIL (b);
|
|
R_RETURN_IF_FAIL (c);
|
|
R_RETURN_IF_FAIL (!r_big_is_zero (b));
|
|
|
|
RNumBig *tmp = r_big_new ();
|
|
|
|
/* c = (a / b) */
|
|
r_big_div (c, a, b);
|
|
|
|
/* tmp = (c * b) */
|
|
r_big_mul (tmp, c, b);
|
|
|
|
/* d = a - tmp */
|
|
r_big_sub (d, a, tmp);
|
|
|
|
r_big_free (tmp);
|
|
}
|
|
|
|
R_API void r_big_and(RNumBig *c, RNumBig *a, RNumBig *b) {
|
|
R_RETURN_IF_FAIL (a);
|
|
R_RETURN_IF_FAIL (b);
|
|
R_RETURN_IF_FAIL (c);
|
|
R_RETURN_IF_FAIL (a->sign > 0);
|
|
R_RETURN_IF_FAIL (b->sign > 0);
|
|
|
|
int i;
|
|
for (i = 0; i < R_BIG_ARRAY_SIZE; i++) {
|
|
c->array[i] = (a->array[i] & b->array[i]);
|
|
}
|
|
}
|
|
|
|
R_API void r_big_or(RNumBig *c, RNumBig *a, RNumBig *b) {
|
|
R_RETURN_IF_FAIL (a);
|
|
R_RETURN_IF_FAIL (b);
|
|
R_RETURN_IF_FAIL (c);
|
|
R_RETURN_IF_FAIL (a->sign > 0);
|
|
R_RETURN_IF_FAIL (b->sign > 0);
|
|
|
|
int i;
|
|
for (i = 0; i < R_BIG_ARRAY_SIZE; i++) {
|
|
c->array[i] = (a->array[i] | b->array[i]);
|
|
}
|
|
}
|
|
|
|
R_API void r_big_xor(RNumBig *c, RNumBig *a, RNumBig *b) {
|
|
R_RETURN_IF_FAIL (a);
|
|
R_RETURN_IF_FAIL (b);
|
|
R_RETURN_IF_FAIL (c);
|
|
R_RETURN_IF_FAIL (a->sign > 0);
|
|
R_RETURN_IF_FAIL (b->sign > 0);
|
|
|
|
int i;
|
|
for (i = 0; i < R_BIG_ARRAY_SIZE; i++) {
|
|
c->array[i] = (a->array[i] ^ b->array[i]);
|
|
}
|
|
}
|
|
|
|
R_API void r_big_lshift(RNumBig *b, RNumBig *a, size_t nbits) {
|
|
R_RETURN_IF_FAIL (a);
|
|
R_RETURN_IF_FAIL (b);
|
|
R_RETURN_IF_FAIL (a->sign > 0);
|
|
R_RETURN_IF_FAIL (b->sign > 0);
|
|
|
|
r_big_assign (b, a);
|
|
/* Handle shift in multiples of word-size */
|
|
const int nbits_pr_word = (R_BIG_WORD_SIZE * 8);
|
|
int nwords = nbits / nbits_pr_word;
|
|
if (nwords != 0) {
|
|
_lshift_word (b, nwords);
|
|
nbits -= (nwords * nbits_pr_word);
|
|
}
|
|
|
|
if (nbits != 0) {
|
|
int i;
|
|
for (i = (R_BIG_ARRAY_SIZE - 1); i > 0; i--) {
|
|
b->array[i] = (b->array[i] << nbits) | (b->array[i - 1] >> ((8 * R_BIG_WORD_SIZE) - nbits));
|
|
}
|
|
b->array[i] <<= nbits;
|
|
}
|
|
}
|
|
|
|
R_API void r_big_rshift(RNumBig *b, RNumBig *a, size_t nbits) {
|
|
R_RETURN_IF_FAIL (a);
|
|
R_RETURN_IF_FAIL (b);
|
|
R_RETURN_IF_FAIL (a->sign > 0);
|
|
R_RETURN_IF_FAIL (b->sign > 0);
|
|
|
|
r_big_assign (b, a);
|
|
/* Handle shift in multiples of word-size */
|
|
const int nbits_pr_word = (R_BIG_WORD_SIZE * 8);
|
|
int nwords = nbits / nbits_pr_word;
|
|
if (nwords != 0) {
|
|
_rshift_word (b, nwords);
|
|
nbits -= (nwords * nbits_pr_word);
|
|
}
|
|
|
|
if (nbits != 0) {
|
|
int i;
|
|
for (i = 0; i < (R_BIG_ARRAY_SIZE - 1); i++) {
|
|
b->array[i] = (b->array[i] >> nbits) | (b->array[i + 1] << ((8 * R_BIG_WORD_SIZE) - nbits));
|
|
}
|
|
b->array[i] >>= nbits;
|
|
}
|
|
}
|
|
|
|
R_API int r_big_cmp(RNumBig *a, RNumBig *b) {
|
|
R_RETURN_VAL_IF_FAIL (a, 0);
|
|
R_RETURN_VAL_IF_FAIL (b, 0);
|
|
|
|
if (a->sign != b->sign)
|
|
return a->sign > 0? 1: -1;
|
|
|
|
int i = R_BIG_ARRAY_SIZE;
|
|
do {
|
|
i -= 1; /* Decrement first, to start with last array element */
|
|
if (a->array[i] > b->array[i]) {
|
|
return 1 * a->sign;
|
|
}
|
|
if (a->array[i] < b->array[i]) {
|
|
return -1 * a->sign;
|
|
}
|
|
} while (i != 0);
|
|
|
|
return 0;
|
|
}
|
|
|
|
R_API int r_big_is_zero(RNumBig *a) {
|
|
R_RETURN_VAL_IF_FAIL (a, -1);
|
|
|
|
int i;
|
|
for (i = 0; i < R_BIG_ARRAY_SIZE; i++) {
|
|
if (a->array[i]) {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
return 1;
|
|
}
|
|
|
|
R_API void r_big_inc(RNumBig *a) {
|
|
R_RETURN_IF_FAIL (a);
|
|
RNumBig *tmp = r_big_new ();
|
|
|
|
r_big_from_int (tmp, 1);
|
|
r_big_add (a, a, tmp);
|
|
|
|
r_big_free (tmp);
|
|
}
|
|
|
|
R_API void r_big_dec(RNumBig *a) {
|
|
R_RETURN_IF_FAIL (a);
|
|
RNumBig *tmp = r_big_new ();
|
|
|
|
r_big_from_int (tmp, 1);
|
|
r_big_sub (a, a, tmp);
|
|
|
|
r_big_free (tmp);
|
|
}
|
|
|
|
R_API void r_big_powm(RNumBig *c, RNumBig *a, RNumBig *b, RNumBig *m) {
|
|
R_RETURN_IF_FAIL (a);
|
|
R_RETURN_IF_FAIL (b);
|
|
R_RETURN_IF_FAIL (c);
|
|
R_RETURN_IF_FAIL (m);
|
|
|
|
RNumBig *bcopy = r_big_new ();
|
|
RNumBig *acopy = r_big_new ();
|
|
|
|
r_big_assign (bcopy, b);
|
|
r_big_assign (acopy, a);
|
|
r_big_mod (acopy, acopy, m);
|
|
r_big_from_int (c, 1);
|
|
|
|
while (!r_big_is_zero (bcopy)) {
|
|
if (r_big_to_int (bcopy) % 2 == 1) {
|
|
r_big_mul (c, c, acopy);
|
|
r_big_mod (c, c, m);
|
|
}
|
|
_rshift_one_bit (bcopy);
|
|
r_big_mul (acopy, acopy, acopy);
|
|
r_big_mod (acopy, acopy, m);
|
|
}
|
|
|
|
r_big_free (bcopy);
|
|
r_big_free (acopy);
|
|
}
|
|
|
|
R_API void r_big_isqrt(RNumBig *b, RNumBig *a) {
|
|
R_RETURN_IF_FAIL (a);
|
|
R_RETURN_IF_FAIL (b);
|
|
|
|
RNumBig *tmp = r_big_new ();
|
|
RNumBig *low = r_big_new ();
|
|
RNumBig *high = r_big_new ();
|
|
RNumBig *mid = r_big_new ();
|
|
|
|
r_big_assign (high, a);
|
|
r_big_rshift (mid, high, 1);
|
|
r_big_inc (mid);
|
|
|
|
while (r_big_cmp (high, low) > 0) {
|
|
r_big_mul (tmp, mid, mid);
|
|
if (r_big_cmp (tmp, a) > 0) {
|
|
r_big_assign (high, mid);
|
|
r_big_dec (high);
|
|
} else {
|
|
r_big_assign (low, mid);
|
|
}
|
|
r_big_sub (mid, high, low);
|
|
_rshift_one_bit (mid);
|
|
r_big_add (mid, mid, low);
|
|
r_big_inc (mid);
|
|
}
|
|
r_big_assign (b, low);
|
|
|
|
r_big_free (tmp);
|
|
r_big_free (low);
|
|
r_big_free (high);
|
|
r_big_free (mid);
|
|
}
|
|
|
|
/* Private / Static functions. */
|
|
static void _rshift_word(RNumBig *a, int nwords) {
|
|
/* Naive method: */
|
|
R_RETURN_IF_FAIL (a);
|
|
R_RETURN_IF_FAIL (nwords >= 0);
|
|
|
|
size_t i;
|
|
if (nwords >= R_BIG_ARRAY_SIZE) {
|
|
for (i = 0; i < R_BIG_ARRAY_SIZE; i++) {
|
|
a->array[i] = 0;
|
|
}
|
|
return;
|
|
}
|
|
|
|
for (i = 0; i < R_BIG_ARRAY_SIZE - nwords; i++) {
|
|
a->array[i] = a->array[i + nwords];
|
|
}
|
|
for (; i < R_BIG_ARRAY_SIZE; i++) {
|
|
a->array[i] = 0;
|
|
}
|
|
}
|
|
|
|
static void _lshift_word(RNumBig *a, int nwords) {
|
|
R_RETURN_IF_FAIL (a);
|
|
R_RETURN_IF_FAIL (nwords >= 0);
|
|
|
|
int i;
|
|
/* Shift whole words */
|
|
for (i = (R_BIG_ARRAY_SIZE - 1); i >= nwords; i--) {
|
|
a->array[i] = a->array[i - nwords];
|
|
}
|
|
/* Zero pad shifted words. */
|
|
for (; i >= 0; i--) {
|
|
a->array[i] = 0;
|
|
}
|
|
}
|
|
|
|
static void _lshift_one_bit(RNumBig *a) {
|
|
R_RETURN_IF_FAIL (a);
|
|
|
|
int i;
|
|
for (i = (R_BIG_ARRAY_SIZE - 1); i > 0; i--) {
|
|
a->array[i] = (a->array[i] << 1) | (a->array[i - 1] >> ((8 * R_BIG_WORD_SIZE) - 1));
|
|
}
|
|
a->array[0] <<= 1;
|
|
}
|
|
|
|
static void _rshift_one_bit(RNumBig *a) {
|
|
R_RETURN_IF_FAIL (a);
|
|
|
|
int i;
|
|
for (i = 0; i < (R_BIG_ARRAY_SIZE - 1); i++) {
|
|
a->array[i] = (a->array[i] >> 1) | (a->array[i + 1] << ((8 * R_BIG_WORD_SIZE) - 1));
|
|
}
|
|
a->array[R_BIG_ARRAY_SIZE - 1] >>= 1;
|
|
}
|
|
|
|
static void _r_big_zero_out(RNumBig *a) {
|
|
R_RETURN_IF_FAIL (a);
|
|
|
|
size_t i;
|
|
for (i = 0; i < R_BIG_ARRAY_SIZE; i++) {
|
|
a->array[i] = 0;
|
|
}
|
|
a->sign = 1; /* hack to avoid -0 */
|
|
}
|
|
|
|
#endif
|