-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathprime.cpp
261 lines (221 loc) · 9.03 KB
/
prime.cpp
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
/*
* prime.cpp
*
* Created on: 03.05.2014
* Author: mad
*/
#include "prime.h"
const unsigned int nFractionalBits = 24;
unsigned int nTargetInitialLength = 7; // initial chain length target
unsigned int nTargetMinLength = 6; // minimum chain length target
static const unsigned int TARGET_FRACTIONAL_MASK = (1u<<nFractionalBits) - 1;
static const unsigned int TARGET_LENGTH_MASK = ~TARGET_FRACTIONAL_MASK;
unsigned int TargetGetLength(unsigned int nBits)
{
return ((nBits & TARGET_LENGTH_MASK) >> nFractionalBits);
}
unsigned int TargetGetFractional(unsigned int nBits)
{
return (nBits & TARGET_FRACTIONAL_MASK);
}
std::string TargetToString(unsigned int nBits)
{
char tmp[20];
sprintf(tmp, "%02x.%06x", TargetGetLength(nBits), TargetGetFractional(nBits));
return std::string(tmp);
}
static void TargetIncrementLength(unsigned int& nBits)
{
nBits += (1 << nFractionalBits);
}
static void TargetDecrementLength(unsigned int& nBits)
{
if (TargetGetLength(nBits) > nTargetMinLength)
nBits -= (1 << nFractionalBits);
}
static const mpz_class mpzTwo = 2;
// Check Fermat probable primality test (2-PRP): 2 ** (n-1) = 1 (mod n)
// true: n is probable prime
// false: n is composite; set fractional length in the nLength output
static bool FermatProbablePrimalityTestFast(const mpz_class& n, unsigned int& nLength, CPrimalityTestParams& testParams, bool fFastFail = false)
{
mpz_class& mpzNMinusOne = testParams.mpzNMinusOne;
mpz_class& mpzE = testParams.mpzE;
mpz_class& mpzBase = testParams.mpzBase;
mpz_class& mpzR = testParams.mpzR;
mpz_class& mpzR2 = testParams.mpzR2;
mpz_class& mpzFrac = testParams.mpzFrac;
mpzNMinusOne = n - 1;
unsigned int nTrailingZeros = mpz_scan1(mpzNMinusOne.get_mpz_t(), 0);
if ((nTrailingZeros > 9))
nTrailingZeros = 9;
// Euler's criterion: 2 ** ((n-1)/2) needs to be either -1 or 1 (mod n)
mpz_tdiv_q_2exp(mpzE.get_mpz_t(), mpzNMinusOne.get_mpz_t(), nTrailingZeros);
unsigned int nShiftCount = (1U << (nTrailingZeros - 1)) - 1;
mpzBase = mpzTwo << nShiftCount;
mpz_powm(mpzR.get_mpz_t(), mpzBase.get_mpz_t(), mpzE.get_mpz_t(), n.get_mpz_t());
if ((mpzR == 1 || mpzR == mpzNMinusOne))
return true;
if ((fFastFail))
return false;
// Failed test, calculate fractional length
mpzR2 = mpzR * mpzR;
mpzR = mpzR2 % n; // derive Fermat test remainder
mpzFrac = n - mpzR;
mpzFrac <<= nFractionalBits;
mpzFrac /= n;
unsigned int nFractionalLength = mpzFrac.get_ui();
if ((nFractionalLength >= (1 << nFractionalBits)))
return false;
nLength = (nLength & TARGET_LENGTH_MASK) | nFractionalLength;
return false;
}
// Test probable primality of n = 2p +/- 1 based on Euler, Lagrange and Lifchitz
// fSophieGermain:
// true: n = 2p+1, p prime, aka Cunningham Chain of first kind
// false: n = 2p-1, p prime, aka Cunningham Chain of second kind
// Return values
// true: n is probable prime
// false: n is composite; set fractional length in the nLength output
static bool EulerLagrangeLifchitzPrimalityTestFast(const mpz_class& n, bool fSophieGermain, unsigned int& nLength, CPrimalityTestParams& testParams, bool fFastFail = false)
{
mpz_class& mpzNMinusOne = testParams.mpzNMinusOne;
mpz_class& mpzE = testParams.mpzE;
mpz_class& mpzBase = testParams.mpzBase;
mpz_class& mpzR = testParams.mpzR;
mpz_class& mpzR2 = testParams.mpzR2;
mpz_class& mpzFrac = testParams.mpzFrac;
mpzNMinusOne = n - 1;
unsigned int nTrailingZeros = mpz_scan1(mpzNMinusOne.get_mpz_t(), 0);
if ((nTrailingZeros > 9))
nTrailingZeros = 9;
mpz_tdiv_q_2exp(mpzE.get_mpz_t(), mpzNMinusOne.get_mpz_t(), nTrailingZeros);
unsigned int nShiftCount = (1U << (nTrailingZeros - 1)) - 1;
mpzBase = mpzTwo << nShiftCount;
mpz_powm(mpzR.get_mpz_t(), mpzBase.get_mpz_t(), mpzE.get_mpz_t(), n.get_mpz_t());
unsigned int nMod8 = n.get_ui() % 8;
bool fPassedTest = false;
if (fSophieGermain && nMod8 == 7) // Euler & Lagrange
fPassedTest = (mpzR == 1);
else if (fSophieGermain && nMod8 == 3) // Lifchitz
fPassedTest = (mpzR == mpzNMinusOne);
else if (!fSophieGermain && nMod8 == 5) // Lifchitz
fPassedTest = (mpzR == mpzNMinusOne);
else if (!fSophieGermain && nMod8 == 1) // LifChitz
fPassedTest = (mpzR == 1);
else
return false;
if ((fPassedTest))
return true;
if ((fFastFail))
return false;
// Failed test, calculate fractional length
mpzR2 = mpzR * mpzR;
mpzR = mpzR2 % n; // derive Fermat test remainder
mpzFrac = n - mpzR;
mpzFrac <<= nFractionalBits;
mpzFrac /= n;
unsigned int nFractionalLength = mpzFrac.get_ui();
if ((nFractionalLength >= (1 << nFractionalBits)))
return false;
nLength = (nLength & TARGET_LENGTH_MASK) | nFractionalLength;
return false;
}
// Test Probable Cunningham Chain for: n
// fSophieGermain:
// true - Test for Cunningham Chain of first kind (n, 2n+1, 4n+3, ...)
// false - Test for Cunningham Chain of second kind (n, 2n-1, 4n-3, ...)
static void ProbableCunninghamChainTestFast(const mpz_class& n, bool fSophieGermain, unsigned int& nProbableChainLength, CPrimalityTestParams& testParams, int base)
{
nProbableChainLength = base << nFractionalBits;
mpz_class &N = testParams.mpzN;
N = n;
if (base > 0) {
int X = (1 << base) - 1;
N <<= base;
N += (fSophieGermain? X : (-X));
}
// Fermat test for n first
if (!FermatProbablePrimalityTestFast(N, nProbableChainLength, testParams, true))
return;
// Euler-Lagrange-Lifchitz test for the following numbers in chain
for (unsigned int nChainSeq = base+1; true; nChainSeq++)
{
TargetIncrementLength(nProbableChainLength);
N <<= 1;
N += (fSophieGermain? 1 : (-1));
bool fFastFail = nChainSeq < 4;
if (!EulerLagrangeLifchitzPrimalityTestFast(N, fSophieGermain, nProbableChainLength, testParams, fFastFail))
break;
}
}
// Test Probable BiTwin Chain for: mpzOrigin
// Test the numbers in the optimal order for any given chain length
// Gives the correct length of a BiTwin chain even for short chains
static void ProbableBiTwinChainTestFast(const mpz_class& mpzOrigin, unsigned int& nProbableChainLength, CPrimalityTestParams& testParams, int base)
{
mpz_class& mpzOriginMinusOne = testParams.mpzOriginMinusOne;
mpz_class& mpzOriginPlusOne = testParams.mpzOriginPlusOne;
nProbableChainLength = (base-base%2) << nFractionalBits;
base /= 2;
int X = (1 << base) - 1;
// Fermat test for origin-1 first
mpzOriginMinusOne = mpzOrigin - 1;
if (base > 0) {
mpzOriginMinusOne <<= base;
mpzOriginMinusOne += X;
}
if (!FermatProbablePrimalityTestFast(mpzOriginMinusOne, nProbableChainLength, testParams, true))
return;
TargetIncrementLength(nProbableChainLength);
// Fermat test for origin+1
mpzOriginPlusOne = mpzOrigin + 1;
if (base > 0) {
mpzOriginPlusOne <<= base;
mpzOriginPlusOne -= X;
}
if (!FermatProbablePrimalityTestFast(mpzOriginPlusOne, nProbableChainLength, testParams, true))
return;
TargetIncrementLength(nProbableChainLength);
// Euler-Lagrange-Lifchitz test for the following numbers in chain
for (unsigned int nChainSeq = base+2; true; nChainSeq += 2)
{
mpzOriginMinusOne <<= 1;
mpzOriginMinusOne++;
bool fFastFail = nChainSeq < 4;
if (!EulerLagrangeLifchitzPrimalityTestFast(mpzOriginMinusOne, true, nProbableChainLength, testParams, fFastFail))
break;
TargetIncrementLength(nProbableChainLength);
mpzOriginPlusOne <<= 1;
mpzOriginPlusOne--;
if (!EulerLagrangeLifchitzPrimalityTestFast(mpzOriginPlusOne, false, nProbableChainLength, testParams, fFastFail))
break;
TargetIncrementLength(nProbableChainLength);
}
}
bool ProbablePrimeChainTestFast(const mpz_class& mpzPrimeChainOrigin, CPrimalityTestParams& testParams, int base)
{
const unsigned int nBits = testParams.nBits;
const unsigned int nCandidateType = testParams.nCandidateType;
mpz_class& mpzOriginMinusOne = testParams.mpzOriginMinusOne;
mpz_class& mpzOriginPlusOne = testParams.mpzOriginPlusOne;
unsigned int& nChainLength = testParams.nChainLength;
nChainLength = 0;
// Test for Cunningham Chain of first kind
if (nCandidateType == 0)
{
mpzOriginMinusOne = mpzPrimeChainOrigin - 1;
ProbableCunninghamChainTestFast(mpzOriginMinusOne, true, nChainLength, testParams, base);
}
else if (nCandidateType == 1)
{
// Test for Cunningham Chain of second kind
mpzOriginPlusOne = mpzPrimeChainOrigin + 1;
ProbableCunninghamChainTestFast(mpzOriginPlusOne, false, nChainLength, testParams, base);
}
else if (nCandidateType == 2)
{
ProbableBiTwinChainTestFast(mpzPrimeChainOrigin, nChainLength, testParams, base);
}
return (nChainLength >= nBits);
}