mirror of
https://gitlab.com/OpenMW/openmw.git
synced 2025-01-26 09:35:28 +00:00
de5a07ee71
git-svn-id: https://openmw.svn.sourceforge.net/svnroot/openmw/trunk@138 ea6a568a-9f4f-0410-981a-c910a81bb256
943 lines
25 KiB
D
943 lines
25 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 (tokenizer.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.tokenizer;
|
|
|
|
import std.string;
|
|
import std.stream;
|
|
import std.stdio;
|
|
import std.utf;
|
|
|
|
import monster.util.string : begins;
|
|
|
|
import monster.vm.error;
|
|
import monster.options;
|
|
|
|
alias Token[] TokenArray;
|
|
|
|
// Check if a character is alpha-numerical or an underscore
|
|
bool validIdentChar(char c)
|
|
{
|
|
if(validFirstIdentChar(c) || numericalChar(c))
|
|
return true;
|
|
return false;
|
|
}
|
|
|
|
// Same as above, except numbers are not allowed as the first
|
|
// character. Will extend to support full Unicode later.
|
|
bool validFirstIdentChar(char c)
|
|
{
|
|
if((c >= 'a' && c <= 'z') ||
|
|
(c >= 'A' && c <= 'Z') ||
|
|
(c == '_') ) return true;
|
|
return false;
|
|
}
|
|
|
|
bool isValidIdent(char[] iname)
|
|
{
|
|
if(iname.length == 0)
|
|
return false;
|
|
|
|
if(!validFirstIdentChar(iname[0]))
|
|
return false;
|
|
|
|
foreach(char c; iname)
|
|
if(!validIdentChar(c)) return false;
|
|
|
|
return true;
|
|
}
|
|
|
|
bool numericalChar(char c)
|
|
{
|
|
return c >= '0' && c <= '9';
|
|
}
|
|
|
|
enum TT
|
|
{
|
|
// Syntax characters
|
|
Semicolon, DDDot, DDot,
|
|
LeftParen, RightParen,
|
|
LeftCurl, RightCurl,
|
|
LeftSquare, RightSquare,
|
|
Dot, Comma, Colon,
|
|
|
|
// Array length symbol
|
|
Dollar,
|
|
|
|
// 'at' sign, @
|
|
Alpha,
|
|
|
|
// Conditional expressions
|
|
IsEqual, NotEqual,
|
|
IsCaseEqual, IsCaseEqual2,
|
|
NotCaseEqual, NotCaseEqual2,
|
|
Less, More,
|
|
LessEq, MoreEq,
|
|
And, Or, Not,
|
|
|
|
// Assignment operators.
|
|
Equals, PlusEq, MinusEq, MultEq, DivEq, RemEq, IDivEq,
|
|
CatEq,
|
|
|
|
// Pre- and postfix increment and decrement operators ++ --
|
|
PlusPlus, MinusMinus,
|
|
|
|
// Arithmetic operators
|
|
Plus, Minus, Mult, Div, Rem, IDiv,
|
|
Cat,
|
|
|
|
// Keywords. Note that we use Class as a separator below, so it
|
|
// must be first in this list. All operator tokens must occur
|
|
// before Class, and all keywords must come after Class.
|
|
Class, Module, Singleton,
|
|
If, Else,
|
|
For, Foreach, ForeachRev,
|
|
Do, While, Until,
|
|
Continue, Break,
|
|
Typeof,
|
|
Return,
|
|
Switch, Select,
|
|
State,
|
|
Struct, Enum,
|
|
Import, Clone, Override, Final, Function, With,
|
|
This, New, Static, Const, Out, Ref, Abstract, Idle,
|
|
Public, Private, Protected, True, False, Native, Null,
|
|
Goto, Var,
|
|
|
|
Last, // Tokens after this do not have a specific string
|
|
// associated with them.
|
|
|
|
StringLiteral, // "something" or 'something'
|
|
IntLiteral, // Anything that starts with a number, except
|
|
// floats
|
|
FloatLiteral, // Any number which contains a period symbol
|
|
Identifier, // user-named identifier
|
|
EOF, // end of file
|
|
EMPTY // empty line (not stored)
|
|
}
|
|
|
|
struct Token
|
|
{
|
|
TT type;
|
|
char[] str;
|
|
Floc loc;
|
|
|
|
// Translated string literal (with resolved escape codes.)
|
|
dchar[] str32;
|
|
|
|
// True if this token was the first on its line.
|
|
bool newline;
|
|
|
|
char[] toString() { return str; }
|
|
|
|
static Token opCall(char[] name, Floc loc)
|
|
{ return Token(TT.Identifier, name, loc); }
|
|
static Token opCall(TT tt, char[] name, Floc loc)
|
|
{
|
|
Token t;
|
|
t.type = tt;
|
|
t.str = name;
|
|
t.loc = loc;
|
|
return t;
|
|
}
|
|
}
|
|
|
|
// Used to look up keywords.
|
|
TT keywordLookup[char[]];
|
|
bool lookupSetup = false;
|
|
|
|
void initTokenizer()
|
|
{
|
|
assert(!lookupSetup);
|
|
|
|
// Insert the keywords into the lookup table
|
|
for(TT t = TT.Class; t < TT.Last; t++)
|
|
{
|
|
char[] tok = tokenList[t];
|
|
assert(tok != "");
|
|
assert((tok in keywordLookup) == null);
|
|
keywordLookup[tok] = t;
|
|
}
|
|
|
|
lookupSetup = true;
|
|
}
|
|
|
|
// Index table of all the tokens
|
|
const char[][] tokenList =
|
|
[
|
|
TT.Semicolon : ";",
|
|
TT.DDDot : "...",
|
|
TT.DDot : "..",
|
|
TT.LeftParen : "(",
|
|
TT.RightParen : ")",
|
|
TT.LeftCurl : "{",
|
|
TT.RightCurl : "}",
|
|
TT.LeftSquare : "[",
|
|
TT.RightSquare : "]",
|
|
TT.Dot : ".",
|
|
TT.Comma : ",",
|
|
TT.Colon : ":",
|
|
|
|
TT.Dollar : "$",
|
|
|
|
TT.Alpha : "@",
|
|
|
|
TT.IsEqual : "==",
|
|
TT.NotEqual : "!=",
|
|
|
|
TT.IsCaseEqual : "=i=",
|
|
TT.IsCaseEqual2 : "=I=",
|
|
TT.NotCaseEqual : "!=i=",
|
|
TT.NotCaseEqual2 : "!=I=",
|
|
|
|
TT.Less : "<",
|
|
TT.More : ">",
|
|
TT.LessEq : "<=",
|
|
TT.MoreEq : ">=",
|
|
TT.And : "&&",
|
|
TT.Or : "||",
|
|
TT.Not : "!",
|
|
|
|
TT.Equals : "=",
|
|
TT.PlusEq : "+=",
|
|
TT.MinusEq : "-=",
|
|
TT.MultEq : "*=",
|
|
TT.DivEq : "/=",
|
|
TT.RemEq : "%=",
|
|
TT.IDivEq : "\\=",
|
|
TT.CatEq : "~=",
|
|
|
|
TT.PlusPlus : "++",
|
|
TT.MinusMinus : "--",
|
|
TT.Cat : "~",
|
|
|
|
TT.Plus : "+",
|
|
TT.Minus : "-",
|
|
TT.Mult : "*",
|
|
TT.Div : "/",
|
|
TT.Rem : "%",
|
|
TT.IDiv : "\\",
|
|
|
|
TT.Class : "class",
|
|
TT.Module : "module",
|
|
TT.Return : "return",
|
|
TT.For : "for",
|
|
TT.This : "this",
|
|
TT.New : "new",
|
|
TT.If : "if",
|
|
TT.Else : "else",
|
|
TT.Foreach : "foreach",
|
|
TT.ForeachRev : "foreach_reverse",
|
|
TT.Do : "do",
|
|
TT.While : "while",
|
|
TT.Until : "until",
|
|
TT.Continue : "continue",
|
|
TT.Break : "break",
|
|
TT.Switch : "switch",
|
|
TT.Select : "select",
|
|
TT.State : "state",
|
|
TT.Struct : "struct",
|
|
TT.Enum : "enum",
|
|
TT.Import : "import",
|
|
TT.Typeof : "typeof",
|
|
TT.Singleton : "singleton",
|
|
TT.Clone : "clone",
|
|
TT.Static : "static",
|
|
TT.Const : "const",
|
|
TT.Abstract : "abstract",
|
|
TT.Override : "override",
|
|
TT.Final : "final",
|
|
TT.Function : "function",
|
|
TT.With : "with",
|
|
TT.Idle : "idle",
|
|
TT.Out : "out",
|
|
TT.Ref : "ref",
|
|
TT.Public : "public",
|
|
TT.Private : "private",
|
|
TT.Protected : "protected",
|
|
TT.True : "true",
|
|
TT.False : "false",
|
|
TT.Native : "native",
|
|
TT.Null : "null",
|
|
TT.Goto : "goto",
|
|
TT.Var : "var",
|
|
|
|
// These are only used in error messages
|
|
TT.StringLiteral : "string literal",
|
|
TT.IntLiteral : "integer literal",
|
|
TT.FloatLiteral : "floating point literal",
|
|
TT.Identifier : "identifier",
|
|
TT.EOF : "end of file",
|
|
TT.EMPTY : "empty line - you should never see this"
|
|
];
|
|
|
|
class Tokenizer
|
|
{
|
|
private:
|
|
// Line buffer. Don't worry, this is perfectly safe. It is used by
|
|
// Stream.readLine, which uses the buffer if it fits and creates a
|
|
// new one if it doesn't. It is only here to optimize memory usage
|
|
// (avoid creating a new buffer for each line), and lines longer
|
|
// than 300 characters will work without problems.
|
|
char[300] buffer;
|
|
char[] line; // The rest of the current line
|
|
Stream inf;
|
|
int lineNum=-1;
|
|
char[] fname;
|
|
bool newline;
|
|
|
|
// Make a token of given type with given string, and remove it from
|
|
// the input line.
|
|
Token retToken(TT type, char[] str)
|
|
{
|
|
Token t;
|
|
t.type = type;
|
|
t.str = str;
|
|
t.newline = newline;
|
|
t.loc.fname = fname;
|
|
t.loc.line = lineNum;
|
|
|
|
// Special case for =I= and !=I=. Treat them the same as =i= and
|
|
// !=i=.
|
|
if(type == TT.IsCaseEqual2) t.type = TT.IsCaseEqual;
|
|
if(type == TT.NotCaseEqual2) t.type = TT.NotCaseEqual;
|
|
|
|
// Treat } as a separator
|
|
if(type == TT.RightCurl) t.newline = true;
|
|
|
|
// Remove the string from 'line', along with any following witespace
|
|
remWord(str);
|
|
return t;
|
|
}
|
|
|
|
// Removes 'str' from the beginning of 'line', or from
|
|
// line[leadIn..$] if leadIn != 0.
|
|
void remWord(char[] str, int leadIn = 0)
|
|
{
|
|
assert(line.length >= leadIn);
|
|
line = line[leadIn..$];
|
|
|
|
assert(line.begins(str));
|
|
line = line[str.length..$].stripl();
|
|
}
|
|
|
|
Token eofToken()
|
|
{
|
|
Token t;
|
|
t.str = "<end of file>";
|
|
t.type = TT.EOF;
|
|
t.newline = true;
|
|
t.loc.line = lineNum;
|
|
t.loc.fname = fname;
|
|
return t;
|
|
}
|
|
|
|
Token empty;
|
|
|
|
public:
|
|
final:
|
|
// Used when reading tokens from a file or a stream
|
|
this(char[] fname, Stream inf, int bom)
|
|
{
|
|
assert(inf !is null);
|
|
|
|
// The BOM (byte order mark) defines the byte order (little
|
|
// endian or big endian) and the encoding (utf8, utf16 or
|
|
// utf32).
|
|
switch(bom)
|
|
{
|
|
case -1:
|
|
// Files without a BOM are interpreted as UTF8
|
|
case BOM.UTF8:
|
|
// UTF8 is the default
|
|
break;
|
|
|
|
case BOM.UTF16LE:
|
|
case BOM.UTF16BE:
|
|
case BOM.UTF32LE:
|
|
case BOM.UTF32BE:
|
|
fail("UTF16 and UTF32 files are not supported yet");
|
|
default:
|
|
fail("Unknown BOM value!");
|
|
}
|
|
|
|
this.inf = inf;
|
|
this.fname = fname;
|
|
|
|
this();
|
|
}
|
|
|
|
// This is used for single-line mode, such as in a console.
|
|
this()
|
|
{
|
|
empty.type = TT.EMPTY;
|
|
}
|
|
|
|
void setLine(char[] ln)
|
|
{
|
|
assert(inf is null, "setLine only supported in line mode");
|
|
line = ln;
|
|
}
|
|
|
|
~this() { if(inf !is null) delete inf; }
|
|
|
|
void fail(char[] msg)
|
|
{
|
|
if(inf !is null)
|
|
// File mode
|
|
throw new MonsterException(format("%s:%s: %s", fname, lineNum, msg));
|
|
else
|
|
// Line mode
|
|
throw new MonsterException(msg);
|
|
}
|
|
|
|
// Various parsing modes
|
|
enum
|
|
{
|
|
Normal, // Normal mode
|
|
Block, // Block comment
|
|
Nest // Nested block comment
|
|
}
|
|
int mode = Normal;
|
|
int nests = 0; // Nest level
|
|
|
|
// Get the next token from the line, if any
|
|
Token getNextFromLine()
|
|
{
|
|
assert(lookupSetup,
|
|
"Internal error: The tokenizer lookup table has not been set up!");
|
|
|
|
restart:
|
|
|
|
if(mode == Block)
|
|
{
|
|
int index = line.find("*/");
|
|
|
|
// If we find a '*/', the comment is done
|
|
if(index != -1)
|
|
{
|
|
mode = Normal;
|
|
|
|
// Cut the comment from the input
|
|
remWord("*/", index);
|
|
}
|
|
else
|
|
{
|
|
// Comment was not terminated on this line, try the next
|
|
line = null;
|
|
}
|
|
}
|
|
else if(mode == Nest)
|
|
{
|
|
// Check for nested /+ and +/ in here, but go to restart if
|
|
// none is found (meaning the comment continues on the next
|
|
// line), or reset mode and go to restart if nest level ever
|
|
// gets to 0.
|
|
|
|
while(line.length >= 2)
|
|
{
|
|
int incInd = -1;
|
|
int decInd = -1;
|
|
// Find the first matching '/+' or '+/
|
|
foreach(int i, char c; line[0..$-1])
|
|
{
|
|
if(c == '/' && line[i+1] == '+')
|
|
{
|
|
incInd = i;
|
|
break;
|
|
}
|
|
else if(c == '+' && line[i+1] == '/')
|
|
{
|
|
decInd = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Add a nest level when '/+' is found
|
|
if(incInd != -1)
|
|
{
|
|
remWord("/+", incInd);
|
|
nests++;
|
|
continue; // Search more in this line
|
|
}
|
|
|
|
// Remove a nest level when '+/' is found
|
|
else if(decInd != -1)
|
|
{
|
|
// Remove the +/ from input
|
|
remWord("+/", decInd);
|
|
|
|
nests--; // Remove a level
|
|
assert(nests >= 0);
|
|
|
|
// Are we done? If so, return to normal mode.
|
|
if(nests == 0)
|
|
{
|
|
mode = Normal;
|
|
break;
|
|
}
|
|
continue;
|
|
}
|
|
|
|
// Nothing found on this line, try the next
|
|
break;
|
|
}
|
|
|
|
// If we're still in nested comment mode, ignore the rest of
|
|
// the line
|
|
if(mode == Nest)
|
|
line = null;
|
|
}
|
|
|
|
// Comment - ignore the rest of the line
|
|
if(line.begins("//"))
|
|
line = null;
|
|
|
|
// If the line is empty at this point, there's nothing more to
|
|
// be done
|
|
if(line == "")
|
|
return empty;
|
|
|
|
// Block comment
|
|
if(line.begins("/*"))
|
|
{
|
|
mode = Block;
|
|
line = line[2..$];
|
|
goto restart;
|
|
}
|
|
|
|
// Nested comment
|
|
if(line.begins("/+"))
|
|
{
|
|
mode = Nest;
|
|
line = line[2..$];
|
|
nests++;
|
|
goto restart;
|
|
}
|
|
|
|
if(line.begins("*/")) fail("Unexpected end of block comment");
|
|
if(line.begins("+/")) fail("Unexpected end of nested comment");
|
|
|
|
// String literals (multi-line literals not implemented yet)
|
|
if(line.begins("\"") || // Standard string: "abc"
|
|
line.begins("r\"") || // Wysiwig string: r"c:\dir"
|
|
line.begins("\\\"") || // ditto: \"c:\dir"
|
|
line.begins("'") ||
|
|
line.begins("r'") || // Equivalent ' versions
|
|
line.begins("\\'"))
|
|
{
|
|
bool found = false;
|
|
bool wysiwig = false;
|
|
|
|
// Quote character that terminates this string.
|
|
char quote;
|
|
|
|
char[] slice = line;
|
|
|
|
// Removes the first num chars from the line
|
|
void skip(int num)
|
|
{
|
|
assert(num <= line.length);
|
|
slice = slice[num..$];
|
|
}
|
|
|
|
// Parse the first quotation
|
|
if(slice[0] == '"' || slice[0] == '\'')
|
|
{
|
|
quote = slice[0];
|
|
skip(1);
|
|
}
|
|
else
|
|
{
|
|
// Check for wysiwig strings
|
|
if(slice[0] == '\\' || slice[0] == 'r')
|
|
wysiwig = true;
|
|
else assert(0);
|
|
|
|
quote = slice[1];
|
|
skip(2);
|
|
}
|
|
|
|
assert(quote == '"' || quote == '\'');
|
|
|
|
// This will store the result
|
|
dchar[] result;
|
|
|
|
// Stores a single character in the result string, and
|
|
// removes a given number of input characters.
|
|
void store(dchar ch, int slen)
|
|
{
|
|
result ~= ch;
|
|
skip(slen);
|
|
}
|
|
|
|
// Convert a given code into 'ch', if it is found.
|
|
void convert(char[] code, dchar ch)
|
|
{
|
|
if(slice.begins(code))
|
|
store(ch, code.length);
|
|
}
|
|
|
|
// Convert given escape character to 'res'
|
|
void escape(char ch, dchar res)
|
|
{
|
|
if(slice.length >= 2 &&
|
|
slice[0] == '\\' &&
|
|
slice[1] == ch)
|
|
store(res, 2);
|
|
}
|
|
|
|
// Interpret string
|
|
while(slice.length)
|
|
{
|
|
int startLen = slice.length;
|
|
|
|
// Convert "" to " (or '' to ' in single-quote strings)
|
|
convert(""~quote~quote, quote);
|
|
|
|
// Interpret backslash escape codes if we're not in
|
|
// wysiwig mode
|
|
if(!wysiwig)
|
|
{
|
|
escape('"', '"'); // \" == literal "
|
|
escape('\'', '\''); // \' == literal '
|
|
escape('\\', '\\'); // \\ == literal \
|
|
|
|
escape('a', 7); // \a == bell
|
|
escape('b', 8); // \b == backspace
|
|
escape('f', 12); // \f == feed form
|
|
escape('n', '\n'); // \n == newline
|
|
escape('r', '\r'); // \r == carriage return
|
|
escape('t', '\t'); // \t == tab
|
|
escape('v', '\v'); // \v == vertical tab
|
|
escape('e', 27); // \e == ANSI escape
|
|
|
|
// Check for numerical escapes
|
|
|
|
// If either of these aren't met, this isn't a valid
|
|
// escape code.
|
|
if(slice.length < 2 ||
|
|
slice[0] != '\\')
|
|
goto nocode;
|
|
|
|
// Checks and converts the digits in slice[] into a
|
|
// character.
|
|
void convertNumber(int skp, int maxLen, int base,
|
|
char[] pattern, char[] name)
|
|
{
|
|
assert(base <= 16);
|
|
|
|
// Skip backslash and other leading characters
|
|
skip(skp);
|
|
|
|
int len; // Number of digits found
|
|
uint result = 0;
|
|
|
|
for(len=0; len<maxLen; len++)
|
|
{
|
|
if(slice.length <= len) break;
|
|
|
|
char digit = slice[len];
|
|
|
|
// Does the digit qualify?
|
|
if(!inPattern(digit, pattern))
|
|
break;
|
|
|
|
// Multiply up the existing number to
|
|
// make room for the digit.
|
|
result *= base;
|
|
|
|
// Convert single digit to a number
|
|
if(digit >= '0' && digit <= '9')
|
|
digit -= '0';
|
|
else if(digit >= 'a' && digit <= 'z')
|
|
digit -= 'a' - 10;
|
|
else if(digit >= 'A' && digit <= 'Z')
|
|
digit -= 'A' - 10;
|
|
assert(digit >= 0 && digit < base);
|
|
|
|
// Add inn the digit
|
|
result += digit;
|
|
}
|
|
|
|
if(len > 0)
|
|
{
|
|
// We got something. Convert it and store
|
|
// it.
|
|
store(result, len);
|
|
}
|
|
else
|
|
fail("Invalid " ~ name ~ " escape code");
|
|
}
|
|
|
|
const Dec = "0-9";
|
|
const Oct = "0-7";
|
|
const Hex = "0-9a-fA-F";
|
|
|
|
// Octal escapes: \0N, \0NN or \0NNN where N are
|
|
// octal digits (0-7). Also accepts \o instead of
|
|
// \0.
|
|
if(slice[1] == '0' || slice[1] == 'o')
|
|
convertNumber(2, 3, 8, Oct, "octal");
|
|
|
|
// Decimal escapes: \N \NN and \NNN, where N are
|
|
// digits and the first digit is not zero.
|
|
else if(inPattern(slice[1], Dec))
|
|
convertNumber(1, 3, 10, Dec, "decimal");
|
|
|
|
// Hex escape codes: \xXX where X are hex digits
|
|
else if(slice[1] == 'x')
|
|
convertNumber(2, 2, 16, Hex, "hex");
|
|
|
|
// Unicode escape codes:
|
|
// \uXXXX
|
|
else if(slice[1] == 'u')
|
|
convertNumber(2, 4, 16, Hex, "Unicode hex");
|
|
|
|
// \UXXXXXXXX
|
|
else if(slice[1] == 'U')
|
|
convertNumber(2, 8, 16, Hex, "Unicode hex");
|
|
|
|
}
|
|
nocode:
|
|
|
|
// If something was converted this round, start again
|
|
// from the top.
|
|
if(startLen != slice.length)
|
|
continue;
|
|
|
|
assert(slice.length > 0);
|
|
|
|
// Nothing was done. Are we at the end of the string?
|
|
if(slice[0] == quote)
|
|
{
|
|
skip(1);
|
|
found = true;
|
|
break;
|
|
}
|
|
|
|
// Unhandled escape code?
|
|
if(slice[0] == '\\' && !wysiwig)
|
|
{
|
|
if(slice.length == 0)
|
|
// Just a single \ at the end of the line
|
|
fail("Multiline string literals not implemented");
|
|
else
|
|
fail("Unhandled escape code: \\" ~ slice[1]);
|
|
}
|
|
|
|
// Nope. It's just a normal character. Decode it from
|
|
// UTF8.
|
|
size_t clen = 0;
|
|
dchar cres;
|
|
cres = decode(slice,clen);
|
|
store(cres, clen);
|
|
}
|
|
if(!found) fail("Unterminated string literal '" ~line~ "'");
|
|
|
|
// Set 'slice' to contain the original string
|
|
slice = line[0..(line.length-slice.length)];
|
|
|
|
// Set up the token
|
|
auto t = retToken(TT.StringLiteral, slice.dup);
|
|
t.str32 = result;
|
|
return t;
|
|
}
|
|
|
|
// Numerical literals - if it starts with a number, we accept
|
|
// it, until it is interupted by an unacceptable character. We
|
|
// also accept numbers on the form .NUM. We do not try to parse
|
|
// the number here.
|
|
if(numericalChar(line[0]) ||
|
|
// Cover the .num case
|
|
( line.length >= 2 && line[0] == '.' &&
|
|
numericalChar(line[1]) ))
|
|
{
|
|
// Treat the rest as we would an identifier - the actual
|
|
// interpretation will be done later. We allow non-numerical
|
|
// tokens in the literal, such as 0x0a or 1_000_000. We must
|
|
// also explicitly allow '.' dots.
|
|
int len = 1;
|
|
bool lastDot = false; // Was the last char a '.'?
|
|
int dots; // Number of dots
|
|
foreach(char ch; line[1..$])
|
|
{
|
|
if(ch == '.')
|
|
{
|
|
// We accept "." but not "..", as this might be an
|
|
// operator.
|
|
if(lastDot)
|
|
{
|
|
// Remove the last dot and exit.
|
|
len--;
|
|
dots--;
|
|
break;
|
|
}
|
|
lastDot = true;
|
|
dots++;
|
|
}
|
|
else
|
|
{
|
|
if(!validIdentChar(ch)) break;
|
|
lastDot = false;
|
|
//lastPer = false;
|
|
}
|
|
|
|
// This was a valid character, count it
|
|
len++;
|
|
}
|
|
if(dots != 0)
|
|
return retToken(TT.FloatLiteral, line[0..len].dup);
|
|
else
|
|
return retToken(TT.IntLiteral, line[0..len].dup);
|
|
}
|
|
|
|
// Check for identifiers
|
|
if(validFirstIdentChar(line[0]))
|
|
{
|
|
// It's an identifier or name, find the length
|
|
int len = 1;
|
|
foreach(char ch; line[1..$])
|
|
{
|
|
if(!validIdentChar(ch)) break;
|
|
len++;
|
|
}
|
|
|
|
char[] id = line[0..len];
|
|
|
|
// We only allow certain identifiers to begin with __, as
|
|
// these are reserved for internal use.
|
|
if(id.begins("__"))
|
|
if(id != "__STACK__")
|
|
fail("Identifier " ~ id ~ " is not allowed to begin with __");
|
|
|
|
// Check if this is a keyword
|
|
if(id in keywordLookup)
|
|
{
|
|
TT t = keywordLookup[id];
|
|
assert(t >= TT.Class && t < TT.Last,
|
|
"Found " ~ id ~ " as a keyword, but with wrong type!");
|
|
return retToken(t, tokenList[t]);
|
|
}
|
|
|
|
// Not a keyword? Then it's an identifier
|
|
return retToken(TT.Identifier, id.dup);
|
|
}
|
|
|
|
// Check for operators and syntax characters. We browse through
|
|
// the entire list, and select the longest match that fits (so
|
|
// we don't risk matching "+" to "+=", for example.)
|
|
TT match;
|
|
int mlen = 0;
|
|
foreach(int i, char[] tok; tokenList[0..TT.Class])
|
|
{
|
|
// Skip =i= and family, if monster.options tells us to
|
|
static if(!ciStringOps)
|
|
{
|
|
if(i == TT.IsCaseEqual || i == TT.IsCaseEqual2 ||
|
|
i == TT.NotCaseEqual || i == TT.NotCaseEqual2)
|
|
continue;
|
|
}
|
|
|
|
if(line.begins(tok) && tok.length >= mlen)
|
|
{
|
|
assert(tok.length > mlen, "Two matching tokens of the same length");
|
|
mlen = tok.length;
|
|
match = cast(TT) i;
|
|
}
|
|
}
|
|
|
|
if(mlen) return retToken(match, tokenList[match]);
|
|
|
|
// Invalid token
|
|
fail("Invalid token " ~ line);
|
|
}
|
|
|
|
// Get the next token from a stream
|
|
Token getNext()
|
|
{
|
|
assert(inf !is null, "getNext() found a null stream");
|
|
|
|
if(lineNum == -1) lineNum = 0;
|
|
|
|
restart:
|
|
newline = false;
|
|
// Get the next line, if the current is empty
|
|
while(line.length == 0)
|
|
{
|
|
// No more information, we're done
|
|
if(inf.eof())
|
|
{
|
|
if(mode == Block) fail("Unterminated block comment");
|
|
if(mode == Nest) fail("Unterminated nested comment");
|
|
return eofToken();
|
|
}
|
|
|
|
// Read a line and remove leading and trailing whitespace
|
|
line = inf.readLine(buffer).strip();
|
|
lineNum++;
|
|
newline = true;
|
|
}
|
|
|
|
assert(line.length > 0);
|
|
|
|
static if(skipHashes)
|
|
{
|
|
// Skip the line if it begins with #.
|
|
if(/*lineNum == 1 && */line.begins("#"))
|
|
{
|
|
line = null;
|
|
goto restart;
|
|
}
|
|
}
|
|
|
|
Token tt = getNextFromLine();
|
|
|
|
// Skip empty lines, don't return them into the token list.
|
|
if(tt.type == TT.EMPTY)
|
|
goto restart;
|
|
|
|
return tt;
|
|
}
|
|
}
|
|
|
|
// Read the entire file into an array of tokens. This includes the EOF
|
|
// token at the end.
|
|
TokenArray tokenizeStream(char[] fname, Stream stream, int bom)
|
|
{
|
|
TokenArray tokenArray;
|
|
|
|
Tokenizer tok = new Tokenizer(fname, stream, bom);
|
|
Token tt;
|
|
do
|
|
{
|
|
tt = tok.getNext();
|
|
tokenArray ~= tt;
|
|
}
|
|
while(tt.type != TT.EOF)
|
|
delete tok;
|
|
|
|
return tokenArray;
|
|
}
|