Skip to content

Commit 6ba39a4

Browse files
committed
fix 31bit performance, improve safety by argument check
1 parent 3958ac4 commit 6ba39a4

File tree

1 file changed

+19
-15
lines changed

1 file changed

+19
-15
lines changed

src/main/java/de/tilman_neumann/jml/factor/pollardRho/PollardRhoBrent31.java

+19-15
Original file line numberDiff line numberDiff line change
@@ -34,7 +34,7 @@ public class PollardRhoBrent31 extends FactorAlgorithm {
3434
private static final boolean DEBUG = false;
3535
private static final SecureRandom RNG = new SecureRandom();
3636

37-
private int N;
37+
private int n;
3838

3939
private Gcd31 gcd = new Gcd31();
4040

@@ -45,17 +45,21 @@ public String getName() {
4545

4646
@Override
4747
public BigInteger findSingleFactor(BigInteger N) {
48-
return BigInteger.valueOf(findSingleFactor(N.intValue()));
48+
if (N.bitLength() > 31) { // this check should be negligible in terms of performance
49+
throw new IllegalArgumentException("N = " + N + " has " + N.bitLength() + " bit, but PollardRho31 only supports arguments <= 31 bit");
50+
}
51+
int factorInt = findSingleFactor(N.intValue());
52+
return BigInteger.valueOf(factorInt);
4953
}
5054

51-
public int findSingleFactor(int N) {
52-
this.N = N;
55+
public int findSingleFactor(int nOriginal) {
56+
this.n = nOriginal<0 ? -nOriginal : nOriginal; // RNG.nextInt(n) below would crash for negative arguments
5357
int G;
5458
int ys, x;
5559
do {
5660
// start with random x0, c from [0, N-1]
57-
int c = RNG.nextInt(N);
58-
int x0 = RNG.nextInt(N);
61+
int c = RNG.nextInt(n);
62+
int x0 = RNG.nextInt(n);
5963
int y = x0;
6064

6165
// Brent: "The probability of the algorithm failing because q_i=0 increases, so it is best not to choose m too large"
@@ -74,26 +78,26 @@ public int findSingleFactor(int N) {
7478
for (int i=1; i<=iMax; i++) {
7579
y = addModN(squareModN(y), c);
7680
final long diff = x<y ? y-x : x-y;
77-
q = (int) ((diff*q) % N);
81+
q = (int) ((diff*q) % n);
7882
}
79-
G = gcd.gcd(q, N);
83+
G = gcd.gcd(q, n);
8084
// if q==0 then G==N -> the loop will be left and restarted with new x0, c
8185
k += m;
8286
if (DEBUG) LOG.debug("r = " + r + ", k = " + k);
8387
} while (k<r && G==1);
8488
r <<= 1;
8589
if (DEBUG) LOG.debug("r = " + r + ", G = " + G);
8690
} while (G==1);
87-
if (G==N) {
91+
if (G==n) {
8892
do {
8993
ys = addModN(squareModN(ys), c);
9094
int diff = x<ys ? ys-x : x-ys;
91-
G = gcd.gcd(diff, N);
95+
G = gcd.gcd(diff, n);
9296
} while (G==1);
9397
if (DEBUG) LOG.debug("G = " + G);
9498
}
95-
} while (G==N);
96-
if (DEBUG) LOG.debug("Found factor of " + N + " = " + G);
99+
} while (G==n);
100+
if (DEBUG) LOG.debug("Found factor of " + nOriginal + " = " + G);
97101
return G;
98102
}
99103

@@ -104,8 +108,8 @@ public int findSingleFactor(int N) {
104108
* @return (a+b) mod N
105109
*/
106110
private int addModN(int a, int b) {
107-
int sum = a+b;
108-
return sum<N ? sum : sum-N;
111+
long sum = a + (long)b; // long is needed for the addition of 31 bit numbers
112+
return (int) (sum<n ? sum : sum-n);
109113
}
110114

111115
/**
@@ -114,6 +118,6 @@ private int addModN(int a, int b) {
114118
* @return
115119
*/
116120
private int squareModN(long x) {
117-
return (int) ((x * x) % N);
121+
return (int) ((x * x) % n);
118122
}
119123
}

0 commit comments

Comments
 (0)