// NeL - MMORPG Framework
// Copyright (C) 2010 Winch Gate Property Limited
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero General Public License as
// published by the Free Software Foundation, either version 3 of the
// License, or (at your option) any later version.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU Affero General Public License for more details.
//
// You should have received a copy of the GNU Affero General Public License
// along with this program. If not, see .
#include "stdmisc.h"
#include "nel/misc/eval_num_expr.h"
#include "nel/misc/debug.h"
#ifdef DEBUG_NEW
#define new DEBUG_NEW
#endif
namespace NLMISC
{
using namespace std;
// ***************************************************************************
CEvalNumExpr::TReturnState CEvalNumExpr::readDecimal (double &value)
{
// Read first integer value
readIntegerNumberDecimal (value);
// Dot ?
char currentChar = *_ExprPtr;
if (currentChar == '.')
{
// Next char
_ExprPtr++;
currentChar = *_ExprPtr;
if (currentChar < '0' || currentChar > '9')
return NumberSyntaxError;
// Read the decimal part
const char *start = _ExprPtr;
double fract;
readIntegerNumberDecimal (fract);
fract /= (double)pow (10.0,(sint)(_ExprPtr-start));
value += fract;
}
return NoError;
}
// ***************************************************************************
void CEvalNumExpr::readIntegerNumberDecimal (double &value)
{
// Registered values
register double regValue = 0;
// Read the first value
char currentChar = *_ExprPtr;
if ((currentChar >= '0') && (currentChar <= '9'))
{
regValue += (currentChar - '0');
_ExprPtr++;
currentChar = *_ExprPtr;
// For each values
while ((currentChar >= '0') && (currentChar <= '9'))
{
regValue *= 10;
regValue += (currentChar - '0');
// Next char
_ExprPtr++;
currentChar = *_ExprPtr;
}
}
// Store value
value = regValue;
}
// ***************************************************************************
CEvalNumExpr::TReturnState CEvalNumExpr::getNextToken (TToken &token)
{
// Get the current char
uint8 currentChar = *_ExprPtr;
// Skip space
while ((currentChar!=0) && (currentChar<=0x20))
{
_ExprPtr++;
currentChar = *_ExprPtr;
}
// Can be an operator ?
if (currentChar <= 128)
{
// Get the operator
_Op = _OperatorArray[currentChar];
// Is an operator ?
if (_Op != NotOperator)
{
// It is an operator
token = Operator;
// Is a 2 characters operator ?
if (_Op != ExtOperator)
{
// Return next character
_ExprPtr++;
return NoError;
}
else
{
// Have a second character ?
char secondChar = *(_ExprPtr+1);
// What kind of 1st character
switch (currentChar)
{
case '!':
if (secondChar == '=')
{
_Op = NotEqual;
_ExprPtr+=2;
return NoError;
}
else
{
_Op = Not;
_ExprPtr++;
return NoError;
}
case '&':
if (secondChar == '&')
{
_Op = LogicalAnd;
_ExprPtr+=2;
return NoError;
}
else
{
_Op = And;
_ExprPtr+=1;
return NoError;
}
case '-':
if (secondChar == '>')
{
_Op = SRightShift;
_ExprPtr+=2;
return NoError;
}
else
{
_Op = Minus;
_ExprPtr++;
return NoError;
}
case '<':
if (secondChar == '<')
{
_Op = ULeftShift;
_ExprPtr+=2;
return NoError;
}
else if (secondChar == '=')
{
_Op = InferiorEqual;
_ExprPtr+=2;
return NoError;
}
else if (secondChar == '-')
{
_Op = SLeftShift;
_ExprPtr+=2;
return NoError;
}
else
{
_Op = Inferior;
_ExprPtr+=1;
return NoError;
}
case '=':
if (secondChar == '=')
{
_Op = Equal;
_ExprPtr+=2;
return NoError;
}
case '>':
if (secondChar == '>')
{
_Op = URightShift;
_ExprPtr+=2;
return NoError;
}
else if (secondChar == '=')
{
_Op = SuperiorEqual;
_ExprPtr+=2;
return NoError;
}
else
{
_Op = Superior;
_ExprPtr+=1;
return NoError;
}
case '^':
if (secondChar == '^')
{
_Op = LogicalXor;
_ExprPtr+=2;
return NoError;
}
else
{
_Op = Xor;
_ExprPtr+=1;
return NoError;
}
case '|':
if (secondChar == '|')
{
_Op = LogicalOr;
_ExprPtr+=2;
return NoError;
}
else
{
_Op = Or;
_ExprPtr+=1;
return NoError;
}
}
// Can't found the operator
return UnknownOperator;
}
}
// Is End, '(', ')', '.' ?
else if (currentChar == 0)
{
token = End;
return NoError;
}
else if (currentChar == '(')
{
_ExprPtr++;
token = Open;
return NoError;
}
else if (currentChar == ')')
{
_ExprPtr++;
token = Close;
return NoError;
}
else if (currentChar == ',')
{
_ExprPtr++;
token = Coma;
return NoError;
}
// Is a number ?
else if (((currentChar >= '0') && (currentChar <= '9')) || (currentChar == '.'))
{
// This is a number
token = Number;
// Have a second character ?
char secondChar = *(_ExprPtr+1);
// Is an hexadecimal value ?
if ((currentChar == '0') && (secondChar == 'x'))
{
// Go to the number
_ExprPtr +=2;
currentChar = *_ExprPtr;
// Registered values
register double regValue = 0;
if ((currentChar >= '0') && (currentChar <= '9'))
{
regValue += (currentChar - '0');
}
else if ((currentChar >= 'a') && (currentChar <= 'f'))
{
regValue += (currentChar - 'a' + 10);
}
else if ((currentChar >= 'A') && (currentChar <= 'F'))
{
regValue += (currentChar - 'A' + 10);
}
else
{
// Number syntax error
return NumberSyntaxError;
}
_ExprPtr++;
currentChar = *_ExprPtr;
// For each values
for(;;)
{
if ((currentChar >= '0') && (currentChar <= '9'))
{
regValue *= 16;
regValue += (currentChar - '0');
}
else if ((currentChar >= 'a') && (currentChar <= 'f'))
{
regValue *= 16;
regValue += (currentChar - 'a' + 10);
}
else if ((currentChar >= 'A') && (currentChar <= 'F'))
{
regValue *= 16;
regValue += (currentChar - 'A' + 10);
}
else
{
// Stop
break;
}
// Next char
_ExprPtr++;
currentChar = *_ExprPtr;
}
// Store value
_Value = regValue;
// Number ok
return NoError;
}
// Is an octal value ?
else if ((currentChar == '0') && (secondChar >= '0') && (secondChar <= '9'))
{
// Go to the number
_ExprPtr ++;
currentChar = *_ExprPtr;
// Registered values
register double regValue = 0;
// Check octal number
if (currentChar > '7')
return NumberSyntaxError;
// Read the first value
regValue += (currentChar - '0');
_ExprPtr++;
currentChar = *_ExprPtr;
// For each values
while ((currentChar >= '0') && (currentChar <= '9'))
{
// Check octal number
if (currentChar > '7')
return NumberSyntaxError;
regValue *= 8;
regValue += (currentChar - '0');
// Next char
_ExprPtr++;
currentChar = *_ExprPtr;
}
// Store value
_Value = regValue;
// Number ok
return NoError;
}
// It is a decimal value
else
{
// Read value
TReturnState state = readDecimal (_Value);
if (state == NoError)
{
// Exponent ?
currentChar = *_ExprPtr;
if ( (currentChar == 'e') || (currentChar == 'E') )
{
// Next char
_ExprPtr++;
// Minus ?
bool negative = false;
if (*_ExprPtr == '-')
{
negative = true;
_ExprPtr++;
}
// Read value
double exponent;
state = readDecimal (exponent);
if (state == NoError)
{
// Negative value ?
if (negative)
exponent = -exponent;
// Raise 10 at the power of
_Value *= pow (10.0, exponent);
}
else
{
return state;
}
}
// Number ok
return NoError;
}
else
{
return state;
}
}
}
// Is a string ?
else if (currentChar == '"')
{
// Look for the end of the string
_ExprPtr++;
currentChar = *_ExprPtr;
const char *start = _ExprPtr;
while ( (currentChar != 0) && (currentChar != '"') )
{
_ExprPtr++;
currentChar = *_ExprPtr;
}
// End reached ?
if (currentChar == 0)
return MustBeDoubleQuote;
// This is a user string, copy the string
uint size = (uint)(_ExprPtr - start);
if (size >= (InternalStringLen-1))
{
_InternalStlString.resize (size);
uint i;
for (i=0; i= 128) || _StringChar[currentChar] )
{
_ExprPtr++;
currentChar = *_ExprPtr;
}
// Is pi ?
if (((_ExprPtr - start) == 2) && (start[0] == 'p') && (start[1] == 'i'))
{
token = Number;
_Value = 3.1415926535897932384626433832795;
return NoError;
}
// Is e ?
else if (((_ExprPtr - start) == 1) && (start[0] == 'e'))
{
token = Number;
_Value = 2.7182818284590452353602874713527;
return NoError;
}
// This is a user string, copy the string
uint size = (uint)(_ExprPtr - start);
if (size >= (InternalStringLen-1))
{
_InternalStlString.resize (size);
uint i;
for (i=0; i= 0 )
{
// The first is the good ?
if ( result == 0 )
{
end = begin;
}
result = strcmp (_InternalStringPtr, _ReservedWord[end]);
if (result <= 0)
{
// The last is the good ?
if ( result == 0 )
{
begin = end;
}
// While there is a middle..
while ((end - begin) > 1)
{
uint middle = begin + (end - begin) / 2;
result = strcmp (_InternalStringPtr, _ReservedWord[middle]);
if (result == 0)
{
begin = middle;
end = middle;
break;
}
else if (result < 0)
{
end = middle;
}
else
{
begin = middle;
}
}
}
// Found ?
if (end == begin)
{
// Return the token
_ReservedWordFound = (TReservedWord)begin;
token = _ReservedWordToken[begin];
// Ok
return NoError;
}
}
// Token
token = String;
return NoError;
}
// ***************************************************************************
CEvalNumExpr::TReturnState CEvalNumExpr::evalExpression (const char *expression, double &result,
int *errorIndex, uint32 userData)
{
// Init the ptr
_ExprPtr = expression;
TToken nextToken;
TReturnState error = evalExpression (result, nextToken, userData);
if (error == NoError)
{
// The end ?
if (nextToken == End)
return NoError;
else
{
if (errorIndex)
*errorIndex = (int)(_ExprPtr - expression);
return MustBeEnd;
}
}
else
{
if (errorIndex)
*errorIndex = (int)(_ExprPtr - expression);
return error;
}
}
// ***************************************************************************
CEvalNumExpr::TReturnState CEvalNumExpr::evalExpression (double &finalResult, TToken &nextToken, uint32 userData)
{
// Array of result
uint exprCount = 0;
double result[InternalOperator];
vector resultSup;
uint opCount = 0;
TOperator resultOp[InternalOperator];
vector resultOpSup;
// Read a token
TReturnState error = getNextToken (nextToken);
if (error != NoError)
return error;
for(;;)
{
// Unary opertor
uint unaryOpCount = 0;
TOperator resultUnaryOp[InternalOperator];
vector resultUnaryOpSup;
// init table
for (uint i = 0; i < (uint)InternalOperator; ++i) resultUnaryOp[i] = NotOperator;
// Current value
double value;
// Unary operator ?
if ( (nextToken == Operator) && ( (_Op == Minus) || (_Op == Not) || (_Op == Tilde) ) )
{
// Push the unary operator
if (unaryOpCountarg1) ? arg0 : arg1;
break;
case Min:
value = (arg0= InternalStringLen-1)
{
internalStlString = _InternalStringPtr;
internalStringPtr = internalStlString.c_str ();
}
else
{
strcpy (internalString ,_InternalStringPtr);
internalStringPtr = internalString;
}
// Read a token
error = getNextToken (nextToken);
if (error == NoError)
{
// Open ?
if (nextToken == Open)
{
// Eval an expression
double arg0;
error = evalExpression (arg0, nextToken, userData);
if (error == NoError)
{
if (nextToken == Coma)
{
// Second argument
double arg1;
error = evalExpression (arg1, nextToken, userData);
if (error == NoError)
{
// Final with close ?
if (nextToken == Close)
{
// Eval the function
error = evalFunction (internalStringPtr, arg0, arg1, value);
if (error != NoError)
return error;
// Get next token
error = getNextToken (nextToken);
if (error != NoError)
return error;
}
else
return MustBeClose;
}
else
return error;
}
else
{
if (nextToken == Close)
{
// Eval the function
error = evalFunction (internalStringPtr, arg0, value);
if (error != NoError)
return error;
// Get next token
error = getNextToken (nextToken);
if (error != NoError)
return error;
}
else
return MustBeClose;
}
}
else
return error;
}
else
{
// This is a user value
error = evalValue (internalStringPtr, value, userData);
if (error != NoError)
return error;
}
}
else
return error;
}
else
{
return MustBeExpression;
}
// Eval unary operator
sint i;
for (i=unaryOpCount-1; i>=0; i--)
{
switch ((i::max());
break;
case Minus:
value = -value;
break;
default:
// Can't be hear after getToken
nlstop;
}
}
// Push the value
if (exprCount < InternalOperator)
result[exprCount] = value;
else
resultSup.push_back (value);
exprCount++;
// Look for an operator
// Operator ?
if (nextToken == Operator)
{
// Yes, push it
if (opCount < InternalOperator)
resultOp[opCount] = _Op;
else
resultOpSup.push_back (_Op);
opCount++;
}
else
{
// Exit the evaluate loop
break;
}
// Next token
error = getNextToken (nextToken);
}
// Reduce the expression
uint index = 1;
while (exprCount != 1)
{
// Reduct ?
TOperator before = (((index-1)>((uint)floor (v1 + 0.5)));
break;
case SLeftShift:
v0 = (double)(((sint)floor (v0 + 0.5))<<((sint)floor (v1 + 0.5)));
break;
case SRightShift:
v0 = (double)(((sint)floor (v0 + 0.5))>>((sint)floor (v1 + 0.5)));
break;
case Inferior:
v0 = (v0v1)?1.0:0.0;
break;
case SuperiorEqual:
v0 = (v0>=v1)?1.0:0.0;
break;
case Equal:
v0 = (v0==v1)?1.0:0.0;
break;
case NotEqual:
v0 = (v0!=v1)?1.0:0.0;
break;
case And:
v0 = (double)(((uint)floor (v0 + 0.5)) & ((uint)floor (v1 + 0.5)));
break;
case Or:
v0 = (double)(((uint)floor (v0 + 0.5)) | ((uint)floor (v1 + 0.5)));
break;
case Xor:
v0 = (double)(((uint)floor (v0 + 0.5)) ^ ((uint)floor (v1 + 0.5)));
break;
case LogicalAnd:
v0 = (double)(uint)((floor (v0 + 0.5) != 0.0) && (floor (v1 + 0.5) != 0.0));
break;
case LogicalOr:
v0 = (double)(uint)((floor (v0 + 0.5) != 0.0) || (floor (v1 + 0.5) != 0.0));
break;
case LogicalXor:
{
bool b0 = floor (v0 + 0.5) != 0.0;
bool b1 = floor (v1 + 0.5) != 0.0;
v0 = (double)(uint)((b0&&!b1) || ((!b0)&&b1));
}
break;
default:
nlstop;
}
// Decal others values
uint i = index;
for (; i 1)
index--;
}
else
index++;
}
finalResult = result[0];
return NoError;
}
// ***************************************************************************
bool CEvalNumExpr::internalCheck ()
{
for (uint i=0; i= 0)
{
nlstop;
return false;
}
return true;
}
// ***************************************************************************
// ASCII TABLE
/*
0 NUL SOH STX ETX EOT ENQ ACK BEL BS TAB LF VT FF CR SO SI
1 DLE DC1 DC2 DC3 DC4 NAK SYN ETB CAN EM SUB ESC FS GS RS US
2 ! " # $ % & ' ( ) * + , - . /
3 0 1 2 3 4 5 6 7 8 9 : ; < = > ?
4 @ A B C D E F G H I J K L M N O
5 P Q R S T U V W X Y Z [ \ ] ^ _
6 ` a b c d e f g h i j k l m n o
7 p q r s t u v w x y z { | } ~
*/
// ***************************************************************************
const CEvalNumExpr::TOperator CEvalNumExpr::_OperatorArray[128] =
{
NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator,
NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator,
NotOperator, ExtOperator, NotOperator, NotOperator, NotOperator, Remainder, ExtOperator, NotOperator, NotOperator, NotOperator, Mul, Plus, NotOperator, ExtOperator, NotOperator, Div,
NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, ExtOperator, ExtOperator, ExtOperator, NotOperator,
NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator,
NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, ExtOperator, NotOperator,
NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator,
NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, NotOperator, ExtOperator, NotOperator, Tilde, NotOperator,
};
// ***************************************************************************
const bool CEvalNumExpr::_StringChar[128] =
{
false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false,
false, false, false, false, false, false, false, false, false, false, false, false, false, false, false, false,
false, false, true, true, true, false, false, true, false, false, false, false, false, false, false, false,
true, true, true, true, true, true, true, true, true, true, true, true, false, false, false, true,
true, true, true, true, true, true, true, true, true, true, true, true, true, true, true, true,
true, true, true, true, true, true, true, true, true, true, true, true, true, true, false, true,
true, true, true, true, true, true, true, true, true, true, true, true, true, true, true, true,
true, true, true, true, true, true, true, true, true, true, true, true, false, true, false, true,
};
// ***************************************************************************
CEvalNumExpr::TReturnState CEvalNumExpr::evalValue (const char *value, double &result, uint32 userData)
{
return UnknownValue;
}
// ***************************************************************************
CEvalNumExpr::TReturnState CEvalNumExpr::evalFunction (const char *funcName, double arg0, double &result)
{
return UnknownFunction;
}
// ***************************************************************************
CEvalNumExpr::TReturnState CEvalNumExpr::evalFunction (const char *funcName, double arg0, double arg1, double &result)
{
return UnknownFunction;
}
// ***************************************************************************
const char *CEvalNumExpr::_ReservedWord[ReservedWordCount] =
{
"abs", // Abs
"acos", // Acos
"asin", // Asin
"atan", // Atan
"atan2", // Atan2
"ceil", // Ceil
"cos", // Cos
"cosh", // Cosh
"exp", // Exp
"exponent", // Exponent
"floor", // Floor
"int", // Int
"log", // Log
"log10", // Log10
"mantissa", // Mantissa
"max", // Max
"min", // Min
"pow", // Pow
"rand", // Rand
"round", // Round
"sin", // Sin
"sinh", // Sinh
"sq", // Sq
"sqrt", // Sqrt
"tan", // Tan
"tanh", // Tanh
};
// ***************************************************************************
const CEvalNumExpr::TToken CEvalNumExpr::_ReservedWordToken[ReservedWordCount] =
{
Function1, // Abs
Function1, // Acos
Function1, // Asin
Function1, // Atan
Function2, // Atan2
Function1, // Ceil
Function1, // Cos
Function1, // Cosh
Function1, // Exp
Function1, // Exponent
Function1, // Floor
Function1, // Int
Function1, // Log
Function1, // Log10
Function1, // Mantissa
Function2, // Max
Function2, // Min
Function2, // Pow
Function2, // Rand
Function1, // Round
Function1, // Sin
Function1, // Sinh
Function1, // Sq
Function1, // Sqrt
Function1, // Tan
Function1, // Tanh
};
// ***************************************************************************
const char *CEvalNumExpr::_ErrorString[ReturnValueCount]=
{
"No error",
"Unknown value",
"Error during user defined value evaluation",
"Unknown function",
"Error during user defined function evaluation",
"Syntax error in a number expression",
"Unknown operator",
"Should be a open parentesis",
"Should be a close parentesis",
"Should be a coma character",
"Should be an expression",
"Should not be an unary operator",
"Should be the end of the expression",
"Should be a double quote",
"Divid by zero",
};
// ***************************************************************************
const char* CEvalNumExpr::getErrorString (TReturnState state) const
{
return _ErrorString[state];
}
// ***************************************************************************
const int CEvalNumExpr::_OperatorPrecedence[]=
{
0, // Not
0, // Tilde
1, // Mul
1, // Div
1, // Remainder
2, // Plus
2, // Minus
3, // ULeftShift
3, // URightShift
3, // SLeftShift
3, // SRightShift
4, // Inferior
4, // InferiorEqual
4, // Superior
4, // SuperiorEqual
5, // Equal
5, // NotEqual
6, // And
7, // Or
8, // Xor
9, // LogicalAnd
10, // LogicalOr
11, // LogicalXor
-1, // OperatorCount
20, // NotOperator
};
// ***************************************************************************
}