1/*==============================================================================
  2| 
  3|  NAME     
  4|
  5|     ppFactor.hpp
  6|
  7|  DESCRIPTION   
  8|
  9|     Header for integer factoring classes.
 10|
 11|     User manual and technical documentation are described in detail in my web page at
 12|     http://seanerikoconnor.freeservers.com/Mathematics/AbstractAlgebra/PrimitivePolynomials/overview.html
 13|
 14|  LEGAL
 15|
 16|     Primpoly Version 16.3 - A Program for Computing Primitive Polynomials.
 17|     Copyright (C) 1999-2025 by Sean Erik O'Connor.  All Rights Reserved.
 18|
 19|     This program is free software: you can redistribute it and/or modify
 20|     it under the terms of the GNU General Public License as published by
 21|     the Free Software Foundation, either version 3 of the License, or
 22|     (at your option) any later version.
 23|
 24|     This program is distributed in the hope that it will be useful,
 25|     but WITHOUT ANY WARRANTY; without even the implied warranty of
 26|     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
 27|     GNU General Public License for more details.
 28|
 29|     You should have received a copy of the GNU General Public License
 30|     along with this program.  If not, see http://www.gnu.org/licenses/.
 31|     
 32|     The author's address is seanerikoconnor!AT!gmail!DOT!com
 33|     with the !DOT! replaced by . and the !AT! replaced by @
 34|
 35==============================================================================*/
 36
 37// Wrap this header file to prevent duplication if it is included
 38// accidentally more than once.
 39#ifndef __PPFACTOR_H__
 40#define __PPFACTOR_H__
 41
 42
 43/*=============================================================================
 44|
 45| NAME
 46|
 47|     FactorError           General factor error, including internal memory errors.
 48|     FactorRangeError      Input range error.
 49|
 50| DESCRIPTION
 51|
 52|     Exception classes for the Factor class
 53|     derived from the STL exception class runtime_error.
 54|
 55+============================================================================*/
 56
 57class FactorError : public runtime_error
 58{
 59    public:
 60        // Throw with error message, file name and line number.
 61         FactorError( const string & description, const string & file, const int & line )
 62         : runtime_error( description + " in file " + file + " at line " + to_string(line) )
 63         {
 64         } ;
 65
 66        // Throw with an error message.
 67        FactorError( const string & description )
 68			: runtime_error( description )
 69        {
 70        } ;
 71
 72        // Default throw with no error message.
 73        FactorError()
 74			: runtime_error( "Factor error:  " )
 75        {
 76        } ;
 77
 78} ; // end class BigIntMathError
 79
 80
 81class FactorRangeError : public FactorError
 82{
 83    public:
 84        // Throw with error message, file name and line number.
 85        FactorRangeError( const string & description, const string & file, const int & line )
 86        : FactorError( description + " in file " + file + " at line " + to_string(line) )
 87        {
 88        };
 89    
 90        // Throw with an error message.
 91        FactorRangeError( const string & description )
 92			: FactorError( description )
 93        {
 94        } ;
 95
 96        // Default throw with no error message.
 97        FactorRangeError()
 98			: FactorError( "Factor range error:  " )
 99        {
100        } ;
101
102} ; // end class FactorRangeError
103
104
105/*=============================================================================
106 |
107 | NAME
108 |
109 |     PrimeFactor
110 |
111 | DESCRIPTION
112 |
113 |     Class to neatly package up unique prime factors to powers.
114 |
115 +============================================================================*/
116
117template <typename IntType>
118class PrimeFactor
119{
120    public:
121        explicit PrimeFactor( IntType prime = static_cast<IntType>( 0u ), int count = 0 )
122            : prime_( prime )
123            , count_( count )
124        {
125        } ;
126
127        ~PrimeFactor()
128        {
129        } ;
130
131        PrimeFactor( const PrimeFactor & factor )
132            : prime_( factor.prime_ )
133            , count_( factor.count_ )
134        {
135        }
136
137        PrimeFactor & operator=( const PrimeFactor & factor )
138        {
139            // Check for assigning to oneself:  just pass back a reference to the unchanged object.
140            if (this == &factor)
141                return *this ;
142
143            prime_ = factor.prime_ ;
144            count_ = factor.count_ ;
145			
146			return *this ;
147        } ;
148    
149        // Print function for a factor.
150        friend ostream & operator<<( ostream & out, const PrimeFactor & factor )
151        {
152            out << factor.prime_ << " ^ " << factor.count_ << " " ;
153        
154            return out ;
155        }
156
157    // Allow direct access to this simple data type for convenience.
158    public:
159        IntType prime_ ;
160        int     count_ ;
161} ;
162
163
164/*=============================================================================
165 |
166 | NAME
167 |
168 |     CompareFactor
169 |
170 | DESCRIPTION
171 |
172 |     Class to sort unique prime factors to powers into order by prime size.
173 |
174 +============================================================================*/
175
176template <typename IntType>
177class CompareFactor
178{
179    public:
180        bool operator()( const PrimeFactor<IntType> & s1, const PrimeFactor<IntType> & s2 )
181        {
182            return s1.prime_ < s2.prime_ ;
183        }
184} ;
185
186
187/*=============================================================================
188 |
189 | NAME
190 |
191 |     Unit
192 |
193 | DESCRIPTION
194 |                                                  0     e
195 |     Class to check for unit factors of the form p  or 1  = 1
196 |
197 +============================================================================*/
198
199template <typename IntType>
200class Unit
201{
202    public:
203        bool operator()( const PrimeFactor<IntType> & s )
204        {
205            return s.count_ == 0u || s.prime_ == static_cast<IntType>( 1u ) ;
206        }
207} ;
208
209
210/*=============================================================================
211|
212| NAME
213|
214|     Factorization
215|
216| DESCRIPTION
217|
218|     Class for single and multiprecision factoring.
219|
220| EXAMPLE
221|
222|     ppuint p{ 2 } ;
223|     ppuint n{ 4 } ;
224|     BigInt p_to_n_minus_1 { power( p, 4 ) - 1 } ;
225|
226|                                       n                                         n
227|     This will give us the factors of p - 1.  We can pass in either p and n or (p  - 1), but p and n might be
228|     the better choice for really large numbers because then we can look up the factorization in a table instead
229|     of computing it.
230|
231|     Factorization<BigInt> factors_of_p_to_n_minus_1( p_to_n_minus_1, FactoringAlgorithm::Automatic, p, n ) ;
232|
233|     And so will this for a smaller sized integer.
234|
235|     Factorization<ppuint> fact( power( p, 4 ) - 1 ) ;
236|
237|     int numDistinctFactors = fact.num() ;
238|     for (int i = 0 ;  i < numDistinctFactors ;  ++i)
239|     {
240|         BigInt primeFactor  = fact[ i ] ;
241|         int    multiplicity = fact.multiplicity( i ) ;
242|     }
243|
244| NOTES
245|
246|     The member functions and friends are documented in detail ppBigInt.cpp
247|
248+============================================================================*/
249
250// Different flavors of factoring algorithm.
251enum class FactoringAlgorithm
252{ 
253    Automatic,
254    TrialDivisionAlgorithm,
255    pollardRhoAlgorithm,
256    FactorTable
257} ;
258
259
260// We need both single precision and multi-precision factoring, so use
261// a template with an integer type.
262template <typename IntType>
263class Factorization
264{
265    public:
266		// Factor n into distinct primes.  Default constructor factors nothing.
267        Factorization( const IntType num = static_cast<IntType>( 1u ),
268                       FactoringAlgorithm type = FactoringAlgorithm::Automatic, ppuint p = 0, ppuint m = 0 ) ;
269
270       ~Factorization() ;
271	   
272        // Copy constructor.
273        Factorization( const Factorization<IntType> & f ) ;
274		
275        // Assignment operator.
276        Factorization<IntType> & operator=( const Factorization<IntType> & g ) ;
277
278	    // Return the number of distinct factors.
279		ppuint numDistinctFactors() const ;
280
281        // Return the ith prime factor along with its multiplicity as an lvalue so we can change either.
282		PrimeFactor<IntType> & operator[]( int i ) ;
283    
284        // Return the ith prime factor.
285        IntType primeFactor( int i ) const ;
286
287        // Return the multiplicity of the ith prime factor.
288        int multiplicity( int i ) const ;
289        
290        // True if p  | (p - 1).
291        //          i
292        bool skipTest( ppuint p, int i ) const ;
293
294        // Factoring with tables.
295        bool factorTable( ppuint p, ppuint n ) ;
296
297        //                                     ---
298        // Factoring by trial division up to \/ n
299        void trialDivision() ;
300
301        // Fast probabilistic factoring method.
302        bool pollardRho( const IntType & c = static_cast<IntType>( 2u )) ;
303
304        // Return a vector of only the distinct prime factors.
305        vector<IntType> getDistinctPrimeFactors() ;
306        
307    // Allow direct access to this simple data type for convenience.
308    public:
309        OperationCount statistics_ ;
310
311    private:
312        // The unfactored remainder.
313        IntType n_ ;
314
315		// Total number of distinct factors.
316		ppuint numFactors_ ;
317
318        // Array of distinct prime factors of n with their multiplicities.
319		vector< PrimeFactor<IntType> > factor_ ;
320
321        // Distinct prime factors.
322        vector<IntType> distinctPrimeFactors_ ;
323} ;
324
325
326/*=============================================================================
327|
328| NAME
329|
330|     Primality
331|
332| DESCRIPTION
333|
334|     What confidence a number is prime?
335|
336+============================================================================*/
337
338enum class Primality
339{
340    Prime = 0,
341    Composite,
342    ProbablyPrime, 
343    Undefined
344
345}  ;
346
347
348/*=============================================================================
349 |
350 | NAME
351 |
352 |     isProbablyPrime
353 |
354 | DESCRIPTION
355 |
356 |     True if n is likely to be prime.  Tests on a random integer x.
357 |
358 +============================================================================*/
359
360template <typename IntType>
361Primality isProbablyPrime( const IntType & n, const IntType & x ) ;
362
363
364/*=============================================================================
365 |
366 | NAME
367 |
368 |     UniformRandomNumber
369 |
370 | DESCRIPTION
371 |
372 |     Return a uniform random integer in [0, range).
373 |
374 | NOTES
375 |
376 |     Use the JKISS random number generator from the article,
377 |         Good Practice in (Pseudo) Random Number Generation for Bioinformatics Applications 
378 |         by David Jones, UCL Bioinformatics Group (E-mail: d.jones@cs.ucl.ac.uk) 
379 |         (Last revised May 7th 2010)
380 |
381 +============================================================================*/
382
383template <typename IntType>
384class UniformRandomIntegers
385{
386    public:
387        // Seed the JKISS generator parameters x, y, z, c, and set up the range of the generator [0, range)
388        explicit UniformRandomIntegers( const IntType range )
389        : range( range )
390        {
391            ++num_of_initializations ;
392            
393            // Reseed at the first call to this class and only occasionally thereafter because it's slow.
394            if (num_of_initializations % how_often_to_reseed == 0)
395            {
396                x = trueRandomNumberFromDevice() ;
397
398                // Seed y must not be zero!
399                while ( !(y = trueRandomNumberFromDevice()) ) ;
400                
401                z = trueRandomNumberFromDevice() ;
402
403                // We don’t really need to set c as well but let's anyway...
404                // NOTE: offset c by 1 to avoid z = c = 0
405                c = trueRandomNumberFromDevice() % JKISS_MAGIC_NUM + 1 ; // Should be less than 698769069
406            }
407         }
408
409        // Destructor has nothing to do.
410        UniformRandomIntegers()
411        {
412        } ;
413
414        IntType rand()
415        {
416            ppuint32 kiss = jKiss() ;
417            
418            // [0, range) falls within our generator's range [0, JKISSMAX).
419            // To preserve a uniform distribution, map numbers in [0, JKISSMAX) down to [0, range) in the following way:
420            // Map [0, range), => [0, range)
421            // Map [range+1, 2 range) => [0, range)
422            // ...
423            // Map [(n-1) range + 1, n range) => [0, range)
424            // Discard numbers in [n range + 1, JKISSMAX) because mapping numbers in this interval < range would inject a non-uniform bias.
425            if (range < IntType( JKISSMAX ))
426            {
427                // It's safe to down convert the integer type.
428                ppuint32 range2{ static_cast<ppuint32>( range ) } ;
429                
430                // Discard the random number unless it falls in a multiple of the range.
431                // Retry with a new random number.  I hope we don't get an infinite loop!
432                ppuint32 within_multiple_of_range{ JKISSMAX - (JKISSMAX % static_cast<ppuint32>(range)) } ;
433                while (kiss > within_multiple_of_range)
434                    kiss = jKiss() ;
435                kiss = jKiss() % range2 ;
436            }
437            // If the range is larger than the maximum of our generator, do nothing.  We don't want to scale up
438            // and have a non-uniform distribution.
439            return static_cast<IntType>( kiss ) ;
440        } ;
441
442        private:
443            static int num_of_initializations ;
444            static const int how_often_to_reseed = 10000 ;
445            static const int JKISS_MAGIC_NUM = 698769068 ;
446    
447            // Default seeds for JKISS.
448            ppuint32 x{ 123456789 } ;
449            ppuint32 y{ 987654321 } ;
450            ppuint32 z{  43219876 } ;
451            ppuint32 c{   6543217 } ;
452
453            // Top end of the jKiss() generator range.
454            const ppuint32 JKISSMAX { numeric_limits<ppuint32>::max() } ;
455    
456            // Return uniform random numbers in the range [0, range).
457            IntType range ;
458
459            // Read from the pseudo device /dev/urandom which return true random integers.
460            unsigned int trueRandomNumberFromDevice()
461            {
462                // The device /dev/urandom only returns bytes, so grab four at a time to make on 32-bit unsigned integer.
463                union RandomDeviceByteToInteger
464                {
465                    ppuint32 integer32 ;
466                    char bytes4[ 4 ] ;
467                } ;
468                
469                RandomDeviceByteToInteger numFromRandomDev ;
470                
471                ifstream fin( "/dev/urandom", ifstream::binary ) ;
472                if (fin.good())
473                {
474                    fin.read( numFromRandomDev.bytes4, 4 ) ;
475                    
476                    if (!fin)
477                    {
478                        ostringstream os ;
479                        os << "Cannot read from random device /dev/urandom" ;
480                        throw FactorError( os.str(), __FILE__, __LINE__ ) ;
481                    }
482                }
483                else
484                {
485                    ostringstream os ;
486                    os << "Cannot open random device /dev/urandom" ;
487                    throw FactorError( os.str(), __FILE__, __LINE__ ) ;
488                }
489
490                return numFromRandomDev.integer32 ;
491            }
492
493            // 32-bit integer random number generator.
494            ppuint32 jKiss()
495            {
496                x = 314527869 * x + 1234567 ;
497
498                y ^= (y <<  5) ;
499                y ^= (y >>  7) ;
500                y ^= (y << 22) ;
501
502                ppuint t{ 4294584393ULL * z + c } ;  // This must be a 64-bit integer type.
503                c = t >> 32 ;
504                z = static_cast<ppuint32>( t ) ;
505
506                return x + y + z ;
507            }
508} ;
509
510
511/*==============================================================================
512|                     Static Class Variables Initialization                    |
513==============================================================================*/
514
515// Static class variables must be initialized outside the class.
516
517template <typename IntType> int UniformRandomIntegers<IntType>::num_of_initializations = 0 ;
518
519
520/*=============================================================================
521|
522| NAME
523|
524|     isAlmostSurelyPrime
525|
526| DESCRIPTION
527|
528|     True if n is probabilistically prime.
529|
530+============================================================================*/
531
532template <typename IntType>
533bool isAlmostSurelyPrime( const IntType & n ) ;
534
535#endif // __PPFACTOR_H__