|
|
@ -22,33 +22,33 @@ namespace Compiler
|
|
|
|
switch (op)
|
|
|
|
switch (op)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
case '(':
|
|
|
|
case '(':
|
|
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
|
|
case 'e': // ==
|
|
|
|
case 'e': // ==
|
|
|
|
case 'n': // !=
|
|
|
|
case 'n': // !=
|
|
|
|
case 'l': // <
|
|
|
|
case 'l': // <
|
|
|
|
case 'L': // <=
|
|
|
|
case 'L': // <=
|
|
|
|
case 'g': // <
|
|
|
|
case 'g': // <
|
|
|
|
case 'G': // >=
|
|
|
|
case 'G': // >=
|
|
|
|
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
|
|
|
|
case '+':
|
|
|
|
case '+':
|
|
|
|
case '-':
|
|
|
|
case '-':
|
|
|
|
|
|
|
|
|
|
|
|
return 2;
|
|
|
|
return 2;
|
|
|
|
|
|
|
|
|
|
|
|
case '*':
|
|
|
|
case '*':
|
|
|
|
case '/':
|
|
|
|
case '/':
|
|
|
|
|
|
|
|
|
|
|
|
return 3;
|
|
|
|
return 3;
|
|
|
|
|
|
|
|
|
|
|
|
case 'm':
|
|
|
|
case 'm':
|
|
|
|
|
|
|
|
|
|
|
|
return 4;
|
|
|
|
return 4;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
@ -59,7 +59,7 @@ namespace Compiler
|
|
|
|
assert (Index<static_cast<int> (mOperands.size()));
|
|
|
|
assert (Index<static_cast<int> (mOperands.size()));
|
|
|
|
return mOperands[mOperands.size()-1-Index];
|
|
|
|
return mOperands[mOperands.size()-1-Index];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
char ExprParser::getOperator() const
|
|
|
|
char ExprParser::getOperator() const
|
|
|
|
{
|
|
|
|
{
|
|
|
|
assert (!mOperators.empty());
|
|
|
|
assert (!mOperators.empty());
|
|
|
@ -70,27 +70,27 @@ namespace Compiler
|
|
|
|
{
|
|
|
|
{
|
|
|
|
return std::find (mOperators.begin(), mOperators.end(), '(')!=mOperators.end();
|
|
|
|
return std::find (mOperators.begin(), mOperators.end(), '(')!=mOperators.end();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ExprParser::popOperator()
|
|
|
|
void ExprParser::popOperator()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
assert (!mOperators.empty());
|
|
|
|
assert (!mOperators.empty());
|
|
|
|
mOperators.resize (mOperators.size()-1);
|
|
|
|
mOperators.resize (mOperators.size()-1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ExprParser::popOperand()
|
|
|
|
void ExprParser::popOperand()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
assert (!mOperands.empty());
|
|
|
|
assert (!mOperands.empty());
|
|
|
|
mOperands.resize (mOperands.size()-1);
|
|
|
|
mOperands.resize (mOperands.size()-1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ExprParser::replaceBinaryOperands()
|
|
|
|
void ExprParser::replaceBinaryOperands()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
char t1 = getOperandType (1);
|
|
|
|
char t1 = getOperandType (1);
|
|
|
|
char t2 = getOperandType();
|
|
|
|
char t2 = getOperandType();
|
|
|
|
|
|
|
|
|
|
|
|
popOperand();
|
|
|
|
popOperand();
|
|
|
|
popOperand();
|
|
|
|
popOperand();
|
|
|
|
|
|
|
|
|
|
|
|
if (t1==t2)
|
|
|
|
if (t1==t2)
|
|
|
|
mOperands.push_back (t1);
|
|
|
|
mOperands.push_back (t1);
|
|
|
|
else if (t1=='f' || t2=='f')
|
|
|
|
else if (t1=='f' || t2=='f')
|
|
|
@ -102,59 +102,59 @@ namespace Compiler
|
|
|
|
void ExprParser::pop()
|
|
|
|
void ExprParser::pop()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
char op = getOperator();
|
|
|
|
char op = getOperator();
|
|
|
|
|
|
|
|
|
|
|
|
switch (op)
|
|
|
|
switch (op)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
case 'm':
|
|
|
|
case 'm':
|
|
|
|
|
|
|
|
|
|
|
|
Generator::negate (mCode, getOperandType());
|
|
|
|
Generator::negate (mCode, getOperandType());
|
|
|
|
popOperator();
|
|
|
|
popOperator();
|
|
|
|
break;
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
case '+':
|
|
|
|
case '+':
|
|
|
|
|
|
|
|
|
|
|
|
Generator::add (mCode, getOperandType (1), getOperandType());
|
|
|
|
Generator::add (mCode, getOperandType (1), getOperandType());
|
|
|
|
popOperator();
|
|
|
|
popOperator();
|
|
|
|
replaceBinaryOperands();
|
|
|
|
replaceBinaryOperands();
|
|
|
|
break;
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
case '-':
|
|
|
|
case '-':
|
|
|
|
|
|
|
|
|
|
|
|
Generator::sub (mCode, getOperandType (1), getOperandType());
|
|
|
|
Generator::sub (mCode, getOperandType (1), getOperandType());
|
|
|
|
popOperator();
|
|
|
|
popOperator();
|
|
|
|
replaceBinaryOperands();
|
|
|
|
replaceBinaryOperands();
|
|
|
|
break;
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
case '*':
|
|
|
|
case '*':
|
|
|
|
|
|
|
|
|
|
|
|
Generator::mul (mCode, getOperandType (1), getOperandType());
|
|
|
|
Generator::mul (mCode, getOperandType (1), getOperandType());
|
|
|
|
popOperator();
|
|
|
|
popOperator();
|
|
|
|
replaceBinaryOperands();
|
|
|
|
replaceBinaryOperands();
|
|
|
|
break;
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
case '/':
|
|
|
|
case '/':
|
|
|
|
|
|
|
|
|
|
|
|
Generator::div (mCode, getOperandType (1), getOperandType());
|
|
|
|
Generator::div (mCode, getOperandType (1), getOperandType());
|
|
|
|
popOperator();
|
|
|
|
popOperator();
|
|
|
|
replaceBinaryOperands();
|
|
|
|
replaceBinaryOperands();
|
|
|
|
break;
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
case 'e':
|
|
|
|
case 'e':
|
|
|
|
case 'n':
|
|
|
|
case 'n':
|
|
|
|
case 'l':
|
|
|
|
case 'l':
|
|
|
|
case 'L':
|
|
|
|
case 'L':
|
|
|
|
case 'g':
|
|
|
|
case 'g':
|
|
|
|
case 'G':
|
|
|
|
case 'G':
|
|
|
|
|
|
|
|
|
|
|
|
Generator::compare (mCode, op, getOperandType (1), getOperandType());
|
|
|
|
Generator::compare (mCode, op, getOperandType (1), getOperandType());
|
|
|
|
popOperator();
|
|
|
|
popOperator();
|
|
|
|
popOperand();
|
|
|
|
popOperand();
|
|
|
|
popOperand();
|
|
|
|
popOperand();
|
|
|
|
mOperands.push_back ('l');
|
|
|
|
mOperands.push_back ('l');
|
|
|
|
break;
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
default:
|
|
|
|
default:
|
|
|
|
|
|
|
|
|
|
|
|
throw std::logic_error ("unknown operator");
|
|
|
|
throw std::logic_error ("unknown operator");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -165,36 +165,36 @@ namespace Compiler
|
|
|
|
mOperands.push_back ('l');
|
|
|
|
mOperands.push_back ('l');
|
|
|
|
Generator::pushInt (mCode, mLiterals, value);
|
|
|
|
Generator::pushInt (mCode, mLiterals, value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ExprParser::pushFloatLiteral (float value)
|
|
|
|
void ExprParser::pushFloatLiteral (float value)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
mNextOperand = false;
|
|
|
|
mNextOperand = false;
|
|
|
|
mOperands.push_back ('f');
|
|
|
|
mOperands.push_back ('f');
|
|
|
|
Generator::pushFloat (mCode, mLiterals, value);
|
|
|
|
Generator::pushFloat (mCode, mLiterals, value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ExprParser::pushBinaryOperator (char c)
|
|
|
|
void ExprParser::pushBinaryOperator (char c)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
while (!mOperators.empty() && getPriority (getOperator())>=getPriority (c))
|
|
|
|
while (!mOperators.empty() && getPriority (getOperator())>=getPriority (c))
|
|
|
|
pop();
|
|
|
|
pop();
|
|
|
|
|
|
|
|
|
|
|
|
mOperators.push_back (c);
|
|
|
|
mOperators.push_back (c);
|
|
|
|
mNextOperand = true;
|
|
|
|
mNextOperand = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ExprParser::close()
|
|
|
|
void ExprParser::close()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
while (getOperator()!='(')
|
|
|
|
while (getOperator()!='(')
|
|
|
|
pop();
|
|
|
|
pop();
|
|
|
|
|
|
|
|
|
|
|
|
popOperator();
|
|
|
|
popOperator();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ExprParser::parseArguments (const std::string& arguments, Scanner& scanner)
|
|
|
|
int ExprParser::parseArguments (const std::string& arguments, Scanner& scanner)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
parseArguments (arguments, scanner, mCode);
|
|
|
|
return parseArguments (arguments, scanner, mCode);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
ExprParser::ExprParser (ErrorHandler& errorHandler, Context& context, Locals& locals,
|
|
|
|
ExprParser::ExprParser (ErrorHandler& errorHandler, Context& context, Locals& locals,
|
|
|
|
Literals& literals, bool argument)
|
|
|
|
Literals& literals, bool argument)
|
|
|
|
: Parser (errorHandler, context), mLocals (locals), mLiterals (literals),
|
|
|
|
: Parser (errorHandler, context), mLocals (locals), mLiterals (literals),
|
|
|
@ -205,11 +205,13 @@ namespace Compiler
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (!mExplicit.empty())
|
|
|
|
if (!mExplicit.empty())
|
|
|
|
return Parser::parseInt (value, loc, scanner);
|
|
|
|
return Parser::parseInt (value, loc, scanner);
|
|
|
|
|
|
|
|
|
|
|
|
mFirst = false;
|
|
|
|
mFirst = false;
|
|
|
|
|
|
|
|
|
|
|
|
if (mNextOperand)
|
|
|
|
if (mNextOperand)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
|
|
|
|
start();
|
|
|
|
|
|
|
|
|
|
|
|
pushIntegerLiteral (value);
|
|
|
|
pushIntegerLiteral (value);
|
|
|
|
mTokenLoc = loc;
|
|
|
|
mTokenLoc = loc;
|
|
|
|
return true;
|
|
|
|
return true;
|
|
|
@ -228,13 +230,15 @@ namespace Compiler
|
|
|
|
return Parser::parseFloat (value, loc, scanner);
|
|
|
|
return Parser::parseFloat (value, loc, scanner);
|
|
|
|
|
|
|
|
|
|
|
|
mFirst = false;
|
|
|
|
mFirst = false;
|
|
|
|
|
|
|
|
|
|
|
|
if (mNextOperand)
|
|
|
|
if (mNextOperand)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
|
|
|
|
start();
|
|
|
|
|
|
|
|
|
|
|
|
pushFloatLiteral (value);
|
|
|
|
pushFloatLiteral (value);
|
|
|
|
mTokenLoc = loc;
|
|
|
|
mTokenLoc = loc;
|
|
|
|
return true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
else
|
|
|
|
{
|
|
|
|
{
|
|
|
|
// no comma was used between arguments
|
|
|
|
// no comma was used between arguments
|
|
|
@ -252,29 +256,31 @@ namespace Compiler
|
|
|
|
mFirst = false;
|
|
|
|
mFirst = false;
|
|
|
|
|
|
|
|
|
|
|
|
if (mNextOperand)
|
|
|
|
if (mNextOperand)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
|
|
|
|
start();
|
|
|
|
|
|
|
|
|
|
|
|
std::string name2 = toLower (name);
|
|
|
|
std::string name2 = toLower (name);
|
|
|
|
|
|
|
|
|
|
|
|
char type = mLocals.getType (name2);
|
|
|
|
char type = mLocals.getType (name2);
|
|
|
|
|
|
|
|
|
|
|
|
if (type!=' ')
|
|
|
|
if (type!=' ')
|
|
|
|
{
|
|
|
|
{
|
|
|
|
Generator::fetchLocal (mCode, type, mLocals.getIndex (name2));
|
|
|
|
Generator::fetchLocal (mCode, type, mLocals.getIndex (name2));
|
|
|
|
mNextOperand = false;
|
|
|
|
mNextOperand = false;
|
|
|
|
mOperands.push_back (type=='f' ? 'f' : 'l');
|
|
|
|
mOperands.push_back (type=='f' ? 'f' : 'l');
|
|
|
|
return true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
type = getContext().getGlobalType (name2);
|
|
|
|
type = getContext().getGlobalType (name2);
|
|
|
|
|
|
|
|
|
|
|
|
if (type!=' ')
|
|
|
|
if (type!=' ')
|
|
|
|
{
|
|
|
|
{
|
|
|
|
Generator::fetchGlobal (mCode, mLiterals, type, name2);
|
|
|
|
Generator::fetchGlobal (mCode, mLiterals, type, name2);
|
|
|
|
mNextOperand = false;
|
|
|
|
mNextOperand = false;
|
|
|
|
mOperands.push_back (type=='f' ? 'f' : 'l');
|
|
|
|
mOperands.push_back (type=='f' ? 'f' : 'l');
|
|
|
|
return true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (mExplicit.empty() && getContext().isId (name))
|
|
|
|
if (mExplicit.empty() && getContext().isId (name))
|
|
|
|
{
|
|
|
|
{
|
|
|
|
mExplicit = name;
|
|
|
|
mExplicit = name;
|
|
|
@ -287,161 +293,183 @@ namespace Compiler
|
|
|
|
scanner.putbackName (name, loc);
|
|
|
|
scanner.putbackName (name, loc);
|
|
|
|
return false;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return Parser::parseName (name, loc, scanner);
|
|
|
|
return Parser::parseName (name, loc, scanner);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool ExprParser::parseKeyword (int keyword, const TokenLoc& loc, Scanner& scanner)
|
|
|
|
bool ExprParser::parseKeyword (int keyword, const TokenLoc& loc, Scanner& scanner)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
mFirst = false;
|
|
|
|
mFirst = false;
|
|
|
|
|
|
|
|
|
|
|
|
if (!mExplicit.empty())
|
|
|
|
if (!mExplicit.empty())
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (mRefOp && mNextOperand)
|
|
|
|
if (mRefOp && mNextOperand)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (keyword==Scanner::K_getdisabled)
|
|
|
|
if (keyword==Scanner::K_getdisabled)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
mTokenLoc = loc;
|
|
|
|
start();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
mTokenLoc = loc;
|
|
|
|
|
|
|
|
|
|
|
|
Generator::getDisabled (mCode, mLiterals, mExplicit);
|
|
|
|
Generator::getDisabled (mCode, mLiterals, mExplicit);
|
|
|
|
mOperands.push_back ('l');
|
|
|
|
mOperands.push_back ('l');
|
|
|
|
mExplicit.clear();
|
|
|
|
mExplicit.clear();
|
|
|
|
mRefOp = false;
|
|
|
|
mRefOp = false;
|
|
|
|
|
|
|
|
|
|
|
|
mNextOperand = false;
|
|
|
|
mNextOperand = false;
|
|
|
|
return true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (keyword==Scanner::K_getdistance)
|
|
|
|
else if (keyword==Scanner::K_getdistance)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
|
|
|
|
start();
|
|
|
|
|
|
|
|
|
|
|
|
mTokenLoc = loc;
|
|
|
|
mTokenLoc = loc;
|
|
|
|
parseArguments ("c", scanner);
|
|
|
|
parseArguments ("c", scanner);
|
|
|
|
|
|
|
|
|
|
|
|
Generator::getDistance (mCode, mLiterals, mExplicit);
|
|
|
|
Generator::getDistance (mCode, mLiterals, mExplicit);
|
|
|
|
mOperands.push_back ('f');
|
|
|
|
mOperands.push_back ('f');
|
|
|
|
mExplicit.clear();
|
|
|
|
mExplicit.clear();
|
|
|
|
mRefOp = false;
|
|
|
|
mRefOp = false;
|
|
|
|
|
|
|
|
|
|
|
|
mNextOperand = false;
|
|
|
|
mNextOperand = false;
|
|
|
|
return true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// check for custom extensions
|
|
|
|
// check for custom extensions
|
|
|
|
if (const Extensions *extensions = getContext().getExtensions())
|
|
|
|
if (const Extensions *extensions = getContext().getExtensions())
|
|
|
|
{
|
|
|
|
{
|
|
|
|
char returnType;
|
|
|
|
char returnType;
|
|
|
|
std::string argumentType;
|
|
|
|
std::string argumentType;
|
|
|
|
|
|
|
|
|
|
|
|
if (extensions->isFunction (keyword, returnType, argumentType, true))
|
|
|
|
if (extensions->isFunction (keyword, returnType, argumentType, true))
|
|
|
|
{
|
|
|
|
{
|
|
|
|
|
|
|
|
start();
|
|
|
|
|
|
|
|
|
|
|
|
mTokenLoc = loc;
|
|
|
|
mTokenLoc = loc;
|
|
|
|
parseArguments (argumentType, scanner);
|
|
|
|
parseArguments (argumentType, scanner);
|
|
|
|
|
|
|
|
|
|
|
|
extensions->generateFunctionCode (keyword, mCode, mLiterals, mExplicit);
|
|
|
|
extensions->generateFunctionCode (keyword, mCode, mLiterals, mExplicit);
|
|
|
|
mOperands.push_back (returnType);
|
|
|
|
mOperands.push_back (returnType);
|
|
|
|
mExplicit.clear();
|
|
|
|
mExplicit.clear();
|
|
|
|
mRefOp = false;
|
|
|
|
mRefOp = false;
|
|
|
|
|
|
|
|
|
|
|
|
mNextOperand = false;
|
|
|
|
mNextOperand = false;
|
|
|
|
return true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return Parser::parseKeyword (keyword, loc, scanner);
|
|
|
|
return Parser::parseKeyword (keyword, loc, scanner);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (mNextOperand)
|
|
|
|
if (mNextOperand)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (keyword==Scanner::K_getsquareroot)
|
|
|
|
if (keyword==Scanner::K_getsquareroot)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
mTokenLoc = loc;
|
|
|
|
start();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
mTokenLoc = loc;
|
|
|
|
parseArguments ("f", scanner);
|
|
|
|
parseArguments ("f", scanner);
|
|
|
|
|
|
|
|
|
|
|
|
Generator::squareRoot (mCode);
|
|
|
|
Generator::squareRoot (mCode);
|
|
|
|
mOperands.push_back ('f');
|
|
|
|
mOperands.push_back ('f');
|
|
|
|
|
|
|
|
|
|
|
|
mNextOperand = false;
|
|
|
|
mNextOperand = false;
|
|
|
|
return true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (keyword==Scanner::K_menumode)
|
|
|
|
else if (keyword==Scanner::K_menumode)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
|
|
|
|
start();
|
|
|
|
|
|
|
|
|
|
|
|
mTokenLoc = loc;
|
|
|
|
mTokenLoc = loc;
|
|
|
|
|
|
|
|
|
|
|
|
Generator::menuMode (mCode);
|
|
|
|
Generator::menuMode (mCode);
|
|
|
|
mOperands.push_back ('l');
|
|
|
|
mOperands.push_back ('l');
|
|
|
|
|
|
|
|
|
|
|
|
mNextOperand = false;
|
|
|
|
mNextOperand = false;
|
|
|
|
return true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (keyword==Scanner::K_random)
|
|
|
|
else if (keyword==Scanner::K_random)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
mTokenLoc = loc;
|
|
|
|
start();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
mTokenLoc = loc;
|
|
|
|
parseArguments ("l", scanner);
|
|
|
|
parseArguments ("l", scanner);
|
|
|
|
|
|
|
|
|
|
|
|
Generator::random (mCode);
|
|
|
|
Generator::random (mCode);
|
|
|
|
mOperands.push_back ('l');
|
|
|
|
mOperands.push_back ('l');
|
|
|
|
|
|
|
|
|
|
|
|
mNextOperand = false;
|
|
|
|
mNextOperand = false;
|
|
|
|
return true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (keyword==Scanner::K_scriptrunning)
|
|
|
|
else if (keyword==Scanner::K_scriptrunning)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
|
|
|
|
start();
|
|
|
|
|
|
|
|
|
|
|
|
mTokenLoc = loc;
|
|
|
|
mTokenLoc = loc;
|
|
|
|
parseArguments ("c", scanner);
|
|
|
|
parseArguments ("c", scanner);
|
|
|
|
|
|
|
|
|
|
|
|
Generator::scriptRunning (mCode);
|
|
|
|
Generator::scriptRunning (mCode);
|
|
|
|
mOperands.push_back ('l');
|
|
|
|
mOperands.push_back ('l');
|
|
|
|
|
|
|
|
|
|
|
|
mNextOperand = false;
|
|
|
|
mNextOperand = false;
|
|
|
|
return true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (keyword==Scanner::K_getdistance)
|
|
|
|
else if (keyword==Scanner::K_getdistance)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
|
|
|
|
start();
|
|
|
|
|
|
|
|
|
|
|
|
mTokenLoc = loc;
|
|
|
|
mTokenLoc = loc;
|
|
|
|
parseArguments ("c", scanner);
|
|
|
|
parseArguments ("c", scanner);
|
|
|
|
|
|
|
|
|
|
|
|
Generator::getDistance (mCode, mLiterals, "");
|
|
|
|
Generator::getDistance (mCode, mLiterals, "");
|
|
|
|
mOperands.push_back ('f');
|
|
|
|
mOperands.push_back ('f');
|
|
|
|
|
|
|
|
|
|
|
|
mNextOperand = false;
|
|
|
|
mNextOperand = false;
|
|
|
|
return true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (keyword==Scanner::K_getsecondspassed)
|
|
|
|
else if (keyword==Scanner::K_getsecondspassed)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
mTokenLoc = loc;
|
|
|
|
start();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
mTokenLoc = loc;
|
|
|
|
|
|
|
|
|
|
|
|
Generator::getSecondsPassed (mCode);
|
|
|
|
Generator::getSecondsPassed (mCode);
|
|
|
|
mOperands.push_back ('f');
|
|
|
|
mOperands.push_back ('f');
|
|
|
|
|
|
|
|
|
|
|
|
mNextOperand = false;
|
|
|
|
mNextOperand = false;
|
|
|
|
return true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (keyword==Scanner::K_getdisabled)
|
|
|
|
else if (keyword==Scanner::K_getdisabled)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
mTokenLoc = loc;
|
|
|
|
start();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
mTokenLoc = loc;
|
|
|
|
|
|
|
|
|
|
|
|
Generator::getDisabled (mCode, mLiterals, "");
|
|
|
|
Generator::getDisabled (mCode, mLiterals, "");
|
|
|
|
mOperands.push_back ('l');
|
|
|
|
mOperands.push_back ('l');
|
|
|
|
|
|
|
|
|
|
|
|
mNextOperand = false;
|
|
|
|
mNextOperand = false;
|
|
|
|
return true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
else
|
|
|
|
{
|
|
|
|
{
|
|
|
|
// check for custom extensions
|
|
|
|
// check for custom extensions
|
|
|
|
if (const Extensions *extensions = getContext().getExtensions())
|
|
|
|
if (const Extensions *extensions = getContext().getExtensions())
|
|
|
|
{
|
|
|
|
{
|
|
|
|
|
|
|
|
start();
|
|
|
|
|
|
|
|
|
|
|
|
char returnType;
|
|
|
|
char returnType;
|
|
|
|
std::string argumentType;
|
|
|
|
std::string argumentType;
|
|
|
|
|
|
|
|
|
|
|
|
if (extensions->isFunction (keyword, returnType, argumentType, false))
|
|
|
|
if (extensions->isFunction (keyword, returnType, argumentType, false))
|
|
|
|
{
|
|
|
|
{
|
|
|
|
mTokenLoc = loc;
|
|
|
|
mTokenLoc = loc;
|
|
|
|
parseArguments (argumentType, scanner);
|
|
|
|
parseArguments (argumentType, scanner);
|
|
|
|
|
|
|
|
|
|
|
|
extensions->generateFunctionCode (keyword, mCode, mLiterals, "");
|
|
|
|
extensions->generateFunctionCode (keyword, mCode, mLiterals, "");
|
|
|
|
mOperands.push_back (returnType);
|
|
|
|
mOperands.push_back (returnType);
|
|
|
|
|
|
|
|
|
|
|
|
mNextOperand = false;
|
|
|
|
mNextOperand = false;
|
|
|
|
return true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
@ -454,7 +482,7 @@ namespace Compiler
|
|
|
|
scanner.putbackKeyword (keyword, loc);
|
|
|
|
scanner.putbackKeyword (keyword, loc);
|
|
|
|
return false;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return Parser::parseKeyword (keyword, loc, scanner);
|
|
|
|
return Parser::parseKeyword (keyword, loc, scanner);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
@ -467,10 +495,10 @@ namespace Compiler
|
|
|
|
mRefOp = true;
|
|
|
|
mRefOp = true;
|
|
|
|
return true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return Parser::parseSpecial (code, loc, scanner);
|
|
|
|
return Parser::parseSpecial (code, loc, scanner);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (code==Scanner::S_comma)
|
|
|
|
if (code==Scanner::S_comma)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
mTokenLoc = loc;
|
|
|
|
mTokenLoc = loc;
|
|
|
@ -481,14 +509,14 @@ namespace Compiler
|
|
|
|
mFirst = false;
|
|
|
|
mFirst = false;
|
|
|
|
return true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// end marker
|
|
|
|
// end marker
|
|
|
|
scanner.putbackSpecial (code, loc);
|
|
|
|
scanner.putbackSpecial (code, loc);
|
|
|
|
return false;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
mFirst = false;
|
|
|
|
mFirst = false;
|
|
|
|
|
|
|
|
|
|
|
|
if (code==Scanner::S_newline)
|
|
|
|
if (code==Scanner::S_newline)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
// end marker
|
|
|
|
// end marker
|
|
|
@ -496,7 +524,7 @@ namespace Compiler
|
|
|
|
scanner.putbackSpecial (code, loc);
|
|
|
|
scanner.putbackSpecial (code, loc);
|
|
|
|
return false;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (code==Scanner::S_minus && mNextOperand)
|
|
|
|
if (code==Scanner::S_minus && mNextOperand)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
// unary
|
|
|
|
// unary
|
|
|
@ -504,7 +532,7 @@ namespace Compiler
|
|
|
|
mTokenLoc = loc;
|
|
|
|
mTokenLoc = loc;
|
|
|
|
return true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (code==Scanner::S_open)
|
|
|
|
if (code==Scanner::S_open)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (mNextOperand)
|
|
|
|
if (mNextOperand)
|
|
|
@ -520,7 +548,7 @@ namespace Compiler
|
|
|
|
return false;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (code==Scanner::S_close && !mNextOperand)
|
|
|
|
if (code==Scanner::S_close && !mNextOperand)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (isOpen())
|
|
|
|
if (isOpen())
|
|
|
@ -528,17 +556,17 @@ namespace Compiler
|
|
|
|
close();
|
|
|
|
close();
|
|
|
|
return true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
mTokenLoc = loc;
|
|
|
|
mTokenLoc = loc;
|
|
|
|
scanner.putbackSpecial (code, loc);
|
|
|
|
scanner.putbackSpecial (code, loc);
|
|
|
|
return false;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (!mNextOperand)
|
|
|
|
if (!mNextOperand)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
mTokenLoc = loc;
|
|
|
|
mTokenLoc = loc;
|
|
|
|
char c = 0; // comparison
|
|
|
|
char c = 0; // comparison
|
|
|
|
|
|
|
|
|
|
|
|
switch (code)
|
|
|
|
switch (code)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
case Scanner::S_plus: pushBinaryOperator ('+'); return true;
|
|
|
|
case Scanner::S_plus: pushBinaryOperator ('+'); return true;
|
|
|
@ -552,7 +580,7 @@ namespace Compiler
|
|
|
|
case Scanner::S_cmpGT: c = 'g'; break;
|
|
|
|
case Scanner::S_cmpGT: c = 'g'; break;
|
|
|
|
case Scanner::S_cmpGE: c = 'G'; break;
|
|
|
|
case Scanner::S_cmpGE: c = 'G'; break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (c)
|
|
|
|
if (c)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (mArgument && !isOpen())
|
|
|
|
if (mArgument && !isOpen())
|
|
|
@ -562,15 +590,15 @@ namespace Compiler
|
|
|
|
scanner.putbackSpecial (code, loc);
|
|
|
|
scanner.putbackSpecial (code, loc);
|
|
|
|
return false;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
pushBinaryOperator (c);
|
|
|
|
pushBinaryOperator (c);
|
|
|
|
return true;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return Parser::parseSpecial (code, loc, scanner);
|
|
|
|
return Parser::parseSpecial (code, loc, scanner);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ExprParser::reset()
|
|
|
|
void ExprParser::reset()
|
|
|
|
{
|
|
|
|
{
|
|
|
|
mOperands.clear();
|
|
|
|
mOperands.clear();
|
|
|
@ -580,8 +608,9 @@ namespace Compiler
|
|
|
|
mFirst = true;
|
|
|
|
mFirst = true;
|
|
|
|
mExplicit.clear();
|
|
|
|
mExplicit.clear();
|
|
|
|
mRefOp = false;
|
|
|
|
mRefOp = false;
|
|
|
|
|
|
|
|
Parser::reset();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
char ExprParser::append (std::vector<Interpreter::Type_Code>& code)
|
|
|
|
char ExprParser::append (std::vector<Interpreter::Type_Code>& code)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (mOperands.empty() && mOperators.empty())
|
|
|
|
if (mOperands.empty() && mOperators.empty())
|
|
|
@ -589,7 +618,7 @@ namespace Compiler
|
|
|
|
getErrorHandler().error ("missing expression", mTokenLoc);
|
|
|
|
getErrorHandler().error ("missing expression", mTokenLoc);
|
|
|
|
return 'l';
|
|
|
|
return 'l';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (mNextOperand || mOperands.empty())
|
|
|
|
if (mNextOperand || mOperands.empty())
|
|
|
|
{
|
|
|
|
{
|
|
|
|
getErrorHandler().error ("syntax error in expression", mTokenLoc);
|
|
|
|
getErrorHandler().error ("syntax error in expression", mTokenLoc);
|
|
|
@ -603,62 +632,90 @@ namespace Compiler
|
|
|
|
|
|
|
|
|
|
|
|
assert (mOperands.size()==1);
|
|
|
|
assert (mOperands.size()==1);
|
|
|
|
return mOperands[0];
|
|
|
|
return mOperands[0];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void ExprParser::parseArguments (const std::string& arguments, Scanner& scanner,
|
|
|
|
int ExprParser::parseArguments (const std::string& arguments, Scanner& scanner,
|
|
|
|
std::vector<Interpreter::Type_Code>& code, bool invert)
|
|
|
|
std::vector<Interpreter::Type_Code>& code, bool invert)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
|
|
|
|
bool optional = false;
|
|
|
|
|
|
|
|
bool optionalCount = 0;
|
|
|
|
|
|
|
|
|
|
|
|
ExprParser parser (getErrorHandler(), getContext(), mLocals, mLiterals, true);
|
|
|
|
ExprParser parser (getErrorHandler(), getContext(), mLocals, mLiterals, true);
|
|
|
|
StringParser stringParser (getErrorHandler(), getContext(), mLiterals);
|
|
|
|
StringParser stringParser (getErrorHandler(), getContext(), mLiterals);
|
|
|
|
|
|
|
|
|
|
|
|
std::stack<std::vector<Interpreter::Type_Code> > stack;
|
|
|
|
std::stack<std::vector<Interpreter::Type_Code> > stack;
|
|
|
|
|
|
|
|
|
|
|
|
for (std::string::const_iterator iter (arguments.begin()); iter!=arguments.end();
|
|
|
|
for (std::string::const_iterator iter (arguments.begin()); iter!=arguments.end();
|
|
|
|
++iter)
|
|
|
|
++iter)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
if (*iter=='S' || *iter=='c')
|
|
|
|
if (*iter=='/')
|
|
|
|
|
|
|
|
{
|
|
|
|
|
|
|
|
optional = true;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
else if (*iter=='S' || *iter=='c')
|
|
|
|
{
|
|
|
|
{
|
|
|
|
stringParser.reset();
|
|
|
|
stringParser.reset();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (optional)
|
|
|
|
|
|
|
|
stringParser.setOptional (true);
|
|
|
|
|
|
|
|
|
|
|
|
if (*iter=='c') stringParser.smashCase();
|
|
|
|
if (*iter=='c') stringParser.smashCase();
|
|
|
|
scanner.scan (stringParser);
|
|
|
|
scanner.scan (stringParser);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (optional && stringParser.isEmpty())
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
if (invert)
|
|
|
|
if (invert)
|
|
|
|
{
|
|
|
|
{
|
|
|
|
std::vector<Interpreter::Type_Code> tmp;
|
|
|
|
std::vector<Interpreter::Type_Code> tmp;
|
|
|
|
stringParser.append (tmp);
|
|
|
|
stringParser.append (tmp);
|
|
|
|
|
|
|
|
|
|
|
|
stack.push (tmp);
|
|
|
|
stack.push (tmp);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
else
|
|
|
|
stringParser.append (code);
|
|
|
|
stringParser.append (code);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (optional)
|
|
|
|
|
|
|
|
++optionalCount;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
else
|
|
|
|
{
|
|
|
|
{
|
|
|
|
parser.reset();
|
|
|
|
parser.reset();
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (optional)
|
|
|
|
|
|
|
|
parser.setOptional (true);
|
|
|
|
|
|
|
|
|
|
|
|
scanner.scan (parser);
|
|
|
|
scanner.scan (parser);
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (optional && parser.isEmpty())
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
|
|
std::vector<Interpreter::Type_Code> tmp;
|
|
|
|
std::vector<Interpreter::Type_Code> tmp;
|
|
|
|
|
|
|
|
|
|
|
|
char type = parser.append (tmp);
|
|
|
|
char type = parser.append (tmp);
|
|
|
|
|
|
|
|
|
|
|
|
if (type!=*iter)
|
|
|
|
if (type!=*iter)
|
|
|
|
Generator::convert (tmp, type, *iter);
|
|
|
|
Generator::convert (tmp, type, *iter);
|
|
|
|
|
|
|
|
|
|
|
|
if (invert)
|
|
|
|
if (invert)
|
|
|
|
stack.push (tmp);
|
|
|
|
stack.push (tmp);
|
|
|
|
else
|
|
|
|
else
|
|
|
|
std::copy (tmp.begin(), tmp.end(), std::back_inserter (code));
|
|
|
|
std::copy (tmp.begin(), tmp.end(), std::back_inserter (code));
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
if (optional)
|
|
|
|
|
|
|
|
++optionalCount;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
while (!stack.empty())
|
|
|
|
while (!stack.empty())
|
|
|
|
{
|
|
|
|
{
|
|
|
|
std::vector<Interpreter::Type_Code>& tmp = stack.top();
|
|
|
|
std::vector<Interpreter::Type_Code>& tmp = stack.top();
|
|
|
|
|
|
|
|
|
|
|
|
std::copy (tmp.begin(), tmp.end(), std::back_inserter (code));
|
|
|
|
std::copy (tmp.begin(), tmp.end(), std::back_inserter (code));
|
|
|
|
|
|
|
|
|
|
|
|
stack.pop();
|
|
|
|
stack.pop();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
return optionalCount;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|