1/*==============================================================================
2|
3| NAME
4|
5| Primpoly.cpp
6|
7| DESCRIPTION
8|
9| Program for finding primitive polynomials of degree n modulo p for
10| any prime p >= 2 and any n >= 2.
11|
12| Useful for generating PN sequences and finite fields for error control coding.
13|
14| Please see the user manual and complete technical documentation at
15|
16| http://seanerikoconnor.freeservers.com/Mathematics/AbstractAlgebra/PrimitivePolynomials/overview.html
17|
18| This is a console app to be run in a terminal window. The OS calls main(), which returns a status to the OS.
19|
20| LEGAL
21|
22| Primpoly Version 16.3 - A Program for Computing Primitive Polynomials.
23| Copyright (C) 1999-2025 by Sean Erik O'Connor. All Rights Reserved.
24|
25| This program is free software: you can redistribute it and/or modify
26| it under the terms of the GNU General Public License as published by
27| the Free Software Foundation, either version 3 of the License, or
28| (at your option) any later version.
29|
30| This program is distributed in the hope that it will be useful,
31| but WITHOUT ANY WARRANTY; without even the implied warranty of
32| MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
33| GNU General Public License for more details.
34|
35| You should have received a copy of the GNU General Public License
36| along with this program. If not, see http://www.gnu.org/licenses/.
37|
38| The author's address is seanerikoconnor!AT!gmail!DOT!com
39| with the !DOT! replaced by . and the !AT! replaced by @
40|
41==============================================================================*/
42
43
44/*------------------------------------------------------------------------------
45| Include Files |
46------------------------------------------------------------------------------*/
47
48#include <cstdlib> // abort()
49#include <iostream> // Basic stream I/O.
50#include <new> // set_new_handler()
51#include <cmath> // Basic math functions e.g. sqrt()
52#include <limits> // Numeric limits.
53#include <complex> // Complex data type and operations.
54#include <fstream> // File stream I/O.
55#include <sstream> // String stream I/O.
56#include <vector> // STL vector class.
57#include <string> // STL string class.
58#include <algorithm> // Iterators.
59#include <stdexcept> // Exceptions.
60#include <cassert> // assert()
61
62using namespace std ; // I don't want to use the std:: prefix everywhere.
63
64#include "ctype.h" // C string functions.
65
66
67/*------------------------------------------------------------------------------
68| PP Include Files |
69------------------------------------------------------------------------------*/
70
71#include "Primpoly.hpp" // Global functions.
72#include "ppArith.hpp" // Basic arithmetic functions.
73#include "ppBigInt.hpp" // Arbitrary precision integer arithmetic.
74#include "ppOperationCount.hpp" // OperationCount collection for factoring and poly finding.
75#include "ppFactor.hpp" // Prime factorization and Euler Phi.
76#include "ppPolynomial.hpp" // Polynomial operations and mod polynomial operations.
77#include "ppParser.hpp" // Parsing of polynomials and I/O services.
78#include "ppUnitTest.hpp" // Complete unit test.
79
80
81/*=============================================================================
82|
83| NAME
84|
85| Message strings.
86|
87| DESCRIPTION
88|
89| Helpful information returned by top level main() function.
90|
91+============================================================================*/
92
93static const string legalNotice
94{
95 "\n"
96 "Primpoly Version 16.3 - A Program for Computing Primitive Polynomials.\n"
97 "Copyright (C) 1999-2025 by Sean Erik O'Connor. All Rights Reserved.\n"
98 "\n"
99 "Primpoly comes with ABSOLUTELY NO WARRANTY; for details see the\n"
100 "GNU General Public License. This is free software, and you are welcome\n"
101 "to redistribute it under certain conditions; see the GNU General Public License\n"
102 "for details.\n\n"
103} ;
104
105static const string helpText
106{
107 "This program generates primitive polynomials of degree n modulo p.\n"
108 "\n"
109 "Usage: Generate a single random polynomial of degree n modulo p where p is a prime >= 2 and n is an integer >= 2\n"
110 " Primpoly p n\n"
111 "Example:\n"
112 " Primpoly 2 4\n"
113 " Self-check passes...\n"
114 " Primitive polynomial modulo 2 of degree 4\n"
115 " x ^ 4 + x + 1, 2\n"
116 "Usage: Test whether a polynomial is primitive modulo p.\n"
117 " Primpoly -t ""<Polynomial to test>, p""\n"
118 " If you leave off the modulus p we default to p = 2\n"
119 "Examples:\n"
120 " Primpoly -t ""x^4 + x + 1, 2""\n"
121 " Self-check passes...\n"
122 " x ^ 4 + x + 1, 2 is primitive!\n"
123 "\n"
124 " Primpoly -t ""x^4 + x + 1""\n"
125 " Self-check passes...\n"
126 " x ^ 4 + x + 1, 2 is primitive!\n"
127 "Usage: Generate all primitive polynomial of degree n modulo p.\n"
128 " Primpoly -a p n\n"
129 "Example:\n"
130 " Primpoly -a 2 4\n"
131 " Self-check passes...\n"
132 " Primitive polynomial modulo 2 of degree 4\n"
133 " x ^ 4 + x + 1, 2\n"
134 " Primitive polynomial modulo 2 of degree 4\n"
135 " x ^ 4 + x ^ 3 + 1, 2\n"
136 "Usage: Same but show computation statistics.\n"
137 " Primpoly -s p n\n"
138 "Examples: \n"
139 "\n"
140 " Primpoly.exe -s 13 19\n"
141 " Self-check passes...\n"
142 " Primitive polynomial modulo 13 of degree 19\n"
143 " x ^ 19 + 9 x + 2, 13\n"
144 "\n"
145 " +--------- OperationCount --------------------------------\n"
146 " |\n"
147 " | Integer factorization: Table lookup + Trial division + Pollard Rho\n"
148 " |\n"
149 " | Number of trial divisions : 0\n"
150 " | Number of gcd's computed : 9027\n"
151 " | Number of primality tests : 2\n"
152 " | Number of squarings: 9026\n"
153 " |\n"
154 " | Polynomial Testing\n"
155 " |\n"
156 " | Total num. degree 19 poly mod 13 : 1461920290375446110677\n"
157 " | Number of possible primitive poly: 6411930599771980992\n"
158 " | Polynomials tested : 120\n"
159 " | Const. coeff. was primitive root : 46\n"
160 " | Free of linear factors : 11\n"
161 " | Irreducible to power >=1 : 1\n"
162 " | Had order r (x^r = integer) : 1\n"
163 " | Passed const. coeff. test : 1\n"
164 " | Had order m (x^m != integer) : 1\n"
165 " |\n"
166 " +-----------------------------------------------------\n"
167 "Usage: Print help message.\n"
168 " Primpoly -h\n"
169 " <Prints this help message.>\n"
170 "\n\n"
171 "Primitive polynomials find many uses in mathematics and communications\n"
172 "engineering:\n"
173 " * Generation of pseudonoise (PN) sequences for spread spectrum\n"
174 " communications and chip fault testing.\n"
175 " * Generating Sobol sequences for high dimensional numerical integration.\n"
176 " * Generation of CRC and Hamming codes.\n"
177 " * Generation of Galois (finite) fields for use in decoding Reed-Solomon\n"
178 " and BCH error correcting codes.\n"
179 "\n"
180 "For detailed technical information, see my web page\n"
181 " http://seanerikoconnor.freeservers.com/Mathematics/AbstractAlgebra/PrimitivePolynomials/overview.html\n"
182 "\n"
183} ;
184
185
186/*=============================================================================
187|
188| NAME
189|
190| main
191|
192| DESCRIPTION
193|
194| Program for finding primitive polynomials of degree n modulo p for
195| any prime p >= 2 and any n >= 2.
196|
197| Useful for generating PN sequences and finite fields for error control coding.
198|
199| Run the program by typing
200|
201| ./Primpoly p n
202|
203| You will get an nth degree primitive polynomial modulo p.
204|
205| To see all the options type
206|
207| ./Primpoly -h
208|
209| Please see the user manual and complete technical documentation at
210|
211| http://seanerikoconnor.freeservers.com/Mathematics/AbstractAlgebra/PrimitivePolynomials/overview.html
212|
213| The author's address is seanerikoconnor!AT!gmail!DOT!com
214| with the !DOT! replaced by . and the !AT! replaced by @
215|
216+============================================================================*/
217
218int main( int argc, const char * argv[] )
219{
220 try
221 {
222 // Make my lawyers happy.
223 cout << legalNotice ;
224
225 // Set up the full parser for both command line parsing and polynomial parsing.
226 PolyParser<PolySymbol, PolyValue> parser ;
227 parser.parseCommandLine( argc, argv ) ;
228
229 // Print the help message only and exit.
230 if (parser.print_help_)
231 {
232 cout << helpText ;
233 return static_cast<int>( ReturnStatus::AskForHelp ) ;
234 }
235
236 #ifdef SELF_CHECK
237 // Do a self check always. We might fail to pass one or more unit tests, or the unit test itself might fail.
238 try
239 {
240 UnitTest unitTest ;
241 if (!unitTest.run())
242 throw PrimpolyError( "Self-check failed!" ) ;
243 else
244 cout << "Self-check passes..." << endl ;
245 }
246 catch (UnitTestError & e)
247 {
248 throw PrimpolyError( static_cast<string>( "Could not run the self-check!\n" ) + " [ " + e.what() + " ] " ) ;
249 }
250 #endif
251
252 // The user input a polynomial. Test it for primitivity.
253 if (parser.test_polynomial_for_primitivity_)
254 {
255 // Test for primitivity with the quick test.
256 Polynomial f( parser.test_polynomial_ ) ;
257 #ifdef DEBUG_PP_PRIMITIVITY
258 cout << "Factoring into primes r = (p^n-1)/(p-1) = " << " for n = " << parser.n << " p = " << parser.p << endl ;
259 #endif // DEBUG_PP_PRIMITIVITY
260 PolyOrder order( f ) ;
261 cout << f << " is " << (order.isPrimitive() ? "" : "NOT") << " primitive!" << endl ;
262
263 if (parser.print_operation_count_)
264 cout << order.statistics_ << endl ;
265
266 // Also do a very slow maximal order test for primitivity, if asked to do so.
267 if (parser.slow_confirm_)
268 {
269 cout << confirmWarning ;
270 cout << " confirmed " << (order.maximal_order() ? "" : "NOT") << " primitive!" << endl ;
271 }
272 }
273 // Find one primitive polynomial at random. Optionally, find all primitive polynomials.
274 else
275 {
276 // Generate and test all possible n th degree, monic, modulo p polynomials
277 // f(x). A polynomial is primitive if passes all the tests successfully.
278 // n
279 // Initialize f(x) to x + (-1). Then, when f(x) passes through function
280 // n
281 // nextTrialPoly for the first time, it will have the correct value, x
282 TrialPolynomial f ;
283 f.initialTrialPoly( parser.n, parser.p ) ;
284 #ifdef DEBUG_PP_PRIMITIVITY
285 cout << "Factoring into primes r = (p^n-1)/(p-1) = " << " for n = " << parser.n << " p = " << parser.p << endl ;
286 #endif // DEBUG_PP_PRIMITIVITY
287 PolyOrder order( f ) ;
288
289 bool is_primitive_poly{ false } ;
290 bool tried_all_poly{ false } ;
291 bool stopTesting{ false } ;
292
293 BigInt num_poly( 0u ) ;
294 BigInt num_primitive_poly( 0u ) ;
295
296 if (parser.list_all_primitive_polynomials_)
297 cout << "\n\nThere are " << order.getNumPrimPoly() << " primitive polynomials modulo " << f.modulus() << " of degree " << f.deg() << "\n\n" ;
298
299 do {
300 ++num_poly ;
301
302 #ifdef DEBUG_PP_PRIMITIVITY
303 cout << "Testing polynomial # " << num_poly << ") p(x) = " << f << " for primitivity" << endl ;
304 #endif // DEBUG_PP_PRIMITIVITY
305
306 order.resetPolynomial( f ) ;
307 is_primitive_poly = order.isPrimitive() ;
308
309 if (is_primitive_poly)
310 {
311 ++num_primitive_poly ;
312 cout << "\n\nPrimitive polynomial modulo " << f.modulus() << " of degree " << f.deg() << "\n\n" ;
313 cout << f ;
314 cout << endl << endl ;
315
316 // Do a very slow maximal order test for primitivity.
317 if (parser.slow_confirm_)
318 {
319 cout << confirmWarning ;
320 if (order.maximal_order())
321 cout << f << " confirmed primitive!" << endl ;
322 else
323 {
324 ostringstream os ;
325 os << "Fast test says " << f << " is a primitive polynomial but slow test disagrees.\n" ;
326 throw PolynomialError( os.str() ) ;
327 }
328 }
329
330 // Early out if we've found all the primitive polynomials.
331 if (num_primitive_poly >= order.getNumPrimPoly())
332 break ;
333 }
334
335 tried_all_poly = (num_poly >= order.getMaxNumPoly()) ;
336 stopTesting = tried_all_poly || (!parser.list_all_primitive_polynomials_ && is_primitive_poly) ;
337
338 f.nextTrialPoly() ; // Try next polynomal in sequence.
339 } while( !stopTesting ) ;
340
341 if (parser.print_operation_count_)
342 cout << order.statistics_ << endl ;
343
344 // Didn't find a primitive polynomial in the find-only-one-primitive-polynomial case, which is an error.
345 if (!parser.list_all_primitive_polynomials_ && !is_primitive_poly)
346 {
347 ostringstream os ;
348 os << "Tested all " << order.getMaxNumPoly() << " possible polynomials, but failed to find a primitive polynomial" ;
349 throw PolynomialError( os.str() ) ;
350 }
351 }
352 return static_cast<int>( ReturnStatus::Success ) ;
353 }
354 // Catch all exceptions and report what happened to the user.
355 // First do the user-defined exceptions.
356 catch( PrimpolyError & e )
357 {
358 cerr << "\nTop Level Error: " << " [ " << e.what() << " ] " << endl << writeToAuthorMessage ;
359 return static_cast<int>( ReturnStatus::InternalError ) ;
360 }
361 catch( ParserError & e )
362 {
363 cerr << "Inputs are incorrect or out of range: " << " [ " << e.what() << " ] " << endl << helpText ;
364 return static_cast<int>( ReturnStatus::RangeError ) ;
365 }
366 catch( FactorError & e )
367 {
368 cerr << "Error in prime factorization: " << " [ " << e.what() << " ] " << endl << writeToAuthorMessage ;
369 return static_cast<int>( ReturnStatus::InternalError ) ;
370 }
371 catch( BigIntRangeError & e )
372 {
373 cerr << "Internal range error in multiple precision arithmetic: " << " [ " << e.what() << " ] " << endl << writeToAuthorMessage ;
374 return static_cast<int>( ReturnStatus::InternalError ) ;
375 }
376 catch( BigIntDomainError & e )
377 {
378 cerr << "Internal domain error in multiple precision arithmetic: " << " [ " << e.what() << " ] " << endl << writeToAuthorMessage ;
379 return static_cast<int>( ReturnStatus::InternalError ) ;
380 }
381 catch( BigIntUnderflow & e )
382 {
383 cerr << "Internal underflow error in multiple precision arithmetic: " << " [ " << e.what() << " ] " << endl << writeToAuthorMessage ;
384 return static_cast<int>( ReturnStatus::InternalError ) ;
385 }
386 catch( BigIntOverflow & e )
387 {
388 cerr << "Internal overflow error in multiple precision arithmetic: " << " [ " << e.what() << " ] " << endl << writeToAuthorMessage ;
389 return static_cast<int>( ReturnStatus::InternalError ) ;
390 }
391 catch( BigIntZeroDivide & e )
392 {
393 cerr << "Internal zero divide error in multiple precision arithmetic: " << " [ " << e.what() << " ] " << endl << writeToAuthorMessage ;
394 return static_cast<int>( ReturnStatus::InternalError ) ;
395 }
396 catch( BigIntMathError & e )
397 {
398 cerr << "Internal math error in multiple precision arithmetic: " << " [ " << e.what() << " ] " << endl << writeToAuthorMessage ;
399 return static_cast<int>( ReturnStatus::InternalError ) ;
400 }
401 catch( ArithModPError & e )
402 {
403 cerr << "Internal modulo p arithmetic error: " << " [ " << e.what() << " ] " << endl << writeToAuthorMessage ;
404 return static_cast<int>( ReturnStatus::InternalError ) ;
405 }
406 catch( PolynomialRangeError & e )
407 {
408 cout << "Error. Polynomial has bad syntax or coefficients are out of range. " << " [ " << e.what() << " ] " << endl << helpText ;
409 return static_cast<int>( ReturnStatus::RangeError ) ;
410 }
411 catch( PolynomialError & e )
412 {
413 cerr << "Error in polynomial arithmetic: " << " [ " << e.what() << " ] " << endl << writeToAuthorMessage ;
414 return static_cast<int>( ReturnStatus::InternalError ) ;
415 }
416 // Standard library exceptions.
417 catch( bad_alloc & e )
418 {
419 cerr << "Error allocating memory: " << " [ " << e.what() << " ] "<< endl ;
420 cerr << "Run on a different computer with more RAM or virtual memory." << writeToAuthorMessage ;
421 return static_cast<int>( ReturnStatus::InternalError ) ;
422 }
423 catch( exception & e )
424 {
425 cerr << "System error: " << e.what() << endl << writeToAuthorMessage ;
426 return static_cast<int>( ReturnStatus::InternalError ) ;
427 }
428 // Catch all other uncaught exceptions, which would otherwise call terminate()
429 // which in turn calls abort() and which would halt this program.
430 //
431 // Limitations:
432 // We can't handle the case where terminate() gets called because the
433 // exception mechanism finds a corrupt stack or catches a destructor
434 // throwing an exception.
435 //
436 // Also we can't catch exceptions which are thrown by initializing or
437 // destructing global variables.
438 catch( ... )
439 {
440 cerr << "Unexpected exception: " << endl << writeToAuthorMessage ;
441 return static_cast<int>( ReturnStatus::InternalError ) ;
442 }
443
444 return static_cast<int>( ReturnStatus::Success ) ;
445
446} //========================== end of function main ========================