263 lines
5.7 KiB
C
263 lines
5.7 KiB
C
#define LUA_LIB
|
|
|
|
#include <lua.h>
|
|
#include <lauxlib.h>
|
|
#include <lualib.h>
|
|
|
|
#include "lgc.h"
|
|
|
|
#ifdef makeshared
|
|
|
|
static void
|
|
mark_shared(lua_State *L) {
|
|
if (lua_type(L, -1) != LUA_TTABLE) {
|
|
luaL_error(L, "Not a table, it's a %s.", lua_typename(L, lua_type(L, -1)));
|
|
}
|
|
Table * t = (Table *)lua_topointer(L, -1);
|
|
if (isshared(t))
|
|
return;
|
|
makeshared(t);
|
|
luaL_checkstack(L, 4, NULL);
|
|
if (lua_getmetatable(L, -1)) {
|
|
luaL_error(L, "Can't share metatable");
|
|
}
|
|
lua_pushnil(L);
|
|
while (lua_next(L, -2) != 0) {
|
|
int i;
|
|
for (i=0;i<2;i++) {
|
|
int idx = -i-1;
|
|
int t = lua_type(L, idx);
|
|
switch (t) {
|
|
case LUA_TTABLE:
|
|
mark_shared(L);
|
|
break;
|
|
case LUA_TNUMBER:
|
|
case LUA_TBOOLEAN:
|
|
case LUA_TLIGHTUSERDATA:
|
|
break;
|
|
case LUA_TFUNCTION:
|
|
if (lua_getupvalue(L, idx, 1) != NULL) {
|
|
luaL_error(L, "Invalid function with upvalue");
|
|
} else if (!lua_iscfunction(L, idx)) {
|
|
LClosure *f = (LClosure *)lua_topointer(L, idx);
|
|
makeshared(f);
|
|
}
|
|
break;
|
|
case LUA_TSTRING:
|
|
lua_sharestring(L, idx);
|
|
break;
|
|
default:
|
|
luaL_error(L, "Invalid type [%s]", lua_typename(L, t));
|
|
break;
|
|
}
|
|
}
|
|
lua_pop(L, 1);
|
|
}
|
|
}
|
|
|
|
static int
|
|
lis_sharedtable(lua_State* L) {
|
|
int b = 0;
|
|
if(lua_type(L, 1) == LUA_TTABLE) {
|
|
Table * t = (Table *)lua_topointer(L, 1);
|
|
b = isshared(t);
|
|
}
|
|
lua_pushboolean(L, b);
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
make_matrix(lua_State *L) {
|
|
// turn off gc , because marking shared will prevent gc mark.
|
|
lua_gc(L, LUA_GCSTOP, 0);
|
|
mark_shared(L);
|
|
Table * t = (Table *)lua_topointer(L, -1);
|
|
lua_pushlightuserdata(L, t);
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
clone_table(lua_State *L) {
|
|
lua_clonetable(L, lua_touserdata(L, 1));
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
lco_stackvalues(lua_State* L) {
|
|
lua_State *cL = lua_tothread(L, 1);
|
|
luaL_argcheck(L, cL, 1, "thread expected");
|
|
int n = 0;
|
|
if(cL != L) {
|
|
luaL_checktype(L, 2, LUA_TTABLE);
|
|
n = lua_gettop(cL);
|
|
if(n > 0) {
|
|
luaL_checkstack(L, n+1, NULL);
|
|
int top = lua_gettop(L);
|
|
lua_xmove(cL, L, n);
|
|
int i=0;
|
|
for(i=1; i<=n; i++) {
|
|
lua_pushvalue(L, top+i);
|
|
lua_seti(L, 2, i);
|
|
}
|
|
lua_xmove(L, cL, n);
|
|
}
|
|
}
|
|
|
|
lua_pushinteger(L, n);
|
|
return 1;
|
|
}
|
|
|
|
|
|
struct state_ud {
|
|
lua_State *L;
|
|
};
|
|
|
|
static int
|
|
close_state(lua_State *L) {
|
|
struct state_ud *ud = (struct state_ud *)luaL_checkudata(L, 1, "BOXMATRIXSTATE");
|
|
if (ud->L) {
|
|
lua_close(ud->L);
|
|
ud->L = NULL;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
get_matrix(lua_State *L) {
|
|
struct state_ud *ud = (struct state_ud *)luaL_checkudata(L, 1, "BOXMATRIXSTATE");
|
|
if (ud->L) {
|
|
const void * v = lua_topointer(ud->L, 1);
|
|
lua_pushlightuserdata(L, (void *)v);
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
static int
|
|
get_size(lua_State *L) {
|
|
struct state_ud *ud = (struct state_ud *)luaL_checkudata(L, 1, "BOXMATRIXSTATE");
|
|
if (ud->L) {
|
|
lua_Integer sz = lua_gc(ud->L, LUA_GCCOUNT, 0);
|
|
sz *= 1024;
|
|
sz += lua_gc(ud->L, LUA_GCCOUNTB, 0);
|
|
lua_pushinteger(L, sz);
|
|
} else {
|
|
lua_pushinteger(L, 0);
|
|
}
|
|
return 1;
|
|
}
|
|
|
|
|
|
static int
|
|
box_state(lua_State *L, lua_State *mL) {
|
|
struct state_ud *ud = (struct state_ud *)lua_newuserdatauv(L, sizeof(*ud), 0);
|
|
ud->L = mL;
|
|
if (luaL_newmetatable(L, "BOXMATRIXSTATE")) {
|
|
lua_pushvalue(L, -1);
|
|
lua_setfield(L, -2, "__index");
|
|
lua_pushcfunction(L, close_state);
|
|
lua_setfield(L, -2, "close");
|
|
lua_pushcfunction(L, get_matrix);
|
|
lua_setfield(L, -2, "getptr");
|
|
lua_pushcfunction(L, get_size);
|
|
lua_setfield(L, -2, "size");
|
|
}
|
|
lua_setmetatable(L, -2);
|
|
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
load_matrixfile(lua_State *L) {
|
|
luaL_openlibs(L);
|
|
const char * source = (const char *)lua_touserdata(L, 1);
|
|
if (source[0] == '@') {
|
|
if (luaL_loadfilex_(L, source+1, NULL) != LUA_OK)
|
|
lua_error(L);
|
|
} else {
|
|
if (luaL_loadstring(L, source) != LUA_OK)
|
|
lua_error(L);
|
|
}
|
|
lua_replace(L, 1);
|
|
if (lua_pcall(L, lua_gettop(L) - 1, 1, 0) != LUA_OK)
|
|
lua_error(L);
|
|
lua_gc(L, LUA_GCCOLLECT, 0);
|
|
lua_pushcfunction(L, make_matrix);
|
|
lua_insert(L, -2);
|
|
lua_call(L, 1, 1);
|
|
return 1;
|
|
}
|
|
|
|
static int
|
|
matrix_from_file(lua_State *L) {
|
|
lua_State *mL = luaL_newstate();
|
|
if (mL == NULL) {
|
|
return luaL_error(L, "luaL_newstate failed");
|
|
}
|
|
const char * source = luaL_checkstring(L, 1);
|
|
int top = lua_gettop(L);
|
|
lua_pushcfunction(mL, load_matrixfile);
|
|
lua_pushlightuserdata(mL, (void *)source);
|
|
if (top > 1) {
|
|
if (!lua_checkstack(mL, top + 1)) {
|
|
return luaL_error(L, "Too many argument %d", top);
|
|
}
|
|
int i;
|
|
for (i=2;i<=top;i++) {
|
|
switch(lua_type(L, i)) {
|
|
case LUA_TBOOLEAN:
|
|
lua_pushboolean(mL, lua_toboolean(L, i));
|
|
break;
|
|
case LUA_TNUMBER:
|
|
if (lua_isinteger(L, i)) {
|
|
lua_pushinteger(mL, lua_tointeger(L, i));
|
|
} else {
|
|
lua_pushnumber(mL, lua_tonumber(L, i));
|
|
}
|
|
break;
|
|
case LUA_TLIGHTUSERDATA:
|
|
lua_pushlightuserdata(mL, lua_touserdata(L, i));
|
|
break;
|
|
case LUA_TFUNCTION:
|
|
if (lua_iscfunction(L, i) && lua_getupvalue(L, i, 1) == NULL) {
|
|
lua_pushcfunction(mL, lua_tocfunction(L, i));
|
|
break;
|
|
}
|
|
return luaL_argerror(L, i, "Only support light C function");
|
|
default:
|
|
return luaL_argerror(L, i, "Type invalid");
|
|
}
|
|
}
|
|
}
|
|
int ok = lua_pcall(mL, top, 1, 0);
|
|
if (ok != LUA_OK) {
|
|
lua_pushstring(L, lua_tostring(mL, -1));
|
|
lua_close(mL);
|
|
lua_error(L);
|
|
}
|
|
return box_state(L, mL);
|
|
}
|
|
|
|
LUAMOD_API int
|
|
luaopen_skynet_sharetable_core(lua_State *L) {
|
|
luaL_checkversion(L);
|
|
luaL_Reg l[] = {
|
|
{ "clone", clone_table },
|
|
{ "stackvalues", lco_stackvalues },
|
|
{ "matrix", matrix_from_file },
|
|
{ "is_sharedtable", lis_sharedtable },
|
|
{ NULL, NULL },
|
|
};
|
|
luaL_newlib(L, l);
|
|
return 1;
|
|
}
|
|
|
|
#else
|
|
|
|
LUAMOD_API int
|
|
luaopen_skynet_sharetable_core(lua_State *L) {
|
|
return luaL_error(L, "No share string table support");
|
|
}
|
|
|
|
#endif |