Blame gen-fib.c

Packit 15dc08
/* Generate Fibonacci table data.
Packit 15dc08
Packit 15dc08
Copyright 2001, 2002, 2004, 2012, 2014, 2015 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
#include <stdio.h>
Packit 15dc08
#include "bootstrap.c"
Packit 15dc08
Packit 15dc08
mpz_t  *f;
Packit 15dc08
int    fnum, fib_limit, luc_limit;
Packit 15dc08
Packit 15dc08
void
Packit 15dc08
generate (int numb_bits)
Packit 15dc08
{
Packit 15dc08
  mpz_t  limit, l;
Packit 15dc08
  int    falloc, i;
Packit 15dc08
Packit 15dc08
  mpz_init2 (limit, numb_bits + 1);
Packit 15dc08
  mpz_setbit (limit, numb_bits);
Packit 15dc08
Packit 15dc08
  /* fib(2n) > 2^n, so use 2n as a limit for the table size */
Packit 15dc08
  falloc = 2 * numb_bits;
Packit 15dc08
  f = (mpz_t*) xmalloc (falloc * sizeof (*f));
Packit 15dc08
Packit 15dc08
  mpz_init_set_ui (f[0], 1L);  /* F[-1] */
Packit 15dc08
  mpz_init_set_ui (f[1], 0L);  /* F[0] */
Packit 15dc08
Packit 15dc08
  mpz_init (l);
Packit 15dc08
Packit 15dc08
  for (i = 2; ; i++)
Packit 15dc08
    {
Packit 15dc08
      assert (i < falloc);
Packit 15dc08
Packit 15dc08
      /* F[i] = F[i-1] + F[i-2] */
Packit 15dc08
      mpz_init (f[i]);
Packit 15dc08
      mpz_add (f[i], f[i-1], f[i-2]);
Packit 15dc08
      if (mpz_cmp (f[i], limit) >= 0)
Packit 15dc08
        break;
Packit 15dc08
Packit 15dc08
      fnum = i+1;
Packit 15dc08
      fib_limit = i-1;
Packit 15dc08
Packit 15dc08
      /* L[i] = F[i]+2*F[i-1] */
Packit 15dc08
      mpz_add (l, f[i], f[i-1]);
Packit 15dc08
      mpz_add (l, l, f[i-1]);
Packit 15dc08
Packit 15dc08
      if (mpz_cmp (l, limit) < 0)
Packit 15dc08
        luc_limit = i-1;
Packit 15dc08
    }
Packit 15dc08
Packit 15dc08
  mpz_clear (limit);
Packit 15dc08
}
Packit 15dc08
Packit 15dc08
Packit 15dc08
void
Packit 15dc08
header (int numb_bits)
Packit 15dc08
{
Packit 15dc08
  printf ("/* This file generated by gen-fib.c - DO NOT EDIT. */\n");
Packit 15dc08
  printf ("\n");
Packit 15dc08
  printf ("#if GMP_NUMB_BITS != %d\n", numb_bits);
Packit 15dc08
  printf ("Error, error, this data is for %d bits\n", numb_bits);
Packit 15dc08
  printf ("#endif\n");
Packit 15dc08
  printf ("\n");
Packit 15dc08
  printf ("#define FIB_TABLE_LIMIT         %d\n", fib_limit);
Packit 15dc08
  printf ("#define FIB_TABLE_LUCNUM_LIMIT  %d\n", luc_limit);
Packit 15dc08
}
Packit 15dc08
Packit 15dc08
void
Packit 15dc08
table (int numb_bits)
Packit 15dc08
{
Packit 15dc08
  int  i;
Packit 15dc08
Packit 15dc08
  printf ("/* This file generated by gen-fib.c - DO NOT EDIT. */\n");
Packit 15dc08
  printf ("\n");
Packit 15dc08
  printf ("#include \"gmp.h\"\n");
Packit 15dc08
  printf ("#include \"gmp-impl.h\"\n");
Packit 15dc08
  printf ("\n");
Packit 15dc08
  printf ("#if GMP_NUMB_BITS != %d\n", numb_bits);
Packit 15dc08
  printf ("Error, error, this data is for %d bits\n", numb_bits);
Packit 15dc08
  printf ("#endif\n");
Packit 15dc08
  printf ("\n");
Packit 15dc08
  printf ("const mp_limb_t\n");
Packit 15dc08
  printf ("__gmp_fib_table[FIB_TABLE_LIMIT+2] = {\n");
Packit 15dc08
Packit 15dc08
  for (i = 0; i < fnum; i++)
Packit 15dc08
    {
Packit 15dc08
      printf ("  CNST_LIMB (0x");
Packit 15dc08
      mpz_out_str (stdout, 16, f[i]);
Packit 15dc08
      printf ("),  /* %d */\n", i-1);
Packit 15dc08
    }
Packit 15dc08
  printf ("};\n");
Packit 15dc08
}
Packit 15dc08
Packit 15dc08
int
Packit 15dc08
main (int argc, char *argv[])
Packit 15dc08
{
Packit 15dc08
  int  limb_bits, nail_bits, numb_bits;
Packit 15dc08
Packit 15dc08
  if (argc != 4)
Packit 15dc08
    {
Packit 15dc08
      fprintf (stderr, "Usage: gen-fib <header|table> <limbbits> <nailbits>\n");
Packit 15dc08
      exit (1);
Packit 15dc08
    }
Packit 15dc08
Packit 15dc08
  limb_bits = atoi (argv[2]);
Packit 15dc08
  nail_bits = atoi (argv[3]);
Packit 15dc08
Packit 15dc08
  if (limb_bits <= 0
Packit 15dc08
      || nail_bits < 0
Packit 15dc08
      || nail_bits >= limb_bits)
Packit 15dc08
    {
Packit 15dc08
      fprintf (stderr, "Invalid limb/nail bits: %d %d\n",
Packit 15dc08
               limb_bits, nail_bits);
Packit 15dc08
      exit (1);
Packit 15dc08
    }
Packit 15dc08
  numb_bits = limb_bits - nail_bits;
Packit 15dc08
Packit 15dc08
  generate (numb_bits);
Packit 15dc08
Packit 15dc08
  if (strcmp (argv[1], "header") == 0)
Packit 15dc08
    header (numb_bits);
Packit 15dc08
  else if (strcmp (argv[1], "table") == 0)
Packit 15dc08
    table (numb_bits);
Packit 15dc08
  else
Packit 15dc08
    {
Packit 15dc08
      fprintf (stderr, "Invalid header/table choice: %s\n", argv[1]);
Packit 15dc08
      exit (1);
Packit 15dc08
    }
Packit 15dc08
Packit 15dc08
  return 0;
Packit 15dc08
}