Blame bootstrap.c

Packit 15dc08
/* Functions needed for bootstrapping the gmp build, based on mini-gmp.
Packit 15dc08
Packit 15dc08
Copyright 2001, 2002, 2004, 2011, 2012 Free Software Foundation, Inc.
Packit 15dc08
Packit 15dc08
This file is part of the GNU MP Library.
Packit 15dc08
Packit 15dc08
The GNU MP Library is free software; you can redistribute it and/or modify
Packit 15dc08
it under the terms of either:
Packit 15dc08
Packit 15dc08
  * the GNU Lesser General Public License as published by the Free
Packit 15dc08
    Software Foundation; either version 3 of the License, or (at your
Packit 15dc08
    option) any later version.
Packit 15dc08
Packit 15dc08
or
Packit 15dc08
Packit 15dc08
  * the GNU General Public License as published by the Free Software
Packit 15dc08
    Foundation; either version 2 of the License, or (at your option) any
Packit 15dc08
    later version.
Packit 15dc08
Packit 15dc08
or both in parallel, as here.
Packit 15dc08
Packit 15dc08
The GNU MP Library is distributed in the hope that it will be useful, but
Packit 15dc08
WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
Packit 15dc08
or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
Packit 15dc08
for more details.
Packit 15dc08
Packit 15dc08
You should have received copies of the GNU General Public License and the
Packit 15dc08
GNU Lesser General Public License along with the GNU MP Library.  If not,
Packit 15dc08
see https://www.gnu.org/licenses/.  */
Packit 15dc08
Packit 15dc08
Packit 15dc08
#include "mini-gmp/mini-gmp.c"
Packit 15dc08
Packit 15dc08
#define MIN(l,o) ((l) < (o) ? (l) : (o))
Packit 15dc08
#define PTR(x)   ((x)->_mp_d)
Packit 15dc08
#define SIZ(x)   ((x)->_mp_size)
Packit 15dc08
Packit 15dc08
#define xmalloc gmp_default_alloc
Packit 15dc08
Packit 15dc08
int
Packit 15dc08
isprime (unsigned long int t)
Packit 15dc08
{
Packit 15dc08
  unsigned long int q, r, d;
Packit 15dc08
Packit 15dc08
  if (t < 32)
Packit 15dc08
    return (0xa08a28acUL >> t) & 1;
Packit 15dc08
  if ((t & 1) == 0)
Packit 15dc08
    return 0;
Packit 15dc08
Packit 15dc08
  if (t % 3 == 0)
Packit 15dc08
    return 0;
Packit 15dc08
  if (t % 5 == 0)
Packit 15dc08
    return 0;
Packit 15dc08
  if (t % 7 == 0)
Packit 15dc08
    return 0;
Packit 15dc08
Packit 15dc08
  for (d = 11;;)
Packit 15dc08
    {
Packit 15dc08
      q = t / d;
Packit 15dc08
      r = t - q * d;
Packit 15dc08
      if (q < d)
Packit 15dc08
	return 1;
Packit 15dc08
      if (r == 0)
Packit 15dc08
	break;
Packit 15dc08
      d += 2;
Packit 15dc08
      q = t / d;
Packit 15dc08
      r = t - q * d;
Packit 15dc08
      if (q < d)
Packit 15dc08
	return 1;
Packit 15dc08
      if (r == 0)
Packit 15dc08
	break;
Packit 15dc08
      d += 4;
Packit 15dc08
    }
Packit 15dc08
  return 0;
Packit 15dc08
}
Packit 15dc08
Packit 15dc08
int
Packit 15dc08
log2_ceil (int n)
Packit 15dc08
{
Packit 15dc08
  int  e;
Packit 15dc08
  assert (n >= 1);
Packit 15dc08
  for (e = 0; ; e++)
Packit 15dc08
    if ((1 << e) >= n)
Packit 15dc08
      break;
Packit 15dc08
  return e;
Packit 15dc08
}
Packit 15dc08
Packit 15dc08
/* Set inv to the inverse of d, in the style of invert_limb, ie. for
Packit 15dc08
   udiv_qrnnd_preinv.  */
Packit 15dc08
void
Packit 15dc08
mpz_preinv_invert (mpz_t inv, mpz_t d, int numb_bits)
Packit 15dc08
{
Packit 15dc08
  mpz_t  t;
Packit 15dc08
  int    norm;
Packit 15dc08
  assert (SIZ(d) > 0);
Packit 15dc08
Packit 15dc08
  norm = numb_bits - mpz_sizeinbase (d, 2);
Packit 15dc08
  assert (norm >= 0);
Packit 15dc08
  mpz_init_set_ui (t, 1L);
Packit 15dc08
  mpz_mul_2exp (t, t, 2*numb_bits - norm);
Packit 15dc08
  mpz_tdiv_q (inv, t, d);
Packit 15dc08
  mpz_set_ui (t, 1L);
Packit 15dc08
  mpz_mul_2exp (t, t, numb_bits);
Packit 15dc08
  mpz_sub (inv, inv, t);
Packit 15dc08
Packit 15dc08
  mpz_clear (t);
Packit 15dc08
}
Packit 15dc08
Packit 15dc08
/* Calculate r satisfying r*d == 1 mod 2^n. */
Packit 15dc08
void
Packit 15dc08
mpz_invert_2exp (mpz_t r, mpz_t a, unsigned long n)
Packit 15dc08
{
Packit 15dc08
  unsigned long  i;
Packit 15dc08
  mpz_t  inv, prod;
Packit 15dc08
Packit 15dc08
  assert (mpz_odd_p (a));
Packit 15dc08
Packit 15dc08
  mpz_init_set_ui (inv, 1L);
Packit 15dc08
  mpz_init (prod);
Packit 15dc08
Packit 15dc08
  for (i = 1; i < n; i++)
Packit 15dc08
    {
Packit 15dc08
      mpz_mul (prod, inv, a);
Packit 15dc08
      if (mpz_tstbit (prod, i) != 0)
Packit 15dc08
	mpz_setbit (inv, i);
Packit 15dc08
    }
Packit 15dc08
Packit 15dc08
  mpz_mul (prod, inv, a);
Packit 15dc08
  mpz_tdiv_r_2exp (prod, prod, n);
Packit 15dc08
  assert (mpz_cmp_ui (prod, 1L) == 0);
Packit 15dc08
Packit 15dc08
  mpz_set (r, inv);
Packit 15dc08
Packit 15dc08
  mpz_clear (inv);
Packit 15dc08
  mpz_clear (prod);
Packit 15dc08
}
Packit 15dc08
Packit 15dc08
/* Calculate inv satisfying r*a == 1 mod 2^n. */
Packit 15dc08
void
Packit 15dc08
mpz_invert_ui_2exp (mpz_t r, unsigned long a, unsigned long n)
Packit 15dc08
{
Packit 15dc08
  mpz_t  az;
Packit 15dc08
  mpz_init_set_ui (az, a);
Packit 15dc08
  mpz_invert_2exp (r, az, n);
Packit 15dc08
  mpz_clear (az);
Packit 15dc08
}