mirror of
https://github.com/ptitSeb/Serious-Engine
synced 2025-01-13 23:31:32 +01:00
1081 lines
28 KiB
Plaintext
1081 lines
28 KiB
Plaintext
%{
|
|
#include <Engine/StdH.h>
|
|
|
|
#include <Engine/Base/Console.h>
|
|
#include <Engine/Base/Shell.h>
|
|
#include "ParsingSymbols.h"
|
|
|
|
#include <Engine/Templates/DynamicStackArray.cpp>
|
|
#include <Engine/Templates/AllocationArray.cpp>
|
|
|
|
%}
|
|
|
|
%{
|
|
// turn off over-helpful bit of bison... --ryan.
|
|
#ifdef __GNUC__
|
|
#define __attribute__(x)
|
|
#endif
|
|
|
|
#define YYERROR_VERBOSE 1
|
|
// if error occurs in parsing
|
|
void yyerror(const char *str)
|
|
{
|
|
// just report the string
|
|
_pShell->ErrorF("%s", str);
|
|
};
|
|
|
|
static BOOL _bExecNextElse = FALSE;
|
|
FLOAT fDummy = -666.0f;
|
|
|
|
static INDEX _iStack = 0;
|
|
static UBYTE _ubStack[1024];
|
|
|
|
INDEX PushExpression(value &v)
|
|
{
|
|
if (v.sttType==STT_FLOAT) {
|
|
FLOAT f = v.fFloat;
|
|
memcpy(_ubStack+_iStack, &f, sizeof(f));
|
|
_iStack+=sizeof(f);
|
|
return sizeof(f);
|
|
} else if (v.sttType==STT_INDEX) {
|
|
INDEX i = v.iIndex;
|
|
memcpy(_ubStack+_iStack, &i, sizeof(i));
|
|
_iStack+=sizeof(i);
|
|
return sizeof(i);
|
|
} else if (v.sttType==STT_STRING) {
|
|
CTString &str = _shell_astrTempStrings.Push();
|
|
str = v.strString;
|
|
CTString *pstr = &str;
|
|
memcpy(_ubStack+_iStack, &pstr, sizeof(pstr));
|
|
_iStack+=sizeof(pstr);
|
|
return sizeof(pstr);
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
BOOL MatchTypes(value &v1, value &v2)
|
|
{
|
|
if (v1.sttType==STT_FLOAT && v2.sttType==STT_FLOAT) {
|
|
return TRUE;
|
|
} else if (v1.sttType==STT_STRING && v2.sttType==STT_STRING) {
|
|
return TRUE;
|
|
} else if (v1.sttType==STT_INDEX && v2.sttType==STT_INDEX) {
|
|
return TRUE;
|
|
} else {
|
|
v1.sttType = STT_ILLEGAL;
|
|
v2.sttType = STT_ILLEGAL;
|
|
_pShell->ErrorF("Type mismatch");
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
void Declaration(
|
|
ULONG ulQualifiers, INDEX istType, CShellSymbol &ssNew,
|
|
INDEX (*pPreFunc)(INDEX), void (*pPostFunc)(INDEX))
|
|
{
|
|
// if external
|
|
if (ulQualifiers&SSF_EXTERNAL) {
|
|
// get it a new value
|
|
if (_shell_ast[istType].st_sttType==STT_INDEX
|
|
||_shell_ast[istType].st_sttType==STT_FLOAT) {
|
|
_pvNextToDeclare = &_shell_afExtFloats.Push();
|
|
} else if (_shell_ast[istType].st_sttType==STT_STRING) {
|
|
_pvNextToDeclare = &_shell_astrExtStrings.Push();
|
|
} else {
|
|
NOTHING;
|
|
}
|
|
}
|
|
|
|
// if not parsing an external declaration
|
|
if (_pvNextToDeclare==NULL) {
|
|
// error
|
|
_pShell->ErrorF("Only external declarations are supported");
|
|
return;
|
|
}
|
|
|
|
// if the symbol is declared already
|
|
if (ssNew.IsDeclared()) {
|
|
// if the declaration is not identical
|
|
if (!ShellTypeIsSame(ssNew.ss_istType, istType) ||
|
|
((ssNew.ss_ulFlags&SSF_CONSTANT)!=(ulQualifiers&SSF_CONSTANT))) {
|
|
// error
|
|
_pShell->ErrorF("Symbol '%s' is already declared diferrently", (const char *) ssNew.ss_strName);
|
|
return;
|
|
}
|
|
|
|
// copy its value
|
|
if (_shell_ast[ssNew.ss_istType].st_sttType==STT_INDEX) {
|
|
*(INDEX*)_pvNextToDeclare = *(INDEX*)ssNew.ss_pvValue;
|
|
} else if (_shell_ast[ssNew.ss_istType].st_sttType==STT_FLOAT) {
|
|
*(FLOAT*)_pvNextToDeclare = *(FLOAT*)ssNew.ss_pvValue;
|
|
} else if (_shell_ast[ssNew.ss_istType].st_sttType==STT_STRING) {
|
|
*(CTString*)_pvNextToDeclare = *(CTString*)ssNew.ss_pvValue;
|
|
} else if (_shell_ast[ssNew.ss_istType].st_sttType==STT_ARRAY) {
|
|
NOTHING; // array values are not retained
|
|
} else if (_shell_ast[ssNew.ss_istType].st_sttType==STT_FUNCTION) {
|
|
NOTHING; // function values are not retained
|
|
} else {
|
|
// error
|
|
_pShell->ErrorF("'%s': old value couldn't be retained", (const char *) ssNew.ss_strName);
|
|
return;
|
|
}
|
|
}
|
|
|
|
// set the type to given type
|
|
if (!ssNew.IsDeclared()) {
|
|
ssNew.ss_istType = ShellTypeMakeDuplicate(istType);
|
|
}
|
|
// set the value for the external symbol if not already set
|
|
if (ssNew.ss_pvValue==NULL || !(ulQualifiers&SSF_EXTERNAL)) {
|
|
ssNew.ss_pvValue = _pvNextToDeclare;
|
|
}
|
|
// remember qualifiers (if already predeclared - keep old flags)
|
|
ssNew.ss_ulFlags |= ulQualifiers;
|
|
// remember pre and post functions
|
|
if (ssNew.ss_pPreFunc==NULL) {
|
|
ssNew.ss_pPreFunc = (BOOL (*)(void *))pPreFunc;
|
|
}
|
|
if (ssNew.ss_pPostFunc==NULL) {
|
|
ssNew.ss_pPostFunc = (void (*)(void *))pPostFunc;
|
|
}
|
|
}
|
|
|
|
void DoComparison(value &vRes, value &v0, value &v1, int token)
|
|
{
|
|
MatchTypes(v0, v1);
|
|
|
|
vRes.sttType = STT_INDEX;
|
|
if (v0.sttType == STT_FLOAT) {
|
|
switch (token) {
|
|
case '<': vRes.iIndex = v0.fFloat <v1.fFloat; break;
|
|
case '>': vRes.iIndex = v0.fFloat >v1.fFloat; break;
|
|
case '=': vRes.iIndex = v0.fFloat==v1.fFloat; break;
|
|
case '!': vRes.iIndex = v0.fFloat!=v1.fFloat; break;
|
|
case '}': vRes.iIndex = v0.fFloat>=v1.fFloat; break;
|
|
case '{': vRes.iIndex = v0.fFloat<=v1.fFloat; break;
|
|
default: ASSERT(FALSE);
|
|
vRes.sttType = STT_INDEX;
|
|
vRes.iIndex = 0;
|
|
}
|
|
} else if (v0.sttType == STT_INDEX) {
|
|
switch (token) {
|
|
case '<': vRes.iIndex = v0.iIndex <v1.iIndex; break;
|
|
case '>': vRes.iIndex = v0.iIndex >v1.iIndex; break;
|
|
case '=': vRes.iIndex = v0.iIndex==v1.iIndex; break;
|
|
case '!': vRes.iIndex = v0.iIndex!=v1.iIndex; break;
|
|
case '}': vRes.iIndex = v0.iIndex>=v1.iIndex; break;
|
|
case '{': vRes.iIndex = v0.iIndex<=v1.iIndex; break;
|
|
default: ASSERT(FALSE);
|
|
vRes.sttType = STT_INDEX;
|
|
vRes.iIndex = 0;
|
|
}
|
|
} else if (v0.sttType == STT_STRING) {
|
|
switch (token) {
|
|
case '<': vRes.iIndex = stricmp(v0.strString, v1.strString) < 0; break;
|
|
case '>': vRes.iIndex = stricmp(v0.strString, v1.strString) > 0; break;
|
|
case '=': vRes.iIndex = stricmp(v0.strString, v1.strString) == 0; break;
|
|
case '!': vRes.iIndex = stricmp(v0.strString, v1.strString) != 0; break;
|
|
case '}': vRes.iIndex = stricmp(v0.strString, v1.strString) >= 0; break;
|
|
case '{': vRes.iIndex = stricmp(v0.strString, v1.strString) <= 0; break;
|
|
default: ASSERT(FALSE);
|
|
vRes.sttType = STT_INDEX;
|
|
vRes.iIndex = 0;
|
|
}
|
|
} else {
|
|
vRes.sttType = STT_INDEX;
|
|
vRes.iIndex = 0;
|
|
}
|
|
}
|
|
%}
|
|
|
|
/* BISON Declarations */
|
|
|
|
// we need to be reentrant!
|
|
%pure_parser
|
|
|
|
%union {
|
|
value val; // for constants and expressions
|
|
arguments arg; // for function input arguments
|
|
ULONG ulFlags; // for declaration qualifiers
|
|
INDEX istType; // for types
|
|
CShellSymbol *pssSymbol; // for symbols
|
|
struct LValue lvLValue;
|
|
INDEX (*pPreFunc)(INDEX); // pre-set function for a variable
|
|
void (*pPostFunc)(INDEX); // post-set function for a variable
|
|
}
|
|
|
|
%{
|
|
extern int yylex(YYSTYPE *lvalp);
|
|
%}
|
|
|
|
%token <val> c_float
|
|
%token <val> c_int
|
|
%token <val> c_string
|
|
%token <val> c_char
|
|
|
|
%token <pssSymbol> identifier
|
|
|
|
%token k_INDEX
|
|
%token k_FLOAT
|
|
%token k_CTString
|
|
%token k_void
|
|
%token k_const
|
|
%token k_user
|
|
%token k_persistent
|
|
%token k_extern
|
|
%token k_pre
|
|
%token k_post
|
|
%token k_help
|
|
%token <ulFlags> k_if
|
|
%token k_else
|
|
%token <ulFlags> k_else_if
|
|
%token SHL
|
|
%token SHR
|
|
%token EQ
|
|
%token NEQ
|
|
%token GEQ
|
|
%token LEQ
|
|
%token LOGAND
|
|
%token LOGOR
|
|
%token block_beg
|
|
%token block_end
|
|
|
|
%type <lvLValue> lvalue
|
|
%type <val> expression
|
|
%type <ulFlags> declaration_qualifiers
|
|
%type <val> opt_string
|
|
%type <istType> type_specifier
|
|
%type <istType> parameter_list
|
|
%type <istType> parameter_list_opt
|
|
%type <pPreFunc> pre_func_opt
|
|
%type <pPostFunc> post_func_opt
|
|
%type <arg> argument_expression_list_opt
|
|
%type <arg> argument_expression_list
|
|
|
|
%right '='
|
|
%left LOGAND LOGOR
|
|
%left '&' '^' '|'
|
|
%left '<' '>' EQ NEQ LEQ GEQ
|
|
%left SHL
|
|
%left SHR
|
|
%left '-' '+'
|
|
%left '*' '/' '%'
|
|
%left TYPECAST
|
|
%left SIGN '!'
|
|
|
|
%start program
|
|
|
|
%%
|
|
|
|
/*/////////////////////////////////////////////////////////
|
|
* Global structure of the source file.
|
|
*/
|
|
program
|
|
: declaration
|
|
| statements
|
|
;
|
|
|
|
block
|
|
: block_beg statements block_end
|
|
| block_beg statements block_end
|
|
;
|
|
|
|
statements
|
|
: /* null */
|
|
| statement statements
|
|
;
|
|
|
|
declaration_qualifiers
|
|
: /* nothing */ {
|
|
$$ = 0;
|
|
}
|
|
| declaration_qualifiers k_const {
|
|
$$ = $1 | SSF_CONSTANT;
|
|
}
|
|
| declaration_qualifiers k_user {
|
|
$$ = $1 | SSF_USER;
|
|
}
|
|
| declaration_qualifiers k_persistent {
|
|
$$ = $1 | SSF_PERSISTENT;
|
|
}
|
|
| declaration_qualifiers k_extern {
|
|
$$ = $1 | SSF_EXTERNAL;
|
|
}
|
|
|
|
opt_string
|
|
: /* nothing */ {
|
|
$$.strString = "";
|
|
}
|
|
| c_string {
|
|
// !!!! remove this option
|
|
//_pShell->ErrorF("Warning: symbol comments are not supported");
|
|
$$.strString = $1.strString;
|
|
}
|
|
|
|
type_specifier
|
|
: k_FLOAT {
|
|
$$ = ShellTypeNewFloat();
|
|
}
|
|
| k_INDEX {
|
|
$$ = ShellTypeNewIndex();
|
|
}
|
|
| k_CTString {
|
|
$$ = ShellTypeNewString();
|
|
}
|
|
| k_void {
|
|
$$ = ShellTypeNewVoid();
|
|
}
|
|
|
|
pre_func_opt
|
|
: {
|
|
$$ = NULL;
|
|
}
|
|
| k_pre ':' identifier {
|
|
if (_shell_ast[$3->ss_istType].st_sttType!=STT_FUNCTION
|
|
||_shell_ast[_shell_ast[$3->ss_istType].st_istBaseType].st_sttType!=STT_INDEX
|
|
||_shell_ast[$3->ss_istType].st_istFirstArgument!=_shell_ast[$3->ss_istType].st_istLastArgument
|
|
||_shell_ast[_shell_ast[$3->ss_istType].st_istFirstArgument].st_sttType!=STT_INDEX) {
|
|
_pShell->ErrorF("'%s' must return 'INDEX' and take 'INDEX' as input", (const char *) $3->ss_strName);
|
|
} else {
|
|
void *pv = $3->ss_pvValue;
|
|
$$ = (INDEX(*)(INDEX))$3->ss_pvValue;
|
|
}
|
|
}
|
|
|
|
post_func_opt
|
|
: {
|
|
$$ = NULL;
|
|
}
|
|
| k_post ':' identifier {
|
|
if (_shell_ast[$3->ss_istType].st_sttType!=STT_FUNCTION
|
|
||_shell_ast[_shell_ast[$3->ss_istType].st_istBaseType].st_sttType!=STT_VOID
|
|
||_shell_ast[$3->ss_istType].st_istFirstArgument!=_shell_ast[$3->ss_istType].st_istLastArgument
|
|
||_shell_ast[_shell_ast[$3->ss_istType].st_istFirstArgument].st_sttType!=STT_INDEX) {
|
|
_pShell->ErrorF("'%s' must return 'void' and take 'INDEX' as input", (const char *) $3->ss_strName);
|
|
} else {
|
|
$$ = (void(*)(INDEX))$3->ss_pvValue;
|
|
}
|
|
}
|
|
;
|
|
|
|
parameter_list_opt
|
|
: /* nothing */ {
|
|
$$ = ShellTypeNewFunction(0);
|
|
ShellTypeAddFunctionArgument($$, ShellTypeNewVoid());
|
|
}
|
|
| parameter_list {
|
|
$$ = $1;
|
|
}
|
|
;
|
|
|
|
parameter_list
|
|
: type_specifier {
|
|
$$ = ShellTypeNewFunction(0);
|
|
ShellTypeAddFunctionArgument($$, $1);
|
|
}
|
|
/*| identifier type_specifier {
|
|
$$ = ShellTypeNewFunction(0);
|
|
ShellTypeAddFunctionArgument($$, $2);
|
|
}*/
|
|
| parameter_list ',' type_specifier {
|
|
$$ = $1;
|
|
ShellTypeAddFunctionArgument($$, $3);
|
|
}
|
|
/*| parameter_list ',' identifier type_specifier {
|
|
$$ = $1;
|
|
ShellTypeAddFunctionArgument($$, $4);
|
|
} */
|
|
;
|
|
|
|
declaration
|
|
: declaration_qualifiers type_specifier identifier pre_func_opt post_func_opt opt_string ';' {
|
|
Declaration($1, $2, *$3, $4, $5);
|
|
ShellTypeDelete($2);
|
|
}
|
|
| declaration_qualifiers type_specifier identifier '(' parameter_list_opt ')' opt_string ';' {
|
|
// take function from the parameter list and set its return type
|
|
_shell_ast[$5].st_istBaseType = $2;
|
|
$2 = $5;
|
|
// declare it
|
|
Declaration($1, $2, *$3, NULL, NULL);
|
|
// free the type (declaration will make a copy)
|
|
ShellTypeDelete($2);
|
|
}
|
|
| declaration_qualifiers type_specifier identifier '[' expression ']' pre_func_opt post_func_opt opt_string ';' {
|
|
if ($5.sttType!=STT_INDEX) {
|
|
_pShell->ErrorF("Array size is not integral");
|
|
}
|
|
$2 = ShellTypeNewArray($2, $5.iIndex);
|
|
Declaration($1, $2, *$3, NULL, NULL);
|
|
ShellTypeDelete($2);
|
|
}
|
|
|
|
statement
|
|
: ';' {
|
|
// dummy
|
|
}
|
|
| block {
|
|
// dummy
|
|
}
|
|
| expression ';' {
|
|
// print its value
|
|
if ($1.sttType == STT_VOID) {
|
|
NOTHING;
|
|
} else if ($1.sttType == STT_FLOAT) {
|
|
CPrintF("%g\n", $1.fFloat);
|
|
} else if ($1.sttType == STT_STRING) {
|
|
CPrintF("\"%s\"\n", $1.strString);
|
|
} else if ($1.sttType == STT_INDEX) {
|
|
CPrintF("%d(0x%08X)\n", $1.iIndex, $1.iIndex);
|
|
} else {
|
|
_pShell->ErrorF("Expression cannot be printed");
|
|
}
|
|
}
|
|
| lvalue '=' expression ';' {
|
|
// if it is constant
|
|
if ($1.lv_pssSymbol->ss_ulFlags&SSF_CONSTANT) {
|
|
_pShell->ErrorF("Symbol '%s' is a constant", (const char *) $1.lv_pssSymbol->ss_strName);
|
|
// if it is not constant
|
|
} else {
|
|
// if it can be changed
|
|
if ($1.lv_pssSymbol->ss_pPreFunc==NULL || $1.lv_pssSymbol->ss_pPreFunc($1.lv_pvAddress)) {
|
|
// if floats
|
|
if ($1.lv_sttType == STT_FLOAT && $3.sttType==STT_FLOAT) {
|
|
// assign value
|
|
*(FLOAT*)$1.lv_pvAddress = $3.fFloat;
|
|
// if indices
|
|
} else if ($1.lv_sttType == STT_INDEX && $3.sttType==STT_INDEX) {
|
|
// assign value
|
|
*(INDEX*)$1.lv_pvAddress = $3.iIndex;
|
|
|
|
// if strings
|
|
} else if ($1.lv_sttType == STT_STRING && $3.sttType==STT_STRING) {
|
|
// assign value
|
|
*(CTString*)$1.lv_pvAddress = $3.strString;
|
|
|
|
// if assigning index to float
|
|
} else if ($1.lv_sttType == STT_FLOAT && $3.sttType==STT_INDEX) {
|
|
*(FLOAT*)$1.lv_pvAddress = $3.iIndex;
|
|
// otherwise
|
|
} else {
|
|
// error
|
|
_pShell->ErrorF("Cannot assign: different types");
|
|
}
|
|
|
|
// call post-change function
|
|
if ($1.lv_pssSymbol->ss_pPostFunc!=NULL) {
|
|
$1.lv_pssSymbol->ss_pPostFunc($1.lv_pvAddress);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
| declaration_qualifiers type_specifier identifier '=' expression ';' {
|
|
Declaration($1, $2, *$3, NULL, NULL);
|
|
ShellTypeDelete($2);
|
|
|
|
CShellSymbol &ssSymbol = *$3;
|
|
// if it is constant
|
|
if (ssSymbol.ss_ulFlags&SSF_CONSTANT) {
|
|
// error
|
|
_pShell->ErrorF("Symbol '%s' is a constant", (const char *) ssSymbol.ss_strName);
|
|
}
|
|
|
|
// get symbol type
|
|
ShellTypeType stt = _shell_ast[$2].st_sttType;
|
|
|
|
// if floats
|
|
if (stt == STT_FLOAT && $5.sttType==STT_FLOAT) {
|
|
// assign value
|
|
*(FLOAT*)ssSymbol.ss_pvValue = $5.fFloat;
|
|
// if indices
|
|
} else if (stt == STT_INDEX && $5.sttType==STT_INDEX) {
|
|
// assign value
|
|
*(INDEX*)ssSymbol.ss_pvValue = $5.iIndex;
|
|
// if strings
|
|
} else if (stt == STT_STRING && $5.sttType==STT_STRING) {
|
|
// assign value
|
|
*(CTString*)ssSymbol.ss_pvValue = $5.strString;
|
|
// !!!! remove this conversion
|
|
} else if (stt == STT_FLOAT && $5.sttType==STT_INDEX) {
|
|
_pShell->ErrorF("Warning: assigning INDEX to FLOAT!");
|
|
*(FLOAT*)ssSymbol.ss_pvValue = $5.iIndex;
|
|
} else {
|
|
_pShell->ErrorF("Symbol '%s' and its initializer have different types", (const char *) ssSymbol.ss_strName);
|
|
}
|
|
}
|
|
| k_help identifier {
|
|
extern void PrintShellSymbolHelp(const CTString &strSymbol);
|
|
PrintShellSymbolHelp($2->ss_strName);
|
|
}
|
|
| k_help identifier '(' ')' {
|
|
extern void PrintShellSymbolHelp(const CTString &strSymbol);
|
|
PrintShellSymbolHelp($2->ss_strName);
|
|
}
|
|
| k_help identifier '[' ']' {
|
|
extern void PrintShellSymbolHelp(const CTString &strSymbol);
|
|
PrintShellSymbolHelp($2->ss_strName);
|
|
}
|
|
| k_if '(' expression ')' {
|
|
_bExecNextBlock = FALSE;
|
|
if ($3.sttType == STT_INDEX) {
|
|
_bExecNextBlock = $3.iIndex!=0;
|
|
} else if ($3.sttType == STT_FLOAT) {
|
|
_bExecNextBlock = $3.fFloat!=0;
|
|
} else {
|
|
_pShell->ErrorF("If expression is not integral");
|
|
}
|
|
$1 = _bExecNextBlock;
|
|
} block {
|
|
_bExecNextElse = !$1;
|
|
_bExecNextBlock = TRUE;
|
|
} opt_else
|
|
;
|
|
|
|
opt_else
|
|
: /* nothing */
|
|
|
|
| k_else_if '(' expression ')' {
|
|
if (_bExecNextElse) {
|
|
_bExecNextBlock = FALSE;
|
|
if ($3.sttType == STT_INDEX) {
|
|
_bExecNextBlock = $3.iIndex!=0;
|
|
} else if ($3.sttType == STT_FLOAT) {
|
|
_bExecNextBlock = $3.fFloat!=0;
|
|
} else {
|
|
_pShell->ErrorF("If expression is not integral");
|
|
}
|
|
$1 = _bExecNextBlock;
|
|
} else {
|
|
_bExecNextBlock = FALSE;
|
|
_bExecNextElse = FALSE;
|
|
$1 = TRUE;
|
|
}
|
|
} block {
|
|
_bExecNextElse = !$1;
|
|
_bExecNextBlock = TRUE;
|
|
} opt_else
|
|
|
|
| k_else {
|
|
_bExecNextBlock = _bExecNextElse;
|
|
} block {
|
|
_bExecNextBlock = TRUE;
|
|
}
|
|
;
|
|
|
|
lvalue
|
|
: identifier {
|
|
CShellSymbol &ssSymbol = *$1;
|
|
const ShellType &stType = _shell_ast[ssSymbol.ss_istType];
|
|
|
|
$$.lv_pssSymbol = &ssSymbol;
|
|
if (!ssSymbol.IsDeclared()) {
|
|
// error
|
|
_pShell->ErrorF("Identifier '%s' is not declared", (const char *) $1->ss_strName);
|
|
fDummy = -666;
|
|
$$.lv_sttType = STT_VOID;
|
|
$$.lv_pvAddress = &fDummy;
|
|
// if the identifier is a float, int or string
|
|
} else if (stType.st_sttType==STT_FLOAT || stType.st_sttType==STT_INDEX || stType.st_sttType==STT_STRING) {
|
|
// get its value and type
|
|
$$.lv_sttType = stType.st_sttType;
|
|
$$.lv_pvAddress = ssSymbol.ss_pvValue;
|
|
// if the identifier is something else
|
|
} else {
|
|
// error
|
|
_pShell->ErrorF("'%s' doesn't have a value", (const char *) $1->ss_strName);
|
|
fDummy = -666.0f;
|
|
$$.lv_sttType = STT_VOID;
|
|
$$.lv_pvAddress = &fDummy;
|
|
}
|
|
}
|
|
| identifier '[' expression ']' {
|
|
CShellSymbol &ssSymbol = *$1;
|
|
const ShellType &stType = _shell_ast[ssSymbol.ss_istType];
|
|
$$.lv_pssSymbol = &ssSymbol;
|
|
|
|
int iIndex = 0;
|
|
// if subscript is index
|
|
if ($3.sttType==STT_INDEX) {
|
|
// get the index
|
|
iIndex = $3.iIndex;
|
|
// if subscript is not index
|
|
} else {
|
|
// error
|
|
_pShell->ErrorF("Array subscript is not integral");
|
|
}
|
|
// if the symbol is array
|
|
if (stType.st_sttType==STT_ARRAY) {
|
|
const ShellType &stBase = _shell_ast[stType.st_istBaseType];
|
|
// if it is float or int array
|
|
if (stBase.st_sttType==STT_FLOAT || stBase.st_sttType==STT_INDEX) {
|
|
// if the index is out of range
|
|
if (iIndex<0 || iIndex>=stType.st_ctArraySize) {
|
|
_pShell->ErrorF("Array member out of range");
|
|
fDummy = -666.0f;
|
|
$$.lv_pvAddress = &fDummy;
|
|
} else {
|
|
// get its value and type
|
|
$$.lv_sttType = stBase.st_sttType;
|
|
$$.lv_pvAddress = (FLOAT*)ssSymbol.ss_pvValue+iIndex;
|
|
}
|
|
}
|
|
} else {
|
|
_pShell->ErrorF("'%s[]' doesn't have a value", (const char *) $1->ss_strName);
|
|
fDummy = -666.0f;
|
|
$$.lv_pvAddress = &fDummy;
|
|
}
|
|
}
|
|
;
|
|
|
|
argument_expression_list_opt
|
|
: /* nothing */ {
|
|
$$.istType = ShellTypeNewFunction(ShellTypeNewVoid());
|
|
ShellTypeAddFunctionArgument($$.istType, ShellTypeNewVoid());
|
|
$$.ctBytes = 0;
|
|
}
|
|
| argument_expression_list {
|
|
$$ = $1;
|
|
}
|
|
;
|
|
|
|
argument_expression_list
|
|
: expression {
|
|
$$.istType = ShellTypeNewFunction(ShellTypeNewVoid());
|
|
ShellTypeAddFunctionArgument($$.istType, ShellTypeNewByType($1.sttType));
|
|
$$.ctBytes = PushExpression($1);
|
|
}
|
|
| argument_expression_list ',' expression {
|
|
$$ = $1;
|
|
ShellTypeAddFunctionArgument($$.istType, ShellTypeNewByType($3.sttType));
|
|
$$.ctBytes += PushExpression($3);
|
|
}
|
|
|
|
expression
|
|
: c_float {
|
|
$$.sttType = STT_FLOAT;
|
|
$$.fFloat = $1.fFloat;
|
|
}
|
|
| c_int {
|
|
$$.sttType = STT_INDEX;
|
|
$$.iIndex = $1.iIndex;
|
|
}
|
|
| c_string {
|
|
$$.sttType = STT_STRING;
|
|
$$.strString = $1.strString;
|
|
}
|
|
| lvalue {
|
|
// get its value
|
|
$$.sttType = $1.lv_sttType;
|
|
if ($1.lv_sttType==STT_VOID) {
|
|
NOTHING;
|
|
} else if ($1.lv_sttType==STT_FLOAT) {
|
|
$$.fFloat = *(FLOAT*)$1.lv_pvAddress;
|
|
} else if ($1.lv_sttType==STT_INDEX) {
|
|
$$.iIndex = *(INDEX*)$1.lv_pvAddress;
|
|
} else if ($1.lv_sttType==STT_STRING) {
|
|
$$.strString = (const char*)*(CTString*)$1.lv_pvAddress;
|
|
} else {
|
|
$$.sttType = STT_FLOAT;
|
|
$$.fFloat = -666.0f;
|
|
_pShell->ErrorF("'%s' is of wrong type", (const char *) $1.lv_pssSymbol->ss_strName);
|
|
}
|
|
}
|
|
/* shift */
|
|
| expression SHL expression {
|
|
|
|
MatchTypes($1, $3);
|
|
|
|
$$.sttType = $1.sttType;
|
|
if ($1.sttType == STT_INDEX) {
|
|
$$.iIndex = $1.iIndex<<$3.iIndex;
|
|
} else {
|
|
_pShell->ErrorF("Wrong arguments for '<<'");
|
|
$$.sttType = STT_INDEX;
|
|
$$.iIndex = -666;
|
|
}
|
|
}
|
|
| expression SHR expression {
|
|
|
|
MatchTypes($1, $3);
|
|
|
|
$$.sttType = $1.sttType;
|
|
if ($1.sttType == STT_INDEX) {
|
|
$$.iIndex = $1.iIndex>>$3.iIndex;
|
|
} else {
|
|
_pShell->ErrorF("Wrong arguments for '>>'");
|
|
$$.sttType = STT_INDEX;
|
|
$$.iIndex = -666;
|
|
}
|
|
}
|
|
/* bitwise operators */
|
|
| expression '&' expression {
|
|
|
|
MatchTypes($1, $3);
|
|
|
|
$$.sttType = $1.sttType;
|
|
if ($1.sttType == STT_FLOAT) {
|
|
_pShell->ErrorF("'&' is illegal for FLOAT values");
|
|
} else if ($1.sttType == STT_INDEX) {
|
|
$$.iIndex = $1.iIndex&$3.iIndex;
|
|
} else {
|
|
$$.sttType = STT_FLOAT;
|
|
$$.fFloat = -666.0f;
|
|
}
|
|
}
|
|
| expression '|' expression {
|
|
|
|
MatchTypes($1, $3);
|
|
|
|
$$.sttType = $1.sttType;
|
|
if ($1.sttType == STT_FLOAT) {
|
|
_pShell->ErrorF("'|' is illegal for FLOAT values");
|
|
} else if ($1.sttType == STT_INDEX) {
|
|
$$.iIndex = $1.iIndex|$3.iIndex;
|
|
} else {
|
|
$$.sttType = STT_FLOAT;
|
|
$$.fFloat = -666.0f;
|
|
}
|
|
}
|
|
| expression '^' expression {
|
|
|
|
MatchTypes($1, $3);
|
|
|
|
$$.sttType = $1.sttType;
|
|
if ($1.sttType == STT_FLOAT) {
|
|
_pShell->ErrorF("'^' is illegal for FLOAT values");
|
|
} else if ($1.sttType == STT_INDEX) {
|
|
$$.iIndex = $1.iIndex^$3.iIndex;
|
|
} else {
|
|
$$.sttType = STT_FLOAT;
|
|
$$.fFloat = -666.0f;
|
|
}
|
|
}
|
|
|
|
/* logical operators */
|
|
| expression LOGAND expression {
|
|
|
|
MatchTypes($1, $3);
|
|
|
|
$$.sttType = $1.sttType;
|
|
if ($1.sttType == STT_FLOAT) {
|
|
_pShell->ErrorF("'&&' is illegal for FLOAT values");
|
|
} else if ($1.sttType == STT_INDEX) {
|
|
$$.iIndex = $1.iIndex&&$3.iIndex;
|
|
} else {
|
|
$$.sttType = STT_FLOAT;
|
|
$$.fFloat = -666.0f;
|
|
}
|
|
}
|
|
| expression LOGOR expression {
|
|
|
|
MatchTypes($1, $3);
|
|
|
|
$$.sttType = $1.sttType;
|
|
if ($1.sttType == STT_FLOAT) {
|
|
_pShell->ErrorF("'||' is illegal for FLOAT values");
|
|
} else if ($1.sttType == STT_INDEX) {
|
|
$$.iIndex = $1.iIndex||$3.iIndex;
|
|
} else {
|
|
$$.sttType = STT_FLOAT;
|
|
$$.fFloat = -666.0f;
|
|
}
|
|
}
|
|
/* addition */
|
|
| expression '+' expression {
|
|
|
|
MatchTypes($1, $3);
|
|
|
|
$$.sttType = $1.sttType;
|
|
if ($1.sttType == STT_FLOAT) {
|
|
$$.fFloat = $1.fFloat+$3.fFloat;
|
|
} else if ($1.sttType == STT_INDEX) {
|
|
$$.iIndex = $1.iIndex+$3.iIndex;
|
|
} else if ($1.sttType == STT_STRING) {
|
|
CTString &strNew = _shell_astrTempStrings.Push();
|
|
strNew = CTString($1.strString)+$3.strString;
|
|
$$.strString = (const char*)strNew;
|
|
} else {
|
|
$$.sttType = STT_FLOAT;
|
|
$$.fFloat = -666.0f;
|
|
}
|
|
}
|
|
/* substraction */
|
|
| expression '-' expression {
|
|
|
|
MatchTypes($1, $3);
|
|
|
|
$$.sttType = $1.sttType;
|
|
if ($1.sttType == STT_FLOAT) {
|
|
$$.fFloat = $1.fFloat-$3.fFloat;
|
|
} else if ($1.sttType == STT_INDEX) {
|
|
$$.iIndex = $1.iIndex-$3.iIndex;
|
|
} else {
|
|
$$.sttType = STT_FLOAT;
|
|
$$.fFloat = -666.0f;
|
|
}
|
|
}
|
|
|
|
/* multiplication */
|
|
| expression '*' expression {
|
|
|
|
MatchTypes($1, $3);
|
|
|
|
$$.sttType = $1.sttType;
|
|
if ($1.sttType == STT_FLOAT) {
|
|
$$.fFloat = $1.fFloat*$3.fFloat;
|
|
} else if ($1.sttType == STT_INDEX) {
|
|
$$.iIndex = $1.iIndex*$3.iIndex;
|
|
} else {
|
|
$$.sttType = STT_FLOAT;
|
|
$$.fFloat = -666.0f;
|
|
}
|
|
|
|
}
|
|
/* division */
|
|
| expression '/' expression {
|
|
|
|
MatchTypes($1, $3);
|
|
|
|
$$.sttType = $1.sttType;
|
|
if ($1.sttType == STT_FLOAT) {
|
|
$$.fFloat = $1.fFloat/$3.fFloat;
|
|
} else if ($1.sttType == STT_INDEX) {
|
|
if ($3.iIndex==0) {
|
|
_pShell->ErrorF("Division by zero!\n");
|
|
$$.iIndex = 0;
|
|
} else {
|
|
$$.iIndex = $1.iIndex/$3.iIndex;
|
|
}
|
|
} else {
|
|
$$.sttType = STT_FLOAT;
|
|
$$.fFloat = -666.0f;
|
|
}
|
|
|
|
}
|
|
|
|
/* modulo */
|
|
| expression '%' expression {
|
|
MatchTypes($1, $3);
|
|
|
|
$$.sttType = $1.sttType;
|
|
if ($1.sttType == STT_FLOAT) {
|
|
_pShell->ErrorF("'%' is illegal for FLOAT values");
|
|
} else if ($1.sttType == STT_INDEX) {
|
|
$$.iIndex = $1.iIndex%$3.iIndex;
|
|
} else {
|
|
$$.sttType = STT_FLOAT;
|
|
$$.fFloat = -666.0f;
|
|
}
|
|
|
|
}
|
|
|
|
/* comparisons */
|
|
| expression '<' expression {
|
|
DoComparison($$, $1, $3, '<');
|
|
}
|
|
| expression '>' expression {
|
|
DoComparison($$, $1, $3, '>');
|
|
}
|
|
| expression EQ expression {
|
|
DoComparison($$, $1, $3, '=');
|
|
}
|
|
| expression NEQ expression {
|
|
DoComparison($$, $1, $3, '!');
|
|
}
|
|
| expression GEQ expression {
|
|
DoComparison($$, $1, $3, '}');
|
|
}
|
|
| expression LEQ expression {
|
|
DoComparison($$, $1, $3, '{');
|
|
}
|
|
|
|
// unary minus
|
|
|
|
| '-' expression %prec SIGN {
|
|
$$.sttType = $2.sttType;
|
|
if ($2.sttType == STT_FLOAT) {
|
|
$$.fFloat = -$2.fFloat;
|
|
} else if ($2.sttType == STT_INDEX) {
|
|
$$.iIndex = -$2.iIndex;
|
|
} else {
|
|
$$.sttType = STT_FLOAT;
|
|
$$.fFloat = -666.0f;
|
|
}
|
|
}
|
|
|
|
// unary plus
|
|
|
|
| '+' expression %prec SIGN {
|
|
$$.sttType = $2.sttType;
|
|
if ($2.sttType == STT_FLOAT) {
|
|
$$.fFloat = $2.fFloat;
|
|
} else if ($2.sttType == STT_INDEX) {
|
|
$$.iIndex = $2.iIndex;
|
|
} else {
|
|
$$.sttType = STT_FLOAT;
|
|
$$.fFloat = -666.0f;
|
|
}
|
|
}
|
|
|
|
| '!' expression %prec SIGN {
|
|
$$.sttType = $2.sttType;
|
|
if ($2.sttType == STT_FLOAT) {
|
|
_pShell->ErrorF("'!' is illegal for FLOAT values");
|
|
$$.fFloat = $2.fFloat;
|
|
} else if ($2.sttType == STT_INDEX) {
|
|
$$.iIndex = !$2.iIndex;
|
|
} else {
|
|
$$.sttType = STT_FLOAT;
|
|
$$.fFloat = -666.0f;
|
|
}
|
|
}
|
|
|
|
| '(' k_FLOAT ')' expression %prec TYPECAST {
|
|
$$.sttType = STT_FLOAT;
|
|
if ($4.sttType == STT_FLOAT) {
|
|
$$.fFloat = $4.fFloat;
|
|
} else if ($4.sttType == STT_INDEX) {
|
|
$$.fFloat = FLOAT($4.iIndex);
|
|
} else if ($4.sttType == STT_STRING) {
|
|
$$.fFloat = atof($4.strString);
|
|
} else {
|
|
_pShell->ErrorF("Cannot convert to FLOAT");
|
|
$$.sttType = STT_VOID;
|
|
}
|
|
}
|
|
|
|
| '(' k_INDEX ')' expression %prec TYPECAST {
|
|
$$.sttType = STT_INDEX;
|
|
if ($4.sttType == STT_FLOAT) {
|
|
$$.iIndex = INDEX($4.fFloat);
|
|
} else if ($4.sttType == STT_INDEX) {
|
|
$$.iIndex = $4.iIndex;
|
|
} else if ($4.sttType == STT_STRING) {
|
|
$$.iIndex = atol($4.strString);
|
|
} else {
|
|
_pShell->ErrorF("Cannot convert to INDEX");
|
|
$$.sttType = STT_VOID;
|
|
}
|
|
}
|
|
|
|
| '(' k_CTString ')' expression %prec TYPECAST {
|
|
CTString &strNew = _shell_astrTempStrings.Push();
|
|
$$.sttType = STT_STRING;
|
|
if ($4.sttType == STT_FLOAT) {
|
|
strNew.PrintF("%g", $4.fFloat);
|
|
} else if ($4.sttType == STT_INDEX) {
|
|
strNew.PrintF("%d", $4.iIndex);
|
|
} else if ($4.sttType == STT_STRING) {
|
|
strNew = $4.strString;
|
|
} else {
|
|
_pShell->ErrorF("Cannot convert to CTString");
|
|
$$.sttType = STT_VOID;
|
|
}
|
|
$$.strString = (const char*)strNew;
|
|
}
|
|
|
|
// function call
|
|
| identifier '(' argument_expression_list_opt ')' {
|
|
// if the identifier is not declared
|
|
if (!$1->IsDeclared()) {
|
|
// error
|
|
_pShell->ErrorF("Identifier '%s' is not declared", (const char *) $1->ss_strName);
|
|
// if the identifier is declared
|
|
} else {
|
|
// get its type
|
|
ShellType &stFunc = _shell_ast[$1->ss_istType];
|
|
|
|
// if the identifier is a function
|
|
if (stFunc.st_sttType==STT_FUNCTION) {
|
|
// determine result type
|
|
ShellType &stResult = _shell_ast[stFunc.st_istBaseType];
|
|
// match argument list result to that result
|
|
_shell_ast[_shell_ast[$3.istType].st_istBaseType].st_sttType = stResult.st_sttType;
|
|
// if types are same
|
|
if (ShellTypeIsSame($3.istType, $1->ss_istType)) {
|
|
bool callfunc = true;
|
|
|
|
// !!! FIXME: maybe just dump the win32 codepath here? This will break on Win64, and maybe break on different compilers/compiler versions, etc.
|
|
#ifdef PLATFORM_WIN32
|
|
#define CALLPARAMS
|
|
#define FUNCSIG void
|
|
#define PUSHPARAMS memcpy(_alloca($3.ctBytes), _ubStack+_iStack-$3.ctBytes, $3.ctBytes);
|
|
#else
|
|
// This is possibly more portable, but no less scary than the alloca hack.
|
|
#define MAXSCRIPTFUNCARGS 5
|
|
void *ARG[MAXSCRIPTFUNCARGS];
|
|
if (($3.ctBytes > sizeof (ARG)))
|
|
{
|
|
_pShell->ErrorF("Function '%s' has too many arguments!", (const char *) $1->ss_strName);
|
|
callfunc = false;
|
|
}
|
|
else
|
|
{
|
|
memcpy(ARG, _ubStack+_iStack-$3.ctBytes, $3.ctBytes);
|
|
memset(((char *) ARG) + $3.ctBytes, '\0', sizeof (ARG) - $3.ctBytes);
|
|
}
|
|
#define PUSHPARAMS
|
|
#define FUNCSIG void*, void*, void*, void*, void*
|
|
#define CALLPARAMS ARG[0], ARG[1], ARG[2], ARG[3], ARG[4]
|
|
#endif
|
|
|
|
if (callfunc) {
|
|
// if void
|
|
if (stResult.st_sttType==STT_VOID) {
|
|
// just call the function
|
|
$$.sttType = STT_VOID;
|
|
PUSHPARAMS;
|
|
((void (*)(FUNCSIG))$1->ss_pvValue)(CALLPARAMS);
|
|
// if index
|
|
} else if (stResult.st_sttType==STT_INDEX) {
|
|
// call the function and return result
|
|
$$.sttType = STT_INDEX;
|
|
PUSHPARAMS;
|
|
$$.iIndex = ((INDEX (*)(FUNCSIG))$1->ss_pvValue)(CALLPARAMS);
|
|
// if float
|
|
} else if (stResult.st_sttType==STT_FLOAT) {
|
|
// call the function and return result
|
|
$$.sttType = STT_FLOAT;
|
|
PUSHPARAMS;
|
|
$$.fFloat = ((FLOAT (*)(FUNCSIG))$1->ss_pvValue)(CALLPARAMS);
|
|
// if string
|
|
} else if (stResult.st_sttType==STT_STRING) {
|
|
// call the function and return result
|
|
$$.sttType = STT_STRING;
|
|
CTString &strNew = _shell_astrTempStrings.Push();
|
|
PUSHPARAMS;
|
|
strNew = ((CTString (*)(FUNCSIG))$1->ss_pvValue)(CALLPARAMS);
|
|
$$.strString = (const char*)strNew;
|
|
} else {
|
|
ASSERT(FALSE);
|
|
$$.sttType = STT_FLOAT;
|
|
$$.fFloat = -666.0f;
|
|
}
|
|
}
|
|
// if types are different
|
|
} else {
|
|
// error
|
|
$$.sttType = STT_VOID;
|
|
_pShell->ErrorF("Wrong parameters for '%s'", (const char *) $1->ss_strName);
|
|
}
|
|
// if the identifier is something else
|
|
} else {
|
|
// error
|
|
$$.sttType = STT_VOID;
|
|
_pShell->ErrorF("Can't call '%s'", (const char *) $1->ss_strName);
|
|
}
|
|
}
|
|
|
|
// pop arguments and free type info
|
|
_iStack-=$3.ctBytes;
|
|
ShellTypeDelete($3.istType);
|
|
}
|
|
// brackets
|
|
| '(' expression ')' {
|
|
$$ = $2;
|
|
}
|
|
;
|
|
|
|
%%
|