2008-11-07 08:45:18 +00:00
|
|
|
/*
|
|
|
|
Monster - an advanced game scripting language
|
|
|
|
Copyright (C) 2007, 2008 Nicolay Korslund
|
|
|
|
Email: <korslund@gmail.com>
|
|
|
|
WWW: http://monster.snaptoad.com/
|
|
|
|
|
|
|
|
This file (scopes.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.scopes;
|
|
|
|
|
2008-11-07 13:14:46 +00:00
|
|
|
import std.stdio;
|
|
|
|
import std.string;
|
2008-11-07 08:45:18 +00:00
|
|
|
|
|
|
|
import monster.util.aa;
|
|
|
|
|
|
|
|
import monster.compiler.statement;
|
|
|
|
import monster.compiler.expression;
|
|
|
|
import monster.compiler.tokenizer;
|
|
|
|
import monster.compiler.types;
|
|
|
|
import monster.compiler.properties;
|
|
|
|
import monster.compiler.functions;
|
|
|
|
import monster.compiler.states;
|
|
|
|
import monster.compiler.variables;
|
|
|
|
|
|
|
|
import monster.vm.mclass;
|
|
|
|
import monster.vm.error;
|
|
|
|
|
|
|
|
// The global scope
|
|
|
|
PackageScope global;
|
|
|
|
|
|
|
|
void initScope()
|
|
|
|
{
|
|
|
|
global = new PackageScope(null, "global");
|
|
|
|
}
|
|
|
|
|
|
|
|
// TODO: Write here which of these should be kept around at runtime,
|
|
|
|
// and which of them can be discarded after compilation.
|
|
|
|
|
|
|
|
abstract class Scope
|
|
|
|
{
|
|
|
|
protected:
|
|
|
|
// The parent scope. For function scopes, this is the scope of the
|
|
|
|
// class it belongs to. For code blocks, loops etc, it is the scope
|
|
|
|
// of the code block outside this one. For classes, this points to
|
|
|
|
// the scope of the parent class, if any, or to the package or
|
|
|
|
// global scope.
|
|
|
|
Scope parent;
|
|
|
|
|
|
|
|
// Properties assigned to this scope (if any).
|
|
|
|
PropertyScope nextProp;
|
|
|
|
|
|
|
|
// Verify that an identifier is not declared in this scope. If the
|
|
|
|
// identifier is found, give a duplicate identifier compiler
|
|
|
|
// error. Recurses through parent scopes.
|
|
|
|
void clearId(Token name)
|
|
|
|
{
|
|
|
|
assert(!isRoot());
|
|
|
|
parent.clearId(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Made protected since it is so easy to confuse with isStateCode(),
|
|
|
|
// and we never actually need it anywhere outside this file.
|
|
|
|
bool isState() { return false; }
|
|
|
|
|
|
|
|
private:
|
|
|
|
// Name of this scope. All scopes must have a name, but for some
|
|
|
|
// types it is set automatically (like a code block scope.) It is
|
|
|
|
// mostly used for debugging.
|
|
|
|
char[] scopeName;
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
this(Scope last, char[] name)
|
|
|
|
{
|
|
|
|
scopeName = name;
|
|
|
|
parent = last;
|
|
|
|
|
|
|
|
assert(last !is this, "scope cannot be it's own parent");
|
|
|
|
|
|
|
|
assert(name != "");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Is this the root scope?
|
|
|
|
final bool isRoot()
|
|
|
|
{
|
|
|
|
if(parent !is null) return false;
|
|
|
|
assert(allowRoot(), toString() ~ " cannot be a root scope");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Is THIS scope of this particular kind?
|
|
|
|
bool isFunc() { return false; }
|
|
|
|
bool isCode() { return false; }
|
|
|
|
bool isLoop() { return false; }
|
|
|
|
bool isClass() { return false; }
|
|
|
|
bool isArray() { return false; }
|
|
|
|
bool isPackage() { return false; }
|
|
|
|
bool isProperty() { return false; }
|
|
|
|
|
|
|
|
// Is this scope allowed to be a root scope (without parent?)
|
|
|
|
bool allowRoot() { return false; }
|
|
|
|
|
|
|
|
// Get a property from this scope
|
|
|
|
void getProperty(Token name, Type ownerType, ref Property p)
|
|
|
|
{
|
|
|
|
assert(0);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the function definition belonging to this scope.
|
|
|
|
Function *getFunction()
|
|
|
|
{
|
|
|
|
assert(!isRoot(), "getFunction called on a root scope");
|
|
|
|
return parent.getFunction();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the class
|
|
|
|
MonsterClass getClass()
|
|
|
|
{
|
|
|
|
assert(!isRoot(), "getClass called on a root scope");
|
|
|
|
return parent.getClass();
|
|
|
|
}
|
|
|
|
|
|
|
|
State* getState()
|
|
|
|
{
|
|
|
|
assert(!isRoot(), "getState called on a root scope");
|
|
|
|
return parent.getState();
|
|
|
|
}
|
|
|
|
|
|
|
|
Expression getArray()
|
|
|
|
{
|
|
|
|
assert(!isRoot(), "getArray called on wrong scope type");
|
|
|
|
return parent.getArray();
|
|
|
|
}
|
|
|
|
|
|
|
|
int getLoopStack()
|
|
|
|
{
|
|
|
|
assert(!isRoot(), "getLoopStack called on wrong scope type");
|
|
|
|
return parent.getLoopStack();
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the break or continue label for the given named loop, or the
|
|
|
|
// innermost loop if name is empty or omitted. Returns null if the
|
|
|
|
// label was not found. Can only be called within loops.
|
|
|
|
LabelStatement getBreak(char[] name = "") { return null; }
|
|
|
|
LabelStatement getContinue(char[] name = "") { return null; }
|
|
|
|
|
|
|
|
// Does this scope have a property with the given name?
|
|
|
|
bool hasProperty(Token name)
|
|
|
|
{
|
|
|
|
assert(!isProperty);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
final bool findProperty(Token name, Type ownerType, ref Property result)
|
|
|
|
{
|
|
|
|
if(hasProperty(name))
|
|
|
|
{
|
|
|
|
getProperty(name, ownerType, result);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
if(nextProp !is null)
|
|
|
|
return nextProp.findProperty(name, ownerType, result);
|
|
|
|
|
|
|
|
// No property in this scope. Check the parent.
|
|
|
|
if(!isRoot) return parent.findProperty(name, ownerType, result);
|
|
|
|
|
|
|
|
// No parent, property not found.
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
Variable* findVar(char[] name)
|
|
|
|
{
|
|
|
|
if(isRoot()) return null;
|
|
|
|
return parent.findVar(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
State* findState(char[] name)
|
|
|
|
{
|
|
|
|
if(isRoot()) return null;
|
|
|
|
return parent.findState(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
Function* findFunc(char[] name)
|
|
|
|
{
|
|
|
|
if(isRoot()) return null;
|
|
|
|
return parent.findFunc(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
void insertLabel(StateLabel *lb)
|
|
|
|
{
|
|
|
|
assert(!isRoot);
|
|
|
|
parent.insertLabel(lb);
|
|
|
|
}
|
|
|
|
|
|
|
|
void insertVar(Variable* dec) { assert(0); }
|
|
|
|
|
|
|
|
// Used for summing up stack level. Redeclared in StackScope.
|
|
|
|
int getTotLocals() { return 0; }
|
|
|
|
int getLocals() { assert(0); }
|
|
|
|
|
|
|
|
// These must be overridden in their respective scopes
|
|
|
|
int addNewDataVar(int varSize) { assert(0); }
|
|
|
|
int addNewLocalVar(int varSize) { assert(0); }
|
|
|
|
|
|
|
|
final:
|
|
|
|
|
|
|
|
bool isStateCode()
|
|
|
|
{ return isInState() && !isInFunc(); }
|
|
|
|
|
|
|
|
// Is this scope OR one of the parent scopes of the given kind?
|
|
|
|
bool isInFunc()
|
|
|
|
{
|
|
|
|
if(isFunc()) return true;
|
|
|
|
if(isRoot()) return false;
|
|
|
|
return parent.isInFunc();
|
|
|
|
}
|
|
|
|
bool isInState()
|
|
|
|
{
|
|
|
|
if(isState()) return true;
|
|
|
|
if(isRoot()) return false;
|
|
|
|
return parent.isInState();
|
|
|
|
}
|
|
|
|
bool isInLoop()
|
|
|
|
{
|
|
|
|
if(isLoop()) return true;
|
|
|
|
if(isRoot()) return false;
|
|
|
|
return parent.isInLoop();
|
|
|
|
}
|
|
|
|
bool isInClass()
|
|
|
|
{
|
|
|
|
if(isClass()) return true;
|
|
|
|
if(isRoot()) return false;
|
|
|
|
return parent.isInClass();
|
|
|
|
}
|
|
|
|
bool isInPackage()
|
|
|
|
{
|
|
|
|
if(isPackage()) return true;
|
|
|
|
if(isRoot()) return false;
|
|
|
|
return parent.isInPackage();
|
|
|
|
}
|
|
|
|
|
|
|
|
char[] toString()
|
|
|
|
{
|
|
|
|
if(parent is null) return scopeName;
|
|
|
|
else return parent.toString() ~ "." ~ scopeName;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
final class StateScope : Scope
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
State* st;
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
this(Scope last, State* s)
|
|
|
|
{
|
|
|
|
st = s;
|
|
|
|
super(last, st.name.str);
|
|
|
|
}
|
|
|
|
|
|
|
|
override:
|
|
|
|
void clearId(Token name)
|
|
|
|
{
|
|
|
|
assert(name.str != "");
|
|
|
|
|
|
|
|
// Check against labels. We are not allowed to shadow labels at
|
|
|
|
// any point.
|
|
|
|
StateLabel *lb;
|
|
|
|
if(st.labels.inList(name.str, lb))
|
|
|
|
fail(format("Identifier '%s' conflicts with label on line %s", name.str,
|
|
|
|
lb.name.loc), name.loc);
|
|
|
|
|
|
|
|
super.clearId(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Insert a label, check for name collisions.
|
|
|
|
void insertLabel(StateLabel *lb)
|
|
|
|
{
|
|
|
|
// Check for name collisions
|
|
|
|
clearId(lb.name);
|
|
|
|
|
|
|
|
st.labels[lb.name.str] = lb;
|
|
|
|
}
|
|
|
|
|
|
|
|
State* getState() { return st; }
|
|
|
|
|
|
|
|
bool isState() { return true; }
|
|
|
|
}
|
|
|
|
|
|
|
|
// A package scope is a scope that can contain classes.
|
|
|
|
final class PackageScope : Scope
|
|
|
|
{
|
|
|
|
// List of classes in this package. This is case insensitive, so we
|
|
|
|
// can look up file names too.
|
|
|
|
HashTable!(char[], MonsterClass, GCAlloc, CITextHash) classes;
|
|
|
|
|
|
|
|
// Lookup by integer index. TODO: This should be in a global scope
|
|
|
|
// rather than per package. We can think about that when we
|
|
|
|
// implement packages.
|
|
|
|
HashTable!(CIndex, MonsterClass) indexList;
|
|
|
|
|
|
|
|
// Forward references. Refers to unloaded and non-existing classes.
|
|
|
|
HashTable!(char[], CIndex) forwards;
|
|
|
|
|
|
|
|
// Unique global index to give the next class. TODO: Ditto
|
|
|
|
CIndex next = 1;
|
|
|
|
|
|
|
|
this(Scope last, char[] name)
|
|
|
|
{
|
|
|
|
super(last, name);
|
|
|
|
|
|
|
|
assert(last is null);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isPackage() { return true; }
|
|
|
|
bool allowRoot() { return true; }
|
|
|
|
|
|
|
|
// Insert a new class into the scope. The class is given a unique
|
|
|
|
// global index. If the class was previously forward referenced, the
|
|
|
|
// forward is replaced and the previously assigned forward index is
|
|
|
|
// used. A class can only be inserted once.
|
|
|
|
void insertClass(MonsterClass cls)
|
|
|
|
{
|
|
|
|
assert(cls !is null);
|
|
|
|
assert(cls.name.str != "");
|
|
|
|
|
|
|
|
// Are we already in the list?
|
|
|
|
MonsterClass c2;
|
|
|
|
if(global.ciInList(cls.name.str, c2))
|
|
|
|
{
|
|
|
|
// That's not allowed. Determine what error message to give.
|
|
|
|
|
|
|
|
if(c2.name.str == cls.name.str)
|
|
|
|
// Exact name collision
|
|
|
|
fail("Cannot load class " ~ cls.name.str ~
|
|
|
|
" because it is already loaded.");
|
|
|
|
|
|
|
|
// Case insensitive match
|
|
|
|
fail("Cannot load class " ~ cls.name.str ~ " because "
|
|
|
|
~ c2.name.str ~
|
|
|
|
" already exists (class names cannot differ only by case.)");
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check that no other identifier with this name exists
|
|
|
|
clearId(cls.name);
|
|
|
|
|
|
|
|
// We're clear. Find an index to use. If the class was forward
|
|
|
|
// referenced, then an index has already been assigned.
|
|
|
|
CIndex ci;
|
|
|
|
|
|
|
|
if(forwards.inList(cls.name.str, ci))
|
|
|
|
{
|
|
|
|
// ci is set, remove the entry from the forwards hashmap
|
|
|
|
assert(ci != 0);
|
|
|
|
forwards.remove(cls.name.str);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
// Get a new index
|
|
|
|
ci = next++;
|
|
|
|
|
|
|
|
assert(!indexList.inList(ci));
|
|
|
|
|
|
|
|
// Assign the index and insert class into both lists
|
|
|
|
cls.gIndex = ci;
|
|
|
|
classes[cls.name.str] = cls;
|
|
|
|
indexList[ci] = cls;
|
|
|
|
|
|
|
|
assert(indexList.inList(ci));
|
|
|
|
}
|
|
|
|
|
|
|
|
// Case insensitive lookups. Used for comparing with file names,
|
|
|
|
// before the actual class is loaded.
|
|
|
|
bool ciInList(char[] name)
|
|
|
|
{ return classes.inList(name); }
|
|
|
|
bool ciInList(char[] name, ref MonsterClass cb)
|
|
|
|
{ return classes.inList(name, cb); }
|
|
|
|
|
|
|
|
// Case sensitive versions. If a class is found that does not match
|
|
|
|
// in case, it is an error.
|
|
|
|
bool csInList(char[] name, ref MonsterClass cb)
|
|
|
|
{
|
|
|
|
return ciInList(name, cb) && cb.name.str == name;
|
|
|
|
//fail(format("Class name mismatch: wanted %s but found %s",
|
|
|
|
// name, cb.name.str));
|
|
|
|
}
|
|
|
|
bool csInList(char[] name)
|
|
|
|
{
|
|
|
|
MonsterClass mc;
|
|
|
|
return csInList(name, mc);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the class. It must exist and the case must match. getClass
|
|
|
|
// will set up the class scope if this is not already done.
|
|
|
|
MonsterClass getClass(char[] name)
|
|
|
|
{
|
|
|
|
MonsterClass mc;
|
|
|
|
if(!csInList(name, mc))
|
|
|
|
{
|
|
|
|
char[] msg = "Class '" ~ name ~ "' not found.";
|
|
|
|
if(ciInList(name, mc))
|
|
|
|
msg ~= " (Perhaps you meant " ~ mc.name.str ~ "?)";
|
|
|
|
fail(msg);
|
|
|
|
}
|
|
|
|
mc.requireScope();
|
|
|
|
return mc;
|
|
|
|
}
|
|
|
|
|
|
|
|
MonsterClass getClass(CIndex ind)
|
|
|
|
{
|
|
|
|
MonsterClass mc;
|
|
|
|
if(!indexList.inList(ind, mc))
|
|
|
|
fail("Invalid class index encountered");
|
|
|
|
mc.requireScope();
|
|
|
|
return mc;
|
|
|
|
}
|
|
|
|
|
|
|
|
override void clearId(Token name)
|
|
|
|
{
|
|
|
|
assert(name.str != "");
|
|
|
|
|
|
|
|
// Type names can never be overwritten, so we check findClass
|
|
|
|
// and the built-in types. We might move the builtin type check
|
|
|
|
// to a "global" scope at some point.
|
|
|
|
if(BasicType.isBasic(name.str) || csInList(name.str))
|
|
|
|
fail("Identifier '"~ name.str~ "' is a type and cannot be redeclared",
|
|
|
|
name.loc);
|
|
|
|
|
|
|
|
assert(isRoot());
|
|
|
|
}
|
|
|
|
|
|
|
|
// Find a parsed class of the given name. Looks in the list of
|
|
|
|
// loaded classes and in the file system. Returns null if the class
|
|
|
|
// cannot be found.
|
|
|
|
private MonsterClass findParsed(char[] name)
|
|
|
|
{
|
|
|
|
MonsterClass result = null;
|
|
|
|
|
|
|
|
// TODO: We must handle package structures etc later.
|
|
|
|
|
|
|
|
// Check if class is already loaded.
|
|
|
|
if(!classes.inList(name, result))
|
|
|
|
{
|
|
|
|
// Class not loaded. Check if the file exists.
|
|
|
|
char[] fname = classToFile(name);
|
|
|
|
if(MonsterClass.findFile(fname))
|
|
|
|
{
|
|
|
|
// File exists. Load it right away. If the class is
|
|
|
|
// already forward referenced, this will be taken care
|
|
|
|
// of automatically by the load process, through
|
|
|
|
// insertClass. The last parameter makes sure findFile
|
|
|
|
// isn't called twice.
|
|
|
|
result = new MonsterClass(name, fname, false);
|
|
|
|
assert(classes.inList(name));
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(result !is null);
|
|
|
|
assert(result.isParsed);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Find a class given its name. The class must be parsed or a file
|
|
|
|
// must exist which can be parsed, otherwise the function
|
|
|
|
// fails. createScope is also called on the class before it'xs
|
|
|
|
// returned.
|
|
|
|
MonsterClass findClass(Token t) { return findClass(t.str, t.loc); }
|
|
|
|
MonsterClass findClass(char[] name, Floc loc = Floc.init)
|
|
|
|
{
|
|
|
|
MonsterClass res = findParsed(name);
|
|
|
|
|
|
|
|
if(res is null)
|
|
|
|
fail("Failed to find class '" ~ name ~ "'", loc);
|
|
|
|
|
|
|
|
res.requireScope();
|
|
|
|
assert(res.isScoped);
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Gets the index of a class, or inserts a forward reference to it
|
|
|
|
// if cannot be found. Subsequent calls for the same name will
|
|
|
|
// insert the same index, and when/if the class is actually loaded
|
|
|
|
// it will get the same index.
|
|
|
|
CIndex getForwardIndex(char[] name)
|
|
|
|
{
|
|
|
|
MonsterClass mc;
|
|
|
|
mc = findParsed(name);
|
|
|
|
if(mc !is null) return mc.getIndex();
|
|
|
|
|
|
|
|
// Called when an existing forward does not exist
|
|
|
|
void inserter(ref CIndex v)
|
|
|
|
{ v = next++; }
|
|
|
|
|
|
|
|
// Return the index in forwards, or create a new one if none
|
|
|
|
// exists.
|
|
|
|
return forwards.get(name, &inserter);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Returns true if a given class has been inserted into the scope.
|
|
|
|
bool isLoaded(CIndex ci)
|
|
|
|
{
|
|
|
|
return indexList.inList(ci);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// A scope that can contain variables.
|
|
|
|
abstract class VarScope : Scope
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
HashTable!(char[], Variable*) variables;
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
this(Scope last, char[] name)
|
|
|
|
{ super(last, name); }
|
|
|
|
|
|
|
|
override:
|
|
|
|
|
|
|
|
// Find a variable, returns the declaration.
|
|
|
|
Variable* findVar(char[] name)
|
|
|
|
{
|
|
|
|
Variable* vd;
|
|
|
|
|
|
|
|
if(variables.inList(name, vd))
|
|
|
|
return vd;
|
|
|
|
|
|
|
|
return super.findVar(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
void clearId(Token name)
|
|
|
|
{
|
|
|
|
assert(name.str != "");
|
|
|
|
|
|
|
|
Variable *vd;
|
|
|
|
if(variables.inList(name.str, vd))
|
|
|
|
fail(format("Identifier '%s' already declared on line %s (as a variable)",
|
|
|
|
name.str, vd.name.loc),
|
|
|
|
name.loc);
|
|
|
|
|
|
|
|
super.clearId(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Insert a variable, checks if it already exists.
|
|
|
|
void insertVar(Variable* dec)
|
|
|
|
{
|
|
|
|
assert(!isStateCode, "called insertVar in state code");
|
|
|
|
|
|
|
|
// Check for name collisions.
|
|
|
|
clearId(dec.name);
|
|
|
|
|
|
|
|
variables[dec.name.str] = dec;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// A class scope. In addition to variables, they can contain states
|
|
|
|
// and functions, and they keep track of the data segment size.
|
|
|
|
final class ClassScope : VarScope
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
// The class information for this class.
|
|
|
|
MonsterClass cls;
|
|
|
|
|
|
|
|
HashTable!(char[], State*) states;
|
|
|
|
HashTable!(char[], Function*) functions;
|
|
|
|
|
|
|
|
int dataSize; // Data segment size for this class
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
this(Scope last, MonsterClass cl)
|
|
|
|
{
|
|
|
|
cls = cl;
|
|
|
|
super(last, cls.name.str);
|
|
|
|
|
|
|
|
// Connect a class property scope with this scope.
|
|
|
|
nextProp = ClassProperties.singleton;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isClass() { return true; }
|
|
|
|
MonsterClass getClass() { return cls; }
|
|
|
|
|
|
|
|
// Add a variable to the data segment, returns the offset.
|
|
|
|
int addNewDataVar(int varSize)
|
|
|
|
{
|
|
|
|
int tmp = dataSize;
|
|
|
|
|
|
|
|
dataSize += varSize;
|
|
|
|
|
|
|
|
return tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
override void clearId(Token name)
|
|
|
|
{
|
|
|
|
assert(name.str != "");
|
|
|
|
|
|
|
|
Function* fd;
|
|
|
|
State* sd;
|
|
|
|
|
|
|
|
if(functions.inList(name.str, fd))
|
|
|
|
{
|
|
|
|
fail(format("Identifier '%s' already declared on line %s (as a function)",
|
|
|
|
name.str, fd.name.loc),
|
|
|
|
name.loc);
|
|
|
|
}
|
|
|
|
|
|
|
|
if(states.inList(name.str, sd))
|
|
|
|
fail(format("Identifier '%s' already declared on line %s (as a state)",
|
|
|
|
name.str, sd.name.loc),
|
|
|
|
name.loc);
|
|
|
|
|
|
|
|
// Let VarScope handle variables and parent scopes
|
|
|
|
super.clearId(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get total data segment size
|
|
|
|
int getDataSize() { return dataSize; }
|
|
|
|
|
|
|
|
// Insert a state
|
|
|
|
void insertState(State* st)
|
|
|
|
{
|
|
|
|
clearId(st.name);
|
|
|
|
|
|
|
|
st.index = states.length;
|
|
|
|
states[st.name.str] = st;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Insert a function.
|
|
|
|
void insertFunc(Function* fd)
|
|
|
|
{
|
|
|
|
// TODO: First check if we are legally overriding an existing
|
|
|
|
// function. If not, we are creating a new identifier and must
|
|
|
|
// call clearId.
|
|
|
|
clearId(fd.name);
|
|
|
|
|
|
|
|
fd.index = functions.length;
|
|
|
|
|
|
|
|
// Store the function definition
|
|
|
|
functions[fd.name.str] = fd;
|
|
|
|
}
|
|
|
|
|
|
|
|
State* findState(char[] name)
|
|
|
|
{
|
|
|
|
State* st;
|
|
|
|
|
|
|
|
if(states.inList(name, st))
|
|
|
|
return st;
|
|
|
|
|
|
|
|
assert(!isRoot());
|
|
|
|
return parent.findState(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
Function* findFunc(char[] name)
|
|
|
|
{
|
|
|
|
Function* fd;
|
|
|
|
|
|
|
|
if(functions.inList(name, fd))
|
|
|
|
return fd;
|
|
|
|
|
|
|
|
assert(!isRoot());
|
|
|
|
return parent.findFunc(name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// A scope that keeps track of the stack
|
|
|
|
abstract class StackScope : VarScope
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
int locals; // The number of local variables declared in this
|
|
|
|
// scope. These must be pop'ed when the scope exits.
|
|
|
|
int sumLocals;// Current position of the stack relative to the stack
|
|
|
|
// pointer. This is set to zero at the start of the
|
|
|
|
// function, and increased for each local variable
|
|
|
|
// that is added. The difference between sumLocals and
|
|
|
|
// locals is that sumLocals counts ALL local variables
|
|
|
|
// declared in the current function (above the current
|
|
|
|
// point), while locals only counts THIS scope.
|
|
|
|
|
|
|
|
int expStack; // Expression stack. Only eeps track of intra-
|
|
|
|
// expression stack values, and must end up at zero
|
|
|
|
// after each statement.
|
|
|
|
|
|
|
|
public:
|
|
|
|
this(Scope last, char[] name)
|
|
|
|
{
|
|
|
|
super(last, name);
|
|
|
|
|
|
|
|
// Local variable position is inherited
|
|
|
|
assert(!isRoot());
|
|
|
|
sumLocals = parent.getTotLocals;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Allocate a local variable on the stack, and return the offset.
|
|
|
|
// The parameter gives the size of the requested variable in ints (4
|
|
|
|
// bytes.)
|
|
|
|
int addNewLocalVar(int varSize)
|
|
|
|
{
|
|
|
|
assert(expStack == 0);
|
|
|
|
|
|
|
|
locals += varSize;
|
|
|
|
|
|
|
|
int tmp = sumLocals;
|
|
|
|
sumLocals += varSize;
|
|
|
|
|
|
|
|
return tmp;
|
|
|
|
}
|
|
|
|
|
|
|
|
void push(int i) { expStack += i; }
|
|
|
|
void push(Type t) { push(t.getSize); }
|
|
|
|
void pop(int i) { expStack -= i; }
|
|
|
|
void pop(Type t) { pop(t.getSize); }
|
|
|
|
|
|
|
|
// Get the number of local variables in the current scope. In
|
|
|
|
// reality it gives the number of ints. A variable 8 bytes long will
|
|
|
|
// count as two variables.
|
|
|
|
int getLocals() { return locals; }
|
|
|
|
|
|
|
|
// Get the total number of local variables for this function. Used
|
|
|
|
// in return statements and by other jumps that might span several
|
|
|
|
// blocks (break and continue.)
|
|
|
|
int getTotLocals() { return sumLocals; }
|
|
|
|
|
|
|
|
// Get instra-expression stack
|
|
|
|
int getExpStack() { return expStack; }
|
|
|
|
|
|
|
|
// Get total stack position, including expression stack values. This
|
|
|
|
// is used by the array lenght symbol $ to find the array index.
|
|
|
|
int getPos() { return getTotLocals() + getExpStack(); }
|
|
|
|
}
|
|
|
|
|
|
|
|
class FuncScope : StackScope
|
|
|
|
{
|
|
|
|
// Function definition, for function scopes
|
|
|
|
private:
|
|
|
|
Function *fnc;
|
|
|
|
|
|
|
|
public:
|
|
|
|
this(Scope last, Function *fd)
|
|
|
|
{
|
|
|
|
super(last, fd.name.str);
|
|
|
|
fnc = fd;
|
|
|
|
}
|
|
|
|
|
|
|
|
override:
|
|
|
|
void insertLabel(StateLabel *lb)
|
|
|
|
{ assert(0, "cannot insert labels in function scopes"); }
|
|
|
|
|
|
|
|
bool isFunc() { return true; }
|
|
|
|
Function *getFunction() { return fnc; }
|
|
|
|
}
|
|
|
|
|
|
|
|
class CodeScope : StackScope
|
|
|
|
{
|
|
|
|
this(Scope last, CodeBlock cb)
|
|
|
|
{
|
|
|
|
char[] name = "codeblock";
|
|
|
|
if(cb.isState) name = "stateblock";
|
|
|
|
|
|
|
|
super(last, name);
|
|
|
|
}
|
|
|
|
|
|
|
|
this(Scope last, char[] name) { super(last, name) ;}
|
|
|
|
|
|
|
|
bool isCode() { return true; }
|
|
|
|
|
|
|
|
LabelStatement getBreak(char[] name = "") { return parent.getBreak(name); }
|
|
|
|
LabelStatement getContinue(char[] name = "") { return parent.getContinue(name); }
|
|
|
|
}
|
|
|
|
|
|
|
|
// Experimental! Used to recompute the array expression for $. NOT a
|
|
|
|
// permanent solution.
|
|
|
|
class ArrayScope : StackScope
|
|
|
|
{
|
|
|
|
private Expression expArray;
|
|
|
|
|
|
|
|
this(Scope last, Expression arr)
|
|
|
|
{
|
|
|
|
super(last, "arrayscope");
|
|
|
|
expArray = arr;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isArray() { return true; }
|
|
|
|
Expression getArray() { return expArray; }
|
|
|
|
}
|
|
|
|
|
|
|
|
// Base class for scopes that have properties. The instances of this
|
|
|
|
// scope are defined in properties.d
|
|
|
|
abstract class PropertyScope : Scope
|
|
|
|
{
|
|
|
|
this(char[] n) { super(null, n); }
|
|
|
|
|
|
|
|
// Override these in base classes.
|
|
|
|
|
|
|
|
Type getPropType(char[] name, Type oType);
|
|
|
|
void getValue(char[] name, Type oType);
|
|
|
|
bool hasProperty(char[] name);
|
|
|
|
bool isStatic(char[] name, Type oType);
|
|
|
|
|
|
|
|
// Most properties are read-only, but override these for exceptions.
|
|
|
|
bool isLValue(char[] name, Type oType) { return false; }
|
|
|
|
void setValue(char[] name, Type oType) { assert(0); }
|
|
|
|
|
|
|
|
override:
|
|
|
|
final:
|
|
|
|
bool isProperty() { return true; }
|
|
|
|
|
|
|
|
// No need for collision checks
|
|
|
|
void clearId(Token name)
|
|
|
|
{ assert(isRoot()); }
|
|
|
|
|
|
|
|
bool allowRoot() { return true; }
|
|
|
|
|
|
|
|
void getProperty(Token name, Type ownerType, ref Property p)
|
|
|
|
{
|
|
|
|
assert(hasProperty(name));
|
|
|
|
|
|
|
|
p.scp = this;
|
|
|
|
p.name = name.str;
|
|
|
|
p.oType = ownerType;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check if we have a given property.
|
|
|
|
bool hasProperty(Token name) { return hasProperty(name.str); }
|
|
|
|
}
|
|
|
|
|
|
|
|
// A reference to a property, used in VariableExpr.
|
|
|
|
struct Property
|
|
|
|
{
|
|
|
|
PropertyScope scp;
|
|
|
|
char[] name; // Name of the property
|
|
|
|
Type oType; // Type of the owner
|
|
|
|
|
|
|
|
// Get the type of this property
|
|
|
|
Type getType() { return scp.getPropType(name, oType); }
|
|
|
|
|
|
|
|
// Push the value (of type getType) onto the stack. Assumes the
|
|
|
|
// owner (of type oType) has already been pushed onto the stack,
|
|
|
|
// unless the property is static.
|
|
|
|
void getValue() { scp.getValue(name, oType); }
|
|
|
|
|
|
|
|
// Pops a value (of type getType) off the stack and sets the
|
|
|
|
// property. Can only be called for lvalues.
|
|
|
|
void setValue() { assert(isLValue); scp.setValue(name, oType); }
|
|
|
|
|
|
|
|
// Can we write to this property?
|
|
|
|
bool isLValue() { return scp.isLValue(name, oType); }
|
|
|
|
|
|
|
|
// Is this property static? If it is, we do not have to push the
|
|
|
|
// owner onto the stack before using the property.
|
|
|
|
bool isStatic() { return scp.isStatic(name, oType); }
|
|
|
|
}
|
|
|
|
|
|
|
|
// Scope inside of loops. Handles break and continue, loop labels, and
|
|
|
|
// some stack stuff.
|
|
|
|
class LoopScope : CodeScope
|
|
|
|
{
|
|
|
|
private:
|
|
|
|
LabelStatement breakLabel, continueLabel;
|
|
|
|
Token loopName; // Loop label name.
|
|
|
|
int loopStack = -1; // Stack position of the array index. Only used in
|
|
|
|
// foreach loops.
|
|
|
|
bool isForeach; // Redundant variable used for integrity checking only
|
|
|
|
|
|
|
|
// Set the label and set up a nice name
|
|
|
|
this(Scope last, char[] name, Token label)
|
|
|
|
{
|
|
|
|
// Include the label name in the scope name
|
|
|
|
if(label.str != "")
|
|
|
|
name ~= ":" ~ label.str;
|
|
|
|
|
|
|
|
super(last, name);
|
|
|
|
|
|
|
|
if(label.str != "")
|
|
|
|
{
|
|
|
|
// This loop has a label, so set it up
|
|
|
|
clearId(label);
|
|
|
|
loopName = label;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
|
|
|
this(Scope last, ForStatement fs)
|
|
|
|
{ this(last, "for-loop", fs.labelName); }
|
|
|
|
this(Scope last, ForeachStatement fs)
|
|
|
|
{
|
|
|
|
this(last, "foreach-loop", fs.labelName);
|
|
|
|
isForeach = true;
|
|
|
|
}
|
|
|
|
this(Scope last, DoWhileStatement fs)
|
|
|
|
{
|
|
|
|
this(last, "do-while-loop", fs.labelName);
|
|
|
|
stackPoint();
|
|
|
|
}
|
|
|
|
this(Scope last, WhileStatement fs)
|
|
|
|
{
|
|
|
|
this(last, "while-loop", fs.labelName);
|
|
|
|
stackPoint();
|
|
|
|
}
|
|
|
|
|
|
|
|
bool isLoop() { return true; }
|
|
|
|
|
|
|
|
// Called when the stack is set up to the level of the loop
|
|
|
|
// interior, after loop variables and such are declared. This
|
|
|
|
// function should only be called for for-loops and foreach-loops,
|
|
|
|
// and will make sure that loop variables are not popped by break
|
|
|
|
// and continue statements.
|
|
|
|
void stackPoint()
|
|
|
|
{
|
|
|
|
assert(breakLabel is null && continueLabel is null && loopStack == -1,
|
|
|
|
"do not call stackPoint multiple times");
|
|
|
|
|
|
|
|
if(isForeach) loopStack = getTotLocals() - 1;
|
|
|
|
breakLabel = new LabelStatement(getTotLocals());
|
|
|
|
continueLabel = new LabelStatement(getTotLocals());
|
|
|
|
}
|
|
|
|
|
|
|
|
override void clearId(Token name)
|
|
|
|
{
|
|
|
|
assert(name.str != "");
|
|
|
|
|
|
|
|
// Check for loop labels as well
|
|
|
|
if(loopName.str == name.str)
|
|
|
|
fail(format("Identifier %s is a loop label on line %s and cannot be redefined.",
|
|
|
|
name.str, loopName.loc),
|
|
|
|
name.loc);
|
|
|
|
|
|
|
|
super.clearId(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get the break or continue label for the given named loop, or the
|
|
|
|
// innermost loop if name is empty or omitted.
|
|
|
|
LabelStatement getBreak(char[] name = "")
|
|
|
|
{
|
|
|
|
if(name == "" || name == loopName.str)
|
|
|
|
return breakLabel;
|
|
|
|
|
|
|
|
return parent.getBreak(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
LabelStatement getContinue(char[] name = "")
|
|
|
|
{
|
|
|
|
if(name == "" || name == loopName.str)
|
|
|
|
return continueLabel;
|
|
|
|
|
|
|
|
return parent.getContinue(name);
|
|
|
|
}
|
|
|
|
|
|
|
|
int getLoopStack()
|
|
|
|
{
|
|
|
|
assert(loopStack != -1 && isForeach,
|
|
|
|
"getLoopStack called for non-foreach scope");
|
|
|
|
return loopStack;
|
|
|
|
}
|
|
|
|
}
|