|  | /*
  ==============================================================================
   This file is part of the JUCE library - "Jules' Utility Class Extensions"
   Copyright 2004-7 by Raw Material Software ltd.
  ------------------------------------------------------------------------------
   JUCE can be redistributed and/or modified under the terms of the
   GNU General Public License, as published by the Free Software Foundation;
   either version 2 of the License, or (at your option) any later version.
   JUCE is distributed in the hope that it will be useful,
   but WITHOUT ANY WARRANTY; without even the implied warranty of
   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
   GNU General Public License for more details.
   You should have received a copy of the GNU General Public License
   along with JUCE; if not, visit www.gnu.org/licenses or write to the
   Free Software Foundation, Inc., 59 Temple Place, Suite 330,
   Boston, MA 02111-1307 USA
  ------------------------------------------------------------------------------
   If you'd like to release a closed-source product which uses JUCE, commercial
   licenses are also available: visit www.rawmaterialsoftware.com/juce for
   more information.
  ==============================================================================
*/
#include "../basics/juce_StandardHeader.h"
BEGIN_JUCE_NAMESPACE
#include "juce_Primes.h"
#include "../basics/juce_Random.h"
//==============================================================================
static void createSmallSieve (const int numBits, BitArray& result) throw()
{
    result.setBit (numBits);
    result.clearBit (numBits); // to enlarge the array
    result.setBit (0);
    int n = 2;
    do
    {
        for (int i = n + n; i < numBits; i += n)
            result.setBit (i);
        n = result.findNextClearBit (n + 1);
    }
    while (n <= (numBits >> 1));
}
static void bigSieve (const BitArray& base,
                      const int numBits,
                      BitArray& result,
                      const BitArray& smallSieve,
                      const int smallSieveSize) throw()
{
    jassert (! base[0]); // must be even!
    result.setBit (numBits);
    result.clearBit (numBits);  // to enlarge the array
    int index = smallSieve.findNextClearBit (0);
    do
    {
        const int prime = (index << 1) + 1;
        BitArray r (base);
        BitArray remainder;
        r.divideBy (prime, remainder);
        int i = prime - remainder.getBitRangeAsInt (0, 32);
        if (r.isEmpty())
            i += prime;
        if ((i & 1) == 0)
            i += prime;
        i = (i - 1) >> 1;
        while (i < numBits)
        {
            result.setBit (i);
            i += prime;
        }
        index = smallSieve.findNextClearBit (index + 1);
    }
    while (index < smallSieveSize);
}
static bool findCandidate (const BitArray& base,
                           const BitArray& sieve,
                           const int numBits,
                           BitArray& result,
                           const int certainty) throw()
{
    for (int i = 0; i < numBits; ++i)
    {
        if (! sieve[i])
        {
            result = base;
            result.add (BitArray ((unsigned int) ((i << 1) + 1)));
            if (Primes::isProbablyPrime (result, certainty))
                return true;
        }
    }
    return false;
}
//==============================================================================
const BitArray Primes::createProbablePrime (const int bitLength,
                                            const int certainty,
                                            const int* randomSeeds,
                                            int numRandomSeeds) throw()
{
    int defaultSeeds[8];
    if (numRandomSeeds <= 0)
    {
        randomSeeds = defaultSeeds;
        numRandomSeeds = 8;
        for (int j = 10; --j >= 0;)
        {
            Random r (0);
            r.setSeedRandomly();
            for (int i = numRandomSeeds; --i >= 0;)
                defaultSeeds[i] ^= r.nextInt() ^ Random::getSystemRandom().nextInt();
        }
    }
    BitArray smallSieve;
    const int smallSieveSize = 15000;
    createSmallSieve (smallSieveSize, smallSieve);
    BitArray p;
    for (int i = numRandomSeeds; --i >= 0;)
    {
        Random::getSystemRandom().setSeed (randomSeeds[i]);
        BitArray p2;
        p2.fillBitsRandomly (0, bitLength);
        p.xorWith (p2);
    }
    Random::getSystemRandom().setSeedRandomly();
    p.setBit (bitLength - 1);
    p.clearBit (0);
    const int searchLen = jmax (1024, (bitLength / 20) * 64);
    while (p.getHighestBit() < bitLength)
    {
        p.add (2 * searchLen);
        BitArray sieve;
        bigSieve (p, searchLen, sieve,
                  smallSieve, smallSieveSize);
        BitArray candidate;
        if (findCandidate (p, sieve, searchLen, candidate, certainty))
            return candidate;
    }
    jassertfalse
    return BitArray();
}
static bool passesMillerRabin (const BitArray& n, int iterations) throw()
{
    const BitArray one (1);
    const BitArray two (2);
    BitArray nMinusOne (n);
    nMinusOne.subtract (one);
    BitArray d (nMinusOne);
    const int s = d.findNextSetBit (0);
    d.shiftBits (-s);
    BitArray smallPrimes;
    int numBitsInSmallPrimes = 0;
    for (;;)
    {
        numBitsInSmallPrimes += 256;
        createSmallSieve (numBitsInSmallPrimes, smallPrimes);
        const int numPrimesFound = numBitsInSmallPrimes - smallPrimes.countNumberOfSetBits();
        if (numPrimesFound > iterations + 1)
            break;
    }
    int smallPrime = 2;
    while (--iterations >= 0)
    {
        smallPrime = smallPrimes.findNextClearBit (smallPrime + 1);
        BitArray r (smallPrime);
        //r.createRandomNumber (nMinusOne);
        r.exponentModulo (d, n);
        if (! (r == one || r == nMinusOne))
        {
            for (int j = 0; j < s; ++j)
            {
                r.exponentModulo (two, n);
                if (r == nMinusOne)
                    break;
            }
            if (r != nMinusOne)
                return false;
        }
    }
    return true;
}
bool Primes::isProbablyPrime (const BitArray& number,
                              const int certainty) throw()
{
    if (! number[0])
        return false;
    if (number.getHighestBit() <= 10)
    {
        const int num = number.getBitRangeAsInt (0, 10);
        for (int i = num / 2; --i > 1;)
            if (num % i == 0)
                return false;
        return true;
    }
    else
    {
        const BitArray screen (2 * 3 * 5 * 7 * 11 * 13 * 17 * 19 * 23);
        if (number.findGreatestCommonDivisor (screen) != BitArray (1))
            return false;
        return passesMillerRabin (number, certainty);
    }
}
END_JUCE_NAMESPACE
 |