Change make kiwmi an object, use pcall for Lua functions

This commit is contained in:
buffet 2019-12-31 15:19:42 +00:00
parent 21f726c3ce
commit ab1e95bef3

View file

@ -71,7 +71,6 @@ lua_callback_register(lua_State *L)
lua_pushvalue(L, -2); lua_pushvalue(L, -2);
lua_settable(L, -3); lua_settable(L, -3);
luaL_setfuncs(L, lua_callback_methods, 0); luaL_setfuncs(L, lua_callback_methods, 0);
lua_pop(L, 1);
return 0; return 0;
} }
@ -126,7 +125,6 @@ kiwmi_view_create(lua_State *L)
luaL_checktype(L, 1, LUA_TLIGHTUSERDATA); // kiwmi_view luaL_checktype(L, 1, LUA_TLIGHTUSERDATA); // kiwmi_view
struct kiwmi_view *view = lua_touserdata(L, 1); struct kiwmi_view *view = lua_touserdata(L, 1);
lua_pop(L, 1);
struct kiwmi_view **view_ud = lua_newuserdata(L, sizeof(*view_ud)); struct kiwmi_view **view_ud = lua_newuserdata(L, sizeof(*view_ud));
luaL_getmetatable(L, "kiwmi_view"); luaL_getmetatable(L, "kiwmi_view");
@ -145,40 +143,40 @@ kiwmi_view_register(lua_State *L)
lua_pushvalue(L, -2); lua_pushvalue(L, -2);
lua_settable(L, -3); lua_settable(L, -3);
luaL_setfuncs(L, kiwmi_view_methods, 0); luaL_setfuncs(L, kiwmi_view_methods, 0);
lua_pop(L, 1);
return 0; return 0;
} }
static struct kiwmi_server *
get_server(lua_State *L)
{
lua_pushliteral(L, "kiwmi_server");
lua_gettable(L, LUA_REGISTRYINDEX);
struct kiwmi_server *server = lua_touserdata(L, -1);
lua_pop(L, 1);
return server;
}
static int static int
l_on(lua_State *L) l_on(lua_State *L)
{ {
luaL_checktype(L, 1, LUA_TSTRING); // type luaL_checkudata(L, 1, "kiwmi_server"); // server
luaL_checktype(L, 2, LUA_TFUNCTION); // callback luaL_checktype(L, 2, LUA_TSTRING); // type
luaL_checktype(L, 3, LUA_TFUNCTION); // callback
// event_handler = registry['kiwmi_events'][type] // event_handler = registry['kiwmi_events'][type]
lua_pushliteral(L, "kiwmi_events"); lua_pushliteral(L, "kiwmi_events");
lua_gettable(L, LUA_REGISTRYINDEX); lua_gettable(L, LUA_REGISTRYINDEX);
lua_pushvalue(L, 1); lua_pushvalue(L, 2);
lua_gettable(L, -2); lua_gettable(L, -2);
lua_CFunction register_callback = lua_tocfunction(L, -1);
lua_pop(L, 2);
luaL_argcheck(L, register_callback, 1, "invalid event"); luaL_argcheck(L, lua_iscfunction(L, -1), 2, "invalid event");
// stack: server type callback kiwmi_events register_callback
lua_remove(L, 4); // remove kiwmi_events
lua_remove(L, 2); // remove type
lua_rotate(L, 1, 1);
// stack: register_callback server callback
// return register_callback(callback) // return register_callback(callback)
return register_callback(L); if (lua_pcall(L, 2, 1, 0)) {
wlr_log(WLR_ERROR, "%s", lua_tostring(L, -1));
return 0;
}
return 1;
} }
static void static void
@ -213,12 +211,15 @@ on_cursor_button_notify(struct wl_listener *listener, void *data)
static int static int
on_cursor_button(lua_State *L) on_cursor_button(lua_State *L)
{ {
struct kiwmi_server *server =
*(struct kiwmi_server **)luaL_checkudata(L, 1, "kiwmi_server");
luaL_checktype(L, 2, LUA_TFUNCTION); // callback
struct lua_callback *lc = malloc(sizeof(*lc)); struct lua_callback *lc = malloc(sizeof(*lc));
if (!lc) { if (!lc) {
return luaL_error(L, "failed to allocate lua_callback"); return luaL_error(L, "failed to allocate lua_callback");
} }
struct kiwmi_server *server = get_server(L);
struct kiwmi_cursor *cursor = server->input.cursor; struct kiwmi_cursor *cursor = server->input.cursor;
wl_list_insert(&server->lua->callbacks, &lc->link); wl_list_insert(&server->lua->callbacks, &lc->link);
@ -238,7 +239,7 @@ on_cursor_button(lua_State *L)
static int static int
l_quit(lua_State *L) l_quit(lua_State *L)
{ {
struct kiwmi_server *server = get_server(L); struct kiwmi_server *server = *(struct kiwmi_server**)luaL_checkudata(L, 1, "kiwmi_server");
wl_display_terminate(server->wl_display); wl_display_terminate(server->wl_display);
@ -248,7 +249,8 @@ l_quit(lua_State *L)
static int static int
l_view_under_cursor(lua_State *L) l_view_under_cursor(lua_State *L)
{ {
struct kiwmi_server *server = get_server(L); struct kiwmi_server *server = *(struct kiwmi_server **)luaL_checkudata(L, 1, "kiwmi_server");
struct kiwmi_cursor *cursor = server->input.cursor; struct kiwmi_cursor *cursor = server->input.cursor;
struct wlr_surface *surface; struct wlr_surface *surface;
@ -264,8 +266,11 @@ l_view_under_cursor(lua_State *L)
&sy); &sy);
if (view) { if (view) {
lua_pushcfunction(L, kiwmi_view_create);
lua_pushlightuserdata(L, view); lua_pushlightuserdata(L, view);
kiwmi_view_create(L); if (lua_pcall(L, 1, 1, 0)) {
wlr_log(WLR_ERROR, "%s", lua_tostring(L, -1));
}
} else { } else {
lua_pushnil(L); lua_pushnil(L);
} }
@ -273,13 +278,41 @@ l_view_under_cursor(lua_State *L)
return 1; return 1;
} }
static const luaL_Reg kiwmi_lib[] = { static const luaL_Reg kiwmi_server_methods[] = {
{"on", l_on}, {"on", l_on},
{"quit", l_quit}, {"quit", l_quit},
{"view_under_cursor", l_view_under_cursor}, {"view_under_cursor", l_view_under_cursor},
{NULL, NULL}, {NULL, NULL},
}; };
static int
kiwmi_server_create(lua_State *L)
{
luaL_checktype(L, 1, LUA_TLIGHTUSERDATA); // kiwmi_server
struct kiwmi_server *server = lua_touserdata(L, 1);
struct kiwmi_server **server_ud = lua_newuserdata(L, sizeof(*server_ud));
luaL_getmetatable(L, "kiwmi_server");
lua_setmetatable(L, -2);
*server_ud = server;
return 1;
}
static int
kiwmi_server_register(lua_State *L)
{
luaL_newmetatable(L, "kiwmi_server");
lua_pushliteral(L, "__index");
lua_pushvalue(L, -2);
lua_settable(L, -3);
luaL_setfuncs(L, kiwmi_server_methods, 0);
return 0;
}
static const luaL_Reg kiwmi_events[] = { static const luaL_Reg kiwmi_events[] = {
{"cursor_button", on_cursor_button}, {"cursor_button", on_cursor_button},
{NULL, NULL}, {NULL, NULL},
@ -301,22 +334,33 @@ luaK_init(struct kiwmi_server *server)
luaL_openlibs(L); luaL_openlibs(L);
// registry['kiwmi_server'] = server
lua_pushliteral(L, "kiwmi_server");
lua_pushlightuserdata(L, server);
lua_settable(L, LUA_REGISTRYINDEX);
// registry['kiwmi_events'] = {'key_down' = l_key_down, ...} // registry['kiwmi_events'] = {'key_down' = l_key_down, ...}
lua_pushliteral(L, "kiwmi_events"); lua_pushliteral(L, "kiwmi_events");
luaL_newlib(L, kiwmi_events); luaL_newlib(L, kiwmi_events);
lua_settable(L, LUA_REGISTRYINDEX); lua_settable(L, LUA_REGISTRYINDEX);
// register types // register types
lua_callback_register(L); int error = false;
kiwmi_view_register(L);
// _G['kiwmi'] = kiwmi_lib lua_pushcfunction(L, lua_callback_register);
luaL_newlib(L, kiwmi_lib); error |= lua_pcall(L, 0, 0, 0);
lua_pushcfunction(L, kiwmi_view_register);
error |= lua_pcall(L, 0, 0, 0);
lua_pushcfunction(L, kiwmi_server_register);
error |= lua_pcall(L, 0, 0, 0);
if (error) {
wlr_log(WLR_ERROR, "%s", lua_tostring(L, -1));
return false;
}
// create kiwmi global
lua_pushcfunction(L, kiwmi_server_create);
lua_pushlightuserdata(L, server);
if (lua_pcall(L, 1, 1, 0)) {
wlr_log(WLR_ERROR, "%s", lua_tostring(L, -1));
return false;
}
lua_setglobal(L, "kiwmi"); lua_setglobal(L, "kiwmi");
lua->L = L; lua->L = L;