Blame gen-jacobitab.c

Packit 15dc08
/* gen-jacobi.c
Packit 15dc08
Packit 15dc08
   Contributed to the GNU project by Niels Möller.
Packit 15dc08
Packit 15dc08
Copyright 2010 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
/* Generate the lookup table needed for fast left-to-right computation
Packit 15dc08
   of the Jacobi symbol. */
Packit 15dc08
Packit 15dc08
#include <assert.h>
Packit 15dc08
#include <stdio.h>
Packit 15dc08
#include <stdlib.h>
Packit 15dc08
Packit 15dc08
static const struct
Packit 15dc08
{
Packit 15dc08
  unsigned char a;
Packit 15dc08
  unsigned char b;
Packit 15dc08
} decode_table[13] = {
Packit 15dc08
  /*  0 */ { 0, 1 },
Packit 15dc08
  /*  1 */ { 0, 3 },
Packit 15dc08
  /*  2 */ { 1, 1 },
Packit 15dc08
  /*  3 */ { 1, 3 },
Packit 15dc08
  /*  4 */ { 2, 1 },
Packit 15dc08
  /*  5 */ { 2, 3 },
Packit 15dc08
  /*  6 */ { 3, 1 },
Packit 15dc08
  /*  7 */ { 3, 3 }, /* d = 1 */
Packit 15dc08
  /*  8 */ { 1, 0 },
Packit 15dc08
  /*  9 */ { 1, 2 },
Packit 15dc08
  /* 10 */ { 3, 0 },
Packit 15dc08
  /* 11 */ { 3, 2 },
Packit 15dc08
  /* 12 */ { 3, 3 }, /* d = 0 */
Packit 15dc08
Packit 15dc08
};
Packit 15dc08
#define JACOBI_A(bits) (decode_table[(bits)>>1].a)
Packit 15dc08
#define JACOBI_B(bits) (decode_table[(bits)>>1].b)
Packit 15dc08
Packit 15dc08
#define JACOBI_E(bits) ((bits) & 1)
Packit 15dc08
#define JACOBI_D(bits) (((bits)>>1) == 7) /* Gives 0 for don't care states. */
Packit 15dc08
Packit 15dc08
static unsigned
Packit 15dc08
encode (unsigned a, unsigned b, unsigned d)
Packit 15dc08
{
Packit 15dc08
  unsigned i;
Packit 15dc08
Packit 15dc08
  assert (d < 2);
Packit 15dc08
  assert (a < 4);
Packit 15dc08
  assert (b < 4);
Packit 15dc08
  assert ( (a | b ) & 1);
Packit 15dc08
Packit 15dc08
  if (a == 3 && b == 3)
Packit 15dc08
    return d ? 7 : 12;
Packit 15dc08
Packit 15dc08
  for (i = 0; i < 12; i++)
Packit 15dc08
    if (decode_table[i].a == a
Packit 15dc08
	&& decode_table[i].b == b)
Packit 15dc08
      return i;
Packit 15dc08
Packit 15dc08
  abort ();
Packit 15dc08
}
Packit 15dc08
Packit 15dc08
int
Packit 15dc08
main (int argc, char **argv)
Packit 15dc08
{
Packit 15dc08
  unsigned bits;
Packit 15dc08
Packit 15dc08
  for (bits = 0; bits < 208; bits++)
Packit 15dc08
    {
Packit 15dc08
      unsigned e, a, b, d_old, d, q;
Packit 15dc08
Packit 15dc08
      if (bits && !(bits & 0xf))
Packit 15dc08
	printf("\n");
Packit 15dc08
Packit 15dc08
      q = bits & 3;
Packit 15dc08
      d = (bits >> 2) & 1;
Packit 15dc08
Packit 15dc08
      e = JACOBI_E (bits >> 3);
Packit 15dc08
      a = JACOBI_A (bits >> 3);
Packit 15dc08
      b = JACOBI_B (bits >> 3);
Packit 15dc08
      d_old = JACOBI_D (bits >> 3);
Packit 15dc08
Packit 15dc08
      if (d != d_old && a == 3 && b == 3)
Packit 15dc08
	e ^= 1;
Packit 15dc08
Packit 15dc08
      if (d == 1)
Packit 15dc08
	{
Packit 15dc08
	  if (b == 2)
Packit 15dc08
	    e ^= (q & (a >> 1)) ^ (q >> 1);
Packit 15dc08
	  a = (a - q * b) & 3;
Packit 15dc08
	}
Packit 15dc08
      else
Packit 15dc08
	{
Packit 15dc08
	  if (a == 2)
Packit 15dc08
	    e ^= (q & (b >> 1)) ^ (q >> 1);
Packit 15dc08
	  b = (b - q * a) & 3;
Packit 15dc08
	}
Packit 15dc08
Packit 15dc08
      printf("%2d,", (encode (a, b, d) << 1) | e);
Packit 15dc08
    }
Packit 15dc08
  printf("\n");
Packit 15dc08
Packit 15dc08
  return 0;
Packit 15dc08
}