Skip to content

Commit 0f3bcaf

Browse files
committed
new Pollard-Rho variant by Dave McGuigan
1 parent 427d459 commit 0f3bcaf

File tree

2 files changed

+102
-8
lines changed

2 files changed

+102
-8
lines changed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,98 @@
1+
/*
2+
* java-math-library is a Java library focused on number theory, but not necessarily limited to it. It is based on the PSIQS 4.0 factoring project.
3+
* Copyright (C) 2018-2025 Tilman Neumann - [email protected]
4+
*
5+
* This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License
6+
* as published by the Free Software Foundation; either version 3 of the License, or (at your option) any later version.
7+
*
8+
* This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied
9+
* warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details.
10+
*
11+
* You should have received a copy of the GNU General Public License along with this program;
12+
* if not, see <http://www.gnu.org/licenses/>.
13+
*/
14+
package de.tilman_neumann.jml.factor.pollardRho;
15+
16+
import static de.tilman_neumann.jml.base.BigIntConstants.*;
17+
18+
import java.math.BigInteger;
19+
import java.security.SecureRandom;
20+
21+
import org.apache.logging.log4j.Logger;
22+
import org.apache.logging.log4j.LogManager;
23+
24+
import de.tilman_neumann.jml.factor.FactorAlgorithm;
25+
26+
/**
27+
* An improvement of the original Pollard-Rho method by Dave McGuigan,
28+
* using a second loop like in Pollard-Rho-Brent.
29+
*
30+
* @author Dave McGuigan
31+
*/
32+
public class PollardRhoTwoLoops extends FactorAlgorithm {
33+
private static final Logger LOG = LogManager.getLogger(PollardRhoTwoLoops.class);
34+
private static final boolean DEBUG = false;
35+
private static final SecureRandom RNG = new SecureRandom();
36+
37+
private BigInteger N;
38+
39+
@Override
40+
public String getName() {
41+
return "PollardRhoTwoLoops";
42+
}
43+
44+
@Override
45+
public BigInteger findSingleFactor(BigInteger N) {
46+
this.N = N;
47+
BigInteger gcd;
48+
int bitLength = N.bitLength();
49+
// get random x0 from [0, N-1]
50+
BigInteger x = new BigInteger(bitLength, RNG);
51+
if (x.compareTo(N)>=0) x=x.subtract(N);
52+
BigInteger xx;
53+
54+
do {
55+
// get random c from [0, N-1]
56+
BigInteger c = new BigInteger(bitLength, RNG);
57+
if (c.compareTo(N)>=0) c=c.subtract(N);
58+
59+
xx = x;
60+
BigInteger xs;
61+
BigInteger xxs;
62+
BigInteger prod = I_1;
63+
int m = 100;
64+
do {
65+
xs = x;
66+
xxs = xx;
67+
for(int i=0; i<m; i++) {
68+
x = squareAddModN(x, c);
69+
xx = squareAddModN(xx, c);
70+
xx = squareAddModN(xx, c);
71+
prod = prod.multiply(x.subtract(xx)).mod(N);
72+
}
73+
gcd = prod.gcd(N);
74+
} while(gcd.equals(I_1));
75+
if (gcd.equals(N)) {
76+
do {
77+
xs = squareAddModN(xs, c);
78+
xxs = squareAddModN(xxs, c);
79+
xxs = squareAddModN(xxs, c);
80+
gcd = N.gcd(xs.subtract(xxs));
81+
} while (gcd.equals(I_1));
82+
}
83+
// leave loop if factor found; otherwise continue with a new random c
84+
} while (gcd.equals(N));
85+
if (DEBUG) LOG.debug("Found factor of " + N + " = " + gcd);
86+
return gcd;
87+
}
88+
89+
/**
90+
* Square and add modulo N, with <code>a, b < N</code>.
91+
* @param y
92+
* @param c
93+
* @return () mod N
94+
*/
95+
private BigInteger squareAddModN(BigInteger y, BigInteger c) {
96+
return y.multiply(y).add(c).mod(N);
97+
}
98+
}

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

+4-8
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
/*
22
* java-math-library is a Java library focused on number theory, but not necessarily limited to it. It is based on the PSIQS 4.0 factoring project.
3-
* Copyright (C) 2018-2024 Tilman Neumann - [email protected]
3+
* Copyright (C) 2018-2025 Tilman Neumann - [email protected]
44
*
55
* This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License
66
* as published by the Free Software Foundation; either version 3 of the License, or (at your option) any later version.
@@ -23,14 +23,10 @@
2323
import de.tilman_neumann.jml.gcd.Gcd31;
2424

2525
/**
26-
* 31-bit implementation of Pollard's Rho method with improvements by Dave McGuigan.
26+
* An improvement of the original Pollard-Rho method by Dave McGuigan,
27+
* using a second loop like in Pollard-Rho-Brent.
2728
*
28-
* Most noteworthy, Dave came up with the second loop in this variant.
29-
*
30-
* Other improvements by Dave used here:
31-
* 1. Use squareAddModN31() instead of nested addModN(squareModN())
32-
* 2. Compute the number of steps before each gcd by m=log(n)
33-
* 3. Use faster "mulMod"
29+
* 31-bit version.
3430
*
3531
* @author Dave McGuigan
3632
*/

0 commit comments

Comments
 (0)