mirror of
https://github.com/TES3MP/openmw-tes3mp.git
synced 2025-01-20 07:23:51 +00:00
de5a07ee71
git-svn-id: https://openmw.svn.sourceforge.net/svnroot/openmw/trunk@138 ea6a568a-9f4f-0410-981a-c910a81bb256
165 lines
4.5 KiB
D
165 lines
4.5 KiB
D
/*
|
|
Monster - an advanced game scripting language
|
|
Copyright (C) 2007-2009 Nicolay Korslund
|
|
Email: <korslund@gmail.com>
|
|
WWW: http://monster.snaptoad.com/
|
|
|
|
This file (block.d) is part of the Monster script language
|
|
package.
|
|
|
|
Monster is distributed as free software: you can redistribute it
|
|
and/or modify it under the terms of the GNU General Public License
|
|
version 3, as published by the Free Software Foundation.
|
|
|
|
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
|
|
version 3 along with this program. If not, see
|
|
http://www.gnu.org/licenses/ .
|
|
|
|
*/
|
|
|
|
module monster.compiler.block;
|
|
|
|
import monster.compiler.tokenizer;
|
|
import monster.compiler.scopes;
|
|
import monster.compiler.assembler;
|
|
|
|
import monster.vm.error;
|
|
|
|
// Base class for all kinds of blocks. A 'block' is a token or
|
|
// collection of tokens that belong together and form a syntactical
|
|
// unit. A block might for example be a statement, a declaration, a
|
|
// block of code, or the entire class.
|
|
abstract class Block
|
|
{
|
|
protected:
|
|
static Token next(ref TokenArray toks)
|
|
{
|
|
if(toks.length == 0)
|
|
fail("Unexpected end of file");
|
|
Token tt = toks[0];
|
|
toks = toks[1..toks.length];
|
|
return tt;
|
|
}
|
|
|
|
static Floc getLoc(TokenArray toks)
|
|
{
|
|
Floc loc;
|
|
if(toks.length) loc = toks[0].loc;
|
|
return loc;
|
|
}
|
|
|
|
static bool isNext(ref TokenArray toks, TT type)
|
|
{
|
|
Floc ln;
|
|
return isNext(toks, type, ln);
|
|
}
|
|
|
|
static bool isNext(ref TokenArray toks, TT type, out Floc loc)
|
|
{
|
|
if( toks.length == 0 ) return false;
|
|
loc = toks[0].loc;
|
|
if( toks[0].type != type ) return false;
|
|
next(toks);
|
|
return true;
|
|
}
|
|
|
|
static bool isNext(ref TokenArray toks, TT type, out Token tok)
|
|
{
|
|
if( toks.length == 0 ) return false;
|
|
if( toks[0].type != type ) return false;
|
|
tok = next(toks);
|
|
return true;
|
|
}
|
|
|
|
// Is the next token a separator, ie. a ; or a new line
|
|
static bool isSep(ref TokenArray toks, TT symbol = TT.Semicolon)
|
|
{
|
|
if( toks.length == 0 ) return true;
|
|
if( toks[0].newline ) return true;
|
|
return isNext(toks, symbol);
|
|
}
|
|
|
|
// Returns true if the next token is on a new line. Does not remove
|
|
// any tokens.
|
|
static bool isNewline(ref TokenArray toks)
|
|
// TT.EMPTY never occurs in the stream, so it's safe to check for
|
|
// it.
|
|
{ return isSep(toks, TT.EMPTY); }
|
|
|
|
// Require either a line break or a given character (default ;)
|
|
static void reqSep(ref TokenArray toks, TT symbol = TT.Semicolon)
|
|
{
|
|
if(!isSep(toks, symbol))
|
|
fail("Expected '" ~ tokenList[symbol] ~ "' or newline", toks);
|
|
}
|
|
|
|
static void reqNext(ref TokenArray toks, TT type, out Token tok)
|
|
{
|
|
if(!isNext(toks, type, tok))
|
|
fail("Expected " ~ tokenList[type], toks);
|
|
}
|
|
|
|
static void reqNext(ref TokenArray toks, TT type, out Floc loc)
|
|
{
|
|
Token t;
|
|
reqNext(toks, type, t);
|
|
loc = t.loc;
|
|
}
|
|
|
|
static void reqNext(ref TokenArray toks, TT type)
|
|
{
|
|
Token t;
|
|
reqNext(toks, type, t);
|
|
}
|
|
|
|
// Skip any matching set of parens (), [] or {}, and anything inside
|
|
// it.
|
|
static void skipParens(ref TokenArray toks, TT type = TT.LeftParen)
|
|
{
|
|
TT endType;
|
|
if(type == TT.LeftParen)
|
|
endType = TT.RightParen;
|
|
else if(type == TT.LeftCurl)
|
|
endType = TT.RightCurl;
|
|
else if(type == TT.LeftSquare)
|
|
endType = TT.RightSquare;
|
|
else assert(0);
|
|
|
|
int count = 0;
|
|
|
|
while(toks.length != 0)
|
|
{
|
|
if(toks[0].type == type)
|
|
count++;
|
|
else if(toks[0].type == endType)
|
|
count--;
|
|
|
|
toks = toks[1..$];
|
|
|
|
if(count <= 0)
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Sets the assembler debug line to the line belonging to this
|
|
// block.
|
|
final void setLine() { tasm.setLine(loc.line); }
|
|
|
|
public:
|
|
// File position where this block was defined
|
|
Floc loc;
|
|
|
|
// Parse a list of tokens and attempt to understand how they belong
|
|
// together. This is the syntactical level.
|
|
void parse(ref TokenArray toks);
|
|
|
|
// This goes through the code, resolves names and types etc,
|
|
// converts expressions into an intermediate form which can be
|
|
// compiled to byte code later. This is basically the semantic level.
|
|
void resolve(Scope sc);
|
|
}
|