You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
140 lines
3.9 KiB
140 lines
3.9 KiB
/*****************************************************************************
|
|
* expr_evaluator.cpp
|
|
*****************************************************************************
|
|
* Copyright (C) 2004 the VideoLAN team
|
|
* $Id$
|
|
*
|
|
* Authors: Cyril Deguet <asmax@via.ecp.fr>
|
|
*
|
|
* This program is free software; you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation; either version 2 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 General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston MA 02110-1301, USA.
|
|
*****************************************************************************/
|
|
|
|
#include "expr_evaluator.hpp"
|
|
|
|
|
|
void ExprEvaluator::parse( const string &rExpr )
|
|
{
|
|
m_stack.clear();
|
|
|
|
const char *pString = rExpr.c_str();
|
|
list<string> opStack; // operator stack
|
|
string token;
|
|
|
|
// Tokenize the expression
|
|
int begin = 0, end = 0;
|
|
while( pString[begin] )
|
|
{
|
|
// Find the next significant char in the string
|
|
while( pString[begin] == ' ' )
|
|
{
|
|
begin++;
|
|
}
|
|
|
|
if( pString[begin] == '(' )
|
|
{
|
|
// '(' found: push it on the stack and continue
|
|
opStack.push_back( "(" );
|
|
begin++;
|
|
}
|
|
else if( pString[begin] == ')' )
|
|
{
|
|
// ')' found: pop the stack until a '(' is found
|
|
while( !opStack.empty() )
|
|
{
|
|
// Pop the stack
|
|
string lastOp = opStack.back();
|
|
opStack.pop_back();
|
|
if( lastOp == "(" )
|
|
{
|
|
break;
|
|
}
|
|
// Push the operator on the RPN stack
|
|
m_stack.push_back( lastOp );
|
|
}
|
|
begin++;
|
|
}
|
|
else
|
|
{
|
|
// Skip white spaces
|
|
end = begin;
|
|
while( pString[end] && pString[end] != ' ' && pString[end] != ')' )
|
|
{
|
|
end++;
|
|
}
|
|
// Get the next token
|
|
token = rExpr.substr( begin, end - begin );
|
|
begin = end;
|
|
|
|
// TODO compare to a set of operators
|
|
if( token == "not" || token == "or" || token == "and" )
|
|
{
|
|
// Pop the operator stack while the operator has a higher
|
|
// precedence than the top of the stack
|
|
while( !opStack.empty() &&
|
|
hasPrecedency( token, opStack.back() ) )
|
|
{
|
|
// Pop the stack
|
|
string lastOp = opStack.back();
|
|
opStack.pop_back();
|
|
m_stack.push_back( lastOp );
|
|
}
|
|
opStack.push_back( token );
|
|
}
|
|
else
|
|
{
|
|
m_stack.push_back( token );
|
|
}
|
|
}
|
|
}
|
|
// Finish popping the operator stack
|
|
while( !opStack.empty() )
|
|
{
|
|
string lastOp = opStack.back();
|
|
opStack.pop_back();
|
|
m_stack.push_back( lastOp );
|
|
}
|
|
}
|
|
|
|
|
|
string ExprEvaluator::getToken()
|
|
{
|
|
if( !m_stack.empty() )
|
|
{
|
|
string token = m_stack.front();
|
|
m_stack.pop_front();
|
|
return token;
|
|
}
|
|
return "";
|
|
}
|
|
|
|
|
|
bool ExprEvaluator::hasPrecedency( const string &op1, const string &op2 ) const
|
|
{
|
|
// FIXME
|
|
if( op1 == "(" )
|
|
{
|
|
return true;
|
|
}
|
|
if( op1 == "and" )
|
|
{
|
|
return (op2 == "or") || (op2 == "not" );
|
|
}
|
|
if( op1 == "or" )
|
|
{
|
|
return (op2 == "not" );
|
|
}
|
|
return false;
|
|
}
|
|
|
|
|