2016-10-08 06:35:16 +00:00
|
|
|
/*
|
|
|
|
** zcc_expr.cpp
|
|
|
|
**
|
|
|
|
**---------------------------------------------------------------------------
|
|
|
|
** Copyright -2016 Randy Heit
|
|
|
|
** All rights reserved.
|
|
|
|
**
|
|
|
|
** Redistribution and use in source and binary forms, with or without
|
|
|
|
** modification, are permitted provided that the following conditions
|
|
|
|
** are met:
|
|
|
|
**
|
|
|
|
** 1. Redistributions of source code must retain the above copyright
|
|
|
|
** notice, this list of conditions and the following disclaimer.
|
|
|
|
** 2. Redistributions in binary form must reproduce the above copyright
|
|
|
|
** notice, this list of conditions and the following disclaimer in the
|
|
|
|
** documentation and/or other materials provided with the distribution.
|
|
|
|
** 3. The name of the author may not be used to endorse or promote products
|
|
|
|
** derived from this software without specific prior written permission.
|
|
|
|
**
|
|
|
|
** THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
|
|
** IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
|
|
** OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
|
|
** IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
** INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|
|
|
** NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
|
|
** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
|
|
** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
|
|
** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
|
|
** THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
**---------------------------------------------------------------------------
|
|
|
|
**
|
|
|
|
*/
|
|
|
|
|
2016-03-01 15:47:10 +00:00
|
|
|
#include "dobject.h"
|
|
|
|
#include "sc_man.h"
|
|
|
|
#include "c_console.h"
|
|
|
|
#include "c_dispatch.h"
|
|
|
|
#include "w_wad.h"
|
|
|
|
#include "cmdlib.h"
|
|
|
|
#include "m_alloc.h"
|
2016-10-07 23:05:27 +00:00
|
|
|
#include "i_system.h"
|
2016-10-18 08:17:09 +00:00
|
|
|
#include "m_argv.h"
|
2016-10-15 10:15:25 +00:00
|
|
|
#include "v_text.h"
|
2016-03-01 15:47:10 +00:00
|
|
|
#include "zcc_parser.h"
|
|
|
|
#include "zcc_compile.h"
|
|
|
|
|
2016-12-03 12:16:09 +00:00
|
|
|
TArray<FString> Includes;
|
|
|
|
TArray<FScriptPosition> IncludeLocs;
|
|
|
|
|
2016-03-01 15:47:10 +00:00
|
|
|
static FString ZCCTokenName(int terminal);
|
2016-12-03 12:16:09 +00:00
|
|
|
void AddInclude(ZCC_ExprConstant *node)
|
|
|
|
{
|
|
|
|
assert(node->Type == TypeString);
|
|
|
|
if (Includes.Find(*node->StringVal) >= Includes.Size())
|
|
|
|
{
|
|
|
|
Includes.Push(*node->StringVal);
|
|
|
|
IncludeLocs.Push(*node);
|
|
|
|
}
|
|
|
|
}
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
#include "zcc-parse.h"
|
|
|
|
#include "zcc-parse.c"
|
|
|
|
|
|
|
|
struct TokenMapEntry
|
|
|
|
{
|
|
|
|
SWORD TokenType;
|
|
|
|
WORD TokenName;
|
|
|
|
TokenMapEntry(SWORD a, WORD b)
|
|
|
|
: TokenType(a), TokenName(b)
|
|
|
|
{ }
|
|
|
|
};
|
|
|
|
static TMap<SWORD, TokenMapEntry> TokenMap;
|
|
|
|
static SWORD BackTokenMap[YYERRORSYMBOL]; // YYERRORSYMBOL immediately follows the terminals described by the grammar
|
|
|
|
|
|
|
|
#define TOKENDEF2(sc, zcc, name) { TokenMapEntry tme(zcc, name); TokenMap.Insert(sc, tme); } BackTokenMap[zcc] = sc
|
|
|
|
#define TOKENDEF(sc, zcc) TOKENDEF2(sc, zcc, NAME_None)
|
|
|
|
|
|
|
|
static void InitTokenMap()
|
|
|
|
{
|
|
|
|
TOKENDEF ('=', ZCC_EQ);
|
|
|
|
TOKENDEF (TK_MulEq, ZCC_MULEQ);
|
|
|
|
TOKENDEF (TK_DivEq, ZCC_DIVEQ);
|
|
|
|
TOKENDEF (TK_ModEq, ZCC_MODEQ);
|
|
|
|
TOKENDEF (TK_AddEq, ZCC_ADDEQ);
|
|
|
|
TOKENDEF (TK_SubEq, ZCC_SUBEQ);
|
|
|
|
TOKENDEF (TK_LShiftEq, ZCC_LSHEQ);
|
|
|
|
TOKENDEF (TK_RShiftEq, ZCC_RSHEQ);
|
2016-10-21 10:22:42 +00:00
|
|
|
TOKENDEF (TK_URShiftEq, ZCC_URSHEQ);
|
2016-03-01 15:47:10 +00:00
|
|
|
TOKENDEF (TK_AndEq, ZCC_ANDEQ);
|
|
|
|
TOKENDEF (TK_OrEq, ZCC_OREQ);
|
|
|
|
TOKENDEF (TK_XorEq, ZCC_XOREQ);
|
|
|
|
TOKENDEF ('?', ZCC_QUESTION);
|
|
|
|
TOKENDEF (':', ZCC_COLON);
|
|
|
|
TOKENDEF (TK_OrOr, ZCC_OROR);
|
|
|
|
TOKENDEF (TK_AndAnd, ZCC_ANDAND);
|
|
|
|
TOKENDEF (TK_Eq, ZCC_EQEQ);
|
|
|
|
TOKENDEF (TK_Neq, ZCC_NEQ);
|
|
|
|
TOKENDEF (TK_ApproxEq, ZCC_APPROXEQ);
|
|
|
|
TOKENDEF ('<', ZCC_LT);
|
|
|
|
TOKENDEF ('>', ZCC_GT);
|
|
|
|
TOKENDEF (TK_Leq, ZCC_LTEQ);
|
|
|
|
TOKENDEF (TK_Geq, ZCC_GTEQ);
|
|
|
|
TOKENDEF (TK_LtGtEq, ZCC_LTGTEQ);
|
|
|
|
TOKENDEF (TK_Is, ZCC_IS);
|
|
|
|
TOKENDEF (TK_DotDot, ZCC_DOTDOT);
|
2016-10-11 11:11:40 +00:00
|
|
|
TOKENDEF (TK_Ellipsis, ZCC_ELLIPSIS);
|
2016-03-01 15:47:10 +00:00
|
|
|
TOKENDEF ('|', ZCC_OR);
|
|
|
|
TOKENDEF ('^', ZCC_XOR);
|
|
|
|
TOKENDEF ('&', ZCC_AND);
|
|
|
|
TOKENDEF (TK_LShift, ZCC_LSH);
|
|
|
|
TOKENDEF (TK_RShift, ZCC_RSH);
|
2016-10-17 13:52:29 +00:00
|
|
|
TOKENDEF (TK_URShift, ZCC_URSH);
|
2016-03-01 15:47:10 +00:00
|
|
|
TOKENDEF ('-', ZCC_SUB);
|
|
|
|
TOKENDEF ('+', ZCC_ADD);
|
|
|
|
TOKENDEF ('*', ZCC_MUL);
|
|
|
|
TOKENDEF ('/', ZCC_DIV);
|
|
|
|
TOKENDEF ('%', ZCC_MOD);
|
|
|
|
TOKENDEF (TK_Cross, ZCC_CROSSPROD);
|
|
|
|
TOKENDEF (TK_Dot, ZCC_DOTPROD);
|
|
|
|
TOKENDEF (TK_MulMul, ZCC_POW);
|
|
|
|
TOKENDEF (TK_Incr, ZCC_ADDADD);
|
|
|
|
TOKENDEF (TK_Decr, ZCC_SUBSUB);
|
|
|
|
TOKENDEF ('.', ZCC_DOT);
|
|
|
|
TOKENDEF ('(', ZCC_LPAREN);
|
|
|
|
TOKENDEF (')', ZCC_RPAREN);
|
|
|
|
TOKENDEF (TK_ColonColon, ZCC_SCOPE);
|
|
|
|
TOKENDEF (';', ZCC_SEMICOLON);
|
|
|
|
TOKENDEF (',', ZCC_COMMA);
|
|
|
|
TOKENDEF (TK_Class, ZCC_CLASS);
|
|
|
|
TOKENDEF (TK_Abstract, ZCC_ABSTRACT);
|
|
|
|
TOKENDEF (TK_Native, ZCC_NATIVE);
|
2016-10-11 11:11:40 +00:00
|
|
|
TOKENDEF (TK_Action, ZCC_ACTION);
|
2016-03-01 15:47:10 +00:00
|
|
|
TOKENDEF (TK_Replaces, ZCC_REPLACES);
|
|
|
|
TOKENDEF (TK_Static, ZCC_STATIC);
|
|
|
|
TOKENDEF (TK_Private, ZCC_PRIVATE);
|
|
|
|
TOKENDEF (TK_Protected, ZCC_PROTECTED);
|
|
|
|
TOKENDEF (TK_Latent, ZCC_LATENT);
|
2016-11-11 19:05:07 +00:00
|
|
|
TOKENDEF (TK_Virtual, ZCC_VIRTUAL);
|
2017-01-21 00:07:44 +00:00
|
|
|
TOKENDEF (TK_VarArg, ZCC_VARARG);
|
2016-11-11 19:05:07 +00:00
|
|
|
TOKENDEF (TK_Override, ZCC_OVERRIDE);
|
2016-03-01 15:47:10 +00:00
|
|
|
TOKENDEF (TK_Final, ZCC_FINAL);
|
|
|
|
TOKENDEF (TK_Meta, ZCC_META);
|
|
|
|
TOKENDEF (TK_Deprecated, ZCC_DEPRECATED);
|
|
|
|
TOKENDEF (TK_ReadOnly, ZCC_READONLY);
|
|
|
|
TOKENDEF ('{', ZCC_LBRACE);
|
|
|
|
TOKENDEF ('}', ZCC_RBRACE);
|
|
|
|
TOKENDEF (TK_Struct, ZCC_STRUCT);
|
2017-01-15 15:55:30 +00:00
|
|
|
TOKENDEF (TK_Property, ZCC_PROPERTY);
|
|
|
|
TOKENDEF (TK_Transient, ZCC_TRANSIENT);
|
2016-03-01 15:47:10 +00:00
|
|
|
TOKENDEF (TK_Enum, ZCC_ENUM);
|
|
|
|
TOKENDEF2(TK_SByte, ZCC_SBYTE, NAME_sByte);
|
|
|
|
TOKENDEF2(TK_Byte, ZCC_BYTE, NAME_Byte);
|
|
|
|
TOKENDEF2(TK_Short, ZCC_SHORT, NAME_Short);
|
|
|
|
TOKENDEF2(TK_UShort, ZCC_USHORT, NAME_uShort);
|
2016-11-22 18:20:31 +00:00
|
|
|
TOKENDEF2(TK_Int8, ZCC_SBYTE, NAME_int8);
|
|
|
|
TOKENDEF2(TK_UInt8, ZCC_BYTE, NAME_uint8);
|
|
|
|
TOKENDEF2(TK_Int16, ZCC_SHORT, NAME_int16);
|
|
|
|
TOKENDEF2(TK_UInt16, ZCC_USHORT, NAME_uint16);
|
2016-03-01 15:47:10 +00:00
|
|
|
TOKENDEF2(TK_Int, ZCC_INT, NAME_Int);
|
|
|
|
TOKENDEF2(TK_UInt, ZCC_UINT, NAME_uInt);
|
|
|
|
TOKENDEF2(TK_Bool, ZCC_BOOL, NAME_Bool);
|
|
|
|
TOKENDEF2(TK_Float, ZCC_FLOAT, NAME_Float);
|
|
|
|
TOKENDEF2(TK_Double, ZCC_DOUBLE, NAME_Double);
|
|
|
|
TOKENDEF2(TK_String, ZCC_STRING, NAME_String);
|
2016-10-27 23:14:21 +00:00
|
|
|
TOKENDEF2(TK_Vector2, ZCC_VECTOR2, NAME_Vector2);
|
|
|
|
TOKENDEF2(TK_Vector3, ZCC_VECTOR3, NAME_Vector3);
|
2016-03-01 15:47:10 +00:00
|
|
|
TOKENDEF2(TK_Name, ZCC_NAME, NAME_Name);
|
|
|
|
TOKENDEF2(TK_Map, ZCC_MAP, NAME_Map);
|
|
|
|
TOKENDEF2(TK_Array, ZCC_ARRAY, NAME_Array);
|
2016-12-03 12:16:09 +00:00
|
|
|
TOKENDEF2(TK_Include, ZCC_INCLUDE, NAME_Include);
|
2016-03-01 15:47:10 +00:00
|
|
|
TOKENDEF (TK_Void, ZCC_VOID);
|
|
|
|
TOKENDEF (TK_True, ZCC_TRUE);
|
|
|
|
TOKENDEF (TK_False, ZCC_FALSE);
|
|
|
|
TOKENDEF ('[', ZCC_LBRACKET);
|
|
|
|
TOKENDEF (']', ZCC_RBRACKET);
|
|
|
|
TOKENDEF (TK_In, ZCC_IN);
|
|
|
|
TOKENDEF (TK_Out, ZCC_OUT);
|
|
|
|
TOKENDEF (TK_Optional, ZCC_OPTIONAL);
|
|
|
|
TOKENDEF (TK_Super, ZCC_SUPER);
|
2016-10-21 22:50:04 +00:00
|
|
|
TOKENDEF (TK_Null, ZCC_NULLPTR);
|
2016-03-01 15:47:10 +00:00
|
|
|
TOKENDEF ('~', ZCC_TILDE);
|
|
|
|
TOKENDEF ('!', ZCC_BANG);
|
|
|
|
TOKENDEF (TK_SizeOf, ZCC_SIZEOF);
|
|
|
|
TOKENDEF (TK_AlignOf, ZCC_ALIGNOF);
|
|
|
|
TOKENDEF (TK_Continue, ZCC_CONTINUE);
|
|
|
|
TOKENDEF (TK_Break, ZCC_BREAK);
|
|
|
|
TOKENDEF (TK_Return, ZCC_RETURN);
|
|
|
|
TOKENDEF (TK_Do, ZCC_DO);
|
|
|
|
TOKENDEF (TK_For, ZCC_FOR);
|
|
|
|
TOKENDEF (TK_While, ZCC_WHILE);
|
|
|
|
TOKENDEF (TK_Until, ZCC_UNTIL);
|
|
|
|
TOKENDEF (TK_If, ZCC_IF);
|
|
|
|
TOKENDEF (TK_Else, ZCC_ELSE);
|
|
|
|
TOKENDEF (TK_Switch, ZCC_SWITCH);
|
|
|
|
TOKENDEF (TK_Case, ZCC_CASE);
|
|
|
|
TOKENDEF2(TK_Default, ZCC_DEFAULT, NAME_Default);
|
|
|
|
TOKENDEF (TK_Const, ZCC_CONST);
|
|
|
|
TOKENDEF (TK_Stop, ZCC_STOP);
|
|
|
|
TOKENDEF (TK_Wait, ZCC_WAIT);
|
|
|
|
TOKENDEF (TK_Fail, ZCC_FAIL);
|
|
|
|
TOKENDEF (TK_Loop, ZCC_LOOP);
|
|
|
|
TOKENDEF (TK_Goto, ZCC_GOTO);
|
|
|
|
TOKENDEF (TK_States, ZCC_STATES);
|
2016-11-04 08:56:03 +00:00
|
|
|
TOKENDEF2(TK_State, ZCC_STATE, NAME_State);
|
|
|
|
TOKENDEF2(TK_Color, ZCC_COLOR, NAME_Color);
|
|
|
|
TOKENDEF2(TK_Sound, ZCC_SOUND, NAME_Sound);
|
2016-12-05 12:24:42 +00:00
|
|
|
TOKENDEF2(TK_Let, ZCC_LET, NAME_let);
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
TOKENDEF (TK_Identifier, ZCC_IDENTIFIER);
|
|
|
|
TOKENDEF (TK_StringConst, ZCC_STRCONST);
|
|
|
|
TOKENDEF (TK_NameConst, ZCC_NAMECONST);
|
|
|
|
TOKENDEF (TK_IntConst, ZCC_INTCONST);
|
|
|
|
TOKENDEF (TK_UIntConst, ZCC_UINTCONST);
|
|
|
|
TOKENDEF (TK_FloatConst, ZCC_FLOATCONST);
|
|
|
|
TOKENDEF (TK_NonWhitespace, ZCC_NWS);
|
|
|
|
|
2016-03-03 21:34:42 +00:00
|
|
|
TOKENDEF (TK_Bright, ZCC_BRIGHT);
|
|
|
|
TOKENDEF (TK_Slow, ZCC_SLOW);
|
|
|
|
TOKENDEF (TK_Fast, ZCC_FAST);
|
|
|
|
TOKENDEF (TK_NoDelay, ZCC_NODELAY);
|
|
|
|
TOKENDEF (TK_Offset, ZCC_OFFSET);
|
|
|
|
TOKENDEF (TK_CanRaise, ZCC_CANRAISE);
|
2016-06-13 14:50:12 +00:00
|
|
|
TOKENDEF (TK_Light, ZCC_LIGHT);
|
2016-10-23 10:57:21 +00:00
|
|
|
TOKENDEF (TK_Extend, ZCC_EXTEND);
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
#undef TOKENDEF
|
|
|
|
#undef TOKENDEF2
|
|
|
|
|
2016-11-04 08:56:03 +00:00
|
|
|
static void ParseSingleFile(const char *filename, int lump, void *parser, ZCCParseState &state)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
|
|
|
int tokentype;
|
2016-10-07 23:05:27 +00:00
|
|
|
//bool failed;
|
2016-03-01 15:47:10 +00:00
|
|
|
ZCCToken value;
|
2016-10-07 23:05:27 +00:00
|
|
|
FScanner sc;
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-11-04 08:56:03 +00:00
|
|
|
if (filename != nullptr)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2016-11-04 08:56:03 +00:00
|
|
|
lump = Wads.CheckNumForFullName(filename, true);
|
|
|
|
if (lump >= 0)
|
|
|
|
{
|
|
|
|
sc.OpenLumpNum(lump);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Printf("Could not find script lump '%s'\n", filename);
|
|
|
|
return;
|
|
|
|
}
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
2016-11-04 08:56:03 +00:00
|
|
|
else sc.OpenLumpNum(lump);
|
2016-03-01 15:47:10 +00:00
|
|
|
|
2016-10-07 23:05:27 +00:00
|
|
|
state.sc = ≻
|
2016-03-01 15:47:10 +00:00
|
|
|
while (sc.GetToken())
|
|
|
|
{
|
|
|
|
value.SourceLoc = sc.GetMessageLine();
|
2016-03-03 21:34:42 +00:00
|
|
|
switch (sc.TokenType)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
2016-03-03 21:34:42 +00:00
|
|
|
case TK_StringConst:
|
2016-03-01 15:47:10 +00:00
|
|
|
value.String = state.Strings.Alloc(sc.String, sc.StringLen);
|
|
|
|
tokentype = ZCC_STRCONST;
|
2016-03-03 21:34:42 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case TK_NameConst:
|
2016-03-01 15:47:10 +00:00
|
|
|
value.Int = sc.Name;
|
|
|
|
tokentype = ZCC_NAMECONST;
|
2016-03-03 21:34:42 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case TK_IntConst:
|
2016-03-01 15:47:10 +00:00
|
|
|
value.Int = sc.Number;
|
|
|
|
tokentype = ZCC_INTCONST;
|
2016-03-03 21:34:42 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case TK_UIntConst:
|
2016-03-01 15:47:10 +00:00
|
|
|
value.Int = sc.Number;
|
|
|
|
tokentype = ZCC_UINTCONST;
|
2016-03-03 21:34:42 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case TK_FloatConst:
|
2016-03-01 15:47:10 +00:00
|
|
|
value.Float = sc.Float;
|
|
|
|
tokentype = ZCC_FLOATCONST;
|
2016-03-03 21:34:42 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case TK_Identifier:
|
2016-03-01 15:47:10 +00:00
|
|
|
value.Int = FName(sc.String);
|
|
|
|
tokentype = ZCC_IDENTIFIER;
|
2016-03-03 21:34:42 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case TK_NonWhitespace:
|
2016-03-01 15:47:10 +00:00
|
|
|
value.Int = FName(sc.String);
|
|
|
|
tokentype = ZCC_NWS;
|
2016-03-03 21:34:42 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
2016-03-01 15:47:10 +00:00
|
|
|
TokenMapEntry *zcctoken = TokenMap.CheckKey(sc.TokenType);
|
2016-10-07 23:05:27 +00:00
|
|
|
if (zcctoken != nullptr)
|
2016-03-01 15:47:10 +00:00
|
|
|
{
|
|
|
|
tokentype = zcctoken->TokenType;
|
|
|
|
value.Int = zcctoken->TokenName;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
sc.ScriptMessage("Unexpected token %s.\n", sc.TokenName(sc.TokenType).GetChars());
|
2016-03-03 21:34:42 +00:00
|
|
|
goto parse_end;
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
2016-03-03 21:34:42 +00:00
|
|
|
break;
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
ZCCParse(parser, tokentype, value, &state);
|
|
|
|
}
|
2016-03-03 21:34:42 +00:00
|
|
|
parse_end:
|
2016-03-01 15:47:10 +00:00
|
|
|
value.Int = -1;
|
|
|
|
ZCCParse(parser, ZCC_EOF, value, &state);
|
2016-10-07 23:05:27 +00:00
|
|
|
state.sc = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
static void DoParse(int lumpnum)
|
|
|
|
{
|
|
|
|
FScanner sc;
|
|
|
|
void *parser;
|
|
|
|
ZCCToken value;
|
2017-01-23 00:56:15 +00:00
|
|
|
auto baselump = lumpnum;
|
|
|
|
auto fileno = Wads.GetLumpFile(lumpnum);
|
2016-10-07 23:05:27 +00:00
|
|
|
|
|
|
|
parser = ZCCParseAlloc(malloc);
|
|
|
|
ZCCParseState state;
|
2016-11-27 11:18:06 +00:00
|
|
|
|
2016-11-27 16:43:11 +00:00
|
|
|
#ifndef NDEBUG
|
2016-11-27 11:18:06 +00:00
|
|
|
FILE *f = nullptr;
|
|
|
|
const char *tracefile = Args->CheckValue("-tracefile");
|
|
|
|
if (tracefile != nullptr)
|
|
|
|
{
|
|
|
|
f = fopen(tracefile, "w");
|
|
|
|
char prompt = '\0';
|
|
|
|
ZCCParseTrace(f, &prompt);
|
|
|
|
}
|
2016-11-27 16:43:11 +00:00
|
|
|
#endif
|
2016-10-07 23:05:27 +00:00
|
|
|
|
|
|
|
sc.OpenLumpNum(lumpnum);
|
2016-11-04 08:56:03 +00:00
|
|
|
auto saved = sc.SavePos();
|
2016-12-03 12:16:09 +00:00
|
|
|
|
|
|
|
ParseSingleFile(nullptr, lumpnum, parser, state);
|
|
|
|
for (unsigned i = 0; i < Includes.Size(); i++)
|
2016-10-07 23:05:27 +00:00
|
|
|
{
|
2016-12-03 12:16:09 +00:00
|
|
|
lumpnum = Wads.CheckNumForFullName(Includes[i], true);
|
|
|
|
if (lumpnum == -1)
|
2016-10-07 23:05:27 +00:00
|
|
|
{
|
2016-12-03 12:16:09 +00:00
|
|
|
IncludeLocs[i].Message(MSG_ERROR, "Include script lump %s not found", Includes[i].GetChars());
|
2016-10-07 23:05:27 +00:00
|
|
|
}
|
2016-12-03 12:16:09 +00:00
|
|
|
else
|
2016-11-04 08:56:03 +00:00
|
|
|
{
|
2017-01-23 00:56:15 +00:00
|
|
|
auto fileno2 = Wads.GetLumpFile(lumpnum);
|
|
|
|
if (fileno == 0 && fileno2 != 0)
|
|
|
|
{
|
|
|
|
I_FatalError("File %s is overriding core lump %s.",
|
|
|
|
Wads.GetWadFullName(Wads.GetLumpFile(baselump)), Includes[i].GetChars());
|
|
|
|
}
|
|
|
|
|
2016-12-03 12:16:09 +00:00
|
|
|
ParseSingleFile(nullptr, lumpnum, parser, state);
|
2016-11-04 08:56:03 +00:00
|
|
|
}
|
2016-10-07 23:05:27 +00:00
|
|
|
}
|
2016-12-03 12:16:09 +00:00
|
|
|
Includes.Clear();
|
|
|
|
Includes.ShrinkToFit();
|
|
|
|
IncludeLocs.Clear();
|
|
|
|
IncludeLocs.ShrinkToFit();
|
2016-10-07 23:05:27 +00:00
|
|
|
|
|
|
|
value.Int = -1;
|
|
|
|
value.SourceLoc = sc.GetMessageLine();
|
2016-03-01 15:47:10 +00:00
|
|
|
ZCCParse(parser, 0, value, &state);
|
|
|
|
ZCCParseFree(parser, free);
|
|
|
|
|
2016-10-15 10:15:25 +00:00
|
|
|
// If the parser fails, there is no point starting the compiler, because it'd only flood the output with endless errors.
|
|
|
|
if (FScriptPosition::ErrorCounter > 0)
|
|
|
|
{
|
2016-10-25 14:53:14 +00:00
|
|
|
I_Error("%d errors while parsing %s", FScriptPosition::ErrorCounter, Wads.GetLumpFullPath(lumpnum).GetChars());
|
2016-10-15 10:15:25 +00:00
|
|
|
}
|
|
|
|
|
2016-11-27 16:43:11 +00:00
|
|
|
#ifndef NDEBUG
|
2016-11-27 11:18:06 +00:00
|
|
|
if (f != nullptr)
|
2016-10-08 09:54:33 +00:00
|
|
|
{
|
|
|
|
fclose(f);
|
|
|
|
}
|
2016-11-27 16:43:11 +00:00
|
|
|
#endif
|
2016-10-18 08:17:09 +00:00
|
|
|
|
|
|
|
// Make a dump of the AST before running the compiler for diagnostic purposes.
|
|
|
|
if (Args->CheckParm("-dumpast"))
|
|
|
|
{
|
|
|
|
FString ast = ZCC_PrintAST(state.TopNode);
|
2016-11-04 09:35:14 +00:00
|
|
|
FString filename = Wads.GetLumpFullPath(lumpnum);
|
|
|
|
filename.ReplaceChars(":\\/?|", '.');
|
|
|
|
filename << ".ast";
|
|
|
|
FILE *ff = fopen(filename, "w");
|
2016-10-18 08:17:09 +00:00
|
|
|
if (ff != NULL)
|
|
|
|
{
|
|
|
|
fputs(ast.GetChars(), ff);
|
|
|
|
fclose(ff);
|
|
|
|
}
|
2016-10-08 09:54:33 +00:00
|
|
|
}
|
|
|
|
|
2016-10-08 20:16:10 +00:00
|
|
|
PSymbolTable symtable;
|
2016-11-08 10:12:56 +00:00
|
|
|
ZCCCompiler cc(state, NULL, symtable, GlobalSymbols, lumpnum);
|
2016-03-01 15:47:10 +00:00
|
|
|
cc.Compile();
|
2016-10-15 10:15:25 +00:00
|
|
|
|
|
|
|
if (FScriptPosition::ErrorCounter > 0)
|
|
|
|
{
|
|
|
|
// Abort if the compiler produced any errors. Also do not compile further lumps, because they very likely miss some stuff.
|
2016-10-25 14:53:14 +00:00
|
|
|
I_Error("%d errors, %d warnings while compiling %s", FScriptPosition::ErrorCounter, FScriptPosition::WarnCounter, Wads.GetLumpFullPath(lumpnum).GetChars());
|
2016-10-15 10:15:25 +00:00
|
|
|
}
|
|
|
|
else if (FScriptPosition::WarnCounter > 0)
|
|
|
|
{
|
|
|
|
// If we got warnings, but no errors, print the information but continue.
|
2016-10-25 14:53:14 +00:00
|
|
|
Printf(TEXTCOLOR_ORANGE "%d warnings while compiling %s", FScriptPosition::WarnCounter, Wads.GetLumpFullPath(lumpnum).GetChars());
|
2016-10-15 10:15:25 +00:00
|
|
|
}
|
|
|
|
|
2016-03-01 15:47:10 +00:00
|
|
|
}
|
|
|
|
|
2016-10-07 23:05:27 +00:00
|
|
|
void ParseScripts()
|
|
|
|
{
|
2016-11-07 10:30:41 +00:00
|
|
|
if (TokenMap.CountUsed() == 0)
|
|
|
|
{
|
|
|
|
InitTokenMap();
|
|
|
|
}
|
2016-10-07 23:05:27 +00:00
|
|
|
int lump, lastlump = 0;
|
2016-10-15 10:15:25 +00:00
|
|
|
FScriptPosition::ResetErrorCounter();
|
2016-11-26 18:23:22 +00:00
|
|
|
|
2016-10-07 23:05:27 +00:00
|
|
|
while ((lump = Wads.FindLump("ZSCRIPT", &lastlump)) != -1)
|
|
|
|
{
|
|
|
|
DoParse(lump);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
2016-03-01 15:47:10 +00:00
|
|
|
CCMD(parse)
|
|
|
|
{
|
|
|
|
if (argv.argc() == 2)
|
|
|
|
{
|
|
|
|
DoParse(argv[1]);
|
|
|
|
}
|
|
|
|
}
|
2016-10-07 23:05:27 +00:00
|
|
|
*/
|
2016-03-01 15:47:10 +00:00
|
|
|
|
|
|
|
static FString ZCCTokenName(int terminal)
|
|
|
|
{
|
|
|
|
if (terminal == ZCC_EOF)
|
|
|
|
{
|
|
|
|
return "end of file";
|
|
|
|
}
|
|
|
|
int sc_token;
|
|
|
|
if (terminal > 0 && terminal < (int)countof(BackTokenMap))
|
|
|
|
{
|
|
|
|
sc_token = BackTokenMap[terminal];
|
|
|
|
if (sc_token == 0)
|
|
|
|
{ // This token was not initialized. Whoops!
|
|
|
|
sc_token = -terminal;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{ // This should never happen.
|
|
|
|
sc_token = -terminal;
|
|
|
|
}
|
|
|
|
return FScanner::TokenName(sc_token);
|
|
|
|
}
|
|
|
|
|
|
|
|
ZCC_TreeNode *ZCC_AST::InitNode(size_t size, EZCCTreeNodeType type, ZCC_TreeNode *basis)
|
|
|
|
{
|
|
|
|
ZCC_TreeNode *node = (ZCC_TreeNode *)SyntaxArena.Alloc(size);
|
|
|
|
node->SiblingNext = node;
|
|
|
|
node->SiblingPrev = node;
|
|
|
|
node->NodeType = type;
|
|
|
|
if (basis != NULL)
|
|
|
|
{
|
|
|
|
node->SourceName = basis->SourceName;
|
2016-10-13 18:45:52 +00:00
|
|
|
node->SourceLump = basis->SourceLump;
|
2016-03-01 15:47:10 +00:00
|
|
|
node->SourceLoc = basis->SourceLoc;
|
|
|
|
}
|
|
|
|
return node;
|
|
|
|
}
|
|
|
|
|
|
|
|
ZCC_TreeNode *ZCCParseState::InitNode(size_t size, EZCCTreeNodeType type)
|
|
|
|
{
|
|
|
|
ZCC_TreeNode *node = ZCC_AST::InitNode(size, type, NULL);
|
2016-10-07 23:05:27 +00:00
|
|
|
node->SourceName = Strings.Alloc(sc->ScriptName);
|
2016-10-13 18:45:52 +00:00
|
|
|
node->SourceLump = sc->LumpNum;
|
2016-03-01 15:47:10 +00:00
|
|
|
return node;
|
|
|
|
}
|
2016-11-23 20:50:21 +00:00
|
|
|
|
|
|
|
// Appends a sibling to this node's sibling list.
|
|
|
|
void AppendTreeNodeSibling(ZCC_TreeNode *thisnode, ZCC_TreeNode *sibling)
|
|
|
|
{
|
|
|
|
if (thisnode == nullptr)
|
|
|
|
{
|
|
|
|
// Some bad syntax can actually get here, so better abort so that the user can see the error which caused this.
|
|
|
|
I_FatalError("Internal script compiler error. Execution aborted.");
|
|
|
|
}
|
|
|
|
if (sibling == nullptr)
|
|
|
|
{
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
ZCC_TreeNode *&SiblingPrev = thisnode->SiblingPrev;
|
|
|
|
ZCC_TreeNode *&SiblingNext = thisnode->SiblingNext;
|
|
|
|
|
|
|
|
// Check integrity of our sibling list.
|
|
|
|
assert(SiblingPrev->SiblingNext == thisnode);
|
|
|
|
assert(SiblingNext->SiblingPrev == thisnode);
|
|
|
|
|
|
|
|
// Check integrity of new sibling list.
|
|
|
|
assert(sibling->SiblingPrev->SiblingNext == sibling);
|
|
|
|
assert(sibling->SiblingNext->SiblingPrev == sibling);
|
|
|
|
|
|
|
|
ZCC_TreeNode *siblingend = sibling->SiblingPrev;
|
|
|
|
SiblingPrev->SiblingNext = sibling;
|
|
|
|
sibling->SiblingPrev = SiblingPrev;
|
|
|
|
SiblingPrev = siblingend;
|
|
|
|
siblingend->SiblingNext = thisnode;
|
|
|
|
}
|