summaryrefslogtreecommitdiff
path: root/includes
diff options
context:
space:
mode:
Diffstat (limited to 'includes')
-rw-r--r--includes/BLua.h335
1 files changed, 335 insertions, 0 deletions
diff --git a/includes/BLua.h b/includes/BLua.h
new file mode 100644
index 0000000..addd081
--- /dev/null
+++ b/includes/BLua.h
@@ -0,0 +1,335 @@
+#pragma once
+
+#include <typeinfo>
+
+extern "C" {
+#include <lua.h>
+#include <lauxlib.h>
+}
+
+#include <Exceptions.h>
+#include <Handle.h>
+
+namespace Balau {
+
+class Lua;
+
+class LuaExport {
+ public:
+ virtual ~LuaExport() { }
+};
+
+class LuaObject {
+ public:
+ LuaObject() : m_wantsDestruct(false), m_pushed(false) { }
+ virtual void push(Lua & L) throw (GeneralException);
+ void pushDestruct(Lua & L) throw (GeneralException);
+ template<class T>
+ static T * getMe(Lua & L, int idx = 1);
+ protected:
+ virtual void pushMembers(Lua & L) = 0;
+ void pushMe(Lua & L, void * obj = 0, const char * name = NULL, bool isObj = true);
+ static void pushIt(Lua & L, const char * name, lua_CFunction func);
+ static void pushMeta(Lua & L, const char * name, lua_CFunction func);
+ static void * getMeInternal(Lua & L, int idx);
+ friend class Lua;
+ private:
+ bool m_wantsDestruct, m_pushed;
+};
+
+typedef int (*openlualib_t)(lua_State * L);
+
+class Lua {
+ public:
+ Lua();
+ Lua(lua_State * __L) : L(__L) { }
+ Lua(const Lua &) throw (GeneralException) { throw GeneralException("Error: can't duplicate a Lua object."); }
+
+ typedef int (*lua_CallWrapper)(lua_State *, lua_CFunction);
+
+ int ref(int t = -2) { luaL_ref(L, t); }
+ void unref(int ref, int t = -1) { luaL_unref(L, t, ref); }
+
+ void open_base();
+ void open_table();
+ void open_string();
+ void open_math();
+ void open_debug();
+ void open_bit();
+ void open_jit();
+ int wrap_open(openlualib_t open) { int n = gettop(); int r = open(L); while (n < gettop()) remove(n); }
+ void openlib(const String & libname, const struct luaL_reg *l, int nup) { luaL_openlib(L, libname.to_charp(), l, nup); }
+
+ void setCallWrap(lua_CallWrapper wrapper);
+ void declareFunc(const char * funcName, lua_CFunction f, int tableIdx = LUA_GLOBALSINDEX);
+
+ void call(const char * funcName, int tableIdx = LUA_GLOBALSINDEX, int nArgs = 0);
+ void call(int nArgs = 0) { resume(nArgs); }
+
+ void push() { checkstack(); lua_pushnil(L); }
+ void push(lua_Number n) { checkstack(); lua_pushnumber(L, n); }
+ void push(const String & s) { checkstack(); lua_pushlstring(L, s.to_charp(), s.strlen()); }
+ void push(bool b) { checkstack(); lua_pushboolean(L, b); }
+ void push(const char * str, int size = -1) { if (size < 0) size = strlen(str); checkstack(); lua_pushlstring(L, str, size); }
+ void push(void * p) { checkstack(); lua_pushlightuserdata(L, p); }
+ void push(lua_CFunction f, int n = 0) { checkstack(); lua_pushcclosure(L, f, n); }
+ void pop(int idx = 1) { lua_pop(L, idx); }
+ int checkstack(int extra = 1) { return lua_checkstack(L, extra); }
+
+ int next(int t = -2) { return lua_next(L, t); }
+ void copy(int n = -1) { checkstack(); lua_pushvalue(L, n); }
+ void remove(int n = 1) { lua_remove(L, n); }
+ void insert(int n = 1) { checkstack(); lua_insert(L, n); }
+ void replace(int n = 1) { lua_replace(L, n); }
+ void newtable() { checkstack(); lua_newtable(L); }
+ void * newuser(size_t s) { checkstack(); return lua_newuserdata(L, s); }
+ void settable(int tableIdx = -3, bool raw = false);
+ void gettable(int tableIdx = -2, bool raw = false);
+ void rawseti(int idx, int tableIdx = -2) { lua_rawseti(L, tableIdx, idx); }
+ void rawgeti(int idx, int tableIdx = -1) { lua_rawgeti(L, tableIdx, idx); }
+ void setvar() { lua_settable(L, LUA_GLOBALSINDEX); }
+ int gettop() { return lua_gettop(L); }
+ void getglobal(const char * name) throw (GeneralException);
+ void pushLuaContext();
+ void error(const char * msg);
+ void error(const String & msg) { error(msg.to_charp()); }
+
+ int type(int i = -1) { return lua_type(L, i); }
+ bool isnil(int i = -1) { return lua_isnil(L, i); }
+ bool isboolean(int i = -1) { return lua_isboolean(L, i); }
+ bool isnumber(int i = -1) { return lua_isnumber(L, i); }
+ bool isstring(int i = -1) { return lua_isstring(L, i); }
+ bool istable(int i = -1) { return lua_istable(L, i); }
+ bool isfunction(int i = -1) { return lua_isfunction(L, i); }
+ bool iscfunction(int i = -1) { return lua_iscfunction(L, i); }
+ bool isuserdata(int i = -1) { return lua_isuserdata(L, i); }
+ bool islightuserdata(int i = -1) { return lua_islightuserdata(L, i); }
+ bool isobject(int i = -1);
+
+ bool toboolean(int i = -1) { return lua_toboolean(L, i); }
+ lua_Number tonumber(int i = -1) { return lua_tonumber(L, i); }
+ String tostring(int i = -1);
+ lua_CFunction tocfunction(int i = -1) { return lua_tocfunction(L, i); }
+ void * touserdata(int i = -1) { return lua_touserdata(L, i); }
+
+ String escapeString(const String &);
+ void load(IO<Handle> in, bool docall = true) throw (GeneralException);
+ void load(const String &, bool docall = true) throw (GeneralException);
+ void dumpvars(IO<Handle> out, const String & prefix, int idx = -1);
+ Lua thread(bool saveit = true);
+ Lua thread(const String &, int nargs = 0, bool saveit = true);
+ Lua thread(IO<Handle> in, int nargs = 0, bool saveit = true);
+ int yield(int nresults = 0) { return lua_yield(L, nresults); }
+ bool resume(int nargs = 0) throw (GeneralException);
+ bool resume(const String &, int nargs = 0);
+ bool resume(IO<Handle> in, int nargs = 0);
+ void showstack(int level = M_INFO);
+ void showerror();
+ int getmetatable(int i = -1) { checkstack(); return lua_getmetatable(L, i); }
+ int setmetatable(int i = -2) { return lua_setmetatable(L, i); }
+ int sethook(lua_Hook func, int mask, int count) { return lua_sethook(L, func, mask, count); }
+ void weaken();
+
+ template<class T>
+ T * recast(int n = 1) {
+ LuaExport * b;
+ T * r;
+
+ b = (LuaExport *) LuaObject::getMeInternal(*this, n);
+ if (!b)
+ error("LuaExport base object required; got null.");
+
+ r = dynamic_cast<T *>(b);
+
+ if (!r)
+ error(String("Object not compatible; expecting ") + typeid(r).name() + " but got *" + typeid(*b).name() + " instead.");
+
+ return r;
+ }
+
+ lua_State * getState() { return L; }
+ protected:
+
+ private:
+ void dumpvars_r(IO<Handle> out, int idx, int depth = 0) throw (GeneralException);
+
+ lua_State * L;
+
+ friend class LuaStatics;
+};
+
+class LuaException : public GeneralException {
+ public:
+ LuaException(String fn) : GeneralException(fn) { }
+ protected:
+ LuaException() { }
+};
+
+enum Lua_types_t {
+ BLUA_OBJECT = 0x01,
+ BLUA_TABLE = 0x02,
+ BLUA_BOOLEAN = 0x04,
+ BLUA_NUMBER = 0x08,
+ BLUA_STRING = 0x10,
+ BLUA_FUNCTION = 0x20,
+ BLUA_NIL = 0x40,
+ BLUA_USERDATA = 0x80,
+ BLUA_ANY = 0xff,
+};
+
+#define MAX_TYPE 8
+
+#define MAXARGS 32
+
+struct lua_functypes_t {
+ int number;
+ const char * name;
+ int minargs, maxargs;
+ int argtypes[MAXARGS];
+};
+
+#define DECLARE_METHOD(classname, enumvar) static int method_##enumvar(lua_State * L) { \
+ return LuaHelpers<classname>::method_multiplex( \
+ enumvar, \
+ L, \
+ sLua_##classname::classname##_proceed, \
+ 0, \
+ classname##_methods, \
+ true); \
+ }
+
+#define DECLARE_FUNCTION(classname, enumvar) static int function_##enumvar(lua_State * L) { \
+ return LuaHelpers<classname>::method_multiplex( \
+ enumvar, \
+ L, \
+ 0, \
+ sLua_##classname::classname##_proceed_statics, \
+ classname##_functions, \
+ false); \
+ }
+
+#define PUSH_METHOD(classname, enumvar) pushit( \
+ L, \
+ classname##_methods[enumvar].name, \
+ sLua_##classname::method_##enumvar)
+
+#define PUSH_METAMETHOD(classname, enumvar) pushmeta( \
+ L, \
+ String("__") + classname##_methods[enumvar].name, \
+ sLua_##classname::method_##enumvar)
+
+#define PUSH_FUNCTION(classname, enumvar) L->declarefunc( \
+ classname##_functions[enumvar].name, \
+ sLua_##classname::function_##enumvar)
+
+#define PUSH_SUBFUNCTION(classname, enumvar, array) L->declarefunc( \
+ classname##_functions[enumvar].name, \
+ sLua_##classname::function_##enumvar, \
+ array)
+
+
+#define CHECK_METHODS(classname) { \
+ int i = 0; \
+ while (classname##_methods[i].number != -1) { \
+ if (i != classname##_methods[i].number) { \
+ throw GeneralException("Data of " #classname "_methods inconsistants!"); \
+ } \
+ i++; \
+ } \
+}
+
+#define CHECK_FUNCTIONS(classname) { \
+ int i = 0; \
+ while (classname##_functions[i].number != -1) { \
+ if (i != classname##_functions[i].number) { \
+ throw GeneralException("Data of " #classname "_functions inconsistants!"); \
+ } \
+ i++; \
+ } \
+}
+
+template <class T>
+T * LuaObject::getMe(Lua & L, int idx) { return L.recast<T>(idx); }
+
+template <class T>
+class LuaHelpers {
+ public:
+ static int method_multiplex(int caller, lua_State * __L, int (*proceed)(Lua & L, int n, T * obj, int caller), int (*proceed_static)(Lua & L, int n, int caller), lua_functypes_t * tab, bool method) {
+ Lua L(__L);
+ int add = method ? 1 : 0;
+ int n = L.gettop() - add;
+ T * obj = 0;
+ int i, j, mask;
+ bool invalid = false, arg_valid;
+
+ if (method)
+ obj = LuaObject::getMe<T>(L);
+
+ if ((n < tab[caller].minargs) || (n > tab[caller].maxargs)) {
+ invalid = true;
+ } else {
+ for (i = 0; i < tab[caller].maxargs && !invalid; i++) {
+ if (n >= (i + 1)) {
+ arg_valid = false;
+ for (j = 0; j < MAX_TYPE && !arg_valid; j++) {
+ mask = 1 << j;
+ if (tab[caller].argtypes[i] & mask) {
+ switch(mask) {
+ case BLUA_OBJECT:
+ if (L.istable(i + 1 + add)) {
+ L.push("__obj");
+ L.gettable(i + 1 + add);
+ arg_valid = L.isuserdata();
+ L.pop();
+ } else {
+ arg_valid = L.isnil(i + 1 + add);
+ }
+ break;
+ case BLUA_TABLE:
+ arg_valid = L.istable(i + 1 + add);
+ break;
+ case BLUA_BOOLEAN:
+ arg_valid = L.isboolean(i + 1 + add);
+ break;
+ case BLUA_NUMBER:
+ arg_valid = L.isnumber(i + 1 + add);
+ break;
+ case BLUA_STRING:
+ arg_valid = L.isstring(i + 1 + add);
+ break;
+ case BLUA_FUNCTION:
+ arg_valid = L.isfunction(i + 1 + add);
+ break;
+ case BLUA_NIL:
+ arg_valid = L.isnil(i + 1 + add);
+ break;
+ case BLUA_USERDATA:
+ arg_valid = L.isuserdata(i + 1 + add) || L.islightuserdata(i + 1 + add);
+ break;
+ }
+ }
+ }
+ invalid = !arg_valid;
+ }
+ }
+ }
+
+ if (invalid) {
+ if (method) {
+ L.error(String("Invalid arguments to method `") + typeid(T).name() + "::" + tab[caller].name + "'");
+ } else {
+ L.error(String("Invalid arguments to function `") + typeid(T).name() + " " + tab[caller].name + "'");
+ }
+ }
+
+ if (method) {
+ return proceed(L, n, obj, caller);
+ } else {
+ return proceed_static(L, n, caller);
+ }
+ }
+};
+
+template<class T> T * lua_recast(Lua & L, int n = 1) { return L.recast<T>(n); }
+
+};