t-mul.c
上传用户:qaz666999
上传日期:2022-08-06
资源大小:2570k
文件大小:6k
源码类别:

数学计算

开发平台:

Unix_Linux

  1. /* Test mpz_cmp, mpz_mul.
  2. Copyright 1991, 1993, 1994, 1996, 1997, 2000, 2001, 2002, 2003, 2004 Free
  3. Software Foundation, Inc.
  4. This file is part of the GNU MP Library.
  5. The GNU MP Library is free software; you can redistribute it and/or modify
  6. it under the terms of the GNU Lesser General Public License as published by
  7. the Free Software Foundation; either version 3 of the License, or (at your
  8. option) any later version.
  9. The GNU MP Library is distributed in the hope that it will be useful, but
  10. WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
  11. or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU Lesser General Public
  12. License for more details.
  13. You should have received a copy of the GNU Lesser General Public License
  14. along with the GNU MP Library.  If not, see http://www.gnu.org/licenses/.  */
  15. #include <stdio.h>
  16. #include <stdlib.h>
  17. #include "gmp.h"
  18. #include "gmp-impl.h"
  19. #include "longlong.h"
  20. #include "tests.h"
  21. void debug_mp __GMP_PROTO ((mpz_t));
  22. static void refmpz_mul __GMP_PROTO ((mpz_t, const mpz_t, const mpz_t));
  23. void dump_abort __GMP_PROTO ((int, char *, mpz_t, mpz_t, mpz_t, mpz_t));
  24. #define FFT_MIN_BITSIZE 100000
  25. char *extra_fft;
  26. void
  27. one (int i, mpz_t multiplicand, mpz_t multiplier)
  28. {
  29.   mpz_t product, ref_product;
  30.   mpz_init (product);
  31.   mpz_init (ref_product);
  32.   /* Test plain multiplication comparing results against reference code.  */
  33.   mpz_mul (product, multiplier, multiplicand);
  34.   refmpz_mul (ref_product, multiplier, multiplicand);
  35.   if (mpz_cmp (product, ref_product))
  36.     dump_abort (i, "incorrect plain product",
  37. multiplier, multiplicand, product, ref_product);
  38.   /* Test squaring, comparing results against plain multiplication  */
  39.   mpz_mul (product, multiplier, multiplier);
  40.   mpz_set (multiplicand, multiplier);
  41.   mpz_mul (ref_product, multiplier, multiplicand);
  42.   if (mpz_cmp (product, ref_product))
  43.     dump_abort (i, "incorrect square product",
  44. multiplier, multiplier, product, ref_product);
  45.   mpz_clear (product);
  46.   mpz_clear (ref_product);
  47. }
  48. int
  49. main (int argc, char **argv)
  50. {
  51.   mpz_t op1, op2;
  52.   int i;
  53.   int fft_max_2exp;
  54.   gmp_randstate_ptr rands;
  55.   mpz_t bs;
  56.   unsigned long bsi, size_range, fsize_range;
  57.   tests_start ();
  58.   rands = RANDS;
  59.   extra_fft = getenv ("GMP_CHECK_FFT");
  60.   fft_max_2exp = 0;
  61.   if (extra_fft != NULL)
  62.     fft_max_2exp = atoi (extra_fft);
  63.   if (fft_max_2exp <= 1) /* compat with old use of GMP_CHECK_FFT */
  64.     fft_max_2exp = 22; /* default limit, good for any machine */
  65.   mpz_init (bs);
  66.   mpz_init (op1);
  67.   mpz_init (op2);
  68.   fsize_range = 4 << 8; /* a fraction 1/256 of size_range */
  69.   for (i = 0;; i++)
  70.     {
  71.       size_range = fsize_range >> 8;
  72.       fsize_range = fsize_range * 33 / 32;
  73.       if (size_range > fft_max_2exp)
  74. break;
  75.       mpz_urandomb (bs, rands, size_range);
  76.       mpz_rrandomb (op1, rands, mpz_get_ui (bs));
  77.       if (i & 1)
  78. mpz_urandomb (bs, rands, size_range);
  79.       mpz_rrandomb (op2, rands, mpz_get_ui (bs));
  80.       mpz_urandomb (bs, rands, 4);
  81.       bsi = mpz_get_ui (bs);
  82.       if ((bsi & 0x3) == 0)
  83. mpz_neg (op1, op1);
  84.       if ((bsi & 0xC) == 0)
  85. mpz_neg (op2, op2);
  86.       /* printf ("%d %dn", SIZ (op1), SIZ (op2)); */
  87.       one (i, op2, op1);
  88.     }
  89.   for (i = -50; i < 0; i++)
  90.     {
  91.       mpz_urandomb (bs, rands, 32);
  92.       size_range = mpz_get_ui (bs) % fft_max_2exp;
  93.       mpz_urandomb (bs, rands, size_range);
  94.       mpz_rrandomb (op1, rands, mpz_get_ui (bs) + FFT_MIN_BITSIZE);
  95.       mpz_urandomb (bs, rands, size_range);
  96.       mpz_rrandomb (op2, rands, mpz_get_ui (bs) + FFT_MIN_BITSIZE);
  97.       /* printf ("%d: %d %dn", i, SIZ (op1), SIZ (op2)); */
  98.       fflush (stdout);
  99.       one (-1, op2, op1);
  100.     }
  101.   mpz_clear (bs);
  102.   mpz_clear (op1);
  103.   mpz_clear (op2);
  104.   tests_end ();
  105.   exit (0);
  106. }
  107. static void
  108. refmpz_mul (mpz_t w, const mpz_t u, const mpz_t v)
  109. {
  110.   mp_size_t usize = u->_mp_size;
  111.   mp_size_t vsize = v->_mp_size;
  112.   mp_size_t wsize;
  113.   mp_size_t sign_product;
  114.   mp_ptr up, vp;
  115.   mp_ptr wp;
  116.   mp_size_t talloc;
  117.   sign_product = usize ^ vsize;
  118.   usize = ABS (usize);
  119.   vsize = ABS (vsize);
  120.   if (usize == 0 || vsize == 0)
  121.     {
  122.       SIZ (w) = 0;
  123.       return;
  124.     }
  125.   talloc = usize + vsize;
  126.   up = u->_mp_d;
  127.   vp = v->_mp_d;
  128.   wp = __GMP_ALLOCATE_FUNC_LIMBS (talloc);
  129.   if (usize > vsize)
  130.     refmpn_mul (wp, up, usize, vp, vsize);
  131.   else
  132.     refmpn_mul (wp, vp, vsize, up, usize);
  133.   wsize = usize + vsize;
  134.   wsize -= wp[wsize - 1] == 0;
  135.   MPZ_REALLOC (w, wsize);
  136.   MPN_COPY (PTR(w), wp, wsize);
  137.   SIZ(w) = sign_product < 0 ? -wsize : wsize;
  138.   __GMP_FREE_FUNC_LIMBS (wp, talloc);
  139. }
  140. void
  141. dump_abort (int i, char *s,
  142.             mpz_t op1, mpz_t op2, mpz_t product, mpz_t ref_product)
  143. {
  144.   mp_size_t b, e;
  145.   fprintf (stderr, "ERROR: %s in test %dn", s, i);
  146.   fprintf (stderr, "op1          = "); debug_mp (op1);
  147.   fprintf (stderr, "op2          = "); debug_mp (op2);
  148.   fprintf (stderr, "    product  = "); debug_mp (product);
  149.   fprintf (stderr, "ref_product  = "); debug_mp (ref_product);
  150.   for (b = 0; b < ABSIZ(ref_product); b++)
  151.     if (PTR(ref_product)[b] != PTR(product)[b])
  152.       break;
  153.   for (e = ABSIZ(ref_product) - 1; e >= 0; e--)
  154.     if (PTR(ref_product)[e] != PTR(product)[e])
  155.       break;
  156.   printf ("ERRORS in %ld--%ldn", b, e);
  157.   abort();
  158. }
  159. void
  160. debug_mp (mpz_t x)
  161. {
  162.   size_t siz = mpz_sizeinbase (x, 16);
  163.   if (siz > 65)
  164.     {
  165.       mpz_t q;
  166.       mpz_init (q);
  167.       mpz_tdiv_q_2exp (q, x, 4 * (mpz_sizeinbase (x, 16) - 25));
  168.       gmp_fprintf (stderr, "%ZX...", q);
  169.       mpz_tdiv_r_2exp (q, x, 4 * 25);
  170.       gmp_fprintf (stderr, "%025ZX [%d]n", q, (int) siz);
  171.       mpz_clear (q);
  172.     }
  173.   else
  174.     {
  175.       gmp_fprintf (stderr, "%ZXn", x);
  176.     }
  177. }