1/*==============================================================================
   2|
   3|  NAME
   4|
   5|     ppArith.cpp
   6|
   7|  DESCRIPTION
   8|
   9|     Miscellaneous integer multiple precision math routines.
  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/*------------------------------------------------------------------------------
  38|                                Include Files                                 |
  39------------------------------------------------------------------------------*/
  40
  41#include <cstdlib>      // abort()
  42#include <iostream>     // Basic stream I/O.
  43#include <new>          // set_new_handler()
  44#include <cmath>        // Basic math functions e.g. sqrt()
  45#include <complex>      // Complex data type and operations.
  46#include <fstream>      // File stream I/O.
  47#include <sstream>      // String stream I/O.
  48#include <vector>       // STL vector class.
  49#include <string>       // STL string class.
  50#include <algorithm>    // Iterators.
  51#include <stdexcept>    // Exceptions.
  52#include <cassert>      // assert()
  53
  54using namespace std ;
  55
  56/*------------------------------------------------------------------------------
  57|                                PP Include Files                              |
  58------------------------------------------------------------------------------*/
  59
  60#include "Primpoly.hpp"		  // Global functions.
  61#include "ppArith.hpp"		  // Basic arithmetic functions.
  62#include "ppBigInt.hpp"         // Arbitrary precision integer arithmetic.
  63#include "ppOperationCount.hpp" // OperationCount collection for factoring and poly finding.
  64#include "ppFactor.hpp"         // Prime factorization and Euler Phi.
  65#include "ppPolynomial.hpp"	  // Polynomial operations and mod polynomial operations.
  66#include "ppParser.hpp"	      // Parsing of polynomials and I/O services.
  67#include "ppUnitTest.hpp"       // Complete unit test.
  68
  69/*=============================================================================
  70 | 
  71 | NAME
  72 | 
  73 |     ModP::operator()
  74 | 
  75 | DESCRIPTION
  76 | 
  77 |     Computes k = n mod p where 0 <= k < p for both positive and negative 
  78 |     arguments n for p >= 1.
  79 |     Takes a surprising proportion of the total compute time according to the 
  80 |     profiler, so worth optimizing.
  81 | 
  82 | EXAMPLE
  83 | 
  84 |     The C language gives -5 % 3 = - ( -(-5) mod 3 ) = -( 5 mod 3 ) = -2.
  85 |     The result is nonzero, so we add p=3 to give 1.
  86 | 
  87 |     C computes -3 % 3 = - ( -(-3) mod 3  ) = 0, which we leave unchanged.
  88 | 
  89 | METHOD
  90 | 
  91 |      For n >= 0, the C language defines r = n mod p by the equation
  92 | 
  93 |          n = kp + r    0 <= r < p
  94 | 
  95 |      but when n < 0, C returns the quantity
  96 | 
  97 |          r = - ( (-n) mod p )
  98 | 
  99 |      To put the result into the correct range 0 to p-1, add p to r if
 100 |      r is non-zero.
 101 | 
 102 |      By the way, dear old FORTRAN's MOD function does the same thing.
 103 | 
 104 +============================================================================*/
 105
 106template <typename UIntType, typename SIntType>
 107inline UIntType ModP<UIntType,SIntType>::operator()( SIntType n )
 108{
 109    if (p_ <= 0)
 110    {
 111        ostringstream os ;
 112        os << "ModP::operator()  The modulus p = " << p_ << " <= 0" ;
 113        throw ArithModPError( os.str(), __FILE__, __LINE__ ) ;
 114    }
 115    
 116    if (n >= 0)
 117    {
 118        // Avoid integer division in this special case.
 119        if (p_ == 2)
 120            return n - ((n >> 1) << 1) ;
 121        else
 122            return n % p_ ;
 123    }
 124    // Both types must be signed if we want n % p to compute correctly for n < 0.
 125     else
 126        return( (n % static_cast<SIntType>(p_)) + p_ ) ;
 127}
 128
 129/*=============================================================================
 130|
 131| NAME
 132|
 133|     PowerMod::operator()
 134|
 135| DESCRIPTION
 136|
 137|               n 
 138|     Compute  a  (modulo p)  for generic integer types.
 139|                                                           0
 140|     where a >= 0, p >= 2, n >= 0.  Other cases including 0
 141|     will throw an ArithModPError.
 142|
 143|     We have two versions:  one is generic for any integer type, and
 144|     the other is specialized for ppuint.
 145|
 146| EXAMPLE
 147|
 148|     BigInt a = 2 ;
 149|     BigInt n = 3 ;
 150|     BigInt p = 7 ;
 151|     try 
 152|     {
 153|         PowerMod<BigInt> powermod( p ) ;
 154|         BigInt pwr = powermod( a, n ) ;
 155|     } 
 156|     catch( ArithModPError e )
 157|     { 
 158|         cout << "should have gotten pwr = 1\n" ;
 159|     }
 160|
 161| METHOD
 162|
 163|     Multiplication by repeated squaring.
 164|
 165+============================================================================*/
 166
 167// Generic.
 168template <typename IntType>
 169IntType PowerMod<IntType>::operator()( const IntType & a, const IntType & n )
 170{
 171    IntType a1 = a ;
 172
 173    //  Out of range conditions.
 174    if (a  <  static_cast<IntType>( 0u ) || 
 175        n  <  static_cast<IntType>( 0u ) || 
 176        p_ <= static_cast<IntType>( 1u ) || 
 177       (a  == static_cast<IntType>( 0u ) && n == static_cast<IntType>( 0u )))
 178    {
 179        ostringstream os ;
 180        os << "PowerMod::operator()  One or more parameters out of range:  a = "
 181           << a << " n = " << n  << " p_ = " << p_ ;
 182        throw ArithModPError( os.str(),  __FILE__, __LINE__ ) ;
 183    }
 184
 185    //  Quick return for 0 ^ n, a^0 and a^1.
 186    if (a == static_cast<IntType>( 0u ))
 187        return static_cast<IntType>( 0u ) ;
 188
 189    if (n == static_cast<IntType>( 0u ))
 190        return static_cast<IntType>( 1u ) ;
 191
 192    if (n == static_cast<IntType>( 1u ))
 193        return a % static_cast<IntType>( p_ ) ;
 194
 195    int bitNum = n.maxBitNumber() ; // Index to highest bit.
 196
 197    #ifdef DEBUG_PP_ARITH
 198    cout << "initial max bitNum = " << bitNum << endl ;
 199    cout << "a = " << a << endl ;
 200    #endif
 201    
 202    // Find the index of the leading 1 bit.
 203    while (!n.testBit( bitNum ))
 204        --bitNum ;
 205
 206    #ifdef DEBUG_PP_ARITH
 207    cout << "after skipping leading 0 bits, bitNum = " << bitNum << endl ;
 208    #endif
 209
 210    if (bitNum == -1)
 211    {
 212        ostringstream os ;
 213        os << "PowerMod::operator() " << "bitNum == -1" ;
 214        throw ArithModPError( os.str(), __FILE__, __LINE__ ) ;
 215    }
 216
 217    #ifdef DEBUG_PP_ARITH
 218    cout << "\nAfter skipping zero bits, bitNum = " << bitNum << endl ;
 219    #endif
 220
 221    //  Exponentiation by repeated squaring.  Discard the leading 1 bit.
 222    //  Thereafter, square for every 0 bit;  square and multiply by a for
 223    //  every 1 bit.
 224    while ( --bitNum >= 0 )
 225    {
 226        a1 = (a1 * a1) % static_cast<IntType>( p_ ) ; // Square mod p.
 227
 228        if (n.testBit( bitNum ))
 229            a1 = (a1 * a) % static_cast<IntType>( p_ ) ; // Times a mod p.
 230
 231        #ifdef DEBUG_PP_ARITH
 232        cout << "S " ;
 233        if (n.testBit( bitNum ))
 234            cout << "X " ;
 235        cout << "Bit num = " << bitNum << " a1 = " << a1 << endl ;
 236        #endif
 237    }
 238
 239    #ifdef DEBUG_PP_ARITH
 240        cout << "Out of the loop bitNum = " << bitNum << " a1 = " << a1 << endl ;
 241    #endif
 242
 243    return a1 ;
 244}
 245
 246
 247
 248/*=============================================================================
 249 | 
 250 | NAME
 251 | 
 252 |     addMod()
 253 | 
 254 | DESCRIPTION
 255 | 
 256 |     Computes (x + y) mod p for generic integer types, up to the maximum
 257 |     unsigned type.  We extend the range to the limit by testing and compensating
 258 |     for carries internally.
 259 |
 260 |     Example:
 261 |         a           = 4294967290
 262 |         n           = 4294967294
 263 |         (a + b) % n = 4294967286
 264 | 
 265 +============================================================================*/
 266
 267template <typename IntType>
 268IntType addMod( IntType a, IntType b, IntType n )
 269{
 270    #ifdef DEBUG_PP_ARITH
 271    cout << "addMod" << endl ;
 272    cout << "    sizeof IntType = " << 8 * sizeof( IntType ) << " bits" << endl ;
 273    cout << "    a = " << a << " b = " << b << " n = " << n << endl ;
 274    #endif
 275
 276    // Assume the positive arguments might not already be in the range [0, n).
 277    a %= n ;
 278    b %= n ;
 279
 280    // Add with carry.  Carry bit thrown away by the language.
 281    IntType c = a + b ;
 282
 283    #ifdef DEBUG_PP_ARITH
 284    cout << "    a mod n = " << a << " b mod n = " << b << endl ;
 285    cout << "    c = a + b (discarding carry bit) = " << c << endl ;
 286    #endif
 287
 288    // Test for a carry.  Most computer languages don't give us access to
 289    // the carry bit, so we must infer whether a carry occurred during addition.
 290    // Logically, 
 291    //     (no carry for a+b)  =>  a+b <= a && a+b <= b
 292    // The contrapositive is
 293    //      a+b < a || a+b < b => carry
 294    if (c < a || c < b)
 295    {
 296        // Lemma 1.  If a carry occured, n < c < 2n.
 297        // Pf.
 298        //                                       m
 299        //     When a carry occurs, c = a + b > 2  > n.  But 0 <= a, b < n.  So n < c and a + b < n + n = 2n.
 300        //
 301        // Lemma 2.  If a carry occured, we can simply compute c - n as usual, discarding the carry.
 302        // Pf.
 303        //     Since a carry occurred, we an write
 304        //                  m
 305        //     c = a + b = 2  + c'
 306        //
 307        //     Let's assume the computer hardware does subtraction in two's complement.
 308        //     Now use two's complement arithmetic on c' - n and see the results.
 309        //
 310        //     c' + TwosComplement( n ) <discarding carry bit> = 
 311        //
 312        //     c' + (~n + 1) <discarding carry bit> =
 313        //             m           m
 314        //     (c' + (2 - n)) mod 2  = 
 315        //           m   m     m           m          m           m
 316        //     (c - 2 + 2  + (2 - n)) mod 2  = (c + (2 - n)) mod 2 =
 317        //
 318        //     c + TwosComplement( n ) <discarding carry bit> =
 319        //
 320        //     c - n done the usual way in computer hardware.
 321        //
 322        // Th. If a carry occurs, c mod n = c - n.
 323        // Pf. See Lemma 1 and 2.
 324        //
 325        // e.g.  Let m = 4 bits and
 326        //
 327        //       a  =   1 1 0 1 = 13
 328        //       b  =   1 1 0 1 = 13
 329        //       n  =   1 1 1 0 = 14
 330        //
 331        //       c  = 1 1 0 1 0
 332        //       c' =   1 0 1 0
 333        //      ~n  =   0 0 0 1 + 1 = 
 334        //              0 0 1 0
 335        //       c' -  n = c' + ~n =
 336        //              1 0 1 0
 337        //           +  0 0 1 0
 338        //        =     1 1 0 0 = 12 = 26 mod 14
 339        
 340        c -= n ;
 341
 342        #ifdef DEBUG_PP_ARITH
 343        cout << "    Carry!" << " c < a = " << (c < a ) << "|| c < b = " << (c < b) << endl ;
 344        #endif
 345    }
 346    else
 347    {
 348        c %= n ;
 349
 350        #ifdef DEBUG_PP_ARITH
 351        cout << "    No carry" << endl ;
 352        #endif
 353    }
 354
 355    return c ;
 356}
 357
 358
 359
 360/*=============================================================================
 361 | 
 362 | NAME
 363 | 
 364 |     timesTwoMod()
 365 | 
 366 | DESCRIPTION
 367 | 
 368 |     Computes (2 * x) mod p for generic integer types, up to the maximum
 369 |     unsigned type.  We extend the range to the limit by testing and compensating
 370 |     for carries internally.
 371 |
 372 |     Example:
 373 |         a           = 10376293541461622782
 374 |         n           = 18446744073709551610
 375 |         (2 * a) % n =  2305843009213693954
 376 | 
 377 +============================================================================*/
 378
 379template <typename IntType>
 380IntType timesTwoMod( IntType a, IntType n )
 381{
 382    #ifdef DEBUG_PP_ARITH
 383    cout << "timesTwoMod" << endl ;
 384    cout << "    sizeof IntType = " << 8 * sizeof( IntType ) << " bits" << endl ;
 385    cout << "    a = " << a << " n = " << n << endl ;
 386    #endif
 387
 388    // Assume the positive argument might not already be in the range [0, n).
 389    a %= n ;
 390
 391    #ifdef DEBUG_PP_ARITH
 392    cout << "    a mod n = " << a << endl ;
 393    #endif
 394
 395    // High bit mask.
 396    IntType mask = ((IntType)1 << (8 * sizeof( IntType ) - 1)) ;
 397
 398    IntType c = (a << 1) ; // Lose the carry bit.
 399
 400    #ifdef DEBUG_PP_ARITH
 401    cout << "    mask = " << hex << mask << dec << endl ;
 402    cout << "    c = 2 a (losing the carry bit) = " << c << endl ;
 403    #endif
 404
 405    // High bit is set so (2 a) will have a carry.
 406    if (mask & a)
 407    {
 408        // See notes in function addMod() above for why this works correctly.
 409        c -= n ;
 410
 411        #ifdef DEBUG_PP_ARITH
 412        cout << "    Carry!" << endl ;
 413        #endif
 414    }
 415    else
 416    {
 417        c %= n ;
 418
 419        #ifdef DEBUG_PP_ARITH
 420        cout << "    No carry" << endl ;
 421        #endif
 422    }
 423
 424    return c ;
 425}
 426
 427
 428
 429/*=============================================================================
 430 | 
 431 | NAME
 432 | 
 433 |     multiplyMod()
 434 | 
 435 | DESCRIPTION
 436 | 
 437 |     Computes (x * y) mod p for generic integer types, up to the maximum
 438 |     unsigned type.  We extend the range this far by testing and compensating
 439 |     for carries internally.
 440 |
 441 |     Multiply a * b mod n 
 442 |
 443 |                   m-1         m-2
 444 |     a b = a (b   2    + b    2    + ... + b  2 + b )
 445 |               m-1        m-2               1      0
 446 |
 447 |     Expanding using Horner's rule,
 448 |
 449 |     a b = (0 2 + a b   ) 2 + a b   ) 2 + ... + a b ) 2 + a b
 450 |                     m-1         m-2               1         0
 451 |
 452 |     where b  = 0 or 1
 453 |            i
 454 |     At each step call specialized functions to do r = 2 a mod n and r = r + a mod n
 455 |     which will work even when carries occur.
 456 |
 457 |     The process takes O( ln n ) operations.  An example,
 458 |
 459 |          a = 4294967290 
 460 |          b = 4294967290
 461 |          n = 4294967294
 462 |         (a * b) % n = 16
 463 | 
 464 |     Reference:  "Comments on 'A Computer Algorithm for Calculating the Product
 465 |     A B Modulo M'" K. R. Sloane, IEEE Transactions on Computers, Vol C-34, No. 3,
 466 |     March 1985.
 467 | 
 468 +============================================================================*/
 469
 470template<typename IntType>
 471IntType multiplyMod( const IntType a, const IntType b, const IntType n )
 472{
 473    // High bit mask.
 474    const int numBits = 8 * sizeof( IntType ) ;
 475    IntType mask = ((IntType)1 << (numBits - 1)) ;
 476
 477    IntType r { 0 } ;
 478
 479    for (int i = numBits-1 ;  i >= 0 ;  --i, mask >>= 1)
 480    {
 481        r = timesTwoMod( r, n ) ;
 482
 483        if (mask & b)
 484            r = addMod( r, a, n ) ;
 485    }
 486
 487    return r ;
 488}
 489
 490
 491
 492/*=============================================================================
 493 | 
 494 | NAME
 495 | 
 496 |     PowerMod::operator()
 497 | 
 498 | DESCRIPTION
 499 | 
 500 |     Specialized for ppuint type.
 501 | 
 502 +============================================================================*/
 503
 504template<>
 505ppuint PowerMod<ppuint>::operator()( const ppuint & a, const ppuint & n )
 506{
 507    //  mask = 1000 ... 000.  That is, all bits of mask are zero except for the
 508    //  most significant bit of the computer word holding its value.
 509    ppuint mask{ static_cast<ppuint>(1) << (8 * sizeof( ppuint ) - 1) } ;
 510    int  bit_count{ 0 } ;  // Number of bits in exponent to the right of the leading bit.
 511    ppuint n1{ n } ;
 512    ppuint product{ a }  ;
 513
 514    // Out of range conditions. 
 515    if (p_ <= 1 || (a == 0 && n1 == 0))
 516    {
 517        ostringstream os ;
 518        os << "PowerMod<ppuint>::operator() parameters are out of range:  a = " << a << " n = " << n  << " p_ = " << p_ ;
 519        throw ArithModPError( os.str(), __FILE__, __LINE__ ) ;
 520    }
 521
 522    //                    n   0      1
 523    //  Quick return for 0 , a  and a
 524    if (a == 0)
 525        return 0 ;
 526
 527    if (n == 0)
 528        return 1 ;
 529
 530    if (n == 1)
 531        return a % p_ ;
 532
 533    #ifdef DEBUG_PP_ARITH
 534    cout << "a                    = " << a << endl ;
 535    cout << "n                    = " << n << endl ;
 536    cout << "p                    = " << p_ << endl ;
 537    cout << "n1 (before shifting) = " << n1 << endl ;
 538    #endif
 539
 540    // Advance the leading bit of the exponent up to the word's left hand boundary.  
 541    // Count how many bits were to the right of the leading bit.
 542    while (! (n1 & mask))
 543    {
 544        n1 <<= 1 ;
 545        ++bit_count ;
 546    }
 547
 548    bit_count = (8 * sizeof( ppuint )) - bit_count ;
 549
 550    #ifdef DEBUG_PP_ARITH
 551    cout << "n1        = " << n1 << endl ;
 552    cout << "mask      = " << mask << endl ;
 553    cout << "bit_count = " << bit_count << endl ;
 554    #endif
 555
 556    // Exponentiation by repeated squaring.  Discard the leading 1 bit.
 557    // Thereafter, square for every 0 bit;  square and multiply by x for every 1 bit.
 558    while ( --bit_count > 0 )
 559    {
 560        #ifdef DEBUG_PP_ARITH
 561        cout << "product (before squaring) = " << product << " n1 = " << n1 << endl ;
 562        #endif
 563
 564        // Expose the next bit.
 565        n1 <<= 1 ;
 566
 567        // Out of range conditions. 
 568        if (product > BigInt::getBase() || a > BigInt::getBase())
 569        {
 570            // Square modulo p.
 571            product = multiplyMod( product, product, p_ ) ;
 572
 573            //  Leading bit is 1: multiply by a modulo p.
 574            if (n1 & mask)
 575                product = multiplyMod( a, product, p_ ) ;
 576        }
 577        else
 578        {
 579            // Square modulo p.
 580            product = (product * product) % p_ ;
 581
 582            //  Leading bit is 1: multiply by a modulo p.
 583            if (n1 & mask)
 584                product = (a * product) % p_ ;
 585        }
 586
 587        #ifdef DEBUG_PP_ARITH
 588        cout << "S " ;
 589        if (n1 & mask)
 590            cout << "X " ;
 591        cout << "product = " << product << " n1 = " << n1 << endl ;
 592        #endif
 593    }
 594
 595    return product ;
 596}
 597
 598
 599
 600/*=============================================================================
 601|
 602| NAME
 603|
 604|     IsPrimitiveRoot::operator()
 605|
 606| DESCRIPTION
 607|
 608|     Returns true if a is a primitive root of prime p, and false otherwise.
 609|     Throws ArithModPError if a < 1 or p < 2 or p is even.
 610|     p must be a prime number, but we only test that p is 2 or odd.
 611|
 612| EXAMPLE
 613|
 614|     ppuint p{ 7 } ;  ppuint a{ 3 } ;
 615|     IsPrimitiveRoot isRoot( p ) ;
 616|     try 
 617|     {
 618|        bool isPrim = isRoot( a ) ;
 619|     } 
 620|     catch( ArithModPError e )
 621|     {  
 622|         cout << "out of range" << endl ; 
 623|     }
 624|                                                     1      2      3
 625|     3 is a primitive root of the prime p = 7 since 3 = 3, 3 = 2, 3 = 6,
 626|      4      5                                       p-1    6
 627|     3 = 4, 3 = 5 (mod 7); all not equal to 1 until 3    = 3 = 1 (mod 7).
 628|
 629|     So 3 is a primitive root of 7 because it has maximal period.  a = 2
 630|                                                      3
 631|	  isn't a primitive root of p = 7 because already 2 = 1 (mod 7).
 632|
 633| METHOD
 634|
 635|    From number theory, a is a primitive root of the prime p iff
 636|     (p-1)/q
 637|    a        != 1 (mod p) for all prime divisors q of (p-1).
 638|                            (p-1)
 639|    Don't need to check if a     = 1 (mod p):  Fermat's little
 640|    theorem guarantees it.
 641|
 642+============================================================================*/
 643
 644bool IsPrimitiveRoot::operator()( ppuint a )
 645{
 646    PowerMod<ppuint> powermod( p_ ) ;
 647
 648    //  a = 0 (mod p);  Zero can't be a primitive root of p.
 649    if (a == 0)
 650        return false ;
 651
 652
 653    //  Error for out of range inputs, including p
 654    //  being an even number greater than 2.
 655    if (p_ < 2 || a < 1 || (p_ > 2 && (p_ % 2 == 0)))
 656    {
 657        ostringstream os ;
 658        os << "IsPrimitiveRoot::operator()  Inputs are out of range: p = " << p_ << " a = " << a ;
 659        throw ArithModPError( os.str(), __FILE__, __LINE__ ) ;
 660    }
 661
 662    //  Special cases:
 663    //  1 is the only primitive root of 2;  i.e. 1 generates the unit elements
 664    //  of GF( 2 );  2 is the only primitive root of 3, and 2 and 3 are the only
 665    //  primitive roots of 5.
 666    //
 667    if ( (p_ == 2  &&  a == 1) ||
 668         (p_ == 3  &&  a == 2) ||
 669         (p_ == 5  && (a == 2  || a == 3)) ||
 670         (p_ == 7  && (a == 3  || a == 5)) ||
 671         (p_ == 11 && (a == 2  || a == 6   || a == 7 || a == 8)) ||
 672         (p_ == 13 && (a == 2  || a == 6   || a == 7 || a == 11)))
 673    {
 674        return true ;
 675    }
 676
 677    //  Reduce a down modulo p.
 678    a = a % p_ ;
 679
 680    //  a = 0 (mod p);  Zero can't be a primitive root of p.
 681    if (a == 0)
 682        return false ;
 683
 684    //  Factor p-1 into distinct primes.
 685    Factorization<ppuint> factorization( p_ - 1 ) ;
 686
 687    //            p-1
 688    //           a    
 689    //  Test    ---    != 1 (mod p) for all primes q | (p-1).
 690    //           q
 691    //  If so, we have a primitive root of p, otherwise, we exit early.
 692    //
 693    for (unsigned int i = 0 ;  i < factorization.numDistinctFactors() ;  ++i)
 694    {
 695        if (powermod( a, (p_ - 1) / factorization.primeFactor( i )) == 1)
 696        {
 697            return false ;
 698        }
 699    }
 700
 701    return true ;
 702} 
 703
 704
 705
 706/*=============================================================================
 707 | 
 708 | NAME
 709 | 
 710 |     inverse_mod_p
 711 | 
 712 | DESCRIPTION
 713 | 
 714 |                                                             -1
 715 |      Find the inverse of u modulo p.  In other words, find u   (mod p)
 716 |      p >= 2.
 717 | 
 718 | EXAMPLE
 719 |                           -1
 720 |      For p=7, and u = 2, u   = 4 because 2 * 4 = 8 (mod 7) = 1.
 721 | 
 722 | METHOD
 723 | 
 724 |     Do extended Euclid's algorithm on u and v to find u1, u2, and u3 such that
 725 | 
 726 |         u u1 + v u2 = u3 = gcd( u, v ).
 727 | 
 728 |     Now let v = p = prime number, so gcd = u3 = 1.  Then we get
 729 | 
 730 |         u u1 + p ? = 1
 731 | 
 732 |     or u u1 = 0 (mod p) which makes u (mod p) the unique multiplicative
 733 |     inverse of u.
 734 | 
 735 +============================================================================*/
 736
 737ppsint InverseModP::operator()( ppsint u )
 738{
 739    ModP<ppsint,ppsint> mod( p_ ) ;
 740
 741    //  Do Euclid's algorithm to find the quantities.
 742	ppsint t1 = 0 ;
 743	ppsint t3 = 0 ;
 744	ppsint q  = 0 ;
 745
 746    ppsint u1 = 1 ;
 747    ppsint u3 = u ;
 748    ppsint v1 = 0 ;
 749    ppsint v3 = p_ ;
 750
 751	ppsint inv_v = 0 ;
 752
 753	while( v3 != 0)
 754	{
 755        // Casting to do a floor function.
 756        q = static_cast<ppsint>(u3 / v3) ;
 757
 758		t1 = u1 - v1 * q ;
 759		t3 = u3 - v3 * q ;
 760
 761		u1 = v1 ;
 762		u3 = v3 ;
 763
 764		v1 = t1 ;
 765		v3 = t3 ;
 766	}
 767
 768    inv_v = mod( u1 ) ;
 769
 770    //                       -1
 771	// Self check:  does u  u   = 1 (mod p)?
 772    //
 773	if ( mod( u * inv_v ) != 1)
 774	{
 775        ostringstream os ;
 776        os << "InverseModP::operator():  inverse self check failed:  u = " << u << " * u^(-1) = " << inv_v  << " != 1" ;
 777        throw ArithModPError( os.str(), __FILE__, __LINE__ ) ;
 778	}
 779
 780	return inv_v ;
 781}
 782
 783
 784/*=============================================================================
 785 | 
 786 | NAME
 787 | 
 788 |     constCoeffTest
 789 | 
 790 | DESCRIPTION
 791 | 
 792 |                   n
 793 |   Test if a = (-1)  a  (mod p) where a  is the constant coefficient
 794 |                      0                0
 795 |   of polynomial f(x) and a is the number from the orderR() test.
 796 |   Return true if we pass the test, false otherwise. 
 797 | 
 798 | EXAMPLE
 799 |                                  11     3
 800 |     Let p = 5, n = 11, f( x ) = x  + 2 x + 1, and a = 4.
 801 |     Thus a  = 1.
 802 |           0
 803 |     Then return true since
 804 | 
 805 |         11
 806 |     (-1)  * 1 (mod 5) = -1 (mod 5) = 4 (mod 5).
 807 | 
 808 | METHOD
 809 |                         n
 810 |     We test if (a - (-1) a ) mod p = 0.
 811 |                           0
 812 | 
 813 +============================================================================*/
 814
 815bool ArithModP::constCoeffTest( ppsint a, ppsint a0, int n )
 816{
 817    ppsint constant_coeff = a0 ;
 818
 819    ModP<ppuint,ppsint> mod( p_ ) ; // Initialize the functionoid.
 820
 821    if (n % 2 != 0)
 822        constant_coeff = -constant_coeff ;    // (-1)^n < 0 for odd n.
 823
 824    return( (mod( a - constant_coeff ) == 0) ? true : false ) ;
 825} 
 826
 827
 828
 829/*=============================================================================
 830 | 
 831 | NAME
 832 | 
 833 |     constCoeffIsPrimitiveRoot
 834 | 
 835 | DESCRIPTION
 836 | 
 837 |               n
 838 |   Test if (-1)  a  (mod p) is a primitive root of the prime p where
 839 |                  0
 840 |   a  is the constant term of the polynomial f(x).
 841 |    0
 842 | 
 843 | EXAMPLE
 844 |                                  11     3
 845 |     Let p = 7, n = 11, f( x ) = x  + 2 x + 4.  Thus a  = 4.
 846 |                                                      0
 847 |     Then return true since
 848 | 
 849 |         11
 850 |     (-1)  * 4 (mod 7) = -4 (mod 7) = 3 (mod 7), and 3 is a
 851 |                                          1      2      3
 852 |     primitive root of the prime 7 since 3 = 3, 3 = 2, 3 = 6,
 853 |      4      5                                       7-1
 854 |     3 = 4, 3 = 5 (mod 7); all not equal to 1 until 3   = 1 (mod 7).
 855 | 
 856 | METHOD
 857 | 
 858 |                    n
 859 |     We test if (-1) a  mod p is a primitive root mod p.
 860 |                      0
 861 |
 862 +============================================================================*/
 863
 864bool ArithModP::constCoeffIsPrimitiveRoot( ppuint a0, int n )
 865{
 866    ppsint constant_coeff = a0 ;
 867
 868    ModP<ppuint,ppsint> mod( p_ ) ;
 869    IsPrimitiveRoot isroot( p_ ) ;
 870
 871    // (-1)^n < 0 for odd n.
 872    if (n % 2 != 0)
 873        constant_coeff = -constant_coeff ;
 874
 875    return isroot( mod( constant_coeff ) ) ;
 876} 
 877
 878
 879
 880/*=============================================================================
 881|
 882| NAME
 883|
 884|     gcd
 885|
 886| DESCRIPTION
 887|
 888|    Euclid's method for computing greatest common divisor.
 889|
 890| METHOD
 891|
 892|     Iterate the equation 
 893|
 894|        gcd( u, v ) = gcd( v, r )
 895|   
 896|     where r = | u / v | = u - q v
 897|              --     --
 898|     until v is zero, terminating in gcd( u, 0 ) = u.
 899|
 900|     Let's use Mathematica to illustrate the method for large integers,
 901|
 902|     gcd[ u_, v_] :=
 903|         (* Compute GCD using Euclid's method. *)
 904|         Module[ {u2, v2, r}, u2 = u ; v2 = v ; 
 905|                  Print[ "Computing GCD( u = ", u, " v = ", v , " )" ] ;
 906|                  While[ v2 != 0, 
 907|                         r = Mod[u2, v2]; u2 = v2; v2 = r;
 908|                         Print[ "r = ", r, " u2 = ", u2, " v2 = ", v2]];
 909|                         Return[ u2 ]]
 910|
 911|     gcd[ 779953197883173551166308319545, 1282866356929526866866376009397 ]
 912|
 913|     Computing GCD( u = 779953197883173551166308319545 v = 1282866356929526866866376009397 )
 914|        u = 779953197883173551166308319545  v = 1282866356929526866866376009397
 915|        r = 779953197883173551166308319545 u2 = 1282866356929526866866376009397  v2 = 779953197883173551166308319545
 916|        r = 502913159046353315700067689852 u2 =  779953197883173551166308319545  v2 = 502913159046353315700067689852
 917|        r = 277040038836820235466240629693 u2 =  502913159046353315700067689852  v2 = 277040038836820235466240629693
 918|        r = 225873120209533080233827060159 u2 =  277040038836820235466240629693  v2 = 225873120209533080233827060159
 919|        r =  51166918627287155232413569534 u2 =  225873120209533080233827060159  v2 =  51166918627287155232413569534
 920|        r =  21205445700384459304172782023 u2 =   51166918627287155232413569534  v2 =  21205445700384459304172782023
 921|        r =   8756027226518236624068005488 u2 =   21205445700384459304172782023  v2 =   8756027226518236624068005488
 922|        r =   3693391247347986056036771047 u2 =    8756027226518236624068005488  v2 =   3693391247347986056036771047
 923|        r =   1369244731822264511994463394 u2 =    3693391247347986056036771047  v2 =   1369244731822264511994463394
 924|        r =    954901783703457032047844259 u2 =    1369244731822264511994463394  v2 =    954901783703457032047844259
 925|        r = 414342948118807479946619135 u2 =  954901783703457032047844259 v2 = 414342948118807479946619135
 926|        r = 126215887465842072154605989 u2 =  414342948118807479946619135 v2 = 126215887465842072154605989
 927|        r = 35695285721281263482801168 u2 =  126215887465842072154605989 v2 = 35695285721281263482801168
 928|        r = 19130030301998281706202485 u2 =  35695285721281263482801168 v2 = 19130030301998281706202485
 929|        r = 16565255419282981776598683 u2 =  19130030301998281706202485 v2 = 16565255419282981776598683
 930|        r = 2564774882715299929603802 u2 =  16565255419282981776598683 v2 = 2564774882715299929603802
 931|        r = 1176606122991182198975871 u2 =  2564774882715299929603802 v2 = 1176606122991182198975871
 932|        r = 211562636732935531652060 u2 =  1176606122991182198975871 v2 = 211562636732935531652060
 933|        r = 118792939326504540715571 u2 =  211562636732935531652060 v2 = 118792939326504540715571
 934|        r = 92769697406430990936489 u2 =  118792939326504540715571 v2 = 92769697406430990936489
 935|        r = 26023241920073549779082 u2 =  92769697406430990936489 v2 = 26023241920073549779082
 936|        r = 14699971646210341599243 u2 =  26023241920073549779082 v2 = 14699971646210341599243
 937|        r = 11323270273863208179839 u2 =  14699971646210341599243 v2 = 11323270273863208179839
 938|        r = 3376701372347133419404 u2 =  11323270273863208179839 v2 = 3376701372347133419404
 939|        r = 1193166156821807921627 u2 =  3376701372347133419404 v2 = 1193166156821807921627
 940|        r = 990369058703517576150 u2 =  1193166156821807921627 v2 = 990369058703517576150
 941|        r = 202797098118290345477 u2 =  990369058703517576150 v2 = 202797098118290345477
 942|        r = 179180666230356194242 u2 =  202797098118290345477 v2 = 179180666230356194242
 943|        r = 23616431887934151235 u2 =  179180666230356194242 v2 = 23616431887934151235
 944|        r = 13865643014817135597 u2 =  23616431887934151235 v2 = 13865643014817135597
 945|        r = 9750788873117015638 u2 =  13865643014817135597 v2 = 9750788873117015638
 946|        r = 4114854141700119959 u2 =  9750788873117015638 v2 = 4114854141700119959
 947|        r = 1521080589716775720 u2 =  4114854141700119959 v2 = 1521080589716775720
 948|        r = 1072692962266568519 u2 =  1521080589716775720 v2 = 1072692962266568519
 949|        r = 448387627450207201 u2 =  1072692962266568519 v2 = 448387627450207201
 950|        r = 175917707366154117 u2 =  448387627450207201 v2 = 175917707366154117
 951|        r = 96552212717898967 u2 =  175917707366154117 v2 = 96552212717898967
 952|        r = 79365494648255150 u2 =  96552212717898967 v2 = 79365494648255150
 953|        r = 17186718069643817 u2 =  79365494648255150 v2 = 17186718069643817
 954|        r = 10618622369679882 u2 =  17186718069643817 v2 = 10618622369679882
 955|        r = 6568095699963935 u2 =  10618622369679882 v2 = 6568095699963935
 956|        r = 4050526669715947 u2 =  6568095699963935 v2 = 4050526669715947
 957|        r = 2517569030247988 u2 =  4050526669715947 v2 = 2517569030247988
 958|        r = 1532957639467959 u2 =  2517569030247988 v2 = 1532957639467959
 959|        r = 984611390780029 u2 =  1532957639467959 v2 = 984611390780029
 960|        r = 548346248687930 u2 =  984611390780029 v2 = 548346248687930
 961|        r = 436265142092099 u2 =  548346248687930 v2 = 436265142092099
 962|        r = 112081106595831 u2 =  436265142092099 v2 = 112081106595831
 963|        r = 100021822304606 u2 =  112081106595831 v2 = 100021822304606
 964|        r = 12059284291225 u2 = 100021822304606  v2 = 12059284291225
 965|        r = 3547547974806 u2 = 12059284291225  v2 = 3547547974806
 966|        r = 1416640366807 u2 = 3547547974806 v2  = 1416640366807
 967|        r = 714267241192 u2 = 1416640366807 v2  = 714267241192
 968|        r = 702373125615 u2 = 714267241192 v2 =  702373125615
 969|        r = 11894115577 u2 = 702373125615 v2 =  11894115577
 970|        r = 620306572 u2 = 11894115577 v2 =  620306572
 971|        r = 108290709 u2 = 620306572 v2 =  108290709
 972|        r = 78853027 u2 = 108290709 v2 =  78853027
 973|        r = 29437682 u2 = 78853027 v2 =  29437682
 974|        r = 19977663 u2 = 29437682 v2 =  19977663
 975|        r = 9460019 u2 = 19977663 v2 = 9460019
 976|        r = 1057625 u2 = 9460019 v2 = 1057625
 977|        r = 999019 u2 = 1057625 v2 = 999019
 978|        r = 58606 u2 = 999019 v2 = 58606
 979|        r = 2717 u2 = 58606 v2 = 2717
 980|        r = 1549 u2 = 2717 v2 = 1549
 981|        r = 1168 u2 = 1549 v2 = 1168
 982|        r = 381 u2 = 1168 v2 = 381
 983|        r = 25 u2 = 381 v2 = 25
 984|        r = 6 u2 = 25 v2 = 6
 985|        r = 1 u2 = 6 v2 = 1
 986|        r = 0 u2 = 1 v2 = 0
 987|        ---> 1
 988|
 989+============================================================================*/
 990
 991template <typename IntType>
 992IntType gcd( const IntType & u, const IntType & v )
 993{
 994    IntType r ;
 995    IntType u2 { u } ;
 996    IntType v2 { v }  ;
 997
 998    #ifdef DEBUG_PP_ARITH
 999    cout << "gcd:  u = " << u << " v = " << v << endl ;
1000    #endif
1001
1002    while (v2 != static_cast<IntType>(0u))
1003    {
1004        r  = u2 % v2 ;
1005        u2 = v2 ;
1006        v2 = r ;
1007
1008       #ifdef DEBUG_PP_ARITH
1009       cout << "  r = " << r << " u2 = " << u2 << " v2 = " << v2 << endl ;
1010       #endif
1011    }
1012
1013    return u2 ;
1014}
1015
1016
1017/*==============================================================================
1018|                     Forced Template Instantiations                           |
1019==============================================================================*/
1020
1021// C++ doesn't automatically generate templated classes or functions until they
1022// are first used.  So depending on the order of compilation, the linker may say
1023// the templated functions are undefined.
1024//
1025// Therefore, explicitly instantiate ALL templates here:
1026
1027template ppuint   gcd( const ppuint &, const ppuint & ) ;
1028template BigInt   gcd( const BigInt &, const BigInt & ) ;
1029
1030template ppuint addMod( ppuint a, ppuint b, ppuint n ) ;
1031template ppuint timesTwoMod( ppuint, ppuint ) ;
1032template ppuint multiplyMod( const ppuint, const ppuint, const ppuint ) ;
1033
1034// We already specialized this function for ppuint in the source code implementation above, so we can omit
1035// the template instantiation:   template ppuint PowerMod<ppuint>::operator()( const ppuint & a, const ppuint & n ) ;
1036template BigInt  PowerMod<BigInt>::operator()( const BigInt &, const BigInt & ) ;
1037
1038template ModP<ppuint,ppsint>::ModP( ppuint ) ;
1039template ModP<ppuint,ppsint>::ModP( const ModP & ) ;
1040template ppuint ModP<ppuint,ppsint>::operator()( ppsint ) ;