mirror of
https://gitlab.com/OpenMW/openmw.git
synced 2025-01-25 06:35:30 +00:00
de5a07ee71
git-svn-id: https://openmw.svn.sourceforge.net/svnroot/openmw/trunk@138 ea6a568a-9f4f-0410-981a-c910a81bb256
1419 lines
41 KiB
D
1419 lines
41 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 (mclass.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.vm.mclass;
|
|
|
|
import monster.compiler.functions;
|
|
import monster.compiler.types;
|
|
import monster.compiler.scopes;
|
|
import monster.compiler.tokenizer;
|
|
import monster.compiler.statement;
|
|
import monster.compiler.variables;
|
|
import monster.compiler.states;
|
|
import monster.compiler.structs;
|
|
import monster.compiler.block;
|
|
import monster.compiler.enums;
|
|
|
|
import monster.vm.codestream;
|
|
import monster.vm.idlefunction;
|
|
import monster.vm.arrays;
|
|
import monster.vm.error;
|
|
import monster.vm.vm;
|
|
import monster.vm.stack;
|
|
import monster.vm.thread;
|
|
import monster.vm.mobject;
|
|
|
|
import monster.util.flags;
|
|
import monster.util.string;
|
|
import monster.util.list;
|
|
import monster.util.freelist;
|
|
|
|
import std.string;
|
|
import std.stdio;
|
|
import std.stream;
|
|
|
|
typedef void *MClass; // Pointer to C++ equivalent of MonsterClass.
|
|
|
|
typedef int CIndex;
|
|
|
|
enum CFlags
|
|
{
|
|
None = 0x00, // Initial value
|
|
|
|
Parsed = 0x01, // Class has been parsed
|
|
Scoped = 0x02, // Class has been inserted into the scope
|
|
Resolved = 0x04, // Class body has been resolved
|
|
Compiled = 0x08, // Class body has been compiled
|
|
InScope = 0x10, // We are currently inside the createScope
|
|
// function
|
|
Module = 0x20, // This is a module, not a class
|
|
Singleton = 0x40, // This is a singleton. Also set for modules.
|
|
Abstract = 0x80, // No objects can be created from this class
|
|
}
|
|
|
|
// The class that handles 'classes' in Monster.
|
|
final class MonsterClass
|
|
{
|
|
/***********************************************
|
|
* *
|
|
* Static functions *
|
|
* *
|
|
***********************************************/
|
|
|
|
static bool canParse(TokenArray tokens)
|
|
{
|
|
return
|
|
Block.isNext(tokens, TT.Class) ||
|
|
Block.isNext(tokens, TT.Singleton) ||
|
|
Block.isNext(tokens, TT.Abstract) ||
|
|
Block.isNext(tokens, TT.Module);
|
|
}
|
|
|
|
static uint getTotalObjects() { return allObjects.length; }
|
|
|
|
// Sets up the Object class, which is the parent of all other
|
|
// classes.
|
|
static void initialize()
|
|
{
|
|
assert(baseObject is null,
|
|
"MonsterClass.initialize() run more than once");
|
|
assert(global !is null);
|
|
|
|
// The Object class is empty
|
|
baseObject = vm.loadString("abstract class Object;", "Object");
|
|
assert(baseObject !is null);
|
|
|
|
// Set up the class. createScope() etc will make a special case
|
|
// for us when it detects that it is running on baseObject.
|
|
baseObject.requireCompile();
|
|
}
|
|
|
|
private static MonsterClass baseObject;
|
|
|
|
// Returns the class called 'Object'
|
|
static MonsterClass getObject()
|
|
{
|
|
return baseObject;
|
|
}
|
|
|
|
final:
|
|
|
|
/*******************************************************
|
|
* *
|
|
* Variables *
|
|
* *
|
|
*******************************************************/
|
|
|
|
// Index within the parent tree. This might become a list at some
|
|
// point.
|
|
int treeIndex;
|
|
|
|
Token name; // Class name and location
|
|
|
|
CIndex gIndex; // Global index of this class
|
|
|
|
ClassScope sc;
|
|
PackageScope pack;
|
|
|
|
ObjectType objType; // Type for objects of this class
|
|
Type classType; // Type for class references to this class
|
|
|
|
// Pointer to the C++ wrapper class, if any. Could be used for other
|
|
// wrapper languages at well, but only one at a time.
|
|
MClass cppClassPtr;
|
|
|
|
private:
|
|
// List of objects of this class. Includes objects of all subclasses
|
|
// as well.
|
|
PointerList objects;
|
|
|
|
Flags!(CFlags) flags;
|
|
|
|
public:
|
|
|
|
// Create a class belonging to the given package scope. Do not call
|
|
// this yourself, use vm.load* to load classes.
|
|
this(PackageScope psc = null)
|
|
{
|
|
assert(psc !is null,
|
|
"Don't create MonsterClasses directly, use vm.load()");
|
|
|
|
pack = psc;
|
|
}
|
|
|
|
/*******************************************************
|
|
* *
|
|
* Management of member functions *
|
|
* *
|
|
*******************************************************/
|
|
|
|
// Bind a delegate to the name of a native function. TODO: Add
|
|
// optional signature check here at some point?
|
|
void bind(char[] name, dg_callback nf)
|
|
{ bind_locate(name, FuncType.NativeDDel).natFunc_dg = nf; }
|
|
|
|
// Same as above, but binds a function instead of a delegate.
|
|
void bind(char[] name, fn_callback nf)
|
|
{ bind_locate(name, FuncType.NativeDFunc).natFunc_fn = nf; }
|
|
|
|
// Used for C functions
|
|
void bind_c(char[] name, c_callback nf)
|
|
{ bind_locate(name, FuncType.NativeCFunc).natFunc_c = nf; }
|
|
|
|
// Bind an idle function
|
|
void bind(char[] name, IdleFunction idle)
|
|
{ bind_locate(name, FuncType.Idle).idleFunc = idle; }
|
|
|
|
void bindT(alias func)(char[] name="")
|
|
{
|
|
// Get the name from the alias parameter directly, if not
|
|
// specified.
|
|
if(name == "")
|
|
// Sort of a hack. func.stringof won't work (parses as a
|
|
// function call). (&func).stringof parses as "& funcname",
|
|
// but this could be implementation specific.
|
|
name = ((&func).stringof)[2..$];
|
|
|
|
// Let the Function handle the rest
|
|
findFunction(name).bindT!(func)();
|
|
}
|
|
|
|
// Find a function by index. Used internally, and works for all
|
|
// function types.
|
|
Function *findFunction(int index)
|
|
{
|
|
requireScope();
|
|
assert(index >=0 && index < functions.length);
|
|
assert(functions[index] !is null);
|
|
return functions[index];
|
|
}
|
|
|
|
// Find a virtual function by index. ctree is the tree index of the
|
|
// class where the function is defined, findex is the intra-class
|
|
// function index.
|
|
Function *findVirtualFunc(int ctree, int findex)
|
|
{
|
|
requireScope();
|
|
assert(ctree >= 0 && ctree <= treeIndex);
|
|
assert(findex >= 0 && findex < virtuals[ctree].length);
|
|
assert(virtuals[ctree][findex] !is null);
|
|
|
|
return virtuals[ctree][findex];
|
|
}
|
|
|
|
// Find a given callable function, looking up parent classes if
|
|
// necessary.
|
|
Function *findFunction(char[] name)
|
|
{
|
|
requireScope();
|
|
|
|
// Get the function from the scope
|
|
auto ln = sc.lookupName(name);
|
|
|
|
if(!ln.isFunc)
|
|
fail("Function '" ~ name ~ "' not found.");
|
|
|
|
auto fn = ln.func;
|
|
|
|
if(!fn.isNormal && !fn.isNative)
|
|
{
|
|
// Being here is always bad. Now we just need to find
|
|
// out what error message to give.
|
|
if(fn.isAbstract)
|
|
fail(name ~ " is abstract.");
|
|
|
|
if(fn.isIdle)
|
|
fail("Idle function " ~ name ~
|
|
" cannot be called from native code.");
|
|
assert(0);
|
|
}
|
|
|
|
assert(fn !is null);
|
|
return fn;
|
|
}
|
|
|
|
/*******************************************************
|
|
* *
|
|
* Binding of native constructors *
|
|
* *
|
|
*******************************************************/
|
|
|
|
// bindConst binds a native function that is run on all new
|
|
// objects. It is executed before the constructor defined in script
|
|
// code (if any.)
|
|
|
|
// bindNew binds a function that is run on all objects created
|
|
// within script code (with the 'new' expression or the 'clone'
|
|
// property), but not on objects that are created in native code
|
|
// through createObject/createClone. This is handy when you want to
|
|
// bind with a native class, and want to be able to create objects
|
|
// in both places. It's executed before both bindConst and the
|
|
// script constructor.
|
|
|
|
void bindConst(dg_callback nf)
|
|
{
|
|
assert(natConst.ftype == FuncType.Native,
|
|
"Cannot set native constructor for " ~ toString ~ ": already set");
|
|
natConst.ftype = FuncType.NativeDDel;
|
|
natConst.natFunc_dg = nf;
|
|
}
|
|
|
|
void bindConst(fn_callback nf)
|
|
{
|
|
assert(natConst.ftype == FuncType.Native,
|
|
"Cannot set native constructor for " ~ toString ~ ": already set");
|
|
natConst.ftype = FuncType.NativeDFunc;
|
|
natConst.natFunc_fn = nf;
|
|
}
|
|
|
|
void bindConst_c(c_callback nf)
|
|
{
|
|
assert(natConst.ftype == FuncType.Native,
|
|
"Cannot set native constructor for " ~ toString ~ ": already set");
|
|
natConst.ftype = FuncType.NativeCFunc;
|
|
natConst.natFunc_c = nf;
|
|
}
|
|
|
|
void bindNew(dg_callback nf)
|
|
{
|
|
assert(natNew.ftype == FuncType.Native,
|
|
"Cannot set native constructor for " ~ toString ~ ": already set");
|
|
natNew.ftype = FuncType.NativeDDel;
|
|
natNew.natFunc_dg = nf;
|
|
}
|
|
|
|
void bindNew(fn_callback nf)
|
|
{
|
|
assert(natNew.ftype == FuncType.Native,
|
|
"Cannot set native constructor for " ~ toString ~ ": already set");
|
|
natNew.ftype = FuncType.NativeDFunc;
|
|
natNew.natFunc_fn = nf;
|
|
}
|
|
|
|
void bindNew_c(c_callback nf)
|
|
{
|
|
assert(natNew.ftype == FuncType.Native,
|
|
"Cannot set native constructor for " ~ toString ~ ": already set");
|
|
natNew.ftype = FuncType.NativeCFunc;
|
|
natNew.natFunc_c = nf;
|
|
}
|
|
|
|
|
|
/*******************************************************
|
|
* *
|
|
* Management of member variables *
|
|
* *
|
|
*******************************************************/
|
|
|
|
Variable* findVariable(char[] name)
|
|
{
|
|
requireScope();
|
|
|
|
auto ln = sc.lookupName(name);
|
|
|
|
if(!ln.isVar)
|
|
fail("Variable " ~ name ~ " not found");
|
|
|
|
Variable *vb = ln.var;
|
|
|
|
assert(vb.vtype == VarType.Class);
|
|
|
|
return vb;
|
|
}
|
|
|
|
|
|
/*******************************************************
|
|
* *
|
|
* Management of member states *
|
|
* *
|
|
*******************************************************/
|
|
|
|
State* findState(char[] name)
|
|
{
|
|
requireScope();
|
|
|
|
auto ln = sc.lookupName(name);
|
|
if(!ln.isState)
|
|
fail("State " ~ name ~ " not found");
|
|
|
|
State *st = ln.state;
|
|
|
|
return st;
|
|
}
|
|
|
|
// Look up state and label based on indices. We allow lindex to be
|
|
// -1, in which case a null label is returned.
|
|
StateLabelPair findState(int sindex, int lindex)
|
|
{
|
|
requireScope();
|
|
assert(sindex >=0 && sindex < states.length);
|
|
|
|
StateLabelPair res;
|
|
res.state = states[sindex];
|
|
|
|
assert(res.state !is null);
|
|
|
|
if(lindex == -1)
|
|
res.label = null;
|
|
else
|
|
{
|
|
assert(lindex >= 0 && lindex < res.state.labelList.length);
|
|
res.label = res.state.labelList[lindex];
|
|
assert(res.label !is null);
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
// Find a state and a given label within it. Fails if it is not
|
|
// found.
|
|
StateLabelPair findState(char[] name, char[] label)
|
|
{
|
|
requireScope();
|
|
assert(label != "");
|
|
|
|
StateLabelPair pr;
|
|
pr.state = findState(name);
|
|
pr.label = pr.state.findLabel(label);
|
|
|
|
if(pr.label is null)
|
|
fail("State " ~ name ~ " does not have a label named " ~ label);
|
|
|
|
return pr;
|
|
}
|
|
|
|
|
|
/*******************************************************
|
|
* *
|
|
* Object managament *
|
|
* *
|
|
*******************************************************/
|
|
|
|
// Loop through all objects of this type
|
|
int opApply(int delegate(ref MonsterObject v) del)
|
|
{
|
|
int dg(ref void *vp)
|
|
{
|
|
auto mop = cast(MonsterObject*)vp;
|
|
return del(*mop);
|
|
}
|
|
return objects.opApply(&dg);
|
|
}
|
|
|
|
// Get the first object in the list for this class
|
|
MonsterObject* getFirst()
|
|
{ return cast(MonsterObject*)objects.getHead().value; }
|
|
|
|
MonsterObject* getNext(MonsterObject *ob)
|
|
{
|
|
auto iter = (*getListPtr(ob, treeIndex)).getNext();
|
|
if(iter is null) return null;
|
|
return cast(MonsterObject*)iter.value;
|
|
}
|
|
|
|
// Get the singleton object
|
|
MonsterObject* getSing()
|
|
{
|
|
if(!isSingleton)
|
|
fail("Class is not a singleton: " ~ name.str);
|
|
requireCompile();
|
|
assert(singObj !is null);
|
|
return singObj;
|
|
}
|
|
alias getSing getSingleton;
|
|
|
|
MonsterObject* createObject(bool callConst = true)
|
|
{ return createClone(null, callConst); }
|
|
|
|
// Call constructors on an object. If scriptNew is true, also call
|
|
// the natNew bindings (if any)
|
|
void callConstOn(MonsterObject *obj, bool scriptNew = false)
|
|
{
|
|
assert(obj.cls is this);
|
|
|
|
// Needed to make sure execute() exits when the constructor is
|
|
// done.
|
|
if(cthread !is null)
|
|
cthread.fstack.pushExt("callConst");
|
|
|
|
// Call constructors
|
|
foreach(c; tree)
|
|
{
|
|
// Call 'new' callback if the object was created in script
|
|
if(scriptNew && c.natNew.ftype != FuncType.Native)
|
|
c.natNew.call(obj);
|
|
|
|
// Call native constructor
|
|
if(c.natConst.ftype != FuncType.Native)
|
|
c.natConst.call(obj);
|
|
|
|
// Call script constructor
|
|
if(c.scptConst !is null)
|
|
c.scptConst.fn.call(obj);
|
|
}
|
|
|
|
if(cthread !is null)
|
|
cthread.fstack.pop();
|
|
}
|
|
|
|
// Get the whole allocated buffer belonging to this object
|
|
private int[] getDataBlock(MonsterObject *obj)
|
|
{
|
|
assert(obj !is null);
|
|
assert(obj.cls is this);
|
|
return (cast(int*)obj.data.ptr)[0..totalData.length];
|
|
}
|
|
|
|
private vpIter getListPtr(MonsterObject *obj, int i)
|
|
{
|
|
auto ep = cast(ExtraData*) &obj.data[i][$-MonsterObject.exSize];
|
|
return &ep.node;
|
|
}
|
|
|
|
// Create a new object based on an existing object
|
|
MonsterObject* createClone(MonsterObject *source, bool callConst = true)
|
|
{
|
|
requireCompile();
|
|
|
|
if(isModule && singObj !is null)
|
|
fail("Cannot create instances of module " ~ name.str);
|
|
|
|
MonsterObject *obj = allObjects.getNew();
|
|
|
|
obj.state = null;
|
|
obj.cls = this;
|
|
|
|
// Allocate the object data segment from a freelist
|
|
int[] odata = Buffers.getInt(totalData.length);
|
|
|
|
// Copy the data, either from the class (in case of new objects)
|
|
// or from the source (when cloning.)
|
|
if(source !is null)
|
|
{
|
|
assert(!isAbstract);
|
|
assert(source.cls is this);
|
|
|
|
assert(source.data.length == tree.length);
|
|
|
|
// Copy data from the object
|
|
odata[] = getDataBlock(source);
|
|
}
|
|
else
|
|
{
|
|
if(isAbstract)
|
|
fail("Cannot create objects from abstract class " ~ name.str);
|
|
|
|
// Copy init values from the class
|
|
odata[] = totalData[];
|
|
}
|
|
|
|
// Use this to get subslices of the data segment
|
|
int[] slice = odata;
|
|
int[] get(int ints)
|
|
{
|
|
assert(ints <= slice.length);
|
|
int[] res = slice[0..ints];
|
|
slice = slice[ints..$];
|
|
return res;
|
|
}
|
|
|
|
// The beginning of the block is used for the int data[][]
|
|
// array.
|
|
obj.data = cast(int[][]) get(iasize*tree.length);
|
|
|
|
// Set up the a slice for the data segment of each class
|
|
foreach(i, c; tree)
|
|
{
|
|
// Just get the slice - the actual data is already set up.
|
|
obj.data[i] = get(c.dataSize + MonsterObject.exSize);
|
|
|
|
// Insert ourselves into the per-class list. We've already
|
|
// allocated size for a node, we just have to add it to the
|
|
// list.
|
|
auto node = getListPtr(obj, i);
|
|
node.value = obj; // Store the object pointer
|
|
c.objects.insertNode(node);
|
|
}
|
|
|
|
// At this point we should have used up the entire slice
|
|
assert(slice.length == 0);
|
|
|
|
// Set the same state as the source
|
|
if(source !is null)
|
|
obj.setState(source.state, null);
|
|
else
|
|
// Use the default state and label
|
|
obj.setState(defState, defLabel);
|
|
|
|
// Make sure that getDataBlock works
|
|
assert(getDataBlock(obj).ptr == odata.ptr &&
|
|
getDataBlock(obj).length == odata.length);
|
|
|
|
// Call constructors
|
|
if(callConst)
|
|
callConstOn(obj);
|
|
|
|
return obj;
|
|
}
|
|
|
|
// Free an object and its thread
|
|
void deleteObject(MonsterObject *obj)
|
|
{
|
|
assert(obj.cls is this);
|
|
|
|
if(isModule)
|
|
fail("Cannot delete instances of module " ~ name.str);
|
|
|
|
// Shut down any active code in the thread
|
|
obj.clearState();
|
|
|
|
// clearState should also clear the thread
|
|
assert(obj.sthread is null);
|
|
|
|
// This effectively marks the object as dead
|
|
obj.cls = null;
|
|
|
|
foreach_reverse(i, c; tree)
|
|
{
|
|
// TODO: Call destructors here
|
|
|
|
// Remove from class list
|
|
c.objects.removeNode(getListPtr(obj,i));
|
|
}
|
|
|
|
// Return it to the freelist
|
|
allObjects.remove(obj);
|
|
|
|
// Return the data segment
|
|
Buffers.free(getDataBlock(obj));
|
|
}
|
|
|
|
|
|
/*******************************************************
|
|
* *
|
|
* Misc. functions *
|
|
* *
|
|
*******************************************************/
|
|
|
|
bool isParsed() { return flags.has(CFlags.Parsed); }
|
|
bool isScoped() { return flags.has(CFlags.Scoped); }
|
|
bool isResolved() { return flags.has(CFlags.Resolved); }
|
|
bool isCompiled() { return flags.has(CFlags.Compiled); }
|
|
|
|
bool isSingleton() { return flags.has(CFlags.Singleton); }
|
|
bool isModule() { return flags.has(CFlags.Module); }
|
|
bool isAbstract() { return flags.has(CFlags.Abstract); }
|
|
|
|
// Call whenever you require this function to have its scope in
|
|
// order. If the scope is missing, this will call createScope if
|
|
// possible, or fail if the class has not been loaded.
|
|
void requireScope()
|
|
{
|
|
if(isScoped) return;
|
|
if(!isParsed)
|
|
fail("Cannot use class '" ~ name.str ~
|
|
"': not found or forward reference",
|
|
name.loc);
|
|
|
|
createScope();
|
|
}
|
|
|
|
// Called whenever we need a completely compiled class, for example
|
|
// when creating an object. Compiles the class if it isn't done
|
|
// already.
|
|
void requireCompile() { if(!isCompiled) compileBody(); }
|
|
|
|
// Check if this class is a child of cls.
|
|
bool childOf(MonsterClass cls)
|
|
{
|
|
requireScope();
|
|
|
|
int ind = cls.treeIndex;
|
|
// If 'cls' is part of our parent tree, then we are a child.
|
|
return ind < tree.length && tree[ind] is cls;
|
|
}
|
|
|
|
// Check if this class is a parent of cls.
|
|
bool parentOf(MonsterClass cls)
|
|
{ return cls.childOf(this); }
|
|
|
|
// Ditto for a given object
|
|
bool parentOf(MonsterObject *obj)
|
|
{ return obj.cls.childOf(this); }
|
|
|
|
// Get the tree-index of a given parent class
|
|
int upcast(MonsterClass mc)
|
|
{
|
|
requireScope();
|
|
|
|
int ind = mc.treeIndex;
|
|
if(ind < tree.length && tree[ind] is mc)
|
|
return ind;
|
|
|
|
fail("Cannot upcast " ~ toString ~ " to " ~ mc.toString);
|
|
}
|
|
|
|
// Get the given class from a tree index
|
|
MonsterClass upcast(int ind)
|
|
{
|
|
requireScope();
|
|
|
|
if(ind < tree.length) return tree[ind];
|
|
|
|
fail("Cannot upcast " ~toString ~ " to index " ~ .toString(ind));
|
|
}
|
|
|
|
// Get the global index of this class
|
|
CIndex getIndex() { requireScope(); return gIndex; }
|
|
int getTreeIndex() { requireScope(); return treeIndex; }
|
|
char[] getName() { assert(name.str != ""); return name.str; }
|
|
char[] toString() { return getName(); }
|
|
uint numObjects() { return objects.length; }
|
|
|
|
// Used internally. Use the string version below instead if you want
|
|
// to change the default state of a class.
|
|
void setDefaultState(State *st, StateLabel *lb)
|
|
{
|
|
defState = st;
|
|
defLabel = lb;
|
|
}
|
|
|
|
// Set the initial state and label for this class. This will affect
|
|
// all newly created objects, but not cloned objects.
|
|
void setDefaultState(char[] st, char[] lb="")
|
|
{
|
|
if(lb == "")
|
|
{
|
|
setDefaultState(findState(st), null);
|
|
return;
|
|
}
|
|
auto pr = findState(st, lb);
|
|
setDefaultState(pr.state, pr.label);
|
|
}
|
|
|
|
// Converts a stream to tokens and parses it.
|
|
void parse(Stream str, char[] fname, int bom)
|
|
{
|
|
assert(str !is null);
|
|
TokenArray tokens = tokenizeStream(fname, str, bom);
|
|
parse(tokens, fname);
|
|
}
|
|
|
|
// Parses a list of tokens, and do other setup.
|
|
void parse(ref TokenArray tokens, char[] fname)
|
|
{
|
|
assert(!isParsed(), "parse() called on a parsed class " ~ name.str);
|
|
|
|
alias Block.isNext isNext;
|
|
|
|
natConst.ftype = FuncType.Native;
|
|
natConst.name.str = "native constructor";
|
|
natConst.owner = this;
|
|
natNew.ftype = FuncType.Native;
|
|
natNew.name.str = "native 'new' callback";
|
|
natNew.owner = this;
|
|
|
|
// Parse keywords. Uses a few local variables, so let's start a
|
|
// block.
|
|
{
|
|
assert(tokens.length > 0);
|
|
Floc loc = tokens[0].loc;
|
|
|
|
// Check for / set a given keyword flag. Returns true if it
|
|
// was NOT found.
|
|
bool check(TT type, ref bool flag)
|
|
{
|
|
Token tok;
|
|
if(isNext(tokens, type, tok))
|
|
{
|
|
if(flag)
|
|
fail("Keyword '" ~ tok.str ~
|
|
"' was specified twice", tok.loc);
|
|
flag = true;
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
// Flags for the various keywords
|
|
bool absSet;
|
|
bool classSet;
|
|
bool moduleSet;
|
|
bool singleSet;
|
|
bool something;
|
|
|
|
while(true)
|
|
{
|
|
if(check(TT.Class, classSet) &&
|
|
check(TT.Abstract, absSet) &&
|
|
check(TT.Module, moduleSet) &&
|
|
check(TT.Singleton, singleSet))
|
|
// Abort if nothing was found this round
|
|
break;
|
|
|
|
// If we get here at least once, then one of the keywords
|
|
// were present.
|
|
something = true;
|
|
}
|
|
|
|
// Was anything found?
|
|
if(!something)
|
|
fail("File must begin with a class or module statement", tokens);
|
|
|
|
// Module and singleton imply class as well
|
|
if(moduleSet || singleSet)
|
|
classSet = true;
|
|
|
|
// Do error checking
|
|
if(moduleSet && singleSet)
|
|
fail("Cannot specify both 'module' and 'singleton' (module implies singleton)",
|
|
loc);
|
|
if(!classSet)
|
|
fail("Class must start with one of 'class', 'singleton' or 'module'",
|
|
loc);
|
|
|
|
// Module implies singleton
|
|
if(moduleSet)
|
|
singleSet = true;
|
|
|
|
// But singletons cannot be abstract
|
|
if(singleSet && absSet)
|
|
fail("Modules and singletons cannot be abstract", loc);
|
|
|
|
// Process flags
|
|
if(singleSet) flags.set(CFlags.Singleton);
|
|
if(moduleSet) flags.set(CFlags.Module);
|
|
if(absSet) flags.set(CFlags.Abstract);
|
|
}
|
|
|
|
if(!isNext(tokens, TT.Identifier, name))
|
|
fail("Class statement expected identifier", tokens);
|
|
|
|
// Module implies singleton
|
|
assert(isSingleton || !isModule);
|
|
assert(!isSingleton || !isAbstract);
|
|
|
|
// Insert ourselves into the package scope. This will also
|
|
// resolve forward references to this class, if any.
|
|
pack.insertClass(this);
|
|
|
|
// Get the parent classes, if any
|
|
if(isNext(tokens, TT.Colon))
|
|
{
|
|
if(isModule)
|
|
fail("Inheritance not allowed for modules.");
|
|
|
|
Token pName;
|
|
do
|
|
{
|
|
if(!isNext(tokens, TT.Identifier, pName))
|
|
fail("Expected parent class identifier", tokens);
|
|
|
|
parentNames ~= pName;
|
|
}
|
|
while(isNext(tokens, TT.Comma));
|
|
}
|
|
|
|
isNext(tokens, TT.Semicolon);
|
|
|
|
if(parents.length > 1)
|
|
fail("Multiple inheritance is currently not supported", name.loc);
|
|
|
|
// Parse the rest of the file
|
|
while(!isNext(tokens, TT.EOF)) store(tokens);
|
|
|
|
// The tokenizer shouldn't allow more tokens after this point
|
|
assert(tokens.length == 0, "found tokens after end of file");
|
|
|
|
flags.set(CFlags.Parsed);
|
|
}
|
|
|
|
private:
|
|
|
|
/*******************************************************
|
|
* *
|
|
* Private variables *
|
|
* *
|
|
*******************************************************/
|
|
|
|
// Contains the entire class tree for this class, always with
|
|
// ourselves as the last entry. Any class in the list is always
|
|
// preceded by all the classes it inherits from.
|
|
MonsterClass tree[];
|
|
|
|
// List of variables and functions declared in this class, ordered
|
|
// by index.
|
|
Function* functions[];
|
|
Variable* vars[];
|
|
State* states[];
|
|
|
|
// Singleton object - used for singletons and modules only.
|
|
MonsterObject *singObj;
|
|
|
|
// Function table translation list. Same length as tree[]. For each
|
|
// class in the parent tree, this list holds a list equivalent to
|
|
// the functions[] list in that class. The difference is that all
|
|
// overrided functions have been replaced by their successors.
|
|
Function*[][] virtuals;
|
|
|
|
// Default state and label
|
|
State *defState = null;
|
|
StateLabel *defLabel = null;
|
|
|
|
// The total data segment that's assigned to each object. It
|
|
// includes the data segment of all parent objects and some
|
|
// additional internal data.
|
|
int[] totalData;
|
|
|
|
// Data segment size for *this* class, not including parents or
|
|
// extra information.
|
|
public int dataSize;
|
|
|
|
// Total data, sliced up to match the class tree
|
|
int[][] totalSliced;
|
|
|
|
// Direct parents of this class
|
|
public MonsterClass parents[];
|
|
Token parentNames[];
|
|
|
|
// Used at compile time
|
|
VarDeclStatement[] vardecs;
|
|
FuncDeclaration[] funcdecs;
|
|
StateDeclaration[] statedecs;
|
|
StructDeclaration[] structdecs;
|
|
EnumDeclaration[] enumdecs;
|
|
ImportStatement[] imports;
|
|
ClassVarSet[] varsets;
|
|
|
|
// Native constructors, if any
|
|
Function natConst, natNew;
|
|
|
|
// Script constructor, if any
|
|
public Constructor scptConst;
|
|
|
|
/*******************************************************
|
|
* *
|
|
* Various private functions *
|
|
* *
|
|
*******************************************************/
|
|
|
|
// Helper function for the bind() variants
|
|
Function* bind_locate(char[] name, FuncType ft)
|
|
{
|
|
requireScope();
|
|
|
|
// Look the function up in the scope
|
|
auto ln = sc.lookupName(name);
|
|
auto fn = ln.func;
|
|
|
|
if(!ln.isFunc)
|
|
fail("Cannot bind to '" ~ name ~ "': no such function");
|
|
|
|
if(ft == FuncType.Idle)
|
|
{
|
|
if(!fn.isIdle())
|
|
fail("Cannot bind to non-idle function '" ~ name ~ "'");
|
|
}
|
|
else
|
|
{
|
|
if(!fn.isNative())
|
|
fail("Cannot bind to non-native function '" ~ name ~ "'");
|
|
}
|
|
|
|
// Check that the function really belongs to this class. We cannot
|
|
// bind functions belonging to parent classes.
|
|
assert(fn.owner !is null);
|
|
if(fn.owner !is this)
|
|
fail("Cannot bind to function " ~ fn.toString() ~
|
|
" - it is not a direct member of class " ~ toString());
|
|
|
|
fn.ftype = ft;
|
|
|
|
return fn;
|
|
}
|
|
|
|
|
|
/*******************************************************
|
|
* *
|
|
* Compiler-related private functions *
|
|
* *
|
|
*******************************************************/
|
|
|
|
// Identify what kind of block the given set of tokens represent,
|
|
// parse them, and store it in the appropriate list;
|
|
void store(ref TokenArray toks)
|
|
{
|
|
if(FuncDeclaration.canParse(toks))
|
|
{
|
|
auto fd = new FuncDeclaration;
|
|
funcdecs ~= fd;
|
|
fd.parse(toks);
|
|
}
|
|
else if(Constructor.canParse(toks))
|
|
{
|
|
auto fd = new Constructor;
|
|
if(scptConst !is null)
|
|
fail("Class " ~ name.str ~ " cannot have more than one constructor", toks[0].loc);
|
|
scptConst = fd;
|
|
fd.parse(toks);
|
|
}
|
|
else if(ClassVarSet.canParse(toks))
|
|
{
|
|
auto cv = new ClassVarSet;
|
|
cv.parse(toks);
|
|
|
|
// Check if this variable is already set in this class
|
|
foreach(ocv; varsets)
|
|
{
|
|
if(cv.isState && ocv.isState)
|
|
fail(format("State already set on line %s",
|
|
ocv.loc.line), cv.loc);
|
|
else if(ocv.name.str == cv.name.str)
|
|
fail(format("Variable %s is already set on line %s",
|
|
cv.name.str, ocv.loc.line),
|
|
cv.loc);
|
|
}
|
|
|
|
varsets ~= cv;
|
|
}
|
|
else if(VarDeclStatement.canParse(toks))
|
|
{
|
|
auto vd = new VarDeclStatement;
|
|
vd.parse(toks);
|
|
vardecs ~= vd;
|
|
}
|
|
else if(StateDeclaration.canParse(toks))
|
|
{
|
|
auto sd = new StateDeclaration;
|
|
sd.parse(toks);
|
|
statedecs ~= sd;
|
|
}
|
|
else if(StructDeclaration.canParse(toks))
|
|
{
|
|
auto sd = new StructDeclaration;
|
|
sd.parse(toks);
|
|
structdecs ~= sd;
|
|
}
|
|
else if(EnumDeclaration.canParse(toks))
|
|
{
|
|
auto sd = new EnumDeclaration;
|
|
sd.parse(toks);
|
|
enumdecs ~= sd;
|
|
}
|
|
else if(ImportStatement.canParse(toks))
|
|
{
|
|
auto sd = new ImportStatement;
|
|
sd.parse(toks);
|
|
imports ~= sd;
|
|
}
|
|
else
|
|
fail("Illegal type or declaration", toks);
|
|
}
|
|
|
|
// Insert the class into the scope system. All parent classes must
|
|
// be loaded before this is called.
|
|
void createScope()
|
|
{
|
|
// Since debugging self inheritance can be a little icky, add an
|
|
// explicit recursion check.
|
|
assert(!flags.has(CFlags.InScope), "createScope called recursively");
|
|
flags.set(CFlags.InScope);
|
|
|
|
assert(isParsed());
|
|
assert(!isScoped(), "createScope called on already scoped class " ~
|
|
name.str);
|
|
|
|
// Set the scoped flag - this makes sure we are not called
|
|
// recursively below.
|
|
flags.set(CFlags.Scoped);
|
|
|
|
// Transfer the parent list
|
|
parents.length = parentNames.length;
|
|
foreach(int i, pName; parentNames)
|
|
{
|
|
// Find the parent class.
|
|
assert(pack !is null);
|
|
auto sl = pack.lookupClass(pName);
|
|
if(!sl.isClass)
|
|
fail("Cannot inherit from " ~ pName.str ~ ": No such class.",
|
|
pName.loc);
|
|
auto mc = sl.mc;
|
|
assert(mc !is null);
|
|
mc.requireScope();
|
|
|
|
assert(mc !is null);
|
|
assert(mc.isScoped);
|
|
|
|
parents[i] = mc;
|
|
|
|
// Direct self inheritance
|
|
if(mc is this)
|
|
fail("Class " ~ name.str ~ " cannot inherit from itself",
|
|
name.loc);
|
|
|
|
if(mc.isModule)
|
|
fail("Cannot inherit from module " ~ mc.name.str);
|
|
|
|
// If a parent class is not a forward reference and still
|
|
// does not have a scope, it means that it is itself running
|
|
// this function. This can only happen if we are a parent of
|
|
// it.
|
|
if(mc.sc is null)
|
|
fail("Class " ~ name.str ~ " is a parent of itself (through "
|
|
~ mc.name.str ~ ")", name.loc);
|
|
}
|
|
|
|
// For now we only support one parent class.
|
|
assert(parents.length <= 1);
|
|
|
|
// initialize() must already have been called before we get here
|
|
assert(baseObject !is null);
|
|
|
|
// If we don't have a parent, and we aren't Object, then set
|
|
// Object as our parent.
|
|
if(parents.length != 1 && this !is baseObject)
|
|
parents = [baseObject];
|
|
|
|
// So at this point we either have a parent, or we are Object.
|
|
assert(parents.length == 1 ||
|
|
this is baseObject);
|
|
|
|
// Since there are only linear (single) inheritance graphs at
|
|
// the moment, we can just copy our parent's tree list and add
|
|
// ourself to it.
|
|
if(parents.length == 1)
|
|
tree = parents[0].tree;
|
|
else
|
|
tree = null;
|
|
tree = tree ~ this;
|
|
treeIndex = tree.length-1;
|
|
|
|
assert(tree.length > 0);
|
|
assert(tree[$-1] is this);
|
|
assert(tree[treeIndex] is this);
|
|
|
|
// The parent scope is the scope of the parent class, or the
|
|
// package scope if there is no parent.
|
|
Scope parSc;
|
|
if(parents.length != 0) parSc = parents[0].sc;
|
|
else
|
|
{
|
|
// For Object, use the package scope (which should be the
|
|
// global scope)
|
|
assert(this is baseObject);
|
|
assert(pack is global);
|
|
parSc = pack;
|
|
}
|
|
|
|
assert(parSc !is null);
|
|
|
|
// Create the scope for this class
|
|
sc = new ClassScope(parSc, this);
|
|
|
|
// Set the type
|
|
objType = new ObjectType(this);
|
|
classType = objType.getMeta();
|
|
|
|
// Insert custom types first. This will never refer to other
|
|
// identifiers.
|
|
foreach(dec; structdecs)
|
|
dec.insertType(sc);
|
|
foreach(dec; enumdecs)
|
|
dec.insertType(sc);
|
|
|
|
// Resolve imports next. May refer to custom types, but no other
|
|
// ids.
|
|
foreach(dec; imports)
|
|
dec.resolve(sc);
|
|
|
|
// Then resolve the type headers.
|
|
foreach(dec; structdecs)
|
|
dec.resolve(sc);
|
|
foreach(dec; enumdecs)
|
|
dec.resolve(sc);
|
|
|
|
// Resolve variable declarations. They will insert themselves
|
|
// into the scope.
|
|
foreach(dec; vardecs)
|
|
dec.resolve(sc);
|
|
|
|
// Add function declarations to the scope.
|
|
foreach(dec; funcdecs)
|
|
sc.insertFunc(dec.fn);
|
|
|
|
// Ditto for states.
|
|
foreach(dec; statedecs)
|
|
sc.insertState(dec.st);
|
|
|
|
// Resolve function headers.
|
|
foreach(func; funcdecs)
|
|
func.resolve(sc);
|
|
|
|
// Set up the function and state lists
|
|
functions.length = funcdecs.length;
|
|
foreach(fn; funcdecs)
|
|
functions[fn.fn.index] = fn.fn;
|
|
|
|
states.length = statedecs.length;
|
|
foreach(st; statedecs)
|
|
states[st.st.index] = st.st;
|
|
|
|
// Now set up the virtual function table. It's elements
|
|
// correspond to the classes in tree[].
|
|
|
|
if(parents.length)
|
|
{
|
|
// This will get a lot trickier if we allow multiple inheritance
|
|
assert(parents.length == 1);
|
|
|
|
// Set up the virtuals list
|
|
auto pv = parents[0].virtuals;
|
|
virtuals.length = pv.length+1;
|
|
|
|
// We have to copy every single sublist, since we're not
|
|
// allowed to change our parent's data
|
|
foreach(i,l; pv)
|
|
virtuals[i] = l.dup;
|
|
|
|
// Add our own list
|
|
virtuals[$-1] = functions;
|
|
}
|
|
else
|
|
virtuals = [functions];
|
|
|
|
assert(virtuals.length == tree.length);
|
|
|
|
// Trace all our own functions back to their origin, and replace
|
|
// them. Since we've copied our parents list, and assume it is
|
|
// all set up, we only have to worry about our own
|
|
// functions. (For multiple inheritance this might be a bit more
|
|
// troublesome, but definitely doable.)
|
|
foreach(fn; functions)
|
|
{
|
|
auto o = fn.overrides;
|
|
|
|
// And we have to loop backwards through the overrides that
|
|
// o overrides as well.
|
|
while(o !is null)
|
|
{
|
|
// Find the owner class tree index of the function we're
|
|
// overriding
|
|
assert(o.owner !is this);
|
|
int clsInd = o.owner.treeIndex;
|
|
assert(clsInd < tree.length-1);
|
|
assert(tree[clsInd] == o.owner);
|
|
|
|
// Next, get the function index and replace the pointer
|
|
virtuals[clsInd][o.index] = fn;
|
|
|
|
// Get the function that o overrides too, and fix that
|
|
// one as well.
|
|
o = o.overrides;
|
|
}
|
|
}
|
|
|
|
flags.unset(CFlags.InScope);
|
|
}
|
|
|
|
// This calls resolve on the interior of functions and states.
|
|
void resolveBody()
|
|
{
|
|
requireScope();
|
|
|
|
assert(!isResolved, getName() ~ " is already resolved");
|
|
|
|
// Resolve the functions
|
|
foreach(func; funcdecs)
|
|
func.resolveBody();
|
|
|
|
// Including the constructor
|
|
if(scptConst !is null)
|
|
{
|
|
scptConst.resolve(sc);
|
|
assert(scptConst.fn.owner is this);
|
|
}
|
|
|
|
// Resolve states
|
|
foreach(state; statedecs)
|
|
state.resolve(sc);
|
|
|
|
// TODO: Resolve struct functions
|
|
/*
|
|
foredach(stru; structdecs)
|
|
stru.resolveBody(sc);
|
|
*/
|
|
|
|
// Validate all variable types
|
|
foreach(var; vardecs)
|
|
var.validate();
|
|
|
|
// Resolve variable and state overrides. No other declarations
|
|
// depend on these (the values are only relevant at the
|
|
// compilation stage), so we can resolve these last.
|
|
foreach(dec; varsets)
|
|
dec.resolve(sc);
|
|
|
|
flags.set(CFlags.Resolved);
|
|
}
|
|
|
|
alias int[] ia;
|
|
// This is platform dependent:
|
|
static const iasize = ia.sizeof / int.sizeof;
|
|
|
|
// Fill the data segment for this class.
|
|
void getDataSegment(int[] data)
|
|
{
|
|
assert(data.length == dataSize);
|
|
int totSize = 0;
|
|
|
|
foreach(VarDeclStatement vds; vardecs)
|
|
foreach(VarDeclaration vd; vds.vars)
|
|
{
|
|
int size = vd.var.type.getSize();
|
|
int[] val;
|
|
totSize += size;
|
|
|
|
val = vd.getCTimeValue();
|
|
|
|
data[vd.var.number..vd.var.number+size] = val[];
|
|
}
|
|
// Make sure the total size of the variables match the total size
|
|
// requested by variables through addNewDataVar.
|
|
assert(totSize == dataSize, "Data size mismatch in scope");
|
|
}
|
|
|
|
bool compiling = false;
|
|
void compileBody()
|
|
{
|
|
assert(!isCompiled, getName() ~ " is already compiled");
|
|
assert(!compiling, "compileBody called recursively");
|
|
compiling = true;
|
|
|
|
// Resolve the class body if it's not already done
|
|
if(!isResolved) resolveBody();
|
|
|
|
// Require that all parent classes are compiled before us
|
|
foreach(mc; tree[0..$-1])
|
|
mc.requireCompile();
|
|
|
|
// Generate byte code for functions and states.
|
|
foreach(f; funcdecs) f.compile();
|
|
foreach(s; statedecs) s.compile();
|
|
if(scptConst !is null) scptConst.compile();
|
|
|
|
// Get the data segment size for this class
|
|
assert(sc !is null && sc.isClass(), "Class does not have a class scope");
|
|
dataSize = sc.getDataSize;
|
|
|
|
// Calculate the total data size we need to allocate for each
|
|
// object
|
|
uint tsize = 0;
|
|
foreach(c; tree)
|
|
{
|
|
tsize += c.dataSize; // Data segment size
|
|
tsize += MonsterObject.exSize; // Extra data per object
|
|
tsize += iasize; // The size of our entry in the data[]
|
|
// table
|
|
}
|
|
|
|
// Allocate the buffer
|
|
totalData = new int[tsize];
|
|
|
|
// Used below to get subslices of the data segment
|
|
int[] slice = totalData;
|
|
int[] get(int ints)
|
|
{
|
|
assert(ints <= slice.length);
|
|
int[] res = slice[0..ints];
|
|
slice = slice[ints..$];
|
|
return res;
|
|
}
|
|
|
|
// The first part of the buffer is used for storing the obj.data
|
|
// array itself - skip that now.
|
|
get(iasize*tree.length);
|
|
|
|
// Set up the slice list
|
|
totalSliced.length = tree.length;
|
|
foreach(i,c; tree)
|
|
{
|
|
// Data segment slice
|
|
totalSliced[i] = get(c.dataSize);
|
|
|
|
// Skip the extra data
|
|
get(MonsterObject.exSize);
|
|
}
|
|
|
|
// At this point we should have used up the entire slice
|
|
assert(slice.length == 0);
|
|
// Sanity check on the size
|
|
assert(totalSliced[$-1].length == dataSize);
|
|
|
|
// Fill our own data segment
|
|
getDataSegment(totalSliced[$-1]);
|
|
|
|
// The next part is only implemented for single inheritance
|
|
assert(parents.length <= 1);
|
|
if(parents.length == 1)
|
|
{
|
|
auto p = parents[0];
|
|
|
|
// Go through the parent's tree, and copy its data
|
|
// segments. This will make sure we include all cumulative
|
|
// variable changes from past classes.
|
|
assert(p.tree.length == tree.length - 1);
|
|
|
|
foreach(i,c; p.tree)
|
|
{
|
|
assert(tree[i] is c);
|
|
|
|
// Copy updated data segment for c from parent class
|
|
totalSliced[i][] = p.totalSliced[i][];
|
|
}
|
|
|
|
// Apply all variable changes defined in this class
|
|
foreach(vs; varsets)
|
|
{
|
|
if(vs.isState) continue;
|
|
|
|
assert(vs.cls !is null);
|
|
int ind = vs.cls.treeIndex;
|
|
assert(ind < p.tree.length);
|
|
assert(tree[ind] is vs.cls);
|
|
|
|
vs.apply(totalSliced[ind]);
|
|
}
|
|
}
|
|
|
|
flags.set(CFlags.Compiled);
|
|
|
|
// If it's a singleton, set up the object.
|
|
if(isSingleton)
|
|
{
|
|
assert(singObj is null);
|
|
singObj = createObject();
|
|
}
|
|
compiling = false;
|
|
}
|
|
}
|