/* Copyright (C) 2007, 2008 William Hart Copyright (C) 2008 Peter Shrimpton Copyright (C) 2010 Fredrik Johansson Copyright (C) 2015 Dana Jacobsen This file is part of FLINT. FLINT is free software: you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License (LGPL) as published by the Free Software Foundation; either version 2.1 of the License, or (at your option) any later version. See . */ #define ulong ulongxx /* interferes with system includes */ #include #include #undef ulong #define ulong mp_limb_t #include #include "flint.h" #include "ulong_extras.h" static unsigned int nextmod30[] = { 1, 6, 5, 4, 3, 2, 1, 4, 3, 2, 1, 2, 1, 4, 3, 2, 1, 2, 1, 4, 3, 2, 1, 6, 5, 4, 3, 2, 1, 2 }; static unsigned int nextindex[] = { 1, 7, 7, 7, 7, 7, 7, 11, 11, 11, 11, 13, 13, 17, 17, 17, 17, 19, 19, 23, 23, 23, 23, 29, 29, 29, 29, 29, 29, 1 }; /* first 64 primes used for modular arithmetic */ #define N_MODULUS (UWORD(1) << (FLINT_BITS - 1)) #define N_MOD_TAB 64 static const unsigned short n_modular_primes_tab[N_MOD_TAB] = { #if FLINT_BITS == 64 29, 99, 123, 131, 155, 255, 269, 359, 435, 449, 453, 485, 491, 543, 585, 599, 753, 849, 879, 885, 903, 995, 1209, 1251, 1311, 1373, 1403, 1485, 1533, 1535, 1545, 1551, 1575, 1601, 1625, 1655, 1701, 1709, 1845, 1859, 1913, 1995, 2045, 2219, 2229, 2321, 2363, 2385, 2483, 2499, 2523, 2543, 2613, 2639, 2679, 2829, 2931, 3089, 3165, 3189, 3245, 3273, 3291, 3341 #else 11, 45, 65, 95, 129, 135, 165, 209, 219, 221, 239, 245, 281, 303, 345, 351, 359, 389, 393, 395, 413, 435, 461, 513, 519, 549, 555, 573, 575, 585, 591, 611, 623, 629, 683, 689, 701, 729, 785, 791, 813, 843, 851, 869, 879, 893, 905, 921, 953, 963, 965, 969, 993, 1031, 1049, 1073, 1085, 1103, 1143, 1173, 1203, 1221, 1229, 1271 #endif }; static mp_limb_t bsearch_uint(mp_limb_t n, const unsigned int *t, int tlen) { int lo = 0; int hi = tlen-1; while (lo < hi) { int mid = lo + (hi-lo)/2; if (t[mid] <= n) lo = mid+1; else hi = mid; } return t[lo]; } mp_limb_t n_nextprime(mp_limb_t n, int proved) { ulong i, index; /* For tiny inputs, bsearch in small primes table */ if (n < flint_primes_small[FLINT_NUM_PRIMES_SMALL-1]) return bsearch_uint(n, flint_primes_small, FLINT_NUM_PRIMES_SMALL); if (n >= N_MODULUS && n < N_MODULUS + n_modular_primes_tab[N_MOD_TAB-1]) { for (i = 0; i < N_MOD_TAB; i++) if (N_MODULUS + n_modular_primes_tab[i] > n) return N_MODULUS + n_modular_primes_tab[i]; } if (n >= UWORD_MAX_PRIME) { flint_printf("Exception (n_nextprime). No larger single-limb prime exists.\n"); flint_abort(); } index = n % 30; do { n += nextmod30[index]; index = nextindex[index]; } while (!n_is_prime(n)); return n; }