toom-shared.h
上传用户:qaz666999
上传日期:2022-08-06
资源大小:2570k
文件大小:4k
源码类别:

数学计算

开发平台:

Unix_Linux

  1. /* Test for various Toom functions.
  2. Copyright 2009 Free Software Foundation, Inc.
  3. This file is part of the GNU MP Library.
  4. The GNU MP Library is free software; you can redistribute it and/or modify
  5. it under the terms of the GNU Lesser General Public License as published by
  6. the Free Software Foundation; either version 3 of the License, or (at your
  7. option) any later version.
  8. The GNU MP Library is distributed in the hope that it will be useful, but
  9. WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY
  10. or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU Lesser General Public
  11. License for more details.
  12. You should have received a copy of the GNU Lesser General Public License
  13. along with the GNU MP Library.  If not, see http://www.gnu.org/licenses/.  */
  14. #include "gmp.h"
  15. #include "gmp-impl.h"
  16. #include "tests.h"
  17. #include <stdlib.h>
  18. #include <stdio.h>
  19. /* Main file is expected to define mpn_toomMN_mul,
  20.  * mpn_toomMN_mul_itch, MIN_AN, MIN_BN(an), MAX_BN(an) and then
  21.  * include this file. */
  22. /* Sizes are up to 2^SIZE_LOG limbs */
  23. #ifndef SIZE_LOG
  24. #define SIZE_LOG 10
  25. #endif
  26. #ifndef COUNT
  27. #define COUNT 2000
  28. #endif
  29. #define MAX_AN (1L << SIZE_LOG)
  30. #ifndef MAX_BN
  31. #define MAX_BN(an) (an)
  32. #endif
  33. /* For general toomMN_mul, we need
  34.  *
  35.  * MIN_BN(an) = N + floor(((N-1)*an + M - N)/M)
  36.  *
  37.  * MAX_BN(an) = floor(N*(an-1)/(M-1)) - N + 1
  38.  */
  39. int
  40. main (int argc, char **argv)
  41. {
  42.   mp_ptr ap, bp, refp, pp, scratch;
  43.   int count = COUNT;
  44.   int test;
  45.   gmp_randstate_ptr rands;
  46.   TMP_DECL;
  47.   TMP_MARK;
  48.   if (argc > 1)
  49.     {
  50.       char *end;
  51.       count = strtol (argv[1], &end, 0);
  52.       if (*end || count <= 0)
  53. {
  54.   fprintf (stderr, "Invalid test count: %s.n", argv[1]);
  55.   return 1;
  56. }
  57.     }
  58.   tests_start ();
  59.   rands = RANDS;
  60.   ap = TMP_ALLOC_LIMBS (MAX_AN);
  61.   bp = TMP_ALLOC_LIMBS (MAX_BN(MAX_AN));
  62.   refp = TMP_ALLOC_LIMBS (MAX_AN + MAX_BN(MAX_AN));
  63.   pp = 1+TMP_ALLOC_LIMBS (MAX_AN + MAX_BN(MAX_AN)+2);
  64.   scratch
  65.     = 1+TMP_ALLOC_LIMBS (mpn_toomMN_mul_itch (MAX_AN, MAX_BN(MAX_AN))
  66.  + 2);
  67.   for (test = 0; test < count; test++)
  68.     {
  69.       unsigned size_min;
  70.       unsigned size_range;
  71.       mp_size_t an, bn;
  72.       mp_size_t itch;
  73.       mp_limb_t p_before, p_after, s_before, s_after;
  74.       for (size_min = 1; (1L << size_min) < MIN_AN; size_min++)
  75. ;
  76.       /* We generate an in the MIN_AN <= an <= (1 << size_range). */
  77.       size_range = size_min
  78. + gmp_urandomm_ui (rands, SIZE_LOG + 1 - size_min);
  79.       an = MIN_AN
  80. + gmp_urandomm_ui (rands, (1L << size_range) + 1 - MIN_AN);
  81.       bn = MIN_BN(an)
  82. + gmp_urandomm_ui (rands, MAX_BN(an) + 1 - MIN_BN(an));
  83.       mpn_random2 (ap, an);
  84.       mpn_random2 (bp, bn);
  85.       mpn_random2 (pp-1, an + bn + 2);
  86.       p_before = pp[-1];
  87.       p_after = pp[an + bn];
  88.       itch = mpn_toomMN_mul_itch (an, bn);
  89.       ASSERT_ALWAYS (itch <= mpn_toomMN_mul_itch (MAX_AN, MAX_BN(MAX_AN)));
  90.       mpn_random2 (scratch-1, itch+2);
  91.       s_before = scratch[-1];
  92.       s_after = scratch[itch];
  93.       mpn_toomMN_mul (pp, ap, an, bp, bn, scratch);
  94.       refmpn_mul (refp, ap, an, bp, bn);
  95.       if (pp[-1] != p_before || pp[an + bn] != p_after
  96.   || scratch[-1] != s_before || scratch[itch] != s_after
  97.   || mpn_cmp (refp, pp, an + bn) != 0)
  98. {
  99.   printf ("ERROR in test %d, an = %d, bn = %dn",
  100.   test, (int) an, (int) bn);
  101.   if (pp[-1] != p_before)
  102.     {
  103.       printf ("before pp:"); mpn_dump (pp -1, 1);
  104.       printf ("keep:   "); mpn_dump (&p_before, 1);
  105.     }
  106.   if (pp[an + bn] != p_after)
  107.     {
  108.       printf ("after pp:"); mpn_dump (pp + an + bn, 1);
  109.       printf ("keep:   "); mpn_dump (&p_after, 1);
  110.     }
  111.   if (scratch[-1] != s_before)
  112.     {
  113.       printf ("before scratch:"); mpn_dump (scratch-1, 1);
  114.       printf ("keep:   "); mpn_dump (&s_before, 1);
  115.     }
  116.   if (scratch[itch] != s_after)
  117.     {
  118.       printf ("after scratch:"); mpn_dump (scratch + itch, 1);
  119.       printf ("keep:   "); mpn_dump (&s_after, 1);
  120.     }
  121.   mpn_dump (ap, an);
  122.   mpn_dump (bp, bn);
  123.   mpn_dump (pp, an + bn);
  124.   mpn_dump (refp, an + bn);
  125.   abort();
  126. }
  127.     }
  128.   TMP_FREE;
  129.   tests_end ();
  130.   return 0;
  131. }