mirror of
https://github.com/ZDoom/qzdoom.git
synced 2025-01-22 00:41:23 +00:00
653 lines
16 KiB
C++
653 lines
16 KiB
C++
// Emacs style mode select -*- C++ -*-
|
|
//----------------------------------------------------------------------------
|
|
//
|
|
// Copyright(C) 2000 Simon Howard
|
|
//
|
|
// 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., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
//
|
|
//--------------------------------------------------------------------------
|
|
//
|
|
// Operators
|
|
//
|
|
// Handler code for all the operators. The 'other half'
|
|
// of the parsing.
|
|
//
|
|
// By Simon Howard
|
|
//
|
|
//---------------------------------------------------------------------------
|
|
//
|
|
// FraggleScript is from SMMU which is under the GPL. Technically,
|
|
// therefore, combining the FraggleScript code with the non-free
|
|
// ZDoom code is a violation of the GPL.
|
|
//
|
|
// As this may be a problem for you, I hereby grant an exception to my
|
|
// copyright on the SMMU source (including FraggleScript). You may use
|
|
// any code from SMMU in (G)ZDoom, provided that:
|
|
//
|
|
// * For any binary release of the port, the source code is also made
|
|
// available.
|
|
// * The copyright notice is kept on any file containing my code.
|
|
//
|
|
//
|
|
|
|
/* includes ************************/
|
|
#include "t_script.h"
|
|
|
|
|
|
#define evaluate_leftnright(a, b, c) {\
|
|
EvaluateExpression(left, (a), (b)-1); \
|
|
EvaluateExpression(right, (b)+1, (c)); }\
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
|
|
FParser::operator_t FParser::operators[]=
|
|
{
|
|
{"=", &FParser::OPequals, backward},
|
|
{"||", &FParser::OPor, forward},
|
|
{"&&", &FParser::OPand, forward},
|
|
{"|", &FParser::OPor_bin, forward},
|
|
{"&", &FParser::OPand_bin, forward},
|
|
{"==", &FParser::OPcmp, forward},
|
|
{"!=", &FParser::OPnotcmp, forward},
|
|
{"<", &FParser::OPlessthan, forward},
|
|
{">", &FParser::OPgreaterthan, forward},
|
|
{"<=", &FParser::OPlessthanorequal, forward},
|
|
{">=", &FParser::OPgreaterthanorequal, forward},
|
|
|
|
{"+", &FParser::OPplus, forward},
|
|
{"-", &FParser::OPminus, forward},
|
|
{"*", &FParser::OPmultiply, forward},
|
|
{"/", &FParser::OPdivide, forward},
|
|
{"%", &FParser::OPremainder, forward},
|
|
{"~", &FParser::OPnot_bin, forward}, // haleyjd
|
|
{"!", &FParser::OPnot, forward},
|
|
{"++", &FParser::OPincrement, forward},
|
|
{"--", &FParser::OPdecrement, forward},
|
|
{".", &FParser::OPstructure, forward},
|
|
};
|
|
|
|
int FParser::num_operators = sizeof(FParser::operators) / sizeof(FParser::operator_t);
|
|
|
|
/***************** logic *********************/
|
|
|
|
//-----------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void FParser::OPequals(svalue_t &result, int start, int n, int stop)
|
|
{
|
|
DFsVariable *var;
|
|
|
|
var = Script->FindVariable(Tokens[start]);
|
|
|
|
if(var)
|
|
{
|
|
EvaluateExpression(result, n+1, stop);
|
|
var->SetValue (result);
|
|
}
|
|
else
|
|
{
|
|
script_error("unknown variable '%s'\n", Tokens[start]);
|
|
}
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void FParser::OPor(svalue_t &result, int start, int n, int stop)
|
|
{
|
|
int exprtrue = false;
|
|
|
|
// if first is true, do not evaluate the second
|
|
|
|
EvaluateExpression(result, start, n-1);
|
|
|
|
if(intvalue(result))
|
|
exprtrue = true;
|
|
else
|
|
{
|
|
EvaluateExpression(result, n+1, stop);
|
|
exprtrue = !!intvalue(result);
|
|
}
|
|
|
|
result.type = svt_int;
|
|
result.value.i = exprtrue;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void FParser::OPand(svalue_t &result, int start, int n, int stop)
|
|
{
|
|
int exprtrue = true;
|
|
// if first is false, do not eval second
|
|
|
|
EvaluateExpression(result, start, n-1);
|
|
|
|
if(!intvalue(result) )
|
|
exprtrue = false;
|
|
else
|
|
{
|
|
EvaluateExpression(result, n+1, stop);
|
|
exprtrue = !!intvalue(result);
|
|
}
|
|
|
|
result.type = svt_int;
|
|
result.value.i = exprtrue;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void FParser::OPcmp(svalue_t &result, int start, int n, int stop)
|
|
{
|
|
svalue_t left, right;
|
|
|
|
evaluate_leftnright(start, n, stop);
|
|
|
|
result.type = svt_int; // always an int returned
|
|
|
|
if(left.type == svt_string && right.type == svt_string)
|
|
{
|
|
result.value.i = !strcmp(left.string, right.string);
|
|
return;
|
|
}
|
|
|
|
// haleyjd: direct mobj comparison when both are mobj
|
|
if(left.type == svt_mobj && right.type == svt_mobj)
|
|
{
|
|
// we can safely assume reference equivalency for
|
|
// AActor's in all cases since they are static for the
|
|
// duration of a level
|
|
result.value.i = (left.value.mobj == right.value.mobj);
|
|
return;
|
|
}
|
|
|
|
if(left.type == svt_fixed || right.type == svt_fixed)
|
|
{
|
|
result.value.i = (fixedvalue(left) == fixedvalue(right));
|
|
return;
|
|
}
|
|
|
|
result.value.i = (intvalue(left) == intvalue(right));
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void FParser::OPnotcmp(svalue_t &result, int start, int n, int stop)
|
|
{
|
|
OPcmp(result, start, n, stop);
|
|
result.type = svt_int;
|
|
result.value.i = !result.value.i;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void FParser::OPlessthan(svalue_t &result, int start, int n, int stop)
|
|
{
|
|
svalue_t left, right;
|
|
|
|
evaluate_leftnright(start, n, stop);
|
|
result.type = svt_int;
|
|
|
|
// haleyjd: 8-17
|
|
if(left.type == svt_fixed || right.type == svt_fixed)
|
|
result.value.i = (fixedvalue(left) < fixedvalue(right));
|
|
else
|
|
result.value.i = (intvalue(left) < intvalue(right));
|
|
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void FParser::OPgreaterthan(svalue_t &result, int start, int n, int stop)
|
|
{
|
|
svalue_t left, right;
|
|
|
|
evaluate_leftnright(start, n, stop);
|
|
|
|
// haleyjd: 8-17
|
|
result.type = svt_int;
|
|
if(left.type == svt_fixed || right.type == svt_fixed)
|
|
result.value.i = (fixedvalue(left) > fixedvalue(right));
|
|
else
|
|
result.value.i = (intvalue(left) > intvalue(right));
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void FParser::OPnot(svalue_t &result, int start, int n, int stop)
|
|
{
|
|
EvaluateExpression(result, n+1, stop);
|
|
|
|
result.value.i = !intvalue(result);
|
|
result.type = svt_int;
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void FParser::OPplus(svalue_t &result, int start, int n, int stop)
|
|
{
|
|
svalue_t left, right;
|
|
|
|
evaluate_leftnright(start, n, stop);
|
|
|
|
if (left.type == svt_string)
|
|
{
|
|
if (right.type == svt_string)
|
|
{
|
|
result.string.Format("%s%s", left.string.GetChars(), right.string.GetChars());
|
|
}
|
|
else if (right.type == svt_fixed)
|
|
{
|
|
result.string.Format("%s%4.4f", left.string.GetChars(), floatvalue(right));
|
|
}
|
|
else
|
|
{
|
|
result.string.Format("%s%i", left.string.GetChars(), intvalue(right));
|
|
}
|
|
result.type = svt_string;
|
|
}
|
|
// haleyjd: 8-17
|
|
else if(left.type == svt_fixed || right.type == svt_fixed)
|
|
{
|
|
result.type = svt_fixed;
|
|
result.value.f = fixedvalue(left) + fixedvalue(right);
|
|
}
|
|
else
|
|
{
|
|
result.type = svt_int;
|
|
result.value.i = intvalue(left) + intvalue(right);
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void FParser::OPminus(svalue_t &result, int start, int n, int stop)
|
|
{
|
|
svalue_t left, right;
|
|
|
|
// do they mean minus as in '-1' rather than '2-1'?
|
|
if(start == n)
|
|
{
|
|
// kinda hack, hehe
|
|
EvaluateExpression(right, n+1, stop);
|
|
}
|
|
else
|
|
{
|
|
evaluate_leftnright(start, n, stop);
|
|
}
|
|
|
|
// haleyjd: 8-17
|
|
if(left.type == svt_fixed || right.type == svt_fixed)
|
|
{
|
|
result.type = svt_fixed;
|
|
result.value.f = fixedvalue(left) - fixedvalue(right);
|
|
}
|
|
else
|
|
{
|
|
result.type = svt_int;
|
|
result.value.i = intvalue(left) - intvalue(right);
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void FParser::OPmultiply(svalue_t &result,int start, int n, int stop)
|
|
{
|
|
svalue_t left, right;
|
|
|
|
evaluate_leftnright(start, n, stop);
|
|
|
|
// haleyjd: 8-17
|
|
if(left.type == svt_fixed || right.type == svt_fixed)
|
|
{
|
|
result.type = svt_fixed;
|
|
result.value.f = FixedMul(fixedvalue(left), fixedvalue(right));
|
|
}
|
|
else
|
|
{
|
|
result.type = svt_int;
|
|
result.value.i = intvalue(left) * intvalue(right);
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void FParser::OPdivide(svalue_t &result, int start, int n, int stop)
|
|
{
|
|
svalue_t left, right;
|
|
|
|
evaluate_leftnright(start, n, stop);
|
|
|
|
// haleyjd: 8-17
|
|
if(left.type == svt_fixed || right.type == svt_fixed)
|
|
{
|
|
fixed_t fr;
|
|
|
|
if((fr = fixedvalue(right)) == 0)
|
|
script_error("divide by zero\n");
|
|
else
|
|
{
|
|
result.type = svt_fixed;
|
|
result.value.f = FixedDiv(fixedvalue(left), fr);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
int ir;
|
|
|
|
if(!(ir = intvalue(right)))
|
|
script_error("divide by zero\n");
|
|
else
|
|
{
|
|
result.type = svt_int;
|
|
result.value.i = intvalue(left) / ir;
|
|
}
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void FParser::OPremainder(svalue_t &result, int start, int n, int stop)
|
|
{
|
|
svalue_t left, right;
|
|
int ir;
|
|
|
|
evaluate_leftnright(start, n, stop);
|
|
|
|
if(!(ir = intvalue(right)))
|
|
script_error("divide by zero\n");
|
|
else
|
|
{
|
|
result.type = svt_int;
|
|
result.value.i = intvalue(left) % ir;
|
|
}
|
|
}
|
|
|
|
/********** binary operators **************/
|
|
|
|
//-----------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void FParser::OPor_bin(svalue_t &result, int start, int n, int stop)
|
|
{
|
|
svalue_t left, right;
|
|
|
|
evaluate_leftnright(start, n, stop);
|
|
|
|
result.type = svt_int;
|
|
result.value.i = intvalue(left) | intvalue(right);
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void FParser::OPand_bin(svalue_t &result, int start, int n, int stop)
|
|
{
|
|
svalue_t left, right;
|
|
|
|
evaluate_leftnright(start, n, stop);
|
|
|
|
result.type = svt_int;
|
|
result.value.i = intvalue(left) & intvalue(right);
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void FParser::OPnot_bin(svalue_t &result, int start, int n, int stop)
|
|
{
|
|
EvaluateExpression(result, n+1, stop);
|
|
|
|
result.value.i = ~intvalue(result);
|
|
result.type = svt_int;
|
|
}
|
|
|
|
|
|
//-----------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void FParser::OPincrement(svalue_t &result, int start, int n, int stop)
|
|
{
|
|
if(start == n) // ++n
|
|
{
|
|
DFsVariable *var;
|
|
|
|
var = Script->FindVariable(Tokens[stop]);
|
|
if(!var)
|
|
{
|
|
script_error("unknown variable '%s'\n", Tokens[stop]);
|
|
}
|
|
var->GetValue(result);
|
|
|
|
// haleyjd
|
|
if(var->type != svt_fixed)
|
|
{
|
|
result.value.i = intvalue(result) + 1;
|
|
result.type = svt_int;
|
|
var->SetValue (result);
|
|
}
|
|
else
|
|
{
|
|
result.value.f = fixedvalue(result) + FRACUNIT;
|
|
result.type = svt_fixed;
|
|
var->SetValue (result);
|
|
}
|
|
}
|
|
else if(stop == n) // n++
|
|
{
|
|
svalue_t newvalue;
|
|
DFsVariable *var;
|
|
|
|
var = Script->FindVariable(Tokens[start]);
|
|
if(!var)
|
|
{
|
|
script_error("unknown variable '%s'\n", Tokens[start]);
|
|
}
|
|
var->GetValue(result);
|
|
|
|
// haleyjd
|
|
if(var->type != svt_fixed)
|
|
{
|
|
newvalue.type = svt_int;
|
|
newvalue.value.i = intvalue(result) + 1;
|
|
var->SetValue (newvalue);
|
|
}
|
|
else
|
|
{
|
|
newvalue.type = svt_fixed;
|
|
newvalue.value.f = fixedvalue(result) + FRACUNIT;
|
|
var->SetValue (newvalue);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
script_error("incorrect arguments to ++ operator\n");
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void FParser::OPdecrement(svalue_t &result, int start, int n, int stop)
|
|
{
|
|
if(start == n) // ++n
|
|
{
|
|
DFsVariable *var;
|
|
|
|
var = Script->FindVariable(Tokens[stop]);
|
|
if(!var)
|
|
{
|
|
script_error("unknown variable '%s'\n", Tokens[stop]);
|
|
}
|
|
var->GetValue(result);
|
|
|
|
// haleyjd
|
|
if(var->type != svt_fixed)
|
|
{
|
|
result.value.i = intvalue(result) - 1;
|
|
result.type = svt_int;
|
|
var->SetValue (result);
|
|
}
|
|
else
|
|
{
|
|
result.value.f = fixedvalue(result) - FRACUNIT;
|
|
result.type = svt_fixed;
|
|
var->SetValue (result);
|
|
}
|
|
}
|
|
else if(stop == n) // n++
|
|
{
|
|
svalue_t newvalue;
|
|
DFsVariable *var;
|
|
|
|
var = Script->FindVariable(Tokens[start]);
|
|
if(!var)
|
|
{
|
|
script_error("unknown variable '%s'\n", Tokens[start]);
|
|
}
|
|
var->GetValue(result);
|
|
|
|
// haleyjd
|
|
if(var->type != svt_fixed)
|
|
{
|
|
newvalue.type = svt_int;
|
|
newvalue.value.i = intvalue(result) - 1;
|
|
var->SetValue (newvalue);
|
|
}
|
|
else
|
|
{
|
|
newvalue.type = svt_fixed;
|
|
newvalue.value.f = fixedvalue(result) - FRACUNIT;
|
|
var->SetValue (newvalue);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
script_error("incorrect arguments to ++ operator\n");
|
|
}
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void FParser::OPlessthanorequal(svalue_t &result, int start, int n, int stop)
|
|
{
|
|
svalue_t left, right;
|
|
|
|
evaluate_leftnright(start, n, stop);
|
|
|
|
result.type = svt_int;
|
|
|
|
if(left.type == svt_fixed || right.type == svt_fixed)
|
|
result.value.i = (fixedvalue(left) <= fixedvalue(right));
|
|
else
|
|
result.value.i = (intvalue(left) <= intvalue(right));
|
|
}
|
|
|
|
//-----------------------------------------------------------------------------
|
|
//
|
|
//
|
|
//
|
|
//-----------------------------------------------------------------------------
|
|
|
|
void FParser::OPgreaterthanorequal(svalue_t &result, int start, int n, int stop)
|
|
{
|
|
svalue_t left, right;
|
|
|
|
evaluate_leftnright(start, n, stop);
|
|
|
|
result.type = svt_int;
|
|
|
|
if(left.type == svt_fixed || right.type == svt_fixed)
|
|
result.value.i = (fixedvalue(left) >= fixedvalue(right));
|
|
else
|
|
result.value.i = (intvalue(left) >= intvalue(right));
|
|
}
|
|
|