1#ifndef Cpl_Math_IntegerExpressionParser_h_
2#define Cpl_Math_IntegerExpressionParser_h_
41#include "colony_config.h"
48#ifndef OPTION_CPL_MATH_INTEXPR_MAX_STACK_SIZE
49#define OPTION_CPL_MATH_INTEXPR_MAX_STACK_SIZE 20
79 bool eval(
const char* expressionAsText, T& result )
83 m_expr = expressionAsText;
84 m_exprLen = strlen( expressionAsText );
87 if ( !parseExpr( result ) )
100 OPERATOR_BITWISE_XOR,
101 OPERATOR_BITWISE_AND,
102 OPERATOR_BITWISE_SHL,
103 OPERATOR_BITWISE_SHR,
105 OPERATOR_SUBTRACTION,
106 OPERATOR_MULTIPLICATION,
126 Operator(
int opr,
int prec,
int assoc )
129 , associativity( assoc )
134 : op( OPERATOR_NULL )
136 , associativity(
'L' )
151 OperatorValue(
const Operator& opr, T val )
164 int getPrecedence()
const
166 return op.precedence;
172 return op.op == OPERATOR_NULL;
178 static T pow( T x, T n )
200 bool calculate( T& result, T v1, T v2,
const Operator& op )
const
205 case OPERATOR_BITWISE_OR:
209 case OPERATOR_BITWISE_XOR:
213 case OPERATOR_BITWISE_AND:
217 case OPERATOR_BITWISE_SHL:
221 case OPERATOR_BITWISE_SHR:
225 case OPERATOR_ADDITION:
229 case OPERATOR_SUBTRACTION:
233 case OPERATOR_MULTIPLICATION:
238 result = pow( v1, v2 );
241 case OPERATOR_EXPONENT:
242 result = v1 * pow( 10, v2 );
245 case OPERATOR_DIVISION:
257 case OPERATOR_MODULO:
281 return m_index >= m_exprLen;
287 char getCharacter()
const
291 return m_expr[m_index];
298 void unexpected()
const
310 while ( isspace( getCharacter() ) != 0 )
319 bool parseOp( Operator& result )
324 switch ( getCharacter() )
328 result = Operator( OPERATOR_BITWISE_OR, 4,
'L' );
333 result = Operator( OPERATOR_BITWISE_XOR, 5,
'L' );
338 result = Operator( OPERATOR_BITWISE_AND, 6,
'L' );
343 if ( getCharacter() ==
'<' )
346 result = Operator( OPERATOR_BITWISE_SHL, 9,
'L' );
357 if ( getCharacter() ==
'>' )
360 result = Operator( OPERATOR_BITWISE_SHR, 9,
'L' );
371 result = Operator( OPERATOR_ADDITION, 10,
'L' );
376 result = Operator( OPERATOR_SUBTRACTION, 10,
'L' );
381 result = Operator( OPERATOR_DIVISION, 20,
'L' );
386 result = Operator( OPERATOR_MODULO, 20,
'L' );
391 if ( getCharacter() !=
'*' )
393 result = Operator( OPERATOR_MULTIPLICATION, 20,
'L' );
398 result = Operator( OPERATOR_POWER, 30,
'R' );
404 result = Operator( OPERATOR_EXPONENT, 40,
'R' );
409 result = Operator( OPERATOR_EXPONENT, 40,
'R' );
413 result = Operator( OPERATOR_NULL, 0,
'L' );
420 static T toInteger(
char c )
422 if ( c >=
'0' && c <=
'9' )
426 if ( c >=
'a' && c <=
'f' )
428 return c -
'a' + 0xa;
430 if ( c >=
'A' && c <=
'F' )
432 return c -
'A' + 0xa;
440 return toInteger( getCharacter() );
446 for ( T d; (d = getInteger()) <= 9; m_index++ )
448 value = value * 10 + d;
455 m_index = m_index + 2;
457 for ( T h; (h = getInteger()) <= 0xf; m_index++ )
459 value = value * 0x10 + h;
467 if ( m_index + 2 < m_exprLen )
469 char x = m_expr[m_index + 1];
470 char h = m_expr[m_index + 2];
471 return ((x ==
'x' || x ==
'X') && toInteger( h ) <= 0xf);
480 bool parseValue( T& valFinal )
484 switch ( getCharacter() )
493 val = parseDecimal();
506 val = parseDecimal();
511 if ( !parseExpr( val ) )
516 if ( getCharacter() !=
')' )
518 CPL_SYSTEM_TRACE_MSG(
"Cpl::Math", (
"Syntax error: `)' expected at end of expression (index=%d",
527 if ( !parseValue( val ) )
536 if ( !parseValue( val ) )
544 if ( !parseValue( val ) )
548 val = val *
static_cast<T
>(-1);
567 bool parseExpr( T& finalValue )
569 m_stack.
push( OperatorValue( Operator( OPERATOR_NULL, 0,
'L' ), 0 ) );
573 if ( !parseValue( value ) )
582 if ( !parseOp( op ) )
588 OperatorValue topVal;
591 while ( op.precedence < topVal.getPrecedence() ||
592 (op.precedence == topVal.getPrecedence() &&
593 op.associativity ==
'L')
597 if ( topVal.isNull() )
605 if ( !calculate( value, topVal.value, value, topVal.op ) )
612 if ( !m_stack.
peekTop( topVal ) )
620 if ( !m_stack.
push( OperatorValue( op, value ) ) )
627 if ( !parseValue( value ) )
#define OPTION_CPL_MATH_INTEXPR_MAX_STACK_SIZE
Maximum depth of the operator stack used when evaluating an expression.
Definition IntegerExpressionParser.h:49
#define CPL_SYSTEM_TRACE_MSG(sect, var_args)
Macro Wrapper.
Definition Trace.h:411
This template class implements a THREAD SAFE Ring Buffer.
Definition RingBufferMT.h:33
void clearTheStack() noexcept
Empties the Stack.
Definition Stack.h:136
bool pop(ITEM &dst) noexcept
Removes the top item of the stack.
Definition Stack.h:156
bool peekTop(ITEM &dst) const noexcept
Returns the item on the top of the Stack.
Definition Stack.h:180
bool push(const ITEM src) noexcept
Adds an item to the top of the stack.
Definition Stack.h:143
bool isEmpty(void) const noexcept
This method returns true if the Stack is empty.
Definition Stack.h:204
This template class evaluates an null terminate string that represents an integer arithmetic expressi...
Definition IntegerExpressionParser.h:66
bool eval(const char *expressionAsText, T &result)
Evaluates an integer arithmetic expression and returns true if the expression was successfully evalua...
Definition IntegerExpressionParser.h:79
IntegerExpressionParser()
Constructor.
Definition IntegerExpressionParser.h:69
The 'Cpl' namespace is the root name space for the Colony.
Definition Api16.h:20