1/*==============================================================================
2|
3| NAME
4|
5| ppParser.hpp
6|
7| DESCRIPTION
8|
9| Header file for polynomial parser 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 __PPPARSER_H__
40
41
42// Some miscellaneous constants.
43#define MAX_NUM_COMMAND_LINE_ARGS 3
44const ppuint minModulus{ 2 } ;
45const ppuint minDegree{ 2 } ;
46
47
48/*=============================================================================
49 |
50 | NAME
51 |
52 | Action
53 |
54 | DESCRIPTION
55 |
56 | Types of LR parser actions.
57 |
58 +============================================================================*/
59
60enum class Action
61{
62 Shift, Reduce, Accept, Error
63} ;
64
65
66/*=============================================================================
67 |
68 | NAME
69 |
70 | enumToString
71 |
72 | DESCRIPTION
73 |
74 | Helper function to translate enum types into human readable format.
75 |
76 +============================================================================*/
77
78template< typename SymbolType >
79string enumToString( const SymbolType & st ) ;
80
81
82/*=============================================================================
83 |
84 | NAME
85 |
86 | Symbol
87 |
88 | DESCRIPTION
89 |
90 | Define the VALUE of a symbol. Used in the parser's syntax directed
91 | translation and also for the final polynomial value returned from the
92 | parser.
93 |
94 +============================================================================*/
95
96// Tokens and other types of symbols used in parsing.
97template< typename SymbolType, typename ValueType >
98class Symbol
99{
100 public:
101 Symbol() ;
102
103 Symbol( SymbolType type, int state ) ;
104
105 Symbol( SymbolType type, int state, ValueType value ) ;
106
107 Symbol( const Symbol< SymbolType, ValueType > & s ) ;
108
109 // Destructor. Virtual so derived class destructor
110 // automatically calls its base class destructor.
111 virtual ~Symbol()
112 {
113 }
114
115 Symbol< SymbolType, ValueType > & operator=( const Symbol<SymbolType,ValueType> & s ) ;
116
117 // Print function for Symbol class.
118 // Define this templated non-member friend function here, otherwise the compiler will generate
119 // a link error. If the compiler sees only a declaration here, it doesn't know yet it's a template.
120 // See https://isocpp.org/wiki/faq/templates#template-friends
121 friend ostream & operator<<( ostream & out, const Symbol<SymbolType,ValueType> & s )
122 {
123 out << enumToString<SymbolType>( s.type_ ) << " " ;
124 out << s.value_ << " " ;
125 out << "state " << s.state_ ;
126
127 return out ;
128 }
129
130 // Allow direct access to this simple data type for convenience.
131 public:
132 SymbolType type_ ; // Type of terminal or nonterminal.
133 ValueType value_ ; // Value of the symbol.
134 int state_ ; // State or production number.
135} ;
136
137/*=============================================================================
138 |
139 | NAME
140 |
141 | ActionState
142 |
143 | DESCRIPTION
144 |
145 | A pair consisting of action and state for the LR parser.
146 | Actions are (shift s, reduce p, accept, error) and the new state
147 | number to transition to.
148 |
149 +============================================================================*/
150
151class ActionState
152{
153 public:
154 ActionState() ;
155
156 ActionState( Action type, int state ) ;
157
158 // Destructor. Virtual so derived class destructor
159 // automatically calls its base class destructor.
160 virtual ~ActionState()
161 {
162 }
163
164 ActionState( const ActionState & as ) ;
165
166 ActionState & operator=( const ActionState & as ) ;
167
168 friend ostream & operator<<( ostream & out, const ActionState & as ) ;
169
170 // Allow direct access to this simple data type for convenience.
171 public:
172 Action action_ ;
173 int state_ ;
174} ;
175
176/*=============================================================================
177 |
178 | NAME
179 |
180 | ParserError
181 |
182 | DESCRIPTION
183 |
184 | Exceptions for the parser class.
185 |
186 +============================================================================*/
187
188class ParserError : public runtime_error
189{
190 public:
191 // Throw with error message, file name and line number.
192 ParserError( const string & description, const string & file, const int & line )
193 : runtime_error( description + " in file " + file + " at line " + to_string(line) )
194 {
195 } ;
196
197 // Throw with an error message.
198 ParserError( const string & description )
199 : runtime_error( description )
200 {
201 } ;
202
203 // Throw with default error message.
204 ParserError()
205 : runtime_error( "Parser error: " )
206 {
207 } ;
208
209} ;
210
211
212
213/*=============================================================================
214|
215| NAME
216|
217| Parser
218|
219| DESCRIPTION
220|
221| Abstract base class for parsing using an LALR(1) parser.
222|
223| EXAMPLE
224|
225| Example of how to use the PolyParser child class for polynomials,
226|
227| // Create the parser.
228| PolyParser<PolySymbol, PolyValue> parser() ;
229|
230| // Sample polynomial.
231| string s = "2 x ^ 3 + 3 x + 4, 5" ;
232|
233| try
234| {
235| PolyValue v = parser.parse( s ) ;
236| cout << "Parsing the string = " << s << endl ;
237| cout << "We have a polynomial = " << v << endl ;
238| }
239| catch( ParserError & e )
240| {
241| cout << "Parser error: " << e.what() << endl ;
242| }
243|
244| NOTES
245|
246| I used a parser generator I wrote in Common LISP to generate the LALR(1) parse tables. See my
247| notes,
248| http://seanerikoconnor.freeservers.com/ComputerScience/Compiler/ParserGeneratorAndParser/LRandLALRParserGeneratorAndParser.html
249| You can also use Flex and Bison (Yacc) to generate the equivalent tables, but I like my own parser generator.
250|
251| While the LALR(1) parser machinery stays the same in this base class, the grammar,
252| i.e. parse tables and error messages, will be redefined in the child class, also the syntax
253| directed translation. We try to use the base class lexical analyzer for all child classes, if possible.
254|
255+============================================================================*/
256
257template< typename SymbolType, typename ValueType >
258class Parser
259{
260 // The basic member functions to create and initialize the parser,
261 // make copies and do a syntax-directed translation of a string.
262 public:
263 // Default construct the parser.
264 Parser() ;
265
266 // Destructor. Virtual so derived class destructor
267 // automatically calls its base class destructor.
268 virtual ~Parser() ;
269
270 // Copy constructor.
271 Parser( const Parser & parser ) ;
272
273 // Assignment operator.
274 virtual Parser & operator=( const Parser & parser ) ;
275
276 // Parse an input string, doing a syntax-directed translation, returning the value.
277 ValueType parse( string s ) ;
278
279 // Pure virtual functions to be defined only in the child classes, since the lexical analyzer, parse tables and
280 // syntax-directed translation depend uniquely on the grammar.
281 //
282 // Defined within the scope of the Parser class or its derived classes only,
283 // so not visible from outside the parser.
284 protected:
285 // Parse string into tokens.
286 virtual void tokenize( string sentence ) = 0 ;
287
288 // NOTE: call this in the child class constructor. It fills in the ACTION, GOTO, and ERROR_MESSAGE tables.
289 virtual void initializeTables() = 0 ;
290
291 // Reduce using production A -> beta, computing A's value from values of the tokens in beta.
292 virtual void syntaxDirectedTranslation( int productionNum, int topIndex, Symbol<SymbolType,ValueType> & as ) = 0 ;
293
294 // These member values and functions are called directly in the child classes.
295 protected:
296 // Number of states, productions, etc.
297 int num_states_ ; // Includes the 0 state.
298 int num_productions__ ; // Number of productions.
299 int num_terminals_ ; // Number of terminal symbols.
300 int num_non_terminals_ ; // Number of nonterminal symbols.
301
302 void insertAction( int state, SymbolType terminal, Action actionType, int actionState ) ;
303 void insertGoto( int state, SymbolType nonterm, int newState ) ;
304 void insertProduction( int prodNum, SymbolType nonTerm, int rhsLength ) ;
305 void insertErrorMessage( int state, string errorMessage ) ;
306
307 protected:
308 // Parser stack containing symbols tokenized by lexical analyzer.
309 vector< Symbol<SymbolType, ValueType> > input_stack_ ;
310
311 // Parse stack containing symbols and states.
312 vector< Symbol<SymbolType, ValueType> > parse_stack_ ;
313
314 // Action table.
315 // action = ACTION_TABLE[ <state at top of parse stack> ][ <lookahead input token> ]
316 vector< vector<ActionState> > action_table_ ;
317
318 // GOTO table.
319 // For reduce actions,
320 // new goto state to push = GOTO[ current state ][ nonterminal production symbol ]
321 vector< vector<int> > goto_table_ ;
322
323 // Error messages.
324 vector<string> error_message_ ;
325
326 // Productions: Left side non-terminal symbol + production number.
327 vector< Symbol<SymbolType, ValueType> > production_ ;
328} ;
329
330
331
332// --------------- Derived class parsers and their symbols and values ---------
333
334
335
336/*=============================================================================
337 |
338 | NAME
339 |
340 | PolyValue
341 |
342 | DESCRIPTION
343 |
344 | Define a symbol with a polynomial value. Used in the parser's syntax directed
345 | translation and also for the final polynomial value returned from the
346 | parser.
347 |
348 +============================================================================*/
349
350class PolyValue
351{
352 public:
353 PolyValue() ;
354
355 PolyValue( const PolyValue & v ) ;
356
357 PolyValue & operator=( const PolyValue & v ) ;
358
359 friend ostream & operator<<( ostream & , const PolyValue & v ) ;
360
361 // Allow direct access to this simple data type for convenience.
362 public:
363 ppuint scalar_ ; // Number which can be a polynomial coefficient, power of x, or the modulus p.
364 // Let it be signed for maximum flexibility. We'll check if it exceeds the allowed
365 // range for internal primitive polynomial computations in the parser.
366 vector<ppuint> f_ ; // Polynomial f(x).
367} ;
368
369
370/*=============================================================================
371 |
372 | NAME
373 |
374 | PolySymbol
375 |
376 | DESCRIPTION
377 |
378 | Define polynomial symbol (tokens). Used in the parser's syntax directed
379 | translation.
380 |
381 +============================================================================*/
382
383enum class PolySymbol
384{
385 // Terminal symbols or tokens.
386 Dollar = 0, Integer, Comma, Ecks, Plus, Exp,
387 NumTerminals,
388
389 // Nonterminal symbols.
390 S, Mod, Poly, Term, Multiplier, Power,
391 NumSymbols
392} ;
393
394/*=============================================================================
395|
396| NAME
397|
398| PolyParser
399|
400| DESCRIPTION
401|
402| Child class for parsing a polynomial grammar inheriting from abstract
403| base class Parser.
404|
405| EXAMPLE
406|
407| Here is the LALR(1) grammar. Note the use of epsilon productions to
408| make the grammar simpler.
409|
410| Productions:
411|
412| S -> Poly Mod
413| Mod -> comma integer | EPSILON
414| Poly -> Poly + Term | Term
415| Term -> Multiplier Power
416| Multiplier -> integer | EPSILON
417| Power -> x | x ^ integer | EPSILON
418|
419| Terminal symbols:
420|
421| comma + integer ^ x $
422|
423+============================================================================*/
424
425template< typename SymbolType, typename ValueType >
426class PolyParser : public Parser< SymbolType, ValueType >
427{
428 public:
429 // Construct the parser.
430 PolyParser() ;
431
432 // Destructor.
433 ~PolyParser() ;
434
435 // Copy constructor.
436 PolyParser( const PolyParser< SymbolType, ValueType > & parser ) ;
437
438 // Assignment operator.
439 PolyParser< SymbolType, ValueType > & operator=( const PolyParser< SymbolType, ValueType > & parser ) ;
440
441 // Command line argument parsing.
442 void parseCommandLine( int argc, const char * argv[] ) ;
443
444 // Allow direct access to these command line parameters for convenience.
445 bool test_polynomial_for_primitivity_ ;
446 bool list_all_primitive_polynomials_ ;
447 bool print_operation_count_ ;
448 bool print_help_ ;
449 bool slow_confirm_ ;
450 ppuint p ;
451 int n ;
452 Polynomial test_polynomial_ ;
453
454 // Location of the factorization tables.
455 static string current_working_dir_ ;
456
457 private:
458 // Parse string into tokens.
459 void tokenize( string sentence ) ;
460
461 // Fill in the ACTION, GOTO, and ERROR_MESSAGE tables.
462 void initializeTables() ;
463
464 // Syntax directed translation. Reduce using production A -> beta, computing A's value from values of
465 // the tokens in beta.
466 void syntaxDirectedTranslation( int productionNum, int topIndex, Symbol<SymbolType,ValueType> & as ) ;
467
468 // Even though these member variables are declared in the base class as protected, they are not visible in this
469 // child class. Because we have a templated class, we must tell the C++ compiler explicitly that these symbols
470 // are from the base class.
471 using Parser<SymbolType, ValueType>::num_states_ ;
472 using Parser<SymbolType, ValueType>::num_productions__ ;
473 using Parser<SymbolType, ValueType>::num_terminals_ ;
474 using Parser<SymbolType, ValueType>::num_non_terminals_ ;
475
476 using Parser<SymbolType, ValueType>::insertAction ;
477 using Parser<SymbolType, ValueType>::insertGoto ;
478 using Parser<SymbolType, ValueType>::insertProduction ;
479 using Parser<SymbolType, ValueType>::insertErrorMessage ;
480
481 using Parser< SymbolType, ValueType >::input_stack_ ;
482 using Parser< SymbolType, ValueType >::parse_stack_ ;
483 using Parser< SymbolType, ValueType >::action_table_ ;
484 using Parser< SymbolType, ValueType >::goto_table_ ;
485 using Parser< SymbolType, ValueType >::error_message_ ;
486 using Parser< SymbolType, ValueType >::production_ ;
487} ;
488
489// Static member variable needs to be initialized outside of the class declaration.
490template< typename SymbolType, typename ValueType >
491string PolyParser<SymbolType, ValueType>::current_working_dir_ = "" ;
492
493
494// --------------- Derived class parsers and their symbols and values ---------
495
496
497
498/*=============================================================================
499 |
500 | NAME
501 |
502 | FactorizationValue
503 |
504 | DESCRIPTION
505 |
506 | Define a parser symbol which containst the number string in one line of
507 | the factorization table and also the prime factorization itself.
508 | This is used in the parser's syntax directed translation and also for
509 | the final value returned from the parser.
510 |
511 +============================================================================*/
512
513template< typename IntType >
514class FactorizationValue
515{
516 public:
517 FactorizationValue() ;
518
519 FactorizationValue( const FactorizationValue<IntType> & v ) ;
520
521 FactorizationValue( const IntType & p, const int & count ) ;
522
523 FactorizationValue<IntType> & operator=( const FactorizationValue<IntType> & v ) ;
524
525 // Print out the factorization.
526 // Define this templated non-member friend function here, otherwise the compiler will generate
527 // a link error. If the compiler sees only a declaration here, and not the complete implementation,
528 // it doesn't know yet it's a template.
529 // See https://isocpp.org/wiki/faq/templates#template-friends
530 friend ostream & operator<<( ostream & out, const FactorizationValue<IntType> & factorization )
531 {
532 if (factorization.factor_.size() > 0)
533 {
534 for (auto & f : factorization.factor_)
535 out << f ;
536 }
537 if (factorization.numberString_.length() > 0)
538 out << factorization.numberString_ ;
539
540 return out ;
541 }
542
543 static IntType numberStringToInteger( const string & numberString ) ;
544
545 // Allow direct access to data for convenience.
546 public:
547 string numberString_ ; // Sequence of digits representing an integer.
548 vector< PrimeFactor<IntType> > factor_ ; // Vector of distinct prime factors to powers.
549} ;
550
551
552/*=============================================================================
553 |
554 | NAME
555 |
556 | FactorizationSymbols
557 |
558 | DESCRIPTION
559 |
560 | Define factoriziton symbols (tokens). Used in the parser's syntax directed
561 | translation.
562 |
563 +============================================================================*/
564
565enum class FactorizationSymbol
566{
567 // Terminal symbols or tokens.
568 Dollar = 0, Integer, Period, Caret, Backslash,
569 NumTerminals,
570
571 // Nonterminal symbols.
572 S, Factorization, Factor, BigInteger,
573
574 // Count number of total symbols.
575 NumSymbols
576} ;
577
578
579
580/*=============================================================================
581|
582| NAME
583|
584| FactorizationParser
585|
586| DESCRIPTION
587|
588| Child class for parsing a factorization table of Cunningham numbers p ^ n - 1
589| inherited from abstract base class Parser.
590|
591| EXAMPLE
592|
593| Each table for p has one line containing the factorization of p ^ n - 1 of the type
594|
595| n <num factors p1 ^ m1 . p2 ^ m2 ...
596|
597| Dots separate the factors. Long numbers are continued on the next line with backslashes.
598| For example, in the table for p = 3, the line below is for n = 398 and has the 12 factors
599| 2 ^ 4, 3, 3583 ... 11881576435363115110426317067813673631182057431802975257340612015708984828478898969687279497327343
600|
601| 398 12 2^4.3.3583.4588543.34266607.2146612951394313997.8670122322845042\
602| 61471.3742361194240057889227626965547117.118815764353631151\
603| 104263170678136736311820574318029752573406120157089\
604| 84828478898969687279497327343
605|
606| Here is the LALR(1) grammar.
607|
608| Productions:
609|
610| S -> integer integer Factorization
611| Factorization -> Factorization . Factor | Factor
612| Factor -> BigInteger ^ BigInteger | BigInteger
613| BigInteger -> BigInteger \ integer | integer
614|
615| Terminal symbols:
616| integer . ^ \ $
617|
618+============================================================================*/
619
620template< typename SymbolType, typename ValueType >
621class FactorizationParser : public Parser< SymbolType, ValueType >
622{
623 public:
624 // Construct the parser.
625 FactorizationParser() ;
626
627 // Destructor.
628 ~FactorizationParser() ;
629
630 // Copy constructor.
631 FactorizationParser( const FactorizationParser< SymbolType, ValueType > & parser ) ;
632
633 // Assignment operator.
634 FactorizationParser< SymbolType, ValueType > & operator=( const FactorizationParser< SymbolType, ValueType > & parser ) ;
635
636 private:
637 // Parse string into tokens.
638 void tokenize( string sentence ) ;
639
640 // Fill in the ACTION, GOTO, and ERROR_MESSAGE tables.
641 void initializeTables() ;
642
643 // Syntax directed translation. Reduce using production A -> beta, computing A's value from values of
644 // the tokens in beta.
645 void syntaxDirectedTranslation( int productionNum, int topIndex, Symbol<SymbolType,ValueType> & as ) ;
646
647 // Even though these member variables are declared in the base class as protected, they are not visible in this
648 // child class. We have to tell the compiler explicitly that these symbols are base class templated types so it
649 // knows what they are.
650 using Parser<SymbolType, ValueType>::num_states_ ;
651 using Parser<SymbolType, ValueType>::num_productions__ ;
652 using Parser<SymbolType, ValueType>::num_terminals_ ;
653 using Parser<SymbolType, ValueType>::num_non_terminals_ ;
654
655 using Parser<SymbolType, ValueType>::insertAction ;
656 using Parser<SymbolType, ValueType>::insertGoto ;
657 using Parser<SymbolType, ValueType>::insertProduction ;
658 using Parser<SymbolType, ValueType>::insertErrorMessage ;
659
660 using Parser< SymbolType, ValueType >::input_stack_ ;
661 using Parser< SymbolType, ValueType >::parse_stack_ ;
662 using Parser< SymbolType, ValueType >::action_table_ ;
663 using Parser< SymbolType, ValueType >::goto_table_ ;
664 using Parser< SymbolType, ValueType >::error_message_ ;
665 using Parser< SymbolType, ValueType >::production_ ;
666} ;
667
668#endif // __PPPARSER_H__