27
27
*
28
28
* 31 bit version.
29
29
*
30
- * Improvement by Dave McGuigan:
31
- * Use squareAddModN31() instead of nested addModN(squareModN())
30
+ * Improvements by Dave McGuigan:
31
+ * 1. Use squareAddModN31() instead of nested addModN(squareModN())
32
+ * 2. reinitialize q before each inner loop
33
+ * 3. Compute the number of steps before each gcd by m=log(n)
32
34
*
33
35
* @author Tilman Neumann
34
36
*/
@@ -66,15 +68,20 @@ public int findSingleFactor(int nOriginal) {
66
68
int y = x0 ;
67
69
68
70
// Brent: "The probability of the algorithm failing because q_i=0 increases, so it is best not to choose m too large"
69
- final int m = 100 ;
71
+ // DM: failing is a bit strong. q(i)=0 happens often when there are small powers of small factor (i.e.5*5)
72
+ // This occurs because multiple instances of the factor are found with larger blocks. The loop below
73
+ // addresses that by re-doing the last block and checking each individual difference. The "failure" is
74
+ // is that larger blocks have more to re-do. Empirical testing indicates 2*log(N) is better than a fixed choice
75
+ // for large N. In 31 bit versions, it was determined just log(N) is best.
76
+ final int m = Math .max (8 , 32 - Integer .numberOfLeadingZeros (n )); // Don't want it too small
70
77
int r = 1 ;
71
- int q = 1 ;
72
78
do {
73
79
x = y ;
74
80
for (int i =1 ; i <=r ; i ++) {
75
81
y = squareAddModN31 (y , c );
76
82
}
77
83
int k = 0 ;
84
+ int q = 1 ;
78
85
do {
79
86
ys = y ;
80
87
final int iMax = Math .min (m , r -k );
@@ -85,6 +92,7 @@ public int findSingleFactor(int nOriginal) {
85
92
}
86
93
G = gcd .gcd (q , n );
87
94
// if q==0 then G==N -> the loop will be left and restarted with new x0, c
95
+ // after checking each diff separately in the loop below.
88
96
k += m ;
89
97
if (DEBUG ) LOG .debug ("r = " + r + ", k = " + k );
90
98
} while (k <r && G ==1 );
0 commit comments