source-sdk-2013-mapbase/sp/src/vscript/vscript_squirrel.cpp

1764 lines
38 KiB
C++
Raw Normal View History

2020-05-15 19:17:08 +10:00
#include "vscript/ivscript.h"
#include "tier1/utlstring.h"
#include "squirrel.h"
#include "sqstdaux.h"
//#include "sqstdblob.h"
//#include "sqstdsystem.h"
//#include "sqstdio.h"
#include "sqstdmath.h"
#include "sqstdstring.h"
#include <cstdarg>
class SquirrelVM : public IScriptVM
{
public:
virtual bool Init() override;
virtual void Shutdown() override;
virtual bool ConnectDebugger() override;
virtual void DisconnectDebugger() override;
virtual ScriptLanguage_t GetLanguage() override;
virtual const char* GetLanguageName() override;
virtual void AddSearchPath(const char* pszSearchPath) override;
//--------------------------------------------------------
virtual bool Frame(float simTime) override;
//--------------------------------------------------------
// Simple script usage
//--------------------------------------------------------
virtual ScriptStatus_t Run(const char* pszScript, bool bWait = true) override;
//--------------------------------------------------------
// Compilation
//--------------------------------------------------------
virtual HSCRIPT CompileScript(const char* pszScript, const char* pszId = NULL) override;
virtual void ReleaseScript(HSCRIPT) override;
//--------------------------------------------------------
// Execution of compiled
//--------------------------------------------------------
virtual ScriptStatus_t Run(HSCRIPT hScript, HSCRIPT hScope = NULL, bool bWait = true) override;
virtual ScriptStatus_t Run(HSCRIPT hScript, bool bWait) override;
//--------------------------------------------------------
// Scope
//--------------------------------------------------------
virtual HSCRIPT CreateScope(const char* pszScope, HSCRIPT hParent = NULL) override;
virtual void ReleaseScope(HSCRIPT hScript) override;
//--------------------------------------------------------
// Script functions
//--------------------------------------------------------
virtual HSCRIPT LookupFunction(const char* pszFunction, HSCRIPT hScope = NULL) override;
virtual void ReleaseFunction(HSCRIPT hScript) override;
//--------------------------------------------------------
// Script functions (raw, use Call())
//--------------------------------------------------------
virtual ScriptStatus_t ExecuteFunction(HSCRIPT hFunction, ScriptVariant_t* pArgs, int nArgs, ScriptVariant_t* pReturn, HSCRIPT hScope, bool bWait) override;
//--------------------------------------------------------
// External functions
//--------------------------------------------------------
virtual void RegisterFunction(ScriptFunctionBinding_t* pScriptFunction) override;
//--------------------------------------------------------
// External classes
//--------------------------------------------------------
virtual bool RegisterClass(ScriptClassDesc_t* pClassDesc) override;
//--------------------------------------------------------
// External instances. Note class will be auto-registered.
//--------------------------------------------------------
virtual HSCRIPT RegisterInstance(ScriptClassDesc_t* pDesc, void* pInstance) override;
virtual void SetInstanceUniqeId(HSCRIPT hInstance, const char* pszId) override;
virtual void RemoveInstance(HSCRIPT hInstance) override;
virtual void* GetInstanceValue(HSCRIPT hInstance, ScriptClassDesc_t* pExpectedType = NULL) override;
//----------------------------------------------------------------------------
virtual bool GenerateUniqueKey(const char* pszRoot, char* pBuf, int nBufSize) override;
//----------------------------------------------------------------------------
virtual bool ValueExists(HSCRIPT hScope, const char* pszKey) override;
virtual bool SetValue(HSCRIPT hScope, const char* pszKey, const char* pszValue) override;
virtual bool SetValue(HSCRIPT hScope, const char* pszKey, const ScriptVariant_t& value) override;
virtual void CreateTable(ScriptVariant_t& Table) override;
virtual int GetNumTableEntries(HSCRIPT hScope) override;
virtual int GetKeyValue(HSCRIPT hScope, int nIterator, ScriptVariant_t* pKey, ScriptVariant_t* pValue) override;
virtual bool GetValue(HSCRIPT hScope, const char* pszKey, ScriptVariant_t* pValue) override;
virtual void ReleaseValue(ScriptVariant_t& value) override;
virtual bool ClearValue(HSCRIPT hScope, const char* pszKey) override;
//----------------------------------------------------------------------------
virtual void WriteState(CUtlBuffer* pBuffer) override;
virtual void ReadState(CUtlBuffer* pBuffer) override;
virtual void RemoveOrphanInstances() override;
virtual void DumpState() override;
virtual void SetOutputCallback(ScriptOutputFunc_t pFunc) override;
virtual void SetErrorCallback(ScriptErrorFunc_t pFunc) override;
//----------------------------------------------------------------------------
virtual bool RaiseException(const char* pszExceptionText) override;
HSQUIRRELVM vm_ = nullptr;
HSQOBJECT lastError_;
HSQOBJECT vectorClass_;
};
SQUserPointer TYPETAG_VECTOR = "VectorTypeTag";
namespace SQVector
{
SQInteger Destruct(SQUserPointer p, SQInteger size)
{
Vector* v = (Vector*)p;
delete v;
return 0;
}
SQInteger Construct(HSQUIRRELVM vm)
{
// TODO: There must be a nicer way to store the data with the actual instance, there are
// default slots but they are really just pointers anyway and you need to hold a member
// pointer which is yet another pointer dereference
int numParams = sq_gettop(vm);
float x = 0;
float y = 0;
float z = 0;
if ((numParams >= 2 && SQ_FAILED(sq_getfloat(vm, 2, &x))) ||
(numParams >= 3 && SQ_FAILED(sq_getfloat(vm, 3, &y))) ||
(numParams >= 4 && SQ_FAILED(sq_getfloat(vm, 4, &z))))
{
return sq_throwerror(vm, "Expected Vector(float x, float y, float z)");
}
Vector* v = new Vector(x, y, z);
sq_setinstanceup(vm, 1, v);
sq_setreleasehook(vm, 1, &Destruct);
return 0;
}
SQInteger Get(HSQUIRRELVM vm)
{
const char* key = nullptr;
sq_getstring(vm, 2, &key);
if (key == nullptr)
{
return sq_throwerror(vm, "Expected Vector._get(string)");
}
if (key[0] < 'x' || key['0'] > 'z' || key[1] != '\0')
{
return sqstd_throwerrorf(vm, "Unexpected key %s\n", key);
}
Vector* v = nullptr;
if (SQ_FAILED(sq_getinstanceup(vm, 1, (SQUserPointer*)&v, TYPETAG_VECTOR)))
{
return sq_throwerror(vm, "Unable to get Vector");
}
int idx = key[0] - 'x';
sq_pushfloat(vm, (*v)[idx]);
return 1;
}
SQInteger Set(HSQUIRRELVM vm)
{
const char* key = nullptr;
sq_getstring(vm, 2, &key);
if (key == nullptr)
{
return sq_throwerror(vm, "Expected Vector._get(string)");
}
if (key[0] < 'x' || key['0'] > 'z' || key[1] != '\0')
{
return sqstd_throwerrorf(vm, "Unexpected key %s\n", key);
}
Vector* v = nullptr;
if (SQ_FAILED(sq_getinstanceup(vm, 1, (SQUserPointer*)&v, TYPETAG_VECTOR)))
{
return sq_throwerror(vm, "Unable to get Vector");
}
float val = 0;
if (SQ_FAILED(sq_getfloat(vm, 3, &val)))
{
return sqstd_throwerrorf(vm, "Vector.%s expects float", key);
}
int idx = key[0] - 'x';
(*v)[idx] = val;
return 0;
}
SQInteger Add(HSQUIRRELVM vm)
{
Vector* v1 = nullptr;
Vector* v2 = nullptr;
if (sq_gettop(vm) != 2 ||
SQ_FAILED(sq_getinstanceup(vm, 1, (SQUserPointer*)&v1, TYPETAG_VECTOR)) ||
SQ_FAILED(sq_getinstanceup(vm, 2, (SQUserPointer*)&v2, TYPETAG_VECTOR)))
{
return sq_throwerror(vm, "Expected (Vector, Vector)");
}
Vector* ret = new Vector((*v1) + (*v2));
sq_getclass(vm, 1);
sq_createinstance(vm, -1);
sq_setinstanceup(vm, -1, ret);
sq_setreleasehook(vm, -1, &Destruct);
sq_remove(vm, -2);
return 1;
}
SQInteger Sub(HSQUIRRELVM vm)
{
Vector* v1 = nullptr;
Vector* v2 = nullptr;
if (sq_gettop(vm) != 2 ||
SQ_FAILED(sq_getinstanceup(vm, 1, (SQUserPointer*)&v1, TYPETAG_VECTOR)) ||
SQ_FAILED(sq_getinstanceup(vm, 2, (SQUserPointer*)&v2, TYPETAG_VECTOR)))
{
return sq_throwerror(vm, "Expected (Vector, Vector)");
}
Vector* ret = new Vector((*v1) - (*v2));
sq_getclass(vm, 1);
sq_createinstance(vm, -1);
sq_setinstanceup(vm, -1, ret);
sq_setreleasehook(vm, -1, &Destruct);
sq_remove(vm, -2);
return 1;
}
SQInteger Multiply(HSQUIRRELVM vm)
{
Vector* v1 = nullptr;
if (sq_gettop(vm) != 2 ||
SQ_FAILED(sq_getinstanceup(vm, 1, (SQUserPointer*)&v1, TYPETAG_VECTOR)))
{
return sq_throwerror(vm, "Expected (Vector, float) or (Vector, Vector)");
}
SQObjectType paramType = sq_gettype(vm, 2);
float s = 0.0;
Vector* v2 = nullptr;
if ((paramType & SQOBJECT_NUMERIC) &&
SQ_SUCCEEDED(sq_getfloat(vm, 2, &s)))
{
Vector* ret = new Vector((*v1) * s);
sq_getclass(vm, 1);
sq_createinstance(vm, -1);
sq_setinstanceup(vm, -1, ret);
sq_setreleasehook(vm, -1, &Destruct);
sq_remove(vm, -2);
return 1;
}
else if (paramType == OT_INSTANCE &&
SQ_SUCCEEDED(sq_getinstanceup(vm, 1, (SQUserPointer*)&v2, TYPETAG_VECTOR)))
{
Vector* ret = new Vector((*v1) * (*v2));
sq_getclass(vm, 1);
sq_createinstance(vm, -1);
sq_setinstanceup(vm, -1, ret);
sq_setreleasehook(vm, -1, &Destruct);
sq_remove(vm, -2);
return 1;
}
else
{
return sq_throwerror(vm, "Expected (Vector, float) or (Vector, Vector)");
}
}
SQInteger Divide(HSQUIRRELVM vm)
{
Vector* v1 = nullptr;
if (sq_gettop(vm) != 2 ||
SQ_FAILED(sq_getinstanceup(vm, 1, (SQUserPointer*)&v1, TYPETAG_VECTOR)))
{
return sq_throwerror(vm, "Expected (Vector, float) or (Vector, Vector)");
}
SQObjectType paramType = sq_gettype(vm, 2);
float s = 0.0;
Vector* v2 = nullptr;
if ((paramType & SQOBJECT_NUMERIC) &&
SQ_SUCCEEDED(sq_getfloat(vm, 2, &s)))
{
Vector* ret = new Vector((*v1) / s);
sq_getclass(vm, 1);
sq_createinstance(vm, -1);
sq_setinstanceup(vm, -1, ret);
sq_setreleasehook(vm, -1, &Destruct);
sq_remove(vm, -2);
return 1;
}
else if (paramType == OT_INSTANCE &&
SQ_SUCCEEDED(sq_getinstanceup(vm, 1, (SQUserPointer*)&v2, TYPETAG_VECTOR)))
{
Vector* ret = new Vector((*v1) / (*v2));
sq_getclass(vm, 1);
sq_createinstance(vm, -1);
sq_setinstanceup(vm, -1, ret);
sq_setreleasehook(vm, -1, &Destruct);
sq_remove(vm, -2);
return 1;
}
else
{
return sq_throwerror(vm, "Expected (Vector, float) or (Vector, Vector)");
}
}
SQInteger Length(HSQUIRRELVM vm)
{
Vector* v1 = nullptr;
if (sq_gettop(vm) != 1 ||
SQ_FAILED(sq_getinstanceup(vm, 1, (SQUserPointer*)&v1, TYPETAG_VECTOR)))
{
return sq_throwerror(vm, "Expected (Vector)");
}
sq_pushfloat(vm, v1->Length());
return 1;
}
SQInteger LengthSqr(HSQUIRRELVM vm)
{
Vector* v1 = nullptr;
if (sq_gettop(vm) != 1 ||
SQ_FAILED(sq_getinstanceup(vm, 1, (SQUserPointer*)&v1, TYPETAG_VECTOR)))
{
return sq_throwerror(vm, "Expected (Vector)");
}
sq_pushfloat(vm, v1->LengthSqr());
return 1;
}
SQInteger Length2D(HSQUIRRELVM vm)
{
Vector* v1 = nullptr;
if (sq_gettop(vm) != 1 ||
SQ_FAILED(sq_getinstanceup(vm, 1, (SQUserPointer*)&v1, TYPETAG_VECTOR)))
{
return sq_throwerror(vm, "Expected (Vector)");
}
sq_pushfloat(vm, v1->Length2D());
return 1;
}
SQInteger Length2DSqr(HSQUIRRELVM vm)
{
Vector* v1 = nullptr;
if (sq_gettop(vm) != 1 ||
SQ_FAILED(sq_getinstanceup(vm, 1, (SQUserPointer*)&v1, TYPETAG_VECTOR)))
{
return sq_throwerror(vm, "Expected (Vector)");
}
sq_pushfloat(vm, v1->Length2DSqr());
return 1;
}
SQInteger Normalized(HSQUIRRELVM vm)
{
Vector* v1 = nullptr;
if (sq_gettop(vm) != 1 ||
SQ_FAILED(sq_getinstanceup(vm, 1, (SQUserPointer*)&v1, TYPETAG_VECTOR)))
{
return sq_throwerror(vm, "Expected (Vector)");
}
Vector* ret = new Vector(v1->Normalized());
sq_getclass(vm, 1);
sq_createinstance(vm, -1);
sq_setinstanceup(vm, -1, ret);
sq_setreleasehook(vm, -1, &Destruct);
sq_remove(vm, -2);
return 1;
}
SQInteger Dot(HSQUIRRELVM vm)
{
Vector* v1 = nullptr;
Vector* v2 = nullptr;
if (sq_gettop(vm) != 2 ||
SQ_FAILED(sq_getinstanceup(vm, 1, (SQUserPointer*)&v1, TYPETAG_VECTOR)) ||
SQ_FAILED(sq_getinstanceup(vm, 2, (SQUserPointer*)&v2, TYPETAG_VECTOR)))
{
return sq_throwerror(vm, "Expected (Vector, Vector)");
}
sq_pushfloat(vm, v1->Dot(*v2));
return 1;
}
SQInteger Cross(HSQUIRRELVM vm)
{
Vector* v1 = nullptr;
Vector* v2 = nullptr;
if (sq_gettop(vm) != 2 ||
SQ_FAILED(sq_getinstanceup(vm, 1, (SQUserPointer*)&v1, TYPETAG_VECTOR)) ||
SQ_FAILED(sq_getinstanceup(vm, 2, (SQUserPointer*)&v2, TYPETAG_VECTOR)))
{
return sq_throwerror(vm, "Expected (Vector, Vector)");
}
Vector* ret = new Vector(v1->Cross(*v2));
sq_getclass(vm, 1);
sq_createinstance(vm, -1);
sq_setinstanceup(vm, -1, ret);
sq_setreleasehook(vm, -1, &Destruct);
sq_remove(vm, -2);
return 1;
}
static const SQRegFunction funcs[] = {
{_SC("constructor"), Construct,0,nullptr},
{_SC("_get"), Get, 2, _SC(".s")},
{_SC("_set"), Set, 3, _SC(".sn")},
{_SC("_add"), Add, 2, _SC("..")},
{_SC("_sub"), Sub, 2, _SC("..")},
{_SC("_mul"), Multiply, 2, _SC("..")},
{_SC("_div"), Divide, 2, _SC("..")},
{_SC("Length"), Length, 1, _SC(".")},
{_SC("LengthSqr"), LengthSqr, 1, _SC(".")},
{_SC("Length2D"), Length2D, 1, _SC(".")},
{_SC("Length2DSqr"), Length2DSqr, 1, _SC(".")},
{_SC("Normalized"), Normalized, 1, _SC(".")},
{_SC("_div"), Divide, 2, _SC("..")},
{_SC("Dot"), Dot, 2, _SC("..")},
{_SC("Cross"), Cross, 2, _SC("..")},
{nullptr,(SQFUNCTION)0,0,nullptr}
};
HSQOBJECT register_class(HSQUIRRELVM v)
{
sq_pushstring(v, _SC("Vector"), -1);
sq_newclass(v, SQFalse);
sq_settypetag(v, -1, TYPETAG_VECTOR);
SQInteger i = 0;
while (funcs[i].name != 0) {
const SQRegFunction& f = funcs[i];
sq_pushstring(v, f.name, -1);
sq_newclosure(v, f.f, 0);
sq_setparamscheck(v, f.nparamscheck, f.typemask);
sq_setnativeclosurename(v, -1, f.name);
sq_newslot(v, -3, SQFalse);
i++;
}
HSQOBJECT klass;
sq_resetobject(&klass);
sq_getstackobj(v, -1, &klass);
sq_addref(v, &klass);
sq_newslot(v, -3, SQFalse);
return klass;
}
} // SQVector
struct ClassInstanceData
{
void* instance;
ScriptClassDesc_t* desc;
};
bool CreateParamCheck(const ScriptFunctionBinding_t& func, char* output)
{
*output++ = '.';
for (int i = 0; i < func.m_desc.m_Parameters.Count(); ++i)
{
switch (func.m_desc.m_Parameters[i])
{
case FIELD_FLOAT:
*output++ = 'n'; // NOTE: Can be int or float
break;
case FIELD_CSTRING:
*output++ = 's';
break;
case FIELD_VECTOR:
*output++ = 'x'; // Generic instance, we validate on arrival
break;
case FIELD_INTEGER:
*output++ = 'i'; // could use 'n' also which is int or float
break;
case FIELD_BOOLEAN:
*output++ = 'b';
break;
case FIELD_CHARACTER:
*output++ = 's';
break;
case FIELD_HSCRIPT:
*output++ = '.';
break;
default:
Assert(!"Unsupported type");
return false;
};
}
*output++ = 0;
return true;
}
void PushVariant(HSQUIRRELVM vm, const ScriptVariant_t& value)
{
switch (value.m_type)
{
case FIELD_VOID:
sq_pushnull(vm);
break;
case FIELD_FLOAT:
sq_pushfloat(vm, value);
break;
case FIELD_CSTRING:
sq_pushstring(vm, value, -1);
break;
case FIELD_VECTOR:
{
SquirrelVM* pSquirrelVM = (SquirrelVM*)sq_getforeignptr(vm);
assert(pSquirrelVM);
sq_pushobject(vm, pSquirrelVM->vectorClass_);
sq_createinstance(vm, -1);
// Valve, wtf is with this wierd lifetime?
sq_setinstanceup(vm, -1, new Vector(value));
sq_setreleasehook(vm, -1, SQVector::Destruct);
sq_remove(vm, -2);
break;
}
case FIELD_INTEGER:
sq_pushinteger(vm, value.m_int);
break;
case FIELD_BOOLEAN:
sq_pushbool(vm, value.m_bool);
break;
case FIELD_CHARACTER:
{
char buf[2] = { value.m_char, 0 };
sq_pushstring(vm, buf, 1);
break;
}
case FIELD_HSCRIPT:
if (value.m_hScript)
{
sq_pushobject(vm, *((HSQOBJECT*)value.m_hScript));
}
else
{
sq_pushnull(vm);
}
break;
}
}
bool getVariant(HSQUIRRELVM vm, SQInteger idx, ScriptVariant_t& variant)
{
switch (sq_gettype(vm, idx))
{
case OT_NULL:
{
// TODO: Should this be (HSCRIPT)nullptr
variant.m_type = FIELD_VOID;
return true;
}
case OT_INTEGER:
{
SQInteger val;
if (SQ_FAILED(sq_getinteger(vm, idx, &val)))
{
return false;
}
variant = (int)val;
return true;
}
case OT_FLOAT:
{
SQFloat val;
if (SQ_FAILED(sq_getfloat(vm, idx, &val)))
{
return false;
}
variant = (float)val;
return true;
}
case OT_BOOL:
{
SQBool val;
if (SQ_FAILED(sq_getbool(vm, idx, &val)))
{
return false;
}
variant = (bool)val;
return true;
}
case OT_STRING:
{
const char* val;
SQInteger size = 0;
if (SQ_FAILED(sq_getstringandsize(vm, idx, &val, &size)))
{
return false;
}
char* buffer = new char[size + 1];
V_memcpy(buffer, val, size);
buffer[size] = 0;
variant = buffer;
variant.m_flags |= SV_FREE;
return true;
}
case OT_INSTANCE:
{
Vector* v = nullptr;
SQUserPointer tag;
if (SQ_SUCCEEDED(sq_gettypetag(vm, idx, &tag)) &&
tag == TYPETAG_VECTOR &&
SQ_SUCCEEDED(sq_getinstanceup(vm, idx, (SQUserPointer*)&v, TYPETAG_VECTOR)))
{
// TODO: This actually ends up pointing to the same data it seems error prone
variant = new Vector(*v);
variant.m_flags |= SV_FREE;
return true;
}
// fall-through for non-vector
}
default:
{
HSQOBJECT* obj = new HSQOBJECT;
sq_resetobject(obj);
sq_getstackobj(vm, idx, obj);
sq_addref(vm, obj);
variant = (HSCRIPT)obj;
}
};
return true;
}
SQInteger function_stub(HSQUIRRELVM vm)
{
SQInteger top = sq_gettop(vm);
SQUserPointer userptr = nullptr;
sq_getuserpointer(vm, top, &userptr);
Assert(userptr);
ScriptFunctionBinding_t* pFunc = (ScriptFunctionBinding_t*)userptr;
auto nargs = pFunc->m_desc.m_Parameters.Count();
if (nargs > top)
{
// TODO: Handle optional parameters?
return sq_throwerror(vm, "Invalid number of parameters");
}
CUtlVector<ScriptVariant_t> params;
params.SetCount(nargs);
for (int i = 0; i < nargs; ++i)
{
switch (pFunc->m_desc.m_Parameters[i])
{
case FIELD_FLOAT:
{
float val = 0.0;
if (SQ_FAILED(sq_getfloat(vm, i + 2, &val)))
return sq_throwerror(vm, "Expected float");
params[i] = val;
break;
}
case FIELD_CSTRING:
{
const char* val;
if (SQ_FAILED(sq_getstring(vm, i + 2, &val)))
return sq_throwerror(vm, "Expected string");
params[i] = val;
break;
}
case FIELD_VECTOR:
{
Vector* val;
if (SQ_FAILED(sq_getinstanceup(vm, i + 2, (SQUserPointer*)&val, TYPETAG_VECTOR)))
return sq_throwerror(vm, "Expected Vector");
params[i] = *val;
break;
}
case FIELD_INTEGER:
{
SQInteger val = 0;
if (SQ_FAILED(sq_getinteger(vm, i + 2, &val)))
return sq_throwerror(vm, "Expected integer");
params[i] = (int)val;
break;
}
case FIELD_BOOLEAN:
{
SQBool val = 0;
if (SQ_FAILED(sq_getbool(vm, i + 2, &val)))
return sq_throwerror(vm, "Expected bool");
params[i] = (bool)val;
break;
}
case FIELD_CHARACTER:
{
const char* val;
if (SQ_FAILED(sq_getstring(vm, i + 2, &val)))
return sq_throwerror(vm, "Expected string");
params[i] = val[i];
break;
}
case FIELD_HSCRIPT:
{
HSQOBJECT val;
if (SQ_FAILED(sq_getstackobj(vm, i + 2, &val)))
return sq_throwerror(vm, "Expected string");
if (sq_isnull(val))
{
params[i] = (HSCRIPT)nullptr;
}
else
{
HSQOBJECT* pObject = new HSQOBJECT;
*pObject = val;
sq_addref(vm, pObject);
params[i] = (HSCRIPT)pObject;
}
break;
}
default:
Assert(!"Unsupported type");
return false;
}
}
void* instance = nullptr;
if (pFunc->m_flags & SF_MEMBER_FUNC)
{
SQUserPointer self;
sq_getinstanceup(vm, 1, &self, nullptr);
if (!self)
{
return sq_throwerror(vm, "Can't be used on a null instance");
}
instance = ((ClassInstanceData*)self)->instance;
}
ScriptVariant_t retval;
SquirrelVM* pSquirrelVM = (SquirrelVM*)sq_getforeignptr(vm);
assert(pSquirrelVM);
sq_resetobject(&pSquirrelVM->lastError_);
(*pFunc->m_pfnBinding)(pFunc->m_pFunction, instance, params.Base(), nargs,
pFunc->m_desc.m_ReturnType == FIELD_VOID ? nullptr : &retval);
if (!sq_isnull(pSquirrelVM->lastError_))
{
sq_pushobject(vm, pSquirrelVM->lastError_);
sq_resetobject(&pSquirrelVM->lastError_);
return sq_throwobject(vm);
}
PushVariant(vm, retval);
return pFunc->m_desc.m_ReturnType != FIELD_VOID;
}
SQInteger destructor_stub(SQUserPointer p, SQInteger size)
{
auto classInstanceData = (ClassInstanceData*)p;
classInstanceData->desc->m_pfnDestruct(classInstanceData->instance);
delete classInstanceData;
return 0;
}
SQInteger destructor_stub_instance(SQUserPointer p, SQInteger size)
{
auto classInstanceData = (ClassInstanceData*)p;
// We don't call destructor here because this is owned by the game
delete classInstanceData;
return 0;
}
SQInteger constructor_stub(HSQUIRRELVM vm)
{
SQInteger top = sq_gettop(vm);
SQUserPointer userptr = nullptr;
sq_getuserpointer(vm, top, &userptr);
ScriptClassDesc_t* pClassDesc = (ScriptClassDesc_t*)userptr;
if (!pClassDesc->m_pfnConstruct)
{
return sqstd_throwerrorf(vm, "Unable to construct instances of %s", pClassDesc->m_pszScriptName);
}
SquirrelVM* pSquirrelVM = (SquirrelVM*)sq_getforeignptr(vm);
assert(pSquirrelVM);
sq_resetobject(&pSquirrelVM->lastError_);
void* instance = pClassDesc->m_pfnConstruct();
if (!sq_isnull(pSquirrelVM->lastError_))
{
sq_pushobject(vm, pSquirrelVM->lastError_);
sq_resetobject(&pSquirrelVM->lastError_);
return sq_throwobject(vm);
}
auto classInstanceData = new ClassInstanceData;
classInstanceData->instance = instance;
classInstanceData->desc = pClassDesc;
sq_setinstanceup(vm, 1, classInstanceData);
sq_setreleasehook(vm, 1, &destructor_stub);
return 0;
}
struct SquirrelSafeCheck
{
SquirrelSafeCheck(HSQUIRRELVM vm) : vm_{ vm }, top_{ sq_gettop(vm) } {}
~SquirrelSafeCheck()
{
if (top_ != sq_gettop(vm_))
Error("Squirrel VM stack is not consistent\n");
}
HSQUIRRELVM vm_;
SQInteger top_;
};
void printfunc(HSQUIRRELVM SQ_UNUSED_ARG(v), const SQChar* format, ...)
{
va_list args;
va_start(args, format);
char buffer[256];
vsprintf(buffer, format, args);
Msg("vscript: %s\n", buffer);
va_end(args);
}
void errorfunc(HSQUIRRELVM SQ_UNUSED_ARG(v), const SQChar* format, ...)
{
va_list args;
va_start(args, format);
char buffer[256];
vsprintf(buffer, format, args);
Msg("vscript: (ERRORR) %s\n", buffer);
va_end(args);
}
bool SquirrelVM::Init()
{
vm_ = sq_open(1024); //creates a VM with initial stack size 1024
if (vm_ == nullptr)
return false;
sq_setforeignptr(vm_, this);
sq_resetobject(&lastError_);
sq_setprintfunc(vm_, printfunc, errorfunc);
{
sq_pushroottable(vm_);
sqstd_register_mathlib(vm_);
sqstd_register_stringlib(vm_);
vectorClass_ = SQVector::register_class(vm_);
// We exclude these libraries as they create a security risk on the client even
// though I'm sure if someone tried hard enough they could achieve all sorts of
// things with the other APIs, this just makes it a little bit harder for a map
// created by someone in the community causing a bunch of security vulnerablilties.
//
// Pretty sure DoIncludeScript() is already a vulnerability vector here, however
// that also depends on compile errors not showing up and relies on IFilesystem with
// a path prefix.
//
//sqstd_register_bloblib(vm_);
//sqstd_register_iolib(vm_);
//sqstd_register_systemlib(vm_);
sqstd_seterrorhandlers(vm_);
sq_pop(vm_, 1);
}
if (Run(
"class CSimpleCallChainer\n"
"{\n"
" function constructor(prefixString, scopeForThis, exactMatch)\n"
" {\n"
" prefix = prefixString;\n"
" scope = scopeForThis;\n"
" chain = [];\n"
" scope[\"Dispatch\" + prefixString] <- Call.bindenv(this);\n"
" }\n"
"\n"
" function PostScriptExecute()\n"
" {\n"
" local func = null;\n"
" try {\n"
" func = scope[prefix];\n"
" } catch(e) {\n"
" return;\n"
" }\n"
" if (typeof(func) != \"function\")\n"
" return;\n"
" chain.push(func);\n"
" }\n"
"\n"
" function Call()\n"
" {\n"
" foreach (func in chain)\n"
" {\n"
" func.pcall(scope);\n"
" }\n"
" }\n"
" prefix = null;\n"
" scope= null;\n"
" chain = [];\n"
"}") != SCRIPT_DONE)
{
this->Shutdown();
return false;
}
return true;
}
void SquirrelVM::Shutdown()
{
if (vm_)
{
sq_close(vm_);
vm_ = nullptr;
}
}
bool SquirrelVM::ConnectDebugger()
{
return false;
}
void SquirrelVM::DisconnectDebugger()
{
}
ScriptLanguage_t SquirrelVM::GetLanguage()
{
return SL_SQUIRREL;
}
const char* SquirrelVM::GetLanguageName()
{
return "squirrel";
}
void SquirrelVM::AddSearchPath(const char* pszSearchPath)
{
}
bool SquirrelVM::Frame(float simTime)
{
return false;
}
ScriptStatus_t SquirrelVM::Run(const char* pszScript, bool bWait)
{
SquirrelSafeCheck safeCheck(vm_);
if (SQ_FAILED(sq_compilebuffer(vm_, pszScript, strlen(pszScript), "<run>", SQTrue)))
{
return SCRIPT_ERROR;
}
sq_pushroottable(vm_);
if (SQ_FAILED(sq_call(vm_, 1, SQFalse, SQTrue)))
{
sq_pop(vm_, 1);
return SCRIPT_ERROR;
}
sq_pop(vm_, 1);
return SCRIPT_DONE;
}
HSCRIPT SquirrelVM::CompileScript(const char* pszScript, const char* pszId)
{
SquirrelSafeCheck safeCheck(vm_);
// TODO: sq_setcompilererrorhandler
Assert(vm_);
if (pszId == nullptr) pszId = "<unnamed>";
if (SQ_FAILED(sq_compilebuffer(vm_, pszScript, strlen(pszScript), pszId, SQTrue)))
{
return nullptr;
}
HSQOBJECT* obj = new HSQOBJECT;
sq_resetobject(obj);
sq_getstackobj(vm_, -1, obj);
sq_addref(vm_, obj);
sq_pop(vm_, 1);
return (HSCRIPT)obj;
}
void SquirrelVM::ReleaseScript(HSCRIPT hScript)
{
SquirrelSafeCheck safeCheck(vm_);
if (!hScript) return;
HSQOBJECT* obj = (HSQOBJECT*)hScript;
sq_release(vm_, obj);
delete obj;
}
ScriptStatus_t SquirrelVM::Run(HSCRIPT hScript, HSCRIPT hScope, bool bWait)
{
SquirrelSafeCheck safeCheck(vm_);
HSQOBJECT* obj = (HSQOBJECT*)hScript;
sq_pushobject(vm_, *obj);
if (hScope)
{
Assert(hScope != INVALID_HSCRIPT);
HSQOBJECT* scope = (HSQOBJECT*)hScope;
sq_pushobject(vm_, *scope);
}
else
{
sq_pushroottable(vm_);
}
auto result = sq_call(vm_, 1, false, true);
sq_pop(vm_, 1);
if (SQ_FAILED(result))
{
// TODO: sq_getlasterror
return SCRIPT_ERROR;
}
return SCRIPT_DONE;
}
ScriptStatus_t SquirrelVM::Run(HSCRIPT hScript, bool bWait)
{
SquirrelSafeCheck safeCheck(vm_);
HSQOBJECT* obj = (HSQOBJECT*)hScript;
sq_pushobject(vm_, *obj);
sq_pushroottable(vm_);
auto result = sq_call(vm_, 1, false, true);
sq_pop(vm_, 1);
if (SQ_FAILED(result))
{
// TODO: sq_getlasterror
return SCRIPT_ERROR;
}
return SCRIPT_DONE;
}
HSCRIPT SquirrelVM::CreateScope(const char* pszScope, HSCRIPT hParent)
{
SquirrelSafeCheck safeCheck(vm_);
sq_newtable(vm_);
if (hParent)
{
HSQOBJECT* parent = (HSQOBJECT*)hParent;
Assert(hParent != INVALID_HSCRIPT);
sq_pushobject(vm_, *parent);
}
else
{
sq_pushroottable(vm_);
}
if (SQ_FAILED(sq_setdelegate(vm_, -2)))
{
sq_pop(vm_, 2);
return nullptr;
}
HSQOBJECT* obj = new HSQOBJECT;
sq_resetobject(obj);
sq_getstackobj(vm_, -1, obj);
sq_addref(vm_, obj);
sq_pop(vm_, 1);
return (HSCRIPT)obj;
}
void SquirrelVM::ReleaseScope(HSCRIPT hScript)
{
SquirrelSafeCheck safeCheck(vm_);
if (!hScript) return;
HSQOBJECT* obj = (HSQOBJECT*)hScript;
sq_release(vm_, obj);
delete obj;
}
HSCRIPT SquirrelVM::LookupFunction(const char* pszFunction, HSCRIPT hScope)
{
SquirrelSafeCheck safeCheck(vm_);
if (hScope)
{
HSQOBJECT* scope = (HSQOBJECT*)hScope;
Assert(hScope != INVALID_HSCRIPT);
sq_pushobject(vm_, *scope);
}
else
{
sq_pushroottable(vm_);
}
sq_pushstring(vm_, _SC(pszFunction), -1);
HSQOBJECT obj;
sq_resetobject(&obj);
if (sq_get(vm_, -2) == SQ_OK)
{
sq_getstackobj(vm_, -1, &obj);
sq_addref(vm_, &obj);
sq_pop(vm_, 1);
}
sq_pop(vm_, 1);
if (!sq_isclosure(obj))
{
sq_release(vm_, &obj);
return nullptr;
}
HSQOBJECT* pObj = new HSQOBJECT;
*pObj = obj;
return (HSCRIPT)pObj;
}
void SquirrelVM::ReleaseFunction(HSCRIPT hScript)
{
SquirrelSafeCheck safeCheck(vm_);
if (!hScript) return;
HSQOBJECT* obj = (HSQOBJECT*)hScript;
sq_release(vm_, obj);
delete obj;
}
ScriptStatus_t SquirrelVM::ExecuteFunction(HSCRIPT hFunction, ScriptVariant_t* pArgs, int nArgs, ScriptVariant_t* pReturn, HSCRIPT hScope, bool bWait)
{
SquirrelSafeCheck safeCheck(vm_);
if (!hFunction)
return SCRIPT_ERROR;
if (hFunction == INVALID_HSCRIPT)
return SCRIPT_ERROR;
HSQOBJECT* pFunc = (HSQOBJECT*)hFunction;
sq_pushobject(vm_, *pFunc);
if (hScope)
{
Assert(hScope != INVALID_HSCRIPT);
HSQOBJECT* scope = (HSQOBJECT*)hScope;
sq_pushobject(vm_, *scope);
}
else
{
sq_pushroottable(vm_);
}
for (int i = 0; i < nArgs; ++i)
{
PushVariant(vm_, pArgs[i]);
}
bool hasReturn = pReturn != nullptr;
if (SQ_FAILED(sq_call(vm_, nArgs + 1, hasReturn, SQTrue)))
{
sq_pop(vm_, 1);
return SCRIPT_ERROR;
}
if (hasReturn)
{
if (!getVariant(vm_, -1, *pReturn))
{
sq_pop(vm_, 1);
return SCRIPT_ERROR;
}
sq_pop(vm_, 1);
}
sq_pop(vm_, 1);
return SCRIPT_DONE;
}
void SquirrelVM::RegisterFunction(ScriptFunctionBinding_t* pScriptFunction)
{
SquirrelSafeCheck safeCheck(vm_);
Assert(pScriptFunction);
if (!pScriptFunction)
return;
char typemask[64];
if (!CreateParamCheck(*pScriptFunction, typemask))
{
return;
}
sq_pushroottable(vm_);
sq_pushstring(vm_, pScriptFunction->m_desc.m_pszScriptName, -1);
sq_pushuserpointer(vm_, pScriptFunction);
sq_newclosure(vm_, function_stub, 1);
sq_setnativeclosurename(vm_, -1, pScriptFunction->m_desc.m_pszScriptName);
sq_setparamscheck(vm_, pScriptFunction->m_desc.m_Parameters.Count() + 1, typemask);
bool isStatic = false;
sq_newslot(vm_, -3, isStatic);
sq_pop(vm_, 1);
}
bool SquirrelVM::RegisterClass(ScriptClassDesc_t* pClassDesc)
{
SquirrelSafeCheck safeCheck(vm_);
sq_pushroottable(vm_);
sq_pushstring(vm_, pClassDesc->m_pszScriptName, -1);
// Check if class name is already taken
if (sq_get(vm_, -2) == SQ_OK)
{
sq_pop(vm_, 2);
return false;
}
// Register base in case it doesn't exist
if (pClassDesc->m_pBaseDesc)
{
RegisterClass(pClassDesc->m_pBaseDesc);
// Check if the base class can be
sq_pushstring(vm_, pClassDesc->m_pBaseDesc->m_pszScriptName, -1);
if (sq_get(vm_, -2) != SQ_OK)
{
sq_pop(vm_, 1);
return false;
}
}
if (SQ_FAILED(sq_newclass(vm_, pClassDesc->m_pBaseDesc ? SQTrue : SQFalse)))
{
sq_pop(vm_, 2 + (pClassDesc->m_pBaseDesc ? 1 : 0));
return false;
}
sq_pushstring(vm_, "constructor", -1);
sq_pushuserpointer(vm_, pClassDesc);
sq_newclosure(vm_, constructor_stub, 1);
sq_newslot(vm_, -3, SQFalse);
for (int i = 0; i < pClassDesc->m_FunctionBindings.Count(); ++i)
{
auto& scriptFunction = pClassDesc->m_FunctionBindings[i];
char typemask[64];
if (!CreateParamCheck(scriptFunction, typemask))
{
Warning("Unable to create param check for %s.%s\n",
pClassDesc->m_pszClassname, scriptFunction.m_desc.m_pszFunction);
break;
}
sq_pushstring(vm_, scriptFunction.m_desc.m_pszScriptName, -1);
sq_pushuserpointer(vm_, &scriptFunction);
sq_newclosure(vm_, function_stub, 1);
sq_setnativeclosurename(vm_, -1, scriptFunction.m_desc.m_pszScriptName);
sq_setparamscheck(vm_, scriptFunction.m_desc.m_Parameters.Count() + 1, typemask);
bool isStatic = false;
sq_newslot(vm_, -3, isStatic);
}
sq_pushstring(vm_, pClassDesc->m_pszScriptName, -1);
sq_push(vm_, -2);
if (SQ_FAILED(sq_newslot(vm_, -4, SQFalse)))
{
sq_pop(vm_, 4);
return false;
}
sq_pop(vm_, 2);
return true;
}
HSCRIPT SquirrelVM::RegisterInstance(ScriptClassDesc_t* pDesc, void* pInstance)
{
SquirrelSafeCheck safeCheck(vm_);
this->RegisterClass(pDesc);
sq_pushroottable(vm_);
sq_pushstring(vm_, pDesc->m_pszScriptName, -1);
if (sq_get(vm_, -2) != SQ_OK)
{
sq_pop(vm_, 1);
return nullptr;
}
if (SQ_FAILED(sq_createinstance(vm_, -1)))
{
sq_pop(vm_, 2);
return nullptr;
}
ClassInstanceData* classInstanceData = new ClassInstanceData;
classInstanceData->desc = pDesc;
classInstanceData->instance = pInstance;
if (SQ_FAILED(sq_setinstanceup(vm_, -1, classInstanceData)))
{
delete classInstanceData;
sq_pop(vm_, 3);
}
sq_setreleasehook(vm_, -1, &destructor_stub_instance);
HSQOBJECT* obj = new HSQOBJECT;
sq_resetobject(obj);
sq_getstackobj(vm_, -1, obj);
sq_addref(vm_, obj);
sq_pop(vm_, 3);
return (HSCRIPT)obj;
}
void SquirrelVM::SetInstanceUniqeId(HSCRIPT hInstance, const char* pszId)
{
SquirrelSafeCheck safeCheck(vm_);
// Hey valve? WTF do you expect to do here
}
void SquirrelVM::RemoveInstance(HSCRIPT hInstance)
{
SquirrelSafeCheck safeCheck(vm_);
if (!hInstance) return;
HSQOBJECT* obj = (HSQOBJECT*)hInstance;
sq_pushobject(vm_, *obj);
SQUserPointer self;
sq_getinstanceup(vm_, -1, &self, nullptr);
auto classInstanceData = (ClassInstanceData*)self;
delete classInstanceData;
sq_setinstanceup(vm_, -1, nullptr);
sq_pop(vm_, 1);
sq_release(vm_, obj);
delete obj;
}
void* SquirrelVM::GetInstanceValue(HSCRIPT hInstance, ScriptClassDesc_t* pExpectedType)
{
SquirrelSafeCheck safeCheck(vm_);
if (!hInstance) return nullptr;
HSQOBJECT* obj = (HSQOBJECT*)hInstance;
if (pExpectedType)
{
sq_pushroottable(vm_);
sq_pushstring(vm_, pExpectedType->m_pszScriptName, -1);
if (sq_get(vm_, -2) != SQ_OK)
{
sq_pop(vm_, 1);
return nullptr;
}
sq_pushobject(vm_, *obj);
if (!sq_instanceof(vm_))
{
sq_pop(vm_, 3);
return nullptr;
}
sq_pop(vm_, 3);
}
sq_pushobject(vm_, *obj);
SQUserPointer self;
sq_getinstanceup(vm_, -1, &self, nullptr);
sq_pop(vm_, 1);
auto classInstanceData = (ClassInstanceData*)self;
if (!classInstanceData)
{
return nullptr;
}
return classInstanceData->instance;
}
bool SquirrelVM::GenerateUniqueKey(const char* pszRoot, char* pBuf, int nBufSize)
{
static int keyIdx = 0;
// This gets used for script scope, still confused why it needs to be inside IScriptVM
// is it just to be a compatible name for CreateScope?
SquirrelSafeCheck safeCheck(vm_);
V_snprintf(pBuf, nBufSize, "%08X_%s", ++keyIdx, pszRoot);
return true;
}
bool SquirrelVM::ValueExists(HSCRIPT hScope, const char* pszKey)
{
SquirrelSafeCheck safeCheck(vm_);
if (hScope)
{
HSQOBJECT* scope = (HSQOBJECT*)hScope;
Assert(hScope != INVALID_HSCRIPT);
sq_pushobject(vm_, *scope);
}
else
{
sq_pushroottable(vm_);
}
sq_pushstring(vm_, pszKey, -1);
if (sq_get(vm_, -2) != SQ_OK)
{
sq_pop(vm_, 1);
return false;
}
sq_pop(vm_, 2);
return true;
}
bool SquirrelVM::SetValue(HSCRIPT hScope, const char* pszKey, const char* pszValue)
{
SquirrelSafeCheck safeCheck(vm_);
if (hScope)
{
HSQOBJECT* scope = (HSQOBJECT*)hScope;
Assert(hScope != INVALID_HSCRIPT);
sq_pushobject(vm_, *scope);
}
else
{
sq_pushroottable(vm_);
}
sq_pushstring(vm_, pszKey, -1);
sq_pushstring(vm_, pszValue, -1);
sq_newslot(vm_, -3, SQFalse);
sq_pop(vm_, 1);
return true;
}
bool SquirrelVM::SetValue(HSCRIPT hScope, const char* pszKey, const ScriptVariant_t& value)
{
SquirrelSafeCheck safeCheck(vm_);
if (hScope)
{
HSQOBJECT* scope = (HSQOBJECT*)hScope;
Assert(hScope != INVALID_HSCRIPT);
sq_pushobject(vm_, *scope);
}
else
{
sq_pushroottable(vm_);
}
sq_pushstring(vm_, pszKey, -1);
PushVariant(vm_, value);
sq_newslot(vm_, -3, SQFalse);
sq_pop(vm_, 1);
return true;
}
void SquirrelVM::CreateTable(ScriptVariant_t& Table)
{
SquirrelSafeCheck safeCheck(vm_);
sq_newtable(vm_);
HSQOBJECT* obj = new HSQOBJECT;
sq_resetobject(obj);
sq_getstackobj(vm_, -1, obj);
sq_addref(vm_, obj);
sq_pop(vm_, 1);
Table = (HSCRIPT)obj;
}
int SquirrelVM::GetNumTableEntries(HSCRIPT hScope)
{
SquirrelSafeCheck safeCheck(vm_);
if (!hScope)
{
// TODO: This is called hScope but seems like just a table so
// lets not fallback to root table
return 0;
}
HSQOBJECT* scope = (HSQOBJECT*)hScope;
Assert(hScope != INVALID_HSCRIPT);
sq_pushobject(vm_, *scope);
int count = sq_getsize(vm_, -1);
sq_pop(vm_, 1);
return count;
}
int SquirrelVM::GetKeyValue(HSCRIPT hScope, int nIterator, ScriptVariant_t* pKey, ScriptVariant_t* pValue)
{
SquirrelSafeCheck safeCheck(vm_);
// TODO: How does this work does it expect to output to pKey and pValue as an array from nIterator
// elements or does it expect nIterator to be an index in hScrope, if so how does that work without
// without depending on squirrel internals not public API for getting the iterator (which is opaque)
// or do you iterate until that point and output the value? If it should be iterator then this should
// be a HSCRIPT for ease of use.
Assert(!"GetKeyValue is not implemented");
return 0;
}
bool SquirrelVM::GetValue(HSCRIPT hScope, const char* pszKey, ScriptVariant_t* pValue)
{
SquirrelSafeCheck safeCheck(vm_);
Assert(pValue);
if (!pValue)
{
return false;
}
if (hScope)
{
HSQOBJECT* scope = (HSQOBJECT*)hScope;
Assert(hScope != INVALID_HSCRIPT);
sq_pushobject(vm_, *scope);
}
else
{
sq_pushroottable(vm_);
}
sq_pushstring(vm_, pszKey, -1);
if (sq_get(vm_, -2) != SQ_OK)
{
sq_pop(vm_, 1);
return false;
}
if (!getVariant(vm_, -1, *pValue))
{
sq_pop(vm_, 2);
return false;
}
sq_pop(vm_, 2);
return true;
}
void SquirrelVM::ReleaseValue(ScriptVariant_t& value)
{
SquirrelSafeCheck safeCheck(vm_);
if (value.m_type == FIELD_HSCRIPT)
{
HSCRIPT hScript = value;
HSQOBJECT* obj = (HSQOBJECT*)hScript;
sq_release(vm_, obj);
delete obj;
}
else
{
value.Free();
}
// Let's prevent this being called again and giving some UB
value.m_type = FIELD_VOID;
}
bool SquirrelVM::ClearValue(HSCRIPT hScope, const char* pszKey)
{
SquirrelSafeCheck safeCheck(vm_);
if (hScope)
{
HSQOBJECT* scope = (HSQOBJECT*)hScope;
Assert(hScope != INVALID_HSCRIPT);
sq_pushobject(vm_, *scope);
}
else
{
sq_pushroottable(vm_);
}
sq_pushstring(vm_, pszKey, -1);
if (SQ_FAILED(sq_deleteslot(vm_, -2, SQFalse)))
{
sq_pop(vm_, 1);
return false;
}
sq_pop(vm_, 1);
return true;
}
void SquirrelVM::WriteState(CUtlBuffer* pBuffer)
{
SquirrelSafeCheck safeCheck(vm_);
// TODO: How to handle this? Seems dangerous to serialize the entire state
// because it depends on instance pointers that could be anywhere
}
void SquirrelVM::ReadState(CUtlBuffer* pBuffer)
{
SquirrelSafeCheck safeCheck(vm_);
// TODO: How to handle this? Seems dangerous to serialize the entire state
// because it depends on instance pointers that could be anywhere
}
void SquirrelVM::RemoveOrphanInstances()
{
SquirrelSafeCheck safeCheck(vm_);
// TODO: Is this the right thing to do here? It's not really removing orphan instances
sq_collectgarbage(vm_);
}
void SquirrelVM::DumpState()
{
SquirrelSafeCheck safeCheck(vm_);
}
void SquirrelVM::SetOutputCallback(ScriptOutputFunc_t pFunc)
{
SquirrelSafeCheck safeCheck(vm_);
}
void SquirrelVM::SetErrorCallback(ScriptErrorFunc_t pFunc)
{
SquirrelSafeCheck safeCheck(vm_);
}
bool SquirrelVM::RaiseException(const char* pszExceptionText)
{
SquirrelSafeCheck safeCheck(vm_);
sq_pushstring(vm_, pszExceptionText, -1);
sq_resetobject(&lastError_);
sq_getstackobj(vm_, -1, &lastError_);
sq_addref(vm_, &lastError_);
sq_pop(vm_, 1);
return true;
}
IScriptVM* makeSquirrelVM()
{
return new SquirrelVM;
}