1/*==============================================================================
   2|
   3|  NAME
   4|
   5|     ppParser.cpp
   6|
   7|  DESCRIPTION
   8|
   9|     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 !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 <limits>       // numeric_limits
  45#include <cmath>        // Basic math functions e.g. sqrt()
  46#include <complex>      // Complex data type and operations.
  47#include <fstream>      // File stream I/O.
  48#include <sstream>      // String stream I/O.
  49#include <vector>       // STL vector class.
  50#include <string>       // STL string class.
  51#include <algorithm>    // Iterators.
  52#include <stdexcept>    // Exceptions.
  53#include <cassert>      // assert()
  54#include <cstring>      // memset()
  55
  56using namespace std ;
  57
  58// Current working directory, recursive dir search.
  59#ifdef USE_EXPERIMENTAL_FILESYSTEM_GNU_CPP
  60    #include <experimental/filesystem>
  61    using namespace experimental::filesystem ;
  62#else
  63    #include <filesystem>
  64    using namespace filesystem ;
  65#endif
  66
  67#include "ctype.h"       // C string functions.
  68
  69
  70/*------------------------------------------------------------------------------
  71|                                PP Include Files                              |
  72------------------------------------------------------------------------------*/
  73
  74#include "Primpoly.hpp"		    // Global functions.
  75#include "ppArith.hpp"		    // Basic arithmetic functions.
  76#include "ppBigInt.hpp"         // Arbitrary precision integer arithmetic.
  77#include "ppOperationCount.hpp" // OperationCount collection for factoring and poly finding.
  78#include "ppFactor.hpp"         // Prime factorization and Euler Phi.
  79#include "ppPolynomial.hpp"	    // Polynomial operations and mod polynomial operations.
  80#include "ppParser.hpp"	        // Parsing of polynomials and I/O services.
  81#include "ppUnitTest.hpp"       // Complete unit test.
  82
  83
  84/*------------------------- Parser Helper Classes ----------------------------*/
  85
  86
  87/*=============================================================================
  88 |
  89 | NAME
  90 |
  91 |     Symbol
  92 |
  93 | DESCRIPTION
  94 |
  95 |     Default constructor for a parser symbol, which has a type (terminal or non-terminal)
  96 |     and a state (or production number).
  97 |
  98 +============================================================================*/
  99
 100template < typename SymbolType, typename ValueType >
 101Symbol< SymbolType, ValueType >::Symbol()
 102: type_(), value_(), state_()
 103{
 104}
 105
 106
 107/*=============================================================================
 108 |
 109 | NAME
 110 |
 111 |     Symbol
 112 |
 113 | DESCRIPTION
 114 |
 115 |     Constructor for a parser symbol, which has a type (terminal or non-terminal)
 116 |     and a state (or production number).
 117 |
 118 +============================================================================*/
 119
 120template < typename SymbolType, typename ValueType >
 121Symbol< SymbolType, ValueType >::Symbol( SymbolType type, int state )
 122: type_( type ), value_(), state_( state )
 123{
 124}
 125
 126
 127/*=============================================================================
 128 |
 129 | NAME
 130 |
 131 |     Symbol
 132 |
 133 | DESCRIPTION
 134 |
 135 |     Constructor for a parser symbol, which has a type (terminal or non-terminal)
 136 |     a value (see above) and a state (or production number).
 137 |
 138 +============================================================================*/
 139
 140template < typename SymbolType, typename ValueType >
 141Symbol< SymbolType, ValueType >::Symbol( SymbolType type, int state, ValueType value )
 142: type_( type ), state_( state ), value_( value )
 143{
 144}
 145
 146/*=============================================================================
 147 |
 148 | NAME
 149 |
 150 |     Symbol
 151 |
 152 | DESCRIPTION
 153 |
 154 |     Copy constructor.
 155 |
 156 +============================================================================*/
 157
 158template < typename SymbolType, typename ValueType >
 159Symbol< SymbolType, ValueType >::Symbol( const Symbol< SymbolType, ValueType > & s )
 160: type_( s.type_ )
 161, value_( s.value_ )
 162, state_( s.state_ )
 163{
 164}
 165
 166
 167/*=============================================================================
 168 |
 169 | NAME
 170 |
 171 |     operator=
 172 |
 173 | DESCRIPTION
 174 |
 175 |     Assignment operator.
 176 |
 177 +============================================================================*/
 178
 179template < typename SymbolType, typename ValueType >
 180Symbol< SymbolType, ValueType > & Symbol< SymbolType, ValueType >::operator=( const Symbol< SymbolType, ValueType > & s )
 181{
 182    // Check for assigning to oneself:  just pass back a reference to the unchanged object.
 183    if (this == &s)
 184        return *this ;
 185    
 186    type_  = s.type_ ;
 187    value_ = s.value_ ;
 188    state_ = s.state_ ;
 189    
 190    // Pass back the object so we can concatenate assignments.
 191    return *this ;
 192}
 193
 194
 195/*=============================================================================
 196 |
 197 | NAME
 198 |
 199 |     ActionState
 200 |
 201 | DESCRIPTION
 202 |
 203 |     Default constructor for an action/state pair.  Action is shift, reduce, 
 204 |     etc. and state is a number.
 205 |
 206 +============================================================================*/
 207
 208ActionState::ActionState()
 209    : action_( Action::Error )
 210    , state_( 0 )
 211{
 212}
 213
 214
 215/*=============================================================================
 216 |
 217 | NAME
 218 |
 219 |     ActionState
 220 |
 221 | DESCRIPTION
 222 |
 223 |     Constructor for an action/state pair.
 224 |
 225 +============================================================================*/
 226
 227ActionState::ActionState( Action type, int state )
 228    : action_( type )
 229    , state_( state )
 230{
 231}
 232
 233
 234/*=============================================================================
 235 |
 236 | NAME
 237 |
 238 |     ActionState
 239 |
 240 | DESCRIPTION
 241 |
 242 |     Copy constructor for an action/state pair.
 243 |
 244 +============================================================================*/
 245
 246ActionState::ActionState( const ActionState & as )
 247    : action_(  as.action_ )
 248    , state_( as.state_ )
 249{
 250}
 251
 252
 253/*=============================================================================
 254 |
 255 | NAME
 256 |
 257 |     ActionState
 258 |
 259 | DESCRIPTION
 260 |
 261 |     Assignment operator for an action/state pair.
 262 |
 263 +============================================================================*/
 264
 265ActionState & ActionState::operator=( const ActionState & as )
 266{
 267    // Check for assigning to oneself:  just pass back a reference to the unchanged object.
 268    if (this == &as)
 269        return *this ;
 270    
 271    action_  = as.action_ ;
 272    state_ = as.state_ ;
 273    
 274    // Pass back the object so we can concatenate assignments.
 275    return *this ;
 276}
 277
 278
 279/*=============================================================================
 280 |
 281 | NAME
 282 |
 283 |     Operator << for ActionState
 284 |
 285 | DESCRIPTION
 286 |
 287 |     Debug print to an output stream for an action state.
 288 |
 289 +============================================================================*/
 290
 291ostream & operator<<( ostream & out, const ActionState & as )
 292{
 293    string s ;
 294    
 295    out << "Action/State: " ;
 296    switch( as.action_ )
 297    {
 298        case Action::Shift:  s = "Shift" ;   break ;
 299        case Action::Reduce: s = "Reduce" ;  break ;
 300        case Action::Accept: s = "Accept" ;  break ;
 301        case Action::Error:  s = "Error" ;   break ;
 302    }
 303    out << s ;
 304    out << " " << as.state_ ;
 305    
 306    return out ;
 307}
 308
 309
 310/*----------------------------- Parser Base Class ----------------------------*/
 311
 312
 313/*=============================================================================
 314 |
 315 | NAME
 316 |
 317 |     Parser
 318 |
 319 | DESCRIPTION
 320 |
 321 |     Default constructor for the class.
 322 |
 323 +============================================================================*/
 324
 325template < typename SymbolType, typename ValueType >
 326Parser< SymbolType, ValueType >::Parser()
 327    : input_stack_()
 328    , parse_stack_()
 329    , num_states_       ( 0 ) // Includes the 0 state.
 330    , num_productions__  ( 0 ) // Number of productions.
 331    , num_terminals_    ( 0 ) // Number of terminal symbols.
 332    , num_non_terminals_ ( 0 ) // Number of nonterminal symbols.
 333{
 334}
 335
 336/*=============================================================================
 337 |
 338 | NAME
 339 |
 340 |     ~Parser
 341 |
 342 | DESCRIPTION
 343 |
 344 |     Default deeeestructor for the class.
 345 |
 346 +============================================================================*/
 347
 348template < typename SymbolType, typename ValueType >
 349Parser< SymbolType, ValueType >::~Parser()
 350{
 351    // Private data destroy themselves.
 352}
 353
 354
 355/*=============================================================================
 356 |
 357 | NAME
 358 |
 359 |     Parser( const Parser & Parser )
 360 |
 361 | DESCRIPTION
 362 |
 363 |     Copy constructor.
 364 |
 365 +============================================================================*/
 366
 367template < typename SymbolType, typename ValueType >
 368Parser< SymbolType, ValueType >::Parser( const Parser & Parser )
 369: input_stack_( Parser.input_stack_ )
 370, parse_stack_( Parser.parse_stack_ )
 371{
 372}
 373
 374/*=============================================================================
 375 |
 376 | NAME
 377 |
 378 |    operator=( const PolyParser & PolyParser )
 379 |
 380 |
 381 | DESCRIPTION
 382 |
 383 |    Assignment operator.
 384 |
 385 +============================================================================*/
 386
 387template < typename SymbolType, typename ValueType >
 388Parser< SymbolType, ValueType > & Parser< SymbolType, ValueType >::operator=( const Parser & Parser )
 389{
 390    // Check for assigning to oneself:  just pass back a reference to the unchanged object.
 391    if (this == &Parser)
 392        return *this ;
 393    
 394    input_stack_ = Parser.input_stack_ ;
 395    parse_stack_ = Parser.parse_stack_ ;
 396    
 397    // Return reference to this object.
 398    return *this ;
 399}
 400
 401/*=============================================================================
 402 |
 403 | NAME
 404 |
 405 |     insertAction
 406 |
 407 | DESCRIPTION
 408 |
 409 |     Add a new entry to the Action Table.
 410 |
 411 +============================================================================*/
 412
 413template < typename SymbolType, typename ValueType >
 414void Parser< SymbolType, ValueType >::insertAction( int    state,      SymbolType terminal,
 415             Action actionType, int      actionState )
 416{
 417    try
 418    {
 419        // Create a new action.
 420        ActionState as( actionType, actionState ) ;
 421        
 422        // Get space for actionTable[ state ] which is an vector<ActionState> indexed by terminal.
 423        action_table_[ static_cast<int>( state ) ].resize( num_terminals_ ) ;
 424        
 425        // Insert it into the ACTION table.
 426        action_table_[ static_cast<int>( state ) ][ static_cast<int>( terminal ) ] = as ;
 427    }
 428    catch( length_error & e )
 429    {
 430        ostringstream os ;
 431        os << "Parser::insertAction had a length_error exception when trying to initialize action tables" ;
 432        throw ParserError( os.str(), __FILE__, __LINE__ ) ;
 433    }
 434    
 435}
 436
 437/*=============================================================================
 438 |
 439 | NAME
 440 |
 441 |    insertGoto
 442 |
 443 | DESCRIPTION
 444 |
 445 |     Add a new entry to the Goto Table.
 446 |
 447 +============================================================================*/
 448
 449template < typename SymbolType, typename ValueType >
 450void Parser< SymbolType, ValueType >::insertGoto( int state, SymbolType nonterm, int newState )
 451{
 452    try
 453    {
 454        // Insert into the GOTO table.
 455        goto_table_[ static_cast<int>( state ) ].resize( static_cast<int>( PolySymbol::NumSymbols ) ) ;
 456        goto_table_[ static_cast<int>( state ) ][ static_cast<int>( nonterm ) ] = newState ;
 457    }
 458    catch( length_error & e )
 459    {
 460        ostringstream os ;
 461        os << "Parser::insertGoto had a length_error exception while initializing goto tables" ;
 462        throw ParserError( os.str(), __FILE__, __LINE__ ) ;
 463    }
 464    
 465}
 466
 467/*=============================================================================
 468 |
 469 | NAME
 470 |
 471 |    insertProduction
 472 |
 473 | DESCRIPTION
 474 |
 475 |    Add a new entry to the list of productions.
 476 |
 477 +============================================================================*/
 478
 479template < typename SymbolType, typename ValueType >
 480void Parser< SymbolType, ValueType >::insertProduction( int prodNum, SymbolType nonTerm, int rhsLength )
 481{
 482    // Insert into the production table.
 483    production_[ prodNum ].type_  = nonTerm ;
 484    production_[ prodNum ].state_ = rhsLength ;
 485}
 486
 487/*=============================================================================
 488 |
 489 | NAME
 490 |
 491 |     insertErrorMessage
 492 |
 493 | DESCRIPTION
 494 |
 495 |     Insert an error message into the list.
 496 |
 497 +============================================================================*/
 498
 499template < typename SymbolType, typename ValueType >
 500void Parser< SymbolType, ValueType >::insertErrorMessage( int state, string errorMessage )
 501{
 502    error_message_[ state ] = errorMessage ;
 503}
 504
 505/*=============================================================================
 506 | 
 507 | NAME
 508 | 
 509 |     parse
 510 | 
 511 | DESCRIPTION
 512 | 
 513 |      Standard LALR(1) parser algorithm, parameterized with the type of
 514 |      symbols and values of its tokens.
 515 |
 516 |      The initial parser configuration is
 517 | 
 518 |             (s0 | a1 ... an $)
 519 | 
 520 |         a1 ... an is the set of input tokens
 521 |         s0 = 0 is the initial state
 522 | 
 523 |         The parse stack is to the left of the bar and the unprocessed
 524 |         input is to the right.  Now suppose the configuration is
 525 | 
 526 |             (s0 X1 ... Xm-r sm-r Xm-r+1 sm-r+1 ... Xm sm   |   ai ai+1 ... an $)
 527 | 
 528 |         There are four possible things we can do:
 529 | 
 530 |         (1)  Shift the input token and push a new state.
 531 | 
 532 |              ACTION[ sm, ai ] = shift s
 533 |
 534 |                  (s0 X1 ... Xm sm ai s   |   ai+1 ... an $)
 535 | 
 536 |         (2)  Reduce.  Pop the stack, push the reduction and new state.
 537 |              Also do a syntax-directed evaluation using the values
 538 |              of the symbols in the production A -> beta which we just reduced.
 539 | 
 540 |              ACTION[ sm, ai ] = reduce( A -> beta = Xm-r+1 ... Xm)
 541 |
 542 |                  (s0 X1 ... Xm-r sm-r A s   |   ai+1 ... an $)
 543 | 
 544 |              where s = GOTO[ sm-r, A ] and r = length( beta )
 545 | 
 546 |         (3)  Accept.  The sentence is in the grammar;  halt.
 547 | 
 548 |              ACTION[ sm, ai ] = accept
 549 | 
 550 |         (4)  Error state.   Produce an error message using the current
 551 |              parsing state and lookahead symbol ai.
 552 | 
 553 |              ACTION[ sm, ai ] = error
 554 | 
 555 +============================================================================*/
 556
 557template< typename SymbolType, typename ValueType >
 558ValueType Parser< SymbolType, ValueType >::parse( string sentence )
 559{
 560    ValueType retVal ; // Default to f(x) = 0, mod 0.
 561
 562    // Null string check.
 563    if (sentence.size() == 0)
 564        return retVal ;
 565
 566    // Tokenize the input string.  This can throw an exception.
 567    tokenize( sentence ) ;
 568
 569    // Initialize the parse stack to the zero state.
 570    parse_stack_.clear() ;
 571
 572    Symbol<SymbolType, ValueType> s( SymbolType::S, 0 ) ;
 573    parse_stack_.push_back( s ) ;
 574
 575    #ifdef DEBUG_PP_PARSER
 576    cout << "Parser:  sentence = " << sentence << endl ;
 577    #endif
 578
 579    // Parse until we accept, error out or run out of input.
 580    while( input_stack_.size() > 0 )
 581    {
 582        // Grab the current state and the lookahead token.
 583        int                             currentState = parse_stack_.back().state_ ;
 584        Symbol< SymbolType, ValueType > lookahead    = input_stack_.back() ;
 585
 586
 587        // Dump the parse and input stacks for debugging.
 588        #ifdef DEBUG_PP_PARSER
 589        cout << endl << "    Parse stack: " << endl ;
 590
 591        for (auto & i : parse_stack_)
 592            cout << "        " << i << endl ;
 593 
 594        cout << "    --------" << endl ;
 595        
 596        cout << "    Input stack: " << endl ;
 597		auto i = input_stack_.end()-1 ;
 598        while (i >= input_stack_.begin())
 599        {
 600            cout << "        " << *i <<  endl ;
 601			if (i == input_stack_.begin())
 602			    break ;
 603			--i ;
 604        }
 605        #endif
 606
 607        // Look up the action (shift 3), (reduce 2), etc.
 608        ActionState actionState = action_table_[ currentState ][ static_cast<int>( lookahead.type_ ) ] ;
 609
 610        // Debug dump the action.
 611        #ifdef DEBUG_PP_PARSER
 612        cout << endl << actionState << endl ;
 613        #endif
 614
 615        try
 616		{
 617			// Look at the action required.
 618			switch( actionState.action_ )
 619			{
 620				case Action::Shift:
 621				{
 622					//  Shift the next token off the input stack.
 623					if (input_stack_.size() == 0)
 624						// Throw a parser error if parse stack is empty (shouldn't get here).
 625						throw ParserError( "Wanted to shift another token, but the parse stack is empty.", __FILE__, __LINE__ ) ;
 626
 627					input_stack_.pop_back() ;
 628
 629					//  Push the input symbol and new state onto the parse stack.
 630					int newState = actionState.state_ ;
 631
 632					Symbol< SymbolType, ValueType > s( lookahead.type_, newState, lookahead.value_ ) ;
 633					parse_stack_.push_back( s ) ;
 634				}
 635				break ;
 636
 637				case Action::Reduce:
 638				{
 639					// Fetch the production A -> beta, beta = Xm-r+1 ... Xm
 640					int productionNum = actionState.state_ ;
 641
 642					// Get the left hand side non-terminal A.
 643					SymbolType prodNonTerm = production_[ productionNum ].type_ ;
 644
 645					// Get r, the length of beta.  If the production is A -> EPSILON, beta = EPSILON, r = 0.
 646					int rhsProductionLength = production_[ productionNum ].state_ ;
 647
 648					// Carry out the syntax directed translation using
 649					// the symbols of beta and its states
 650					// which are on the top of the stack and the nonterminal A.
 651
 652					// Creat a new symbol/state  A s
 653                    Symbol< SymbolType, ValueType > A( prodNonTerm,  0 ) ;
 654					int topIndex = static_cast<int>( parse_stack_.size() ) - 1 ;
 655					int i = 0 ;
 656                    
 657                    // Carry out the syntax directed translation f() using the values in the right hand
 658                    // side of the production:
 659                    //     A.value = f( Xm-r+1.value ... Xm.value )
 660                    syntaxDirectedTranslation( productionNum, topIndex, A ) ;
 661
 662					// Pop the symbols of beta and its states
 663					//     Xm-r+1 sm-r+1 ... Xm sm
 664					// off the stack.
 665					for (i = 1 ;  i <= rhsProductionLength ;  ++i)
 666					{
 667						if (input_stack_.size() == 0)
 668							return retVal ;
 669						parse_stack_.pop_back() ;
 670					}
 671
 672					// Get the GOTO state s.
 673					currentState = parse_stack_.back().state_ ;
 674					int gotoState = goto_table_[ static_cast<int>( currentState ) ][ static_cast<int>( prodNonTerm ) ] ;
 675
 676					// Push nonterminal A and goto state s.
 677					A.state_ = gotoState ;
 678					parse_stack_.push_back( A ) ;
 679				}
 680				break ;
 681
 682				// Parsed successfully.
 683				case Action::Accept:
 684					#ifdef DEBUG_PP_PARSER
 685					cout << "    Accept:  S.value = " << parse_stack_.back().value_ << endl ;
 686					#endif
 687
 688					return parse_stack_.back().value_ ;
 689				break ;
 690
 691				case Action::Error:
 692				{
 693                    #ifdef DEBUG_PP_PARSER
 694                    cout << "    Error: " << s << endl ;
 695                    #endif
 696
 697					// Throw a parser error with an error message based on the current state.
 698                    // Don't need to print the file name and line number since this is more of an
 699                    // error status than a surprise.
 700                    ostringstream os ;
 701                    os << error_message_[ currentState ] << " in sentence " << sentence ;
 702                    throw ParserError( os.str() ) ;
 703                    
 704					return retVal ;
 705				}
 706				break ;
 707
 708				default:
 709					// Internal error.
 710					return retVal ;
 711			}   // end switch
 712		
 713		}
 714		catch( length_error & e )
 715	    {
 716			ostringstream os ;
 717			os << "Parser::parse had a length_error exception during resizing poly or power terms" ;
 718			throw ParserError( os.str(), __FILE__, __LINE__ ) ;
 719	    }  
 720
 721    } // end while
 722
 723    // Shouldn't get here.
 724    return retVal ;
 725}
 726
 727
 728
 729/*------------------------------- Parser Child Classes ------------------------*/
 730
 731
 732
 733/*=============================================================================
 734 |
 735 | NAME
 736 |
 737 |     operator=( const PolyValue & v )
 738 |
 739 |
 740 | DESCRIPTION
 741 |
 742 |     Assignment operator for PolyValue.
 743 |
 744 +============================================================================*/
 745
 746PolyValue & PolyValue::operator=( const PolyValue & v )
 747{
 748    // Check for assigning to oneself:  just pass back a reference to the unchanged object.
 749    if (this == &v)
 750        return *this ;
 751    
 752    scalar_ = v.scalar_ ;
 753    f_ = v.f_ ;
 754    
 755    // Pass back the object so we can concatenate assignments.
 756    return *this ;
 757}
 758
 759
 760/*=============================================================================
 761 |
 762 | NAME
 763 |
 764 |     Operator << for PolyValue
 765 |
 766 | DESCRIPTION
 767 |
 768 |     Print the scalar and polynomial parts of a PolyValue type.
 769 |
 770 +============================================================================*/
 771
 772ostream & operator<<( ostream & out, const PolyValue & poly )
 773{
 774    out << poly.scalar_ << " f( x ) = " ;
 775    
 776    // Convert into a Polynomial type from a vector, then send to output.
 777    Polynomial p( poly.f_ ) ;
 778    out << p ;
 779        
 780    return out ;
 781}
 782
 783
 784/*=============================================================================
 785 |
 786 | NAME
 787 |
 788 |     PolyValue
 789 |
 790 | DESCRIPTION
 791 |
 792 |     Default constructor for a value which contains a scalar or polynomial.
 793 |
 794 +============================================================================*/
 795
 796PolyValue::PolyValue()
 797: scalar_( 0 )
 798, f_(1, 0) // f(x) = 0
 799{
 800}
 801
 802
 803/*=============================================================================
 804 |
 805 | NAME
 806 |
 807 |     PolyValue
 808 |
 809 | DESCRIPTION
 810 |
 811 |     Copy constructor for PolyValue.
 812 |
 813 +============================================================================*/
 814
 815PolyValue::PolyValue( const PolyValue & v )
 816: scalar_( v.scalar_ )
 817, f_( v.f_ )
 818{
 819}
 820
 821
 822/*=============================================================================
 823|
 824| NAME
 825|
 826|     PolyParser
 827|
 828| DESCRIPTION
 829|
 830|     Default constructor for the class.
 831|
 832+============================================================================*/
 833
 834template < typename SymbolType, typename ValueType >
 835PolyParser< SymbolType, ValueType >::PolyParser()
 836    : Parser< SymbolType, ValueType >()
 837    , test_polynomial_()
 838    , test_polynomial_for_primitivity_( false )
 839    , list_all_primitive_polynomials_( false )
 840    , print_operation_count_( false )
 841    , print_help_( false )
 842    , slow_confirm_( false )
 843    , p( 0 )
 844    , n( 0 )
 845{
 846    // Initialize all the parse tables.
 847    initializeTables() ;
 848}
 849
 850/*=============================================================================
 851 | 
 852 | NAME
 853 | 
 854 |     ~PolyParser
 855 | 
 856 | DESCRIPTION
 857 | 
 858 |     Default deeeestructor for the class.
 859 | 
 860 +============================================================================*/
 861
 862template < typename SymbolType, typename ValueType >
 863PolyParser< SymbolType, ValueType >::~PolyParser()
 864{
 865    // Private data destroy themselves.
 866}
 867
 868
 869
 870/*=============================================================================
 871 |
 872 | NAME
 873 |
 874 |     PolyParser( const PolyParser & PolyParser )
 875 |
 876 | DESCRIPTION
 877 |
 878 |     Copy constructor.
 879 |
 880 +============================================================================*/
 881
 882template < typename SymbolType, typename ValueType >
 883PolyParser<SymbolType, ValueType>::PolyParser( const PolyParser< SymbolType, ValueType > & PolyParser )
 884        : Parser<SymbolType, ValueType>( PolyParser )
 885{
 886}
 887
 888
 889/*=============================================================================
 890 | 
 891 | NAME
 892 | 
 893 |    operator=( const PolyParser & PolyParser )
 894 | 
 895 | 
 896 | DESCRIPTION
 897 | 
 898 |    Assignment operator.
 899 | 
 900 +============================================================================*/
 901        
 902template < typename SymbolType, typename ValueType >
 903PolyParser< SymbolType, ValueType > & PolyParser< SymbolType, ValueType >::operator=( const PolyParser & PolyParser )
 904{
 905    // Check for assigning to oneself:  just pass back a reference to the unchanged object.
 906    if (this == &PolyParser)
 907             return *this ;
 908
 909    input_stack_ = PolyParser.input_stack_ ;
 910    parse_stack_ = PolyParser.parse_stack_ ;
 911
 912    // Return reference to this object.
 913    return *this ;
 914}
 915
 916
 917/*=============================================================================
 918 | 
 919 | NAME
 920 | 
 921 |     initializeTables
 922 | 
 923 | DESCRIPTION
 924 | 
 925 |     Initialize ACTION, GOTO, and ERROR_MESSAGE tables for an LALR(1) parser.
 926 |     And also information about the productions.
 927 |
 928 +============================================================================*/
 929
 930template < typename SymbolType, typename ValueType >
 931    void PolyParser< SymbolType, ValueType >::
 932        initializeTables()
 933{
 934    // Need these so we can preallocate the vector lengths.  Vector's operator[] is unchecked!
 935    num_states_       = 15 ; // Including the 0 state.
 936    num_productions__  = 11 ;
 937    num_terminals_    = static_cast<int>(PolySymbol::NumTerminals) ;
 938    num_non_terminals_ = static_cast<int>(PolySymbol::NumSymbols) - static_cast<int>(PolySymbol::NumTerminals) - 1 ;  // Don't count NumTerminals enum itself.
 939
 940    // Expand the rows of the ACTION table to hold all the states.
 941    action_table_.resize( num_states_ ) ;
 942    
 943    // Expand the rows of the GOTO table to hold all the states.
 944    goto_table_.resize( num_states_ ) ;
 945    
 946    // Expand the rows of the ACTION table to hold all the states.
 947    // Allow space in 0th slot (unused).
 948    production_.resize( num_productions__ + 1 ) ;
 949    
 950    error_message_.resize( num_states_ ) ;
 951    
 952    // TODO:  does resize or operator[] throw any exceptions?  use vector.at( i ) for checked access?
 953    try
 954	{
 955		// For each state and each terminal, insert
 956        //    if Shift, the new state to push on the stack,
 957        //    if Reduce, the production number.
 958		insertAction( 0,    PolySymbol::Integer,    Action::Shift,       3 ) ;
 959		insertAction( 0,    PolySymbol::Ecks,       Action::Reduce,      8 ) ;
 960		insertAction( 0,    PolySymbol::Comma,      Action::Reduce,      8 ) ;
 961		insertAction( 0,    PolySymbol::Dollar,     Action::Reduce,      8 ) ;
 962		insertAction( 0,    PolySymbol::Plus,       Action::Reduce,      8 ) ;
 963
 964		insertAction( 1,    PolySymbol::Dollar,     Action::Accept,      0 ) ;
 965
 966		insertAction( 2,    PolySymbol::Comma,      Action::Shift,       7 ) ;
 967		insertAction( 2,    PolySymbol::Plus,       Action::Shift,       8 ) ;
 968		insertAction( 2,    PolySymbol::Dollar,     Action::Reduce,      3 ) ;
 969
 970		insertAction( 3,    PolySymbol::Ecks,       Action::Reduce,      7 ) ;
 971		insertAction( 3,    PolySymbol::Comma,      Action::Reduce,      7 ) ;
 972		insertAction( 3,    PolySymbol::Dollar,     Action::Reduce,      7 ) ;
 973		insertAction( 3,    PolySymbol::Plus,       Action::Reduce,      7 ) ;
 974
 975		insertAction( 4,    PolySymbol::Comma,      Action::Reduce,      5 ) ;
 976		insertAction( 4,    PolySymbol::Dollar,     Action::Reduce,      5 ) ;
 977		insertAction( 4,    PolySymbol::Plus,       Action::Reduce,      5 ) ;
 978
 979		insertAction( 5,    PolySymbol::Ecks,       Action::Shift,      10 ) ;
 980		insertAction( 5,    PolySymbol::Comma,      Action::Reduce,     11 ) ;
 981		insertAction( 5,    PolySymbol::Dollar,     Action::Reduce,     11 ) ;
 982		insertAction( 5,    PolySymbol::Plus,       Action::Reduce,     11 ) ;
 983
 984		insertAction( 6,    PolySymbol::Dollar,     Action::Reduce,      1 ) ;
 985
 986		insertAction( 7,    PolySymbol::Integer,    Action::Shift,      11 ) ;
 987
 988		insertAction( 8,    PolySymbol::Integer,    Action::Shift,       3 ) ;
 989		insertAction( 8,    PolySymbol::Ecks,       Action::Reduce,      8 ) ;
 990		insertAction( 8,    PolySymbol::Comma,      Action::Reduce,      8 ) ;
 991		insertAction( 8,    PolySymbol::Dollar,     Action::Reduce,      8 ) ;
 992		insertAction( 8,    PolySymbol::Plus,       Action::Reduce,      8 ) ;
 993
 994		insertAction( 9,    PolySymbol::Comma,      Action::Reduce,      6 ) ;
 995		insertAction( 9,    PolySymbol::Dollar,     Action::Reduce,      6 ) ;
 996		insertAction( 9,    PolySymbol::Plus,       Action::Reduce,      6 ) ;
 997
 998		insertAction( 10,   PolySymbol::Comma,      Action::Reduce,      9 ) ;
 999		insertAction( 10,   PolySymbol::Exp,        Action::Shift,      13 ) ;
1000		insertAction( 10,   PolySymbol::Dollar,     Action::Reduce,      9 ) ;
1001		insertAction( 10,   PolySymbol::Plus,       Action::Reduce,      9 ) ;
1002
1003		insertAction( 11,   PolySymbol::Dollar,     Action::Reduce,      2 ) ;
1004
1005		insertAction( 12,   PolySymbol::Comma,      Action::Reduce,      4 ) ;
1006		insertAction( 12,   PolySymbol::Dollar,     Action::Reduce,      4 ) ;
1007		insertAction( 12,   PolySymbol::Plus,       Action::Reduce,      4 ) ;
1008
1009		insertAction( 13,   PolySymbol::Integer,    Action::Shift,      14 ) ;
1010
1011		insertAction( 14,   PolySymbol::Comma,      Action::Reduce,     10 ) ;
1012		insertAction( 14,   PolySymbol::Dollar,     Action::Reduce,     10 ) ;
1013		insertAction( 14,   PolySymbol::Plus,       Action::Reduce,     10 ) ;
1014
1015
1016		// For each state and symbol insert a new state.
1017		insertGoto( 0,  PolySymbol::S,               1 ) ;
1018		insertGoto( 0,  PolySymbol::Poly,            2 ) ;
1019		insertGoto( 0,  PolySymbol::Term,            4 ) ;
1020		insertGoto( 0,  PolySymbol::Multiplier,      5 ) ;
1021
1022		insertGoto( 2,  PolySymbol::Mod,             6 ) ;
1023
1024		insertGoto( 5,  PolySymbol::Power,           9 ) ;
1025
1026		insertGoto( 8,  PolySymbol::Term,           12 ) ;
1027		insertGoto( 8,  PolySymbol::Multiplier,      5 ) ;
1028
1029
1030		// For each production (given by a unique number), insert the single nonterminal
1031        // which starts the production, and the length in symbols of the right hand side.
1032        // We'll need to know which production is being reduced so we can apply the syntax
1033        // directed translation, and pop the production off the stack.
1034		insertProduction(  1,      PolySymbol::S,           2 ) ; // S          -> Poly Mod
1035		insertProduction(  2,      PolySymbol::Mod,         2 ) ; // Mod        -> , integer
1036		insertProduction(  3,      PolySymbol::Mod,         0 ) ; // Mod        -> EPSILON
1037		insertProduction(  4,      PolySymbol::Poly,        3 ) ; // Poly       -> Poly + Term
1038		insertProduction(  5,      PolySymbol::Poly,        1 ) ; // Poly       -> Term
1039		insertProduction(  6,      PolySymbol::Term,        2 ) ; // Term       -> Multiplier Power
1040		insertProduction(  7,      PolySymbol::Multiplier,  1 ) ; // Multiplier -> integer
1041		insertProduction(  8,      PolySymbol::Multiplier,  0 ) ; // Multiplier -> EPSILON
1042		insertProduction(  9,      PolySymbol::Power,       1 ) ; // Power      -> x
1043		insertProduction( 10,      PolySymbol::Power,       3 ) ; // Power      -> x ^ integer
1044		insertProduction( 11,      PolySymbol::Power,       0 ) ; // Power      -> EPSILON
1045
1046        // For each state, provide an error message.
1047		insertErrorMessage(  0, "Expecting to see the start of the polynomial or next term or coefficient" ) ;
1048		insertErrorMessage(  1, "Expecting to see end of the polynomial" ) ;
1049		insertErrorMessage(  2, "Expecting to see mod or + term or , integer or end of polynomial" ) ;
1050		insertErrorMessage(  3, "Expecting to see x or , or end of the polynomial" ) ;
1051		insertErrorMessage(  4, "Expecting to see + or end of the polynomial" ) ;
1052		insertErrorMessage(  5, "Expecting to see a power after a coefficient or x or ," ) ;
1053		insertErrorMessage(  6, "Expecting to see ," ) ;
1054		insertErrorMessage(  7, "Expecting to see mod after ," ) ;
1055		insertErrorMessage(  8, "Expecting to see a term after a + or a term or coefficient" ) ;
1056		insertErrorMessage(  9, "Expecting to see , or end of polynomial after a term" ) ;
1057		insertErrorMessage( 10, "Expecting to see x^ or x or x ^ integer" ) ;
1058		insertErrorMessage( 11, "Expecting to see end of the polynomial after , integer" ) ;
1059		insertErrorMessage( 12, "Expecting to see , end of polynomial or + after a term" ) ;
1060		insertErrorMessage( 13, "Expecting to see an exponent after x ^" ) ;
1061		insertErrorMessage( 14, "Expecting to see , or + end of polynomial after x ^ integer" ) ;
1062	}
1063	catch( length_error & e)
1064	{
1065		ostringstream os ;
1066		os << "PolyParser::initializeTables had a length_error exception while initializing parse tables" ;
1067		throw ParserError( os.str(), __FILE__, __LINE__ ) ;
1068    }  
1069}
1070
1071
1072/*=============================================================================
1073 |
1074 | NAME
1075 |
1076 |    tokenize
1077 |
1078 | DESCRIPTION
1079 |
1080 |    This is the lexical analyzer.  Given an input string, convert it into tokens
1081 |    to fill the input stack.
1082 |
1083 |    Each token has a type, which is a nonterminal symbol, and its value.
1084 |
1085 |    This specific lexer converts a polynomial string.
1086 |
1087 +============================================================================*/
1088
1089template < typename SymbolType, typename ValueType >
1090void PolyParser< SymbolType, ValueType >::tokenize( string sentence )
1091{
1092	Symbol< SymbolType, ValueType > tok( PolySymbol::NumSymbols, 0 ) ;
1093
1094    // Starting position to scan.
1095    unsigned int pos = 0 ;
1096
1097    bool minusSignDetected = false ;
1098
1099    // Clear the token stack.
1100    input_stack_.clear() ;
1101    
1102    #ifdef DEBUG_PP_PARSER
1103    cout << "Lexical analyzer:  sentence = " << sentence << endl ;
1104    #endif
1105
1106    // Scan sentence to end.
1107    while( pos < sentence.length() )
1108    {
1109        // Skip whitespace.
1110        while (pos < sentence.length() && iswspace( sentence[ pos ]))
1111            ++pos ; // Advance to next token.
1112
1113        // Read an integer.
1114        if (pos < sentence.size() && isdigit( sentence[ pos ]))
1115        {
1116            unsigned int num = 0 ;
1117            while( pos < sentence.size() && isdigit( sentence[ pos ]))
1118            {
1119				char asciiDigit[ 2 ] = "\0" ; asciiDigit[ 0 ] = sentence[ pos ] ;
1120                int digit = atoi( asciiDigit ) ;
1121
1122                // Stop reading the next decimal digit if we're about to overflow.
1123                // Not really an exception but more of a user input error.
1124                if (num > (BigInt::getBase() - digit) / 10)
1125                {
1126                    ostringstream os ;
1127                    os << "Error:  number about to overflow in tokenizer at digit = " << digit ;
1128                    throw ParserError( os.str() ) ;
1129                }
1130
1131                num = 10 * num + digit ;
1132                ++pos ; // Advance to next token.
1133            }
1134
1135            tok.type_          = PolySymbol::Integer ;
1136            tok.value_.scalar_ = num ;
1137
1138            // Apply a minus sign to this number.
1139            if (minusSignDetected)
1140            {
1141                // Don't allow a minus sign for polynomial coefficients.
1142                // Not really an exception but more of a user input error.
1143                ostringstream os ;
1144                os << "Error:  negative number for a polynomial coefficient = -" << tok.value_.scalar_ << " is not allowed.  Numbers must be >= 0" ;
1145                throw ParserError( os.str() ) ;
1146
1147                // TODO If we do handle negative coefficients in the future, we'll do something like this:
1148                //     ModP modp( p ) ;
1149                //     tok.value_.scalar_ = modp( -tok.value_.scalar_ ) ;
1150                //     minusSignDetected = false ;
1151            }
1152        }
1153        // A non-digit symbol.
1154        else
1155        {
1156            tok.value_.scalar_ = 0 ;
1157
1158            // Read another type of token.
1159            switch( sentence[ pos ] )
1160            {
1161                case '+' :
1162                    tok.type_ = PolySymbol::Plus ;
1163                break ;
1164
1165                // Make this look like a plus symbol to the parser:  we'll let this lexer
1166                // apply it to the next integer we see.  If there isn't an integer following
1167                // the minus sign, it's an error for the parser anyway!
1168                case '-' :
1169                    tok.type_ = PolySymbol::Plus ;
1170                    minusSignDetected = true ;
1171                break ;
1172
1173                case '^' :
1174                    tok.type_ = PolySymbol::Exp ;
1175                break ;
1176
1177                case 'x' :
1178                case 'X' :
1179                    tok.type_ = PolySymbol::Ecks ;
1180                break ;
1181
1182                case ',' :
1183                    tok.type_ = PolySymbol::Comma ;
1184                break ;
1185                
1186                default:
1187                    // Throw a parser error of unexpected input if we see a bad symbol.
1188                    // Not really an exception but more of a user input error.
1189                    ostringstream os ;
1190                    os << "Error:  Unexpected symbol in lexical analyzer" << sentence[ pos ] ;
1191                    throw ParserError( os.str() ) ;
1192            } 
1193            // ----------------< from user >----------------------
1194
1195            ++pos ; // Advance to next token.
1196        } // end else
1197
1198        // Push token on stack.
1199        input_stack_.push_back( tok ) ;
1200
1201        #ifdef DEBUG_PP_PARSER
1202        cout << "    Push token " << tok << endl ;
1203        #endif
1204
1205    } // end while
1206
1207    // At end of input, push the $ terminator symbol.
1208    tok.type_ = PolySymbol::Dollar ;
1209    tok.value_.scalar_ = 0 ;
1210
1211    input_stack_.push_back( tok ) ;
1212
1213    #ifdef DEBUG_PP_PARSER
1214    cout << "    Last token " << tok << endl ;
1215    #endif
1216
1217    // Reverse all the symbols into our preferred order.
1218    reverse( input_stack_.begin(), input_stack_.end() ) ;
1219}
1220
1221/*=============================================================================
1222 |
1223 | NAME
1224 |
1225 |    syntaxDirectedTranslation
1226 |
1227 | DESCRIPTION
1228 |
1229 |     We have a reduce action   
1230 | 
1231 |         ACTION[ sm, ai ] = reduce( A -> beta = Xm-r+1 ... Xm )  
1232 |
1233 |     using the production
1234 |                        
1235 |         A -> Xm-r+1 ... Xm
1236 |                              
1237 |     Here's the stack before the reduction:
1238 |
1239 |                                parse_stack_[ topIndex-1 ]   parse_stack_[ topIndex ]
1240 |                                                        v   v
1241 |                                                        v   v
1242 |         (s0 X1 ...    Xm-r sm-r      Xm-r+1 sm-r+1 ... Xm sm     |     ai ai+1 ... an $)
1243 | 
1244 |     and after the reduction:
1245 |
1246 |         (s0 X1 ...    Xm-r sm-r      A s                         |      ai+1 ... an $)
1247 |
1248 |     Carry out the syntax directed translation f () using the values in the right hand
1249 |     side of the ith production,                i
1250 |
1251 |         A.value = f ( Xm-r+1.value ... Xm.value )
1252 |                    i 
1253 |
1254 +============================================================================*/
1255
1256template <typename SymbolType, typename ValueType>
1257void PolyParser<SymbolType, ValueType>::syntaxDirectedTranslation( int productionNum, int topIndex, 
1258                                                                   Symbol<SymbolType,ValueType> & symbol )
1259{
1260    switch( productionNum )
1261    {
1262        // Reduce (S -> POLY MOD)
1263        case 1:
1264        {
1265            // I'll explain this syntax-directed translation in detail.  The other cases are similar.
1266            // Looking at the production above, we have the right hand side symbols on the parse stack
1267            // along with their values:
1268            //     parse stack = (POLY MOD |
1269            // Now we compute the value of S (denoted by the variable as) from POLY and MOD.
1270            // S is the final result.  Its polynomial value is the value of POLY and its modulus is the
1271            // value of MOD.
1272            symbol.value_.f_      = parse_stack_[ topIndex - 1 ].value_.f_ ;
1273            symbol.value_.scalar_ = parse_stack_[ topIndex     ].value_.scalar_ ;
1274
1275            // ----------------< debug print from parse tables >----------------
1276            #ifdef DEBUG_PP_PARSER
1277            cout << "    Reduce (S -> POLY MOD)" << " Value = " << symbol.value_ << endl ;
1278            #endif
1279        }
1280        break ;
1281
1282        // Reduce (MOD -> COMMA INTEGER)
1283        case 2:
1284        {
1285            // The value of MOD is INTEGER (we ignore the comma).
1286            symbol.value_.scalar_ = parse_stack_[ topIndex ].value_.scalar_ ;
1287
1288            #ifdef DEBUG_PP_PARSER
1289            cout << "    Reduce (MOD -> COMMA INTEGER)" << " Value = " << symbol.value_ << endl ;
1290            #endif
1291        }
1292        break ;
1293
1294        // Reduce (MOD -> EPSILON)
1295        case 3:
1296        {
1297            // There isn't any value for MOD, so use the default modulus of 2.
1298            symbol.value_.scalar_ = 2 ;
1299
1300            #ifdef DEBUG_PP_PARSER
1301            cout << "    Reduce (MOD -> EPSILON)" << " Value = " << symbol.value_ << endl ;
1302            #endif
1303        }
1304        break ;
1305
1306        // Reduce (POLY -> POLY + TERM)
1307        case 4:
1308        {
1309            // A.poly = POLY + TERM
1310
1311            // Degree of POLY.  e.g. x+1 has degree 1, 3 has degree 0.
1312            int degPoly = static_cast<int>(parse_stack_[ topIndex - 2 ].value_.f_.size()) - 1 ;
1313
1314            // Degree of TERM
1315            int degTerm = static_cast<int>(parse_stack_[ topIndex ].value_.f_.size()) - 1 ;
1316
1317            // A.poly = POLY
1318            symbol.value_.f_ = parse_stack_[ topIndex - 2 ].value_.f_ ;
1319
1320            // Make POLY large enough to handle rhs POLY and TERM.
1321            symbol.value_.f_.resize( degTerm > degPoly ? degTerm+1 : degPoly+1, 0 ) ;
1322
1323            // A.poly += TERM
1324            for (int i = degTerm ;  i >= 0 ;  --i)
1325            {
1326                symbol.value_.f_[ i ] +=
1327                    parse_stack_[ topIndex ].value_.f_[ i ] ;
1328            }
1329
1330            #ifdef DEBUG_PP_PARSER
1331            cout << "    Reduce (POLY -> POLY + TERM)" << " Value = " << symbol.value_ << endl ;
1332            #endif
1333        }
1334        break ;
1335
1336        // Reduce (POLY -> TERM)
1337        case 5:
1338        {
1339            // Degree of POLY
1340            int degPoly = static_cast<int>(symbol.value_.f_.size()) - 1 ;
1341
1342            // Degree of TERM
1343            int degTerm = static_cast<int>(parse_stack_[ topIndex ].value_.f_.size()) - 1 ;
1344
1345            // Make POLY large enough to handle term's power.
1346            symbol.value_.f_.resize( degTerm > degPoly ? degTerm+1 : degPoly+1, 0 ) ;
1347
1348            // Zero the polynomial.
1349            for (int i = static_cast<int>(symbol.value_.f_.size()) - 1 ;  i >= 0 ;  --i)
1350                symbol.value_.f_[ i ] = 0 ;
1351
1352            // Add term.
1353            for (int i = static_cast<int>(symbol.value_.f_.size()) - 1 ;  i >= 0 ;  --i)
1354                symbol.value_.f_[ i ]  += parse_stack_[ topIndex ].value_.f_[ i ] ;
1355
1356            #ifdef DEBUG_PP_PARSER
1357            cout << "    Reduce (POLY -> TERM) " << " Value = " << symbol.value_ << endl ;
1358            #endif
1359        }
1360        break ;
1361
1362        // Reduce (TERM -> MULTIPLIER POWER)
1363        case 6:
1364        {
1365            // Degree of POWER.
1366            unsigned int degPower = static_cast<int>( parse_stack_[ topIndex ].value_.scalar_ ) ;
1367
1368            // Make TERM large enough to handle the power.
1369            symbol.value_.f_.resize( degPower + 1, 0 ) ;
1370
1371            // Zero TERM.
1372            for (int i = static_cast<int>(symbol.value_.f_.size()) - 1 ;  i >= 0 ;  --i)
1373                symbol.value_.f_[ i ] = 0 ;
1374
1375            // TERM = MULTIPLIER x ^ POWER
1376            symbol.value_.f_[ degPower ] =
1377            static_cast<int>( parse_stack_[ topIndex - 1 ].value_.scalar_ ) ;
1378
1379            #ifdef DEBUG_PP_PARSER
1380            cout << "    Reduce (TERM -> MULTIPLIER POWER)" << " Value = " << symbol.value_ << endl ;
1381            #endif
1382        }
1383        break ;
1384
1385        // Reduce (MULTIPLIER -> INTEGER)
1386        case 7:
1387        {
1388            // Set value of MULTIPLIER.
1389            symbol.value_.scalar_ = parse_stack_[ topIndex ].value_.scalar_ ;
1390
1391            #ifdef DEBUG_PP_PARSER
1392            cout << "    Reduce (MULTIPLIER -> INTEGER)" << " Value = " << symbol.value_ << endl ;
1393            #endif
1394        }
1395        break ;
1396
1397        // Reduce (MULTIPLIER -> EPSILON)
1398        case 8:
1399        {
1400            // Multiplier default = 1
1401            symbol.value_.scalar_ = 1 ;
1402
1403            #ifdef DEBUG_PP_PARSER
1404            cout << "    Reduce (MULTIPLIER -> EPSILON)" << " Value = " << symbol.value_ << endl ;
1405            #endif
1406        }
1407        break ;
1408
1409        // Reduce (POWER -> X)
1410        case 9:
1411        {
1412            // POWER is the exponent of x, so we
1413            // interpret as a scalar.
1414            symbol.value_.scalar_ = 1 ;
1415
1416            #ifdef DEBUG_PP_PARSER
1417            cout << "    Reduce  (POWER -> X)" << " Value = " << symbol.value_ << endl ;
1418            #endif
1419        }
1420        break ;
1421
1422        // Reduce (POWER -> X ^ INTEGER)
1423        case 10:
1424        {
1425            // POWER = value of INTEGER.
1426            symbol.value_.scalar_ = parse_stack_[ topIndex ].value_.scalar_ ;
1427
1428            #ifdef DEBUG_PP_PARSER
1429            cout << "    Reduce (POWER -> X ^ INTEGER)" << " Value = " << symbol.value_ << endl ;
1430            #endif
1431        }
1432        break ;
1433
1434        // Reduce (POWER -> EPSILON)
1435        case 11:
1436        {
1437            // Default power is 0.
1438            symbol.value_.scalar_ = 0 ;
1439
1440            #ifdef DEBUG_PP_PARSER
1441            cout << "    Reduce (POWER -> EPSILON)" << "Value = " << symbol.value_ << endl ;
1442            #endif
1443        }
1444        break ;
1445    }
1446}
1447
1448
1449
1450/*------------------------------- Parser Child Classes ------------------------*/
1451
1452
1453
1454/*=============================================================================
1455 |
1456 | NAME
1457 |
1458 |     operator=( const FactorizationValue & v )
1459 |
1460 |
1461 | DESCRIPTION
1462 |
1463 |     Assignment operator for FactorizationValue.
1464 |
1465 +============================================================================*/
1466
1467template<typename IntType>
1468FactorizationValue<IntType> & FactorizationValue<IntType>::operator=( const FactorizationValue<IntType> & v )
1469{
1470    // Check for assigning to oneself:  just pass back a reference to the unchanged object.
1471    if (this == &v)
1472        return *this ;
1473    
1474    numberString_ = v.numberString_ ;
1475    factor_ = v.factor_ ;
1476
1477    // Pass back the object so we can concatenate assignments.
1478    return *this ;
1479}
1480
1481
1482
1483/*=============================================================================
1484 |
1485 | NAME
1486 |
1487 |     numberStringToInteger
1488 |
1489 |
1490 | DESCRIPTION
1491 |
1492 |     Convert a number string (sequence of decimal digits) to an integer.
1493 |
1494 +============================================================================*/
1495
1496template<typename IntType>
1497IntType FactorizationValue<IntType>::numberStringToInteger( const string & numberString )
1498{
1499    // First define a generic integer type.
1500    IntType u ;
1501    
1502    // Create a string stream and load it with the number string.
1503    istringstream is ;
1504    is.clear() ;
1505    is.str( numberString ) ;
1506    
1507    // Read the generic integer from the stream using the >> operator which will see IntType and call either of:
1508    //     The BigInt class method for operator>>
1509    //     The C++ language method for the >> operator using the type ppuint which is unsigned long long or unsigned int
1510    is >> u ;
1511
1512    return u ;
1513}
1514
1515
1516/*=============================================================================
1517 |
1518 | NAME
1519 |
1520 |     FactorizationValue
1521 |
1522 | DESCRIPTION
1523 |
1524 |     Default constructor for a value which contains a scalar or polynomial.
1525 |
1526 +============================================================================*/
1527
1528template<typename IntType>
1529FactorizationValue<IntType>::FactorizationValue()
1530    : numberString_ { "" }
1531	, factor_ { 0 }
1532{
1533}
1534
1535
1536/*=============================================================================
1537 |
1538 | NAME
1539 |
1540 |     FactorizationValue
1541 |
1542 | DESCRIPTION
1543 |
1544 |     Default constructor for a value which contains a scalar or polynomial.
1545 |
1546 +============================================================================*/
1547
1548template<typename IntType>
1549FactorizationValue<IntType>::FactorizationValue( const IntType & p, const int & count )
1550: numberString_{ "" }
1551{
1552    PrimeFactor<IntType> factor( p, count ) ;
1553    factor_.clear() ;
1554    factor_.push_back( factor ) ;
1555}
1556
1557
1558/*=============================================================================
1559 |
1560 | NAME
1561 |
1562 |     FactorizationValue
1563 |
1564 | DESCRIPTION
1565 |
1566 |     Copy constructor for FactorizationValue.
1567 |
1568 +============================================================================*/
1569
1570template<typename IntType>
1571FactorizationValue<IntType>::FactorizationValue( const FactorizationValue<IntType> & v )
1572: numberString_( v.numberString_ )
1573, factor_( v.factor_ )
1574{
1575}
1576
1577
1578/*=============================================================================
1579|
1580| NAME
1581|
1582|     FactorizationParser
1583|
1584| DESCRIPTION
1585|
1586|     Default constructor for the class.
1587|
1588+============================================================================*/
1589
1590template < typename SymbolType, typename ValueType >
1591FactorizationParser< SymbolType, ValueType >::FactorizationParser()
1592    : Parser<SymbolType,ValueType>()
1593{
1594    // Initialize all the parse tables.
1595    initializeTables() ;
1596}
1597
1598
1599/*=============================================================================
1600 | 
1601 | NAME
1602 | 
1603 |     ~FactorizationParser
1604 | 
1605 | DESCRIPTION
1606 | 
1607 |     Default deeeestructor for the class.
1608 | 
1609 +============================================================================*/
1610
1611template < typename SymbolType, typename ValueType >
1612FactorizationParser< SymbolType, ValueType >::~FactorizationParser()
1613{
1614    // Private data destroy themselves.
1615}
1616
1617
1618/*=============================================================================
1619 |
1620 | NAME
1621 |
1622 |     FactorizationParser( const FactorizationParser & FactorizationParser )
1623 |
1624 | DESCRIPTION
1625 |
1626 |     Copy constructor.
1627 |
1628 +============================================================================*/
1629
1630template < typename SymbolType, typename ValueType >
1631FactorizationParser<SymbolType, ValueType>::FactorizationParser( const FactorizationParser< SymbolType, ValueType > & FactorizationParser )
1632        : Parser<SymbolType, ValueType>( FactorizationParser )
1633{
1634}
1635
1636
1637/*=============================================================================
1638 | 
1639 | NAME
1640 | 
1641 |    operator=( const FactorizationParser & FactorizationParser )
1642 | 
1643 | 
1644 | DESCRIPTION
1645 | 
1646 |    Assignment operator.
1647 | 
1648 +============================================================================*/
1649        
1650template < typename SymbolType, typename ValueType >
1651FactorizationParser< SymbolType, ValueType > & FactorizationParser< SymbolType, ValueType >::operator=( const FactorizationParser & FactorizationParser )
1652{
1653    // Check for assigning to oneself:  just pass back a reference to the unchanged object.
1654    if (this == &FactorizationParser)
1655             return *this ;
1656
1657    input_stack_ = FactorizationParser.input_stack_ ;
1658    parse_stack_ = FactorizationParser.parse_stack_ ;
1659
1660    // Return reference to this object.
1661    return *this ;
1662}
1663
1664
1665/*=============================================================================
1666 | 
1667 | NAME
1668 | 
1669 |     initializeTables
1670 | 
1671 | DESCRIPTION
1672 | 
1673 |     Initialize ACTION, GOTO, and ERROR_MESSAGE tables for an LALR(1) parser.
1674 |     And also information about the productions.
1675 |
1676 +============================================================================*/
1677
1678template < typename SymbolType, typename ValueType >
1679void FactorizationParser<SymbolType,ValueType>::initializeTables()
1680{
1681    // Need these so we can preallocate the vector lengths.  Vector's operator[] is unchecked!
1682    num_states_       = 16 ; // Including the 0 state.
1683    num_productions__  =  7 ;
1684    num_terminals_    = static_cast<int>(FactorizationSymbol::NumTerminals) ;
1685    num_non_terminals_ = static_cast<int>(FactorizationSymbol::NumSymbols) - static_cast<int>(FactorizationSymbol::NumTerminals) - 1 ;  // Don't count the NumTerminals enum itself.
1686
1687    // Expand the rows of the ACTION table to hold all the states.
1688    action_table_.resize( num_states_ ) ;
1689    
1690    // Expand the rows of the GOTO table to hold all the states.
1691    goto_table_.resize( num_states_ ) ;
1692    
1693    // Expand the rows of the ACTION table to hold all the states.
1694    // Allow space in 0th slot (unused).
1695    production_.resize( num_productions__ + 1 ) ;
1696    
1697    error_message_.resize( num_states_ ) ;
1698    
1699    // TODO:  does resize or operator[] throw any exceptions?  use vector.at( i ) for checked access?
1700    try
1701	{
1702		// For each state and each terminal, insert
1703        //    if Shift, the new state to push on the stack,
1704        //    if Reduce, the production number.
1705		insertAction(  0,  FactorizationSymbol::Integer,   Action::Shift,   2 ) ;
1706		insertAction(  1,  FactorizationSymbol::Dollar,    Action::Accept,  0 ) ;
1707		insertAction(  2,  FactorizationSymbol::Integer,   Action::Shift,   3 ) ;
1708		insertAction(  3,  FactorizationSymbol::Integer,   Action::Shift,   4 ) ;
1709
1710		insertAction(  4,  FactorizationSymbol::Caret,     Action::Reduce,  7 ) ;
1711		insertAction(  4,  FactorizationSymbol::Dollar,    Action::Reduce,  7 ) ;
1712        insertAction(  4,  FactorizationSymbol::Period,    Action::Reduce,  7 ) ;
1713        insertAction(  4,  FactorizationSymbol::Backslash, Action::Reduce,  7 ) ;
1714
1715		insertAction(  5,  FactorizationSymbol::Dollar,    Action::Reduce,  1 ) ;
1716		insertAction(  5,  FactorizationSymbol::Period,    Action::Shift,   8 ) ;
1717
1718		insertAction(  6,  FactorizationSymbol::Dollar,    Action::Reduce,  3 ) ;
1719		insertAction(  6,  FactorizationSymbol::Period,    Action::Reduce,  3 ) ;
1720
1721		insertAction(  7,  FactorizationSymbol::Caret,     Action::Shift,   9 ) ;
1722        insertAction(  7,  FactorizationSymbol::Dollar,    Action::Reduce,  5 ) ;
1723        insertAction(  7,  FactorizationSymbol::Period,    Action::Reduce,  5 ) ;
1724		insertAction(  7,  FactorizationSymbol::Backslash, Action::Shift,  10 ) ;
1725
1726		insertAction(  8,  FactorizationSymbol::Integer,   Action::Shift,   4 ) ;
1727 
1728		insertAction(  9,  FactorizationSymbol::Integer,   Action::Shift,   4 ) ;
1729
1730		insertAction( 10,  FactorizationSymbol::Integer,   Action::Shift,  14 ) ;
1731
1732		insertAction( 11,  FactorizationSymbol::Dollar,    Action::Reduce,  2 ) ;
1733		insertAction( 11,  FactorizationSymbol::Period,    Action::Reduce,  2 ) ;
1734
1735		insertAction( 13,  FactorizationSymbol::Dollar,    Action::Reduce,  4 ) ;
1736		insertAction( 13,  FactorizationSymbol::Period,    Action::Reduce,  4 ) ;
1737		insertAction( 13,  FactorizationSymbol::Backslash, Action::Shift,  10 ) ;
1738
1739		insertAction( 14,  FactorizationSymbol::Caret,     Action::Reduce,  6 ) ;
1740		insertAction( 14,  FactorizationSymbol::Dollar,    Action::Reduce,  6 ) ;
1741        insertAction( 14,  FactorizationSymbol::Period,    Action::Reduce,  6 ) ;
1742        insertAction( 14,  FactorizationSymbol::Backslash, Action::Reduce,  6 ) ;
1743
1744
1745		// For each state and symbol insert a new state.
1746		insertGoto( 0,  FactorizationSymbol::S,               1 ) ;
1747
1748		insertGoto( 3,  FactorizationSymbol::Factorization,   5 ) ;
1749		insertGoto( 3,  FactorizationSymbol::Factor,          6 ) ;
1750		insertGoto( 3,  FactorizationSymbol::BigInteger,      7 ) ;
1751
1752		insertGoto( 8,  FactorizationSymbol::Factor,         11 ) ;
1753		insertGoto( 8,  FactorizationSymbol::BigInteger,      7 ) ;
1754
1755		insertGoto( 9,  FactorizationSymbol::BigInteger,     13 ) ;
1756
1757
1758		// For each production (given by a unique number), insert the single nonterminal
1759        // which starts the production, and the length in symbols of the right hand side.
1760        // We'll need to know which production is being reduced so we can apply the syntax
1761        // directed translation, and pop the production off the stack.
1762		insertProduction(  1,      FactorizationSymbol::S,               3 ) ; // S -> INTEGER INTEGER FACTORIZATION
1763		insertProduction(  2,      FactorizationSymbol::Factorization,   3 ) ; // FACTORIZATION -> FACTORIZATION PERIOD FACTOR
1764		insertProduction(  3,      FactorizationSymbol::Factorization,   1 ) ; // FACTORIZATION -> FACTOR
1765		insertProduction(  4,      FactorizationSymbol::Factor,          3 ) ; // FACTOR -> BIGINTEGER ^ BIGINTEGER
1766		insertProduction(  5,      FactorizationSymbol::Factor,          1 ) ; // FACTOR -> BIGINTEGER
1767		insertProduction(  6,      FactorizationSymbol::BigInteger,      3 ) ; // BIGINTEGER -> BIGINTEGER \ INTEGER
1768		insertProduction(  7,      FactorizationSymbol::BigInteger,      1 ) ; // BIGINTEGER -> INTEGER
1769
1770        // For each state, provide an error message.
1771        insertErrorMessage(  0, "Expecting to see the power n." ) ;
1772        insertErrorMessage(  1, "Expecting end of input." ) ;
1773        insertErrorMessage(  2, "Expecting to see the number of prime factors." ) ;
1774        insertErrorMessage(  3, "Expecting an integer." ) ;
1775        insertErrorMessage(  4, "Expecting integer continuation \\ or . followed by a factor or ^ followed by a power or end of input." ) ;
1776        insertErrorMessage(  5, "Expecting another factor after the . or the end of the factorization." ) ;
1777        insertErrorMessage(  6, "Expecting a ." ) ;
1778        insertErrorMessage(  7, "Expecting integer continuation \\ or . followed by a factor or a ^ follwed by a power or end of input." ) ;
1779        insertErrorMessage(  8, "Expecting factor or an integer." ) ;
1780        insertErrorMessage(  9, "Expecting an integer." ) ;
1781        insertErrorMessage( 10, "Expecting an integer after the continuation \\." ) ;
1782        insertErrorMessage( 11, "Expecting . and another factor or end of input." ) ;
1783        insertErrorMessage( 13, "Expecting integer continuation \\ or . and next factor or end of input." ) ;
1784        insertErrorMessage( 14, "Expecting . and next factor or ^ and power or end of input after integer continuation \\." ) ;
1785
1786	}
1787	catch( length_error & e)
1788	{
1789		ostringstream os ;
1790		os << "FactorizationParser::initializeTables had a length_error exception while initializing parse tables" ;
1791		throw ParserError( os.str(), __FILE__, __LINE__ ) ;
1792    }  
1793}
1794
1795
1796/*=============================================================================
1797 |
1798 | NAME
1799 |
1800 |    tokenize
1801 |
1802 | DESCRIPTION
1803 |
1804 |    This is the lexical analyzer.  Given an input string, convert it into tokens
1805 |    to fill the input stack.
1806 |
1807 |    Each token has a type, which is a nonterminal symbol, and its value.
1808 |
1809 |    This specific lexer converts one line of a factorization table.
1810 |
1811 +============================================================================*/
1812
1813template <typename SymbolType, typename ValueType> 
1814void FactorizationParser<SymbolType, ValueType>::tokenize( string sentence )
1815{
1816	Symbol<SymbolType, ValueType> tok( FactorizationSymbol::NumSymbols, 0 ) ;
1817
1818    // Starting position to scan.
1819    unsigned int pos = 0 ;
1820
1821    // Clear the token stack.
1822    input_stack_.clear() ;
1823
1824    // Scan sentence to end.
1825    while( pos < sentence.length() )
1826    {
1827        // Skip whitespace.
1828        while (pos < sentence.length() && iswspace( sentence[ pos ]))
1829            ++pos ; // Advance to next token.
1830
1831        #ifdef DEBUG_PP_PARSER
1832        cout << "tokenize:  sentence[ " << pos << " ] = " << sentence[ pos ] << endl ;
1833        #endif
1834
1835        // We see a digit.
1836        if (pos < sentence.size() && isdigit( sentence[ pos ]))
1837        {
1838            string numberString = "" ;
1839
1840            // Read a sequence of digits until the next non-digit.
1841            while( pos < sentence.size() && isdigit( sentence[ pos ]))
1842            {
1843                numberString += sentence[ pos ] ;
1844                ++pos ; // Advance to next token.
1845            }
1846
1847            tok.type_                = FactorizationSymbol::Integer ;
1848            tok.value_.numberString_ = numberString ;
1849        }
1850        // A non-digit symbol.
1851        else
1852        {
1853            tok.value_.numberString_ = "" ;
1854
1855            // Read another type of token.
1856            switch( sentence[ pos ] )
1857            {
1858                case '.' :
1859                    tok.type_ = FactorizationSymbol::Period ;
1860                break ;
1861
1862                case '^' :
1863                    tok.type_ = FactorizationSymbol::Caret ;
1864                break ;
1865
1866                case '\\' :
1867                    tok.type_ = FactorizationSymbol::Backslash ;
1868                break ;
1869
1870                default:
1871                    // Throw a parser error of unexpected input if we see a bad symbol.
1872                    // Not really an exception but more of a user input error.
1873                    ostringstream os ;
1874                    os << "Error:  Unexpected symbol in lexical analyzer" << sentence[ pos ] ;
1875                    throw ParserError( os.str() ) ;
1876            } 
1877
1878            ++pos ; // Advance to next token.
1879        } // end else
1880
1881        // Push token on stack.
1882        input_stack_.push_back( tok ) ;
1883
1884        #ifdef DEBUG_PP_PARSER
1885        cout << "    Push token " << tok << endl ;
1886        #endif
1887
1888    } // end while
1889
1890    // At end of input, push the $ terminator symbol.
1891    tok.type_ = FactorizationSymbol::Dollar ;
1892    tok.value_.numberString_ = "" ;
1893
1894    input_stack_.push_back( tok ) ;
1895
1896    #ifdef DEBUG_PP_PARSER
1897    cout << "    Push last token " << tok << endl ;
1898    #endif
1899
1900    // Reverse all the symbols into our preferred order.
1901    reverse( input_stack_.begin(), input_stack_.end() ) ;
1902}
1903
1904/*=============================================================================
1905 |
1906 | NAME
1907 |
1908 |    syntaxDirectedTranslation
1909 |
1910 | DESCRIPTION
1911 |
1912 |     We have a reduce action   
1913 | 
1914 |         ACTION[ sm, ai ] = reduce( A -> beta = Xm-r+1 ... Xm )  
1915 |
1916 |     using the production
1917 |                        
1918 |         A -> Xm-r+1 ... Xm
1919 |                              
1920 |     Here's the stack before the reduction:
1921 |
1922 |                                parse_stack_[ topIndex-1 ]   parse_stack_[ topIndex ]
1923 |                                                        v   v
1924 |                                                        v   v
1925 |         (s0 X1 ...    Xm-r sm-r      Xm-r+1 sm-r+1 ... Xm sm     |     ai ai+1 ... an $)
1926 | 
1927 |     and after the reduction:
1928 |
1929 |         (s0 X1 ...    Xm-r sm-r      A s                         |      ai+1 ... an $)
1930 |
1931 |     Carry out the syntax directed translation f () using the values in the right hand
1932 |     side of the ith production,                i
1933 |
1934 |         A.value = f ( Xm-r+1.value ... Xm.value )
1935 |                    i 
1936 |
1937 +============================================================================*/
1938
1939template <typename SymbolType, typename ValueType>
1940void FactorizationParser<SymbolType,ValueType>::
1941         syntaxDirectedTranslation( int productionNum, int topIndex, Symbol<SymbolType,ValueType> & symbol )
1942{
1943    switch( productionNum )
1944    {
1945        // Reduce (S -> INTEGER INTEGER FACTORIZATION)
1946        case 1:
1947        {
1948            // I'll explain this syntax-directed translation in detail.  The other cases are similar.
1949            // The parse stack contains the right hand side symbols (types and values) of the production:
1950            //     parse stack = (INTEGER INTEGER FACTORIZATION |
1951            // We compute the value of S (denoted by the variable symbol) from the right hand side.
1952            //
1953            symbol.value_.factor_       = parse_stack_[ topIndex     ].value_.factor_ ; // Factorization.
1954            symbol.value_.numberString_ = parse_stack_[ topIndex - 2 ].value_.numberString_ ; // Value of n.
1955
1956            #ifdef DEBUG_PP_PARSER
1957            cout << "    Reduce (S -> INTEGER INTEGER FACTORIZATION)" << " Value = " << symbol << endl ;
1958            #endif
1959        }
1960        break ;
1961
1962        // Reduce (FACTORIZATION -> FACTORIZATION PERIOD FACTOR)
1963        case 2:
1964        {
1965            // Append the new prime factor to the end of the factorization.
1966            symbol.value_.factor_ = parse_stack_[ topIndex - 2 ].value_.factor_ ;
1967            symbol.value_.factor_.push_back( parse_stack_[ topIndex ].value_.factor_[ 0 ] ) ;
1968
1969            #ifdef DEBUG_PP_PARSER
1970            cout << "    Reduce (FACTORIZATION -> FACTORIZATION PERIOD FACTOR)" << " Value = " << symbol.value_ << endl ;
1971            #endif
1972        }
1973        break ;
1974
1975        // Reduce (FACTORIZATION -> FACTOR)
1976        case 3:
1977        {
1978            // Factorization has only one factor.
1979            symbol.value_.factor_.clear() ;
1980            symbol.value_.factor_.push_back( parse_stack_[ topIndex ].value_.factor_[ 0 ] ) ;
1981            
1982            #ifdef DEBUG_PP_PARSER
1983            cout << "    Reduce (FACTORIZATION -> FACTOR)" << " Value = " << symbol.value_ << endl ;
1984            #endif
1985        }
1986        break ;
1987
1988        // Reduce (FACTOR -> BIGINTEGER ^ BIGINTEGER)
1989        case 4:
1990        {
1991            // Create a new value and load it with the prime factor and its count.
1992            auto primeFactor = ValueType::numberStringToInteger( parse_stack_[ topIndex - 2 ].value_.numberString_ ) ;
1993            auto primeCount  = static_cast<int>( ValueType::numberStringToInteger( parse_stack_[ topIndex ].value_.numberString_ ) ) ;
1994            ValueType value( primeFactor, primeCount ) ;
1995            
1996            // Factorization has only one factor.
1997            symbol.value_.factor_.clear() ;
1998            symbol.value_.factor_.push_back( value.factor_[ 0 ] ) ;
1999
2000            #ifdef DEBUG_PP_PARSER
2001            cout << "    Reduce (FACTOR -> BIGINTEGER ^ BIGINTEGER)" << " Value = " << symbol.value_ << endl ;
2002            #endif
2003        }
2004        break ;
2005
2006        // Reduce (FACTOR -> BIGINTEGER)
2007        case 5:
2008        {
2009            // Create a new value which has a single prime factor.
2010            auto primeFactor = ValueType::numberStringToInteger( parse_stack_[ topIndex ].value_.numberString_ ) ;
2011            ValueType value( primeFactor, 1 ) ;
2012            
2013            // Factorization has only one factor.
2014            symbol.value_.factor_.clear() ;
2015            symbol.value_.factor_.push_back( value.factor_[ 0 ] ) ;
2016
2017            #ifdef DEBUG_PP_PARSER
2018            cout << "    Reduce (FACTOR -> BIGINTEGER) " << " Value = " << symbol.value_ << endl ;
2019            #endif
2020        }
2021        break ;
2022
2023        // Reduce (BIGINTEGER -> BIGINTEGER \ INTEGER)
2024        case 6:
2025        {
2026            // Concatenate the number strings.  Remember, the production right hand side is sitting on the parse stack and the rightmost
2027            // token is at the top of the stack.
2028            symbol.value_.numberString_ = parse_stack_[ topIndex - 2 ].value_.numberString_ + parse_stack_[ topIndex ].value_.numberString_ ;
2029            
2030            #ifdef DEBUG_PP_PARSER
2031            cout << "    Reduce (BIGINTEGER -> BIGINTEGER \\ INTEGER)" << " Value = " << symbol.value_ << endl ;
2032            #endif
2033        }
2034        break ;
2035
2036        // Reduce (BIGINTEGER -> INTEGER)
2037        case 7:
2038        {
2039            // Initial number string.
2040            symbol.value_.numberString_ = parse_stack_[ topIndex ].value_.numberString_ ;
2041            
2042            #ifdef DEBUG_PP_PARSER
2043            cout << "    Reduce (BIGINTEGER -> INTEGER)" << " Value = " << symbol.value_ << endl ;
2044            #endif
2045        }
2046        break ;
2047    }
2048}
2049
2050
2051
2052/*=============================================================================
2053 | 
2054 | NAME
2055 | 
2056 |     parseCommandLine
2057 | 
2058 | DESCRIPTION
2059 | 
2060 |      Parse the command line.
2061 | 
2062 | 
2063 | EXAMPLE
2064 | 
2065 |    pp -h
2066 |    pp -s 2 4
2067 |    pp -t 2 4 x^3+x^2+1                 // No blanks, please!  Looks like
2068 |                                        // several command line arguments.
2069 |    pp -t 2 4 "x^  3  +  x ^ 2  +  1"   // Blanks OK now.
2070 | 
2071 +============================================================================*/
2072
2073template < typename SymbolType, typename PolyValueType >
2074void PolyParser< SymbolType, PolyValueType >::parseCommandLine( int argc, const char * argv[] )
2075{
2076    const char * input_arg_string ;
2077    const char * option_ptr ;
2078
2079    int          num_arg{ 0 } ;
2080    const char * arg_string[ MAX_NUM_COMMAND_LINE_ARGS ] ;
2081    memset( arg_string, 0, sizeof( arg_string ) ) ;
2082
2083    /*  Initialize to defaults. */
2084    test_polynomial_for_primitivity_ = false ;
2085    list_all_primitive_polynomials_  = false ;
2086    print_operation_count_          = false ;
2087    print_help_                    = false ;
2088    slow_confirm_                  = false ;
2089    p                             = 0 ;
2090    n                             = 0 ;
2091
2092    current_working_dir_ = current_path().string() ;
2093    
2094    //  Parse the command line to get the options and their inputs.
2095    for (int input_arg_index = 0 ;  input_arg_index < argc ;
2096         ++input_arg_index)
2097    {
2098        /*  Get next argument string. */
2099        input_arg_string = argv[ input_arg_index ] ;
2100
2101        /* We have an option:  a hyphen followed by a non-null string. */
2102        if (input_arg_string[ 0 ] == '-' && input_arg_string[ 1 ] != '\0')
2103        {
2104            /* Scan all options. */
2105            for (option_ptr = input_arg_string + 1 ;  *option_ptr != '\0' ;
2106                 ++option_ptr)
2107            {
2108                switch( *option_ptr )
2109                {
2110                    /* Test a given polynomial for primitivity. */
2111                    case 't':
2112                        test_polynomial_for_primitivity_ = true ;
2113                    break ;
2114
2115                    /* List all primitive polynomials.  */
2116                    case 'a':
2117                       list_all_primitive_polynomials_ = true ;
2118                    break ;
2119
2120                    /* Print statistics on program operation. */
2121                    case 's':
2122                       print_operation_count_ = true ;
2123                    break ;
2124
2125                    /* Print help. */
2126                    case 'h':
2127                    case 'H':
2128                        print_help_ = true ;
2129                    break ;
2130
2131                    /* Turn on all self-checking (slow!).  */
2132                    case 'c':
2133                        slow_confirm_ = true ;
2134                    break ;
2135
2136                    default:
2137                       ostringstream os ;
2138                       os << "Cannot recognize the option" << *option_ptr ;
2139                       throw ParserError( os.str() ) ;
2140                    break ;
2141                }
2142            }
2143        }
2144        else  /* Not an option, but an argument. */
2145        {
2146            if (num_arg >= MAX_NUM_COMMAND_LINE_ARGS)
2147            {
2148                ostringstream os ;
2149                os << "Too many command line arguments = " << num_arg << " >= MAX_NUM_COMMAND_LINE_ARGS = " << MAX_NUM_COMMAND_LINE_ARGS ;
2150                throw ParserError( os.str() ) ;
2151            }
2152            else
2153                arg_string[ num_arg++ ] = input_arg_string ;
2154        }
2155    }
2156
2157    // User specified a polynomial to test.  First argument is program name, next is
2158    // the polynomial.
2159    if (test_polynomial_for_primitivity_)
2160    {
2161        string testPoly ;
2162        testPoly.assign( arg_string[ 1 ] ) ; // stringify it.
2163        test_polynomial_ = testPoly ;         // We've range checked n and p here already.
2164      
2165        n = test_polynomial_.deg() ;
2166        p = test_polynomial_.modulus() ;
2167    }
2168    // We had the arguments arg_string[0] = <programe name>
2169    //                      arg_string[1]= p
2170    //                      arg_string[2]= n
2171    // When we come here num_arg = 3 after the final num_arg++ above.
2172    else if (num_arg == MAX_NUM_COMMAND_LINE_ARGS)
2173    {
2174        // TODO:  convert numeric decimal string to unsigned int with overflow check.
2175        p = atoi( arg_string[ 1 ] ) ;
2176        n = atoi( arg_string[ 2 ] ) ;
2177    }
2178    else
2179    {
2180        ostringstream os{ "ERROR:  Expecting two arguments, p and n.\n\n" } ;
2181        print_help_ = true ;
2182        throw ParserError( os.str() ) ;
2183    }
2184
2185    // Check ranges on n and p.
2186    if (p < minModulus)
2187    {
2188        ostringstream os ;
2189        os << "Error.  Polynomial modulus p must be >= " << minModulus << endl ;
2190        print_help_ = true ;
2191        throw ParserError( os.str() ) ;
2192    }
2193
2194    if (p >= BigInt::getBase())
2195    {
2196        ostringstream os ;
2197        os << "Error.  Polynomial modulus p must be < " << BigInt::getBase() << endl ;
2198        print_help_ = true ;
2199        throw ParserError( os.str() ) ;
2200    }
2201
2202    if (n < minDegree)
2203    {
2204        ostringstream os ;
2205        os << "Error.  Polynomial degree n must be >= " << minDegree << " but n = " << n << endl ;
2206        print_help_ = true ;
2207        throw ParserError( os.str() ) ;
2208    }
2209
2210    //  Check to see if p is a prime.
2211    if (!isAlmostSurelyPrime( static_cast<ppuint>( p )))
2212        throw ParserError( "ERROR:  p must be a prime number.\n\n" ) ;
2213}
2214
2215
2216/*=============================================================================
2217 |
2218 | NAME
2219 |
2220 |     enumToString
2221 |
2222 | DESCRIPTION
2223 |
2224 |     Helper functions to translate enum types into human readable format.
2225 |     One generic for any SymbolType, the other two template specializations.
2226 |
2227 +============================================================================*/
2228
2229template< typename SymbolType >
2230string enumToString( const SymbolType & st )
2231{
2232    ostringstream os ;
2233    
2234    // Just return the integer number of the enumerated type since we have no
2235    // further information.
2236    os << static_cast<int>( st ) ;
2237    return os.str() ;
2238}
2239
2240// Specialized for PolySymbol.
2241template<>
2242string enumToString<PolySymbol>( const PolySymbol & st )
2243{
2244    string name[] { "$", "Integer", ",", "x", "+", "^", "", "S", "Mod", "Poly", "Term", "Multiplier", "Power", "" } ;
2245    return name[ static_cast<int>( st ) ] ;
2246}
2247
2248// Specialized for FactorizationSymbol.
2249template<>
2250string enumToString<FactorizationSymbol>( const FactorizationSymbol & st )
2251{
2252    string name[] { "$", "Integer", ".", "^", "\\", "", "S", "Factorization", "Factor", "BigInteger", "" } ;
2253    return name[ static_cast<int>( st ) ] ;
2254}
2255
2256/*==============================================================================
2257 |                     Forced Template Instantiations                           |
2258 ==============================================================================*/
2259
2260// C++ doesn't automatically generate templated classes or functions until they
2261// are first used.  So depending on the order of compilation, the linker may say
2262// the templated functions are undefined.
2263//
2264// Therefore, explicitly instantiate ALL templates here:
2265
2266// General purpose LALR(1) parser.  Template versions for both the polynomial grammar and factorization grammar.
2267template PolyValue                          Parser<PolySymbol, PolyValue>::parse( string ) ;
2268template FactorizationValue<BigInt>         Parser<FactorizationSymbol, FactorizationValue<BigInt>>::parse( string ) ;
2269template FactorizationValue<ppuint>         Parser<FactorizationSymbol, FactorizationValue<ppuint>>::parse( string ) ;
2270
2271// Generate all symbols for the polynomial grammar.
2272template                                    Symbol<PolySymbol, PolyValue>::Symbol( PolySymbol, int ) ;
2273template                                    Symbol<PolySymbol, PolyValue>::Symbol( PolySymbol, int, PolyValue )  ;
2274template                                    Symbol<PolySymbol, PolyValue>::~Symbol() ;
2275template                                    Symbol<PolySymbol, PolyValue>::Symbol(    const Symbol< PolySymbol, PolyValue > & ) ;
2276template Symbol<PolySymbol, PolyValue> &    Symbol<PolySymbol, PolyValue>::operator=( const Symbol< PolySymbol, PolyValue > & ) ;
2277
2278// Generate all the symbols for the factorization grammar.
2279template FactorizationValue<BigInt>::FactorizationValue() ;
2280template FactorizationValue<BigInt>::FactorizationValue( const FactorizationValue<BigInt> & ) ;
2281template FactorizationValue<BigInt> & FactorizationValue<BigInt>::operator=( const FactorizationValue<BigInt> & ) ;
2282template FactorizationValue<ppuint>::FactorizationValue() ;
2283template FactorizationValue<ppuint>::FactorizationValue( const FactorizationValue<ppuint> & ) ;
2284template FactorizationValue<ppuint> & FactorizationValue<ppuint>::operator=( const FactorizationValue<ppuint> & ) ;
2285
2286template Symbol<FactorizationSymbol, FactorizationValue<BigInt>>::Symbol( FactorizationSymbol, int ) ;
2287template Symbol<FactorizationSymbol, FactorizationValue<BigInt>>::Symbol( FactorizationSymbol, int, FactorizationValue<BigInt> )  ;
2288template Symbol<FactorizationSymbol, FactorizationValue<BigInt>>::~Symbol() ;
2289template Symbol<FactorizationSymbol, FactorizationValue<BigInt>>::Symbol(    const Symbol<FactorizationSymbol, FactorizationValue<BigInt>> & ) ;
2290template Symbol<FactorizationSymbol, FactorizationValue<BigInt>> &
2291         Symbol<FactorizationSymbol, FactorizationValue<BigInt>>::operator=( const Symbol<FactorizationSymbol, FactorizationValue<BigInt>> & ) ;
2292template Symbol<FactorizationSymbol, FactorizationValue<ppuint>>::Symbol( FactorizationSymbol, int ) ;
2293template Symbol<FactorizationSymbol, FactorizationValue<ppuint>>::Symbol( FactorizationSymbol, int, FactorizationValue<ppuint> )  ;
2294template Symbol<FactorizationSymbol, FactorizationValue<ppuint>>::~Symbol() ;
2295template Symbol<FactorizationSymbol, FactorizationValue<ppuint>>::Symbol(    const Symbol<FactorizationSymbol, FactorizationValue<ppuint>> & ) ;
2296template Symbol<FactorizationSymbol, FactorizationValue<ppuint>> &
2297         Symbol<FactorizationSymbol, FactorizationValue<ppuint>>::operator=( const Symbol<FactorizationSymbol, FactorizationValue<ppuint>> & ) ;
2298
2299// Generate all child class member functions for the polynomial grammar.
2300template                                      PolyParser<PolySymbol, PolyValue>::PolyParser() ;
2301template                                      PolyParser<PolySymbol, PolyValue>::~PolyParser() ;
2302template                                      PolyParser<PolySymbol, PolyValue>::PolyParser( const PolyParser<PolySymbol, PolyValue> & ) ;
2303template PolyParser<PolySymbol, PolyValue> &  PolyParser<PolySymbol, PolyValue>::operator=(  const PolyParser<PolySymbol, PolyValue> & ) ;
2304
2305template void                                 PolyParser<PolySymbol, PolyValue>::parseCommandLine( int, const char * argv[] ) ;
2306
2307// Generate all child class member functions for the factorization grammar.
2308template FactorizationParser<FactorizationSymbol, FactorizationValue<BigInt>>::FactorizationParser() ;
2309template FactorizationParser<FactorizationSymbol, FactorizationValue<BigInt>>::~FactorizationParser() ;
2310template FactorizationParser<FactorizationSymbol, FactorizationValue<BigInt>>::FactorizationParser( const FactorizationParser<FactorizationSymbol,
2311                                                                                                          FactorizationValue<BigInt>> & ) ;
2312template FactorizationParser<FactorizationSymbol, FactorizationValue<BigInt>> &
2313         FactorizationParser<FactorizationSymbol, FactorizationValue<BigInt>>::operator=(  const FactorizationParser<FactorizationSymbol, FactorizationValue<BigInt>> & ) ;
2314template FactorizationParser<FactorizationSymbol, FactorizationValue<ppuint>>::FactorizationParser() ;
2315template FactorizationParser<FactorizationSymbol, FactorizationValue<ppuint>>::~FactorizationParser() ;
2316template FactorizationParser<FactorizationSymbol, FactorizationValue<ppuint>>::FactorizationParser( const FactorizationParser<FactorizationSymbol,
2317                                                                                                          FactorizationValue<ppuint>> & ) ;
2318template FactorizationParser<FactorizationSymbol, FactorizationValue<ppuint>> &
2319         FactorizationParser<FactorizationSymbol, FactorizationValue<ppuint>>::operator=(  const FactorizationParser<FactorizationSymbol, FactorizationValue<ppuint>> & ) ;