aboutsummaryrefslogtreecommitdiffstats
path: root/src
diff options
context:
space:
mode:
Diffstat (limited to 'src')
-rw-r--r--src/buffer_event.c255
-rw-r--r--src/event_buffer.c8
-rw-r--r--src/luaevent.c11
-rw-r--r--src/utility.c45
4 files changed, 315 insertions, 4 deletions
diff --git a/src/buffer_event.c b/src/buffer_event.c
new file mode 100644
index 0000000..4e20353
--- /dev/null
+++ b/src/buffer_event.c
@@ -0,0 +1,255 @@
+/* LuaEvent - Copyright (C) 2007 Thomas Harning <harningt@gmail.com>
+ * Licensed as LGPL - See doc/COPYING for details */
+#include "buffer_event.h"
+#include "luaevent.h"
+#include "utility.h"
+#include <lauxlib.h>
+#include <malloc.h>
+#include "event_buffer.h"
+
+#define BUFFER_EVENT_MT "BUFFER_EVENT_MT"
+
+/* Locations of READ/WRITE buffers in the fenv */
+#define READ_BUFFER_LOCATION 4
+#define WRITE_BUFFER_LOCATION 5
+
+/* Obtains an le_bufferevent structure from a given index */
+static le_bufferevent* buffer_event_get(lua_State* L, int idx) {
+ return (le_bufferevent*)luaL_checkudata(L, idx, BUFFER_EVENT_MT);
+}
+
+/* Obtains an le_bufferevent structure from a given index
+ AND checks that it hadn't been prematurely freed
+*/
+le_bufferevent* buffer_event_check(lua_State* L, int idx) {
+ le_bufferevent* buf = (le_bufferevent*)luaL_checkudata(L, idx, BUFFER_EVENT_MT);
+ if(!buf->ev)
+ luaL_argerror(L, idx, "Attempt to use closed buffer_event object");
+ return buf;
+}
+
+/* Checks if the given index contains an le_buffer object */
+int is_buffer_event(lua_State* L, int idx) {
+ int ret;
+ lua_getmetatable(L, idx);
+ luaL_getmetatable(L, BUFFER_EVENT_MT);
+ ret = lua_rawequal(L, -2, -1);
+ lua_pop(L, 2);
+ return ret;
+}
+
+static void handle_callback(le_bufferevent* le_ev, short what, int callbackIndex) {
+ lua_State* L = le_ev->base->loop_L;
+ le_weak_get(L, le_ev);
+ lua_getfenv(L, -1);
+ lua_rawgeti(L, -1, callbackIndex);
+ lua_remove(L, -2);
+ lua_pushvalue(L, -2);
+ lua_remove(L, -3);
+ /* func, bufferevent */
+ lua_pushinteger(L, what);
+ /* What to do w/ errors...? */
+ lua_pcall(L, 2, 0, 0);
+}
+
+static void buffer_event_readcb(struct bufferevent *ev, void *ptr) {
+ handle_callback((le_bufferevent*)ptr, EVBUFFER_READ, 1);
+}
+
+static void buffer_event_writecb(struct bufferevent *ev, void *ptr) {
+ handle_callback((le_bufferevent*)ptr, EVBUFFER_WRITE, 2);
+}
+
+static void buffer_event_errorcb(struct bufferevent *ev, short what, void *ptr) {
+ handle_callback((le_bufferevent*)ptr, what, 3);
+}
+
+/* LUA: new(fd, read, write, error)
+ Pushes a new bufferevent instance on the stack
+ Accepts: base, fd, read, write, error cb
+ Requires base, fd and error cb
+*/
+static int buffer_event_push(lua_State* L) {
+ le_bufferevent *ev;
+ le_base* base = event_base_get(L, 1);
+ /* NOTE: Should probably reference the socket as well... */
+ int fd = getSocketFd(L, 2);
+ luaL_checktype(L, 5, LUA_TFUNCTION);
+ if(!lua_isnil(L, 3)) luaL_checktype(L, 3, LUA_TFUNCTION);
+ if(!lua_isnil(L, 4)) luaL_checktype(L, 4, LUA_TFUNCTION);
+ ev= (le_bufferevent*)lua_newuserdata(L, sizeof(le_bufferevent));
+ luaL_getmetatable(L, BUFFER_EVENT_MT);
+ lua_setmetatable(L, -2);
+ ev->ev = bufferevent_new(fd, buffer_event_readcb, buffer_event_writecb, buffer_event_errorcb, ev);
+ lua_createtable(L, 5, 0);
+ lua_pushvalue(L, 3);
+ lua_rawseti(L, -2, 1); // Read
+ lua_pushvalue(L, 4);
+ lua_rawseti(L, -2, 2); // Write
+ lua_pushvalue(L, 5);
+ lua_rawseti(L, -2, 3); // Err
+
+ event_buffer_push(L, ev->ev->input);
+ lua_rawseti(L, -2, READ_BUFFER_LOCATION);
+ event_buffer_push(L, ev->ev->output);
+ lua_rawseti(L, -2, WRITE_BUFFER_LOCATION);
+ lua_setfenv(L, -2);
+ ev->base = base;
+ return 1;
+}
+
+/* LUA: __gc and buffer:close()
+ Releases the buffer resources
+*/
+static int buffer_event_gc(lua_State* L) {
+ le_bufferevent* ev = buffer_event_get(L, 1);
+ if(ev->ev) {
+ le_buffer *read, *write;
+ bufferevent_free(ev->ev);
+ ev->ev = NULL;
+ /* Also clear out the associated input/output event_buffers
+ * since they would have already been freed.. */
+ lua_getfenv(L, 1);
+ lua_rawgeti(L, -1, READ_BUFFER_LOCATION);
+ lua_rawgeti(L, -2, WRITE_BUFFER_LOCATION);
+ read = event_buffer_check(L, -2);
+ write = event_buffer_check(L, -1);
+ /* Erase Lua's link to the buffers */
+ lua_pushnil(L);
+ /* LS: ..., fenv, readBuf, writeBuf, nil */
+ lua_rawseti(L, -4, READ_BUFFER_LOCATION);
+ lua_pushnil(L);
+ lua_rawseti(L, -4, WRITE_BUFFER_LOCATION);
+ /* Erase their knowledge of the buffer so that the GC won't try to double-free */
+ read->buffer = NULL;
+ write->buffer = NULL;
+ }
+ return 0;
+}
+
+static int buffer_event_get_read(lua_State* L) {
+ (void)buffer_event_get(L, 1);
+ lua_getfenv(L, 1);
+ lua_rawgeti(L, -1, READ_BUFFER_LOCATION);
+ return 1;
+}
+
+static int buffer_event_get_write(lua_State* L) {
+ (void)buffer_event_get(L, 1);
+ lua_getfenv(L, 1);
+ lua_rawgeti(L, -1, WRITE_BUFFER_LOCATION);
+ return 1;
+}
+
+static int buffer_event_set_read_watermarks(lua_State* L) {
+ int low, high;
+ le_bufferevent* ev = buffer_event_get(L, 1);
+ if(!ev->ev) return 0;
+
+ low = lua_tonumber(L, 2);
+ high = lua_tonumber(L, 3);
+
+ ev->ev->wm_read.low = low;
+ ev->ev->wm_read.high = high;
+ return 0;
+}
+
+static int buffer_event_set_write_watermarks(lua_State* L) {
+ int low, high;
+ le_bufferevent* ev = buffer_event_get(L, 1);
+ if(!ev->ev) return 0;
+
+ low = lua_tonumber(L, 2);
+ high = lua_tonumber(L, 3);
+
+ ev->ev->wm_write.low = low;
+ ev->ev->wm_write.high = high;
+ return 0;
+}
+
+static int buffer_event_get_read_watermarks(lua_State* L) {
+ le_bufferevent* ev = buffer_event_get(L, 1);
+ if(!ev->ev) return 0;
+
+ lua_pushinteger(L, ev->ev->wm_read.low);
+ lua_pushinteger(L, ev->ev->wm_read.high);
+ return 2;
+}
+
+static int buffer_event_get_write_watermarks(lua_State* L) {
+ le_bufferevent* ev = buffer_event_get(L, 1);
+ if(!ev->ev) return 0;
+
+ lua_pushinteger(L, ev->ev->wm_write.low);
+ lua_pushinteger(L, ev->ev->wm_write.high);
+ return 2;
+}
+
+static int buffer_event_set_timeouts(lua_State* L) {
+ int timeout_read, timeout_write;
+ le_bufferevent* ev = buffer_event_get(L, 1);
+ if(!ev->ev) return 0;
+
+ timeout_read = lua_tointeger(L, 2);
+ timeout_write = lua_tointeger(L, 3);
+
+ bufferevent_settimeout(ev->ev, timeout_read, timeout_write);
+ return 0;
+}
+
+static int buffer_event_get_timeouts(lua_State* L) {
+ le_bufferevent* ev = buffer_event_get(L, 1);
+ if(!ev->ev) return 0;
+
+ lua_pushinteger(L, ev->ev->timeout_read);
+ lua_pushinteger(L, ev->ev->timeout_write);
+ return 2;
+}
+
+static int buffer_event_enable(lua_State* L) {
+ le_bufferevent* ev = buffer_event_get(L, 1);
+ if(!ev->ev) return 0;
+
+ lua_pushinteger(L, bufferevent_enable(ev->ev, luaL_checkinteger(L, 2)));
+ return 1;
+}
+
+static int buffer_event_disable(lua_State* L) {
+ le_bufferevent* ev = buffer_event_get(L, 1);
+ if(!ev->ev) return 0;
+
+ lua_pushinteger(L, bufferevent_disable(ev->ev, luaL_checkinteger(L, 2)));
+ return 1;
+}
+
+static luaL_Reg buffer_event_funcs[] = {
+ {"get_read", buffer_event_get_read},
+ {"get_write", buffer_event_get_write},
+ {"set_read_watermarks", buffer_event_set_read_watermarks},
+ {"set_write_watermarks", buffer_event_set_write_watermarks},
+ {"get_read_watermarks", buffer_event_get_read_watermarks},
+ {"get_write_watermarks", buffer_event_get_write_watermarks},
+ {"set_timeouts", buffer_event_set_timeouts},
+ {"get_timeouts", buffer_event_get_timeouts},
+ {"enable", buffer_event_enable},
+ {"disable", buffer_event_disable},
+ {NULL, NULL}
+};
+
+static luaL_Reg funcs[] = {
+ {"new", buffer_event_push},
+ {NULL, NULL}
+};
+
+int buffer_event_register(lua_State* L) {
+ luaL_newmetatable(L, BUFFER_EVENT_MT);
+ lua_pushcfunction(L, buffer_event_gc);
+ lua_setfield(L, -2, "__gc");
+ lua_newtable(L);
+ luaL_register(L, NULL, buffer_event_funcs);
+ lua_setfield(L, -2, "__index");
+ lua_pop(L, 1);
+
+ luaL_register(L, "luaevent.core.bufferevent", funcs);
+ return 1;
+}
diff --git a/src/event_buffer.c b/src/event_buffer.c
index 72c6abe..62dc0e3 100644
--- a/src/event_buffer.c
+++ b/src/event_buffer.c
@@ -18,7 +18,7 @@ static le_buffer* event_buffer_get(lua_State* L, int idx) {
/* Obtains an le_buffer structure from a given index
AND checks that it hadn't been prematurely freed
*/
-static le_buffer* event_buffer_check(lua_State* L, int idx) {
+le_buffer* event_buffer_check(lua_State* L, int idx) {
le_buffer* buf = (le_buffer*)luaL_checkudata(L, idx, EVENT_BUFFER_MT);
if(!buf->buffer)
luaL_argerror(L, idx, "Attempt to use closed event_buffer object");
@@ -26,7 +26,7 @@ static le_buffer* event_buffer_check(lua_State* L, int idx) {
}
/* Checks if the given index contains an le_buffer object */
-static int is_event_buffer(lua_State* L, int idx) {
+int is_event_buffer(lua_State* L, int idx) {
int ret;
lua_getmetatable(L, idx);
luaL_getmetatable(L, EVENT_BUFFER_MT);
@@ -37,7 +37,7 @@ static int is_event_buffer(lua_State* L, int idx) {
/* TODO: Use lightuserdata mapping to locate hanging object instances */
/* Pushes the specified evbuffer object onto the stack, attaching a metatable to it */
-static int event_buffer_push(lua_State* L, struct evbuffer* buffer) {
+int event_buffer_push(lua_State* L, struct evbuffer* buffer) {
le_buffer *buf = (le_buffer*)lua_newuserdata(L, sizeof(le_buffer));
buf->buffer = buffer;
luaL_getmetatable(L, EVENT_BUFFER_MT);
@@ -257,5 +257,5 @@ int event_buffer_register(lua_State* L) {
lua_pop(L, 1);
luaL_register(L, "luaevent.core.buffer", funcs);
- return 0;
+ return 1;
}
diff --git a/src/luaevent.c b/src/luaevent.c
index d352cdc..c9b5430 100644
--- a/src/luaevent.c
+++ b/src/luaevent.c
@@ -4,6 +4,7 @@
#include "luaevent.h"
#include "event_callback.h"
#include "event_buffer.h"
+#include "buffer_event.h"
#include <lua.h>
#include <lauxlib.h>
@@ -109,6 +110,14 @@ static namedInteger consts[] = {
{"EV_READ", EV_READ},
{"EV_WRITE", EV_WRITE},
{"EV_TIMEOUT", EV_TIMEOUT},
+ {"EV_SIGNAL", EV_SIGNAL},
+ {"EV_PERSIST", EV_PERSIST},
+ /* bufferevent */
+ {"EVBUFFER_READ", EVBUFFER_READ},
+ {"EVBUFFER_WRITE", EVBUFFER_WRITE},
+ {"EVBUFFER_EOF", EVBUFFER_EOF},
+ {"EVBUFFER_ERROR", EVBUFFER_ERROR},
+ {"EVBUFFER_TIMEOUT", EVBUFFER_TIMEOUT},
{NULL, 0}
};
@@ -131,6 +140,8 @@ int luaopen_luaevent_core(lua_State* L) {
/* Register external items */
event_callback_register(L);
event_buffer_register(L);
+ buffer_event_register(L);
+ lua_settop(L, 0);
/* Setup metatable */
luaL_newmetatable(L, EVENT_BASE_MT);
lua_newtable(L);
diff --git a/src/utility.c b/src/utility.c
new file mode 100644
index 0000000..d42be1f
--- /dev/null
+++ b/src/utility.c
@@ -0,0 +1,45 @@
+/* LuaEvent - Copyright (C) 2007 Thomas Harning <harningt@gmail.com>
+ * Licensed as LGPL - See doc/COPYING for details */
+#include "utility.h"
+#include <lauxlib.h>
+
+#define WEAK_REF_LOCATION le_register_utility
+
+static void get_weakref_table(lua_State* L) {
+ lua_pushlightuserdata(L, WEAK_REF_LOCATION);
+ lua_gettable(L, LUA_REGISTRYINDEX);
+}
+
+void le_weak_ref(lua_State* L, void* ptr, int idx) {
+ get_weakref_table(L);
+ lua_pushlightuserdata(L, ptr);
+ if(idx < 0) idx-=2;
+ lua_pushvalue(L, idx);
+ lua_settable(L, -3);
+}
+void le_weak_unref(lua_State* L, void* ptr) {
+ get_weakref_table(L);
+ lua_pushlightuserdata(L, ptr);
+ lua_pushnil(L);
+ lua_settable(L, -3);
+}
+
+void le_weak_get(lua_State* L, void* ptr) {
+ get_weakref_table(L);
+ lua_pushlightuserdata(L, ptr);
+ lua_gettable(L, -2);
+}
+
+static void push_weak_table(lua_State* L, const char* mode) {
+ lua_newtable(L);
+ lua_createtable(L,0,1);
+ lua_pushstring(L,mode);
+ lua_setfield(L,-2,"__mode");
+ lua_setmetatable(L,-2);
+}
+
+void le_register_utility(lua_State* L) {
+ lua_pushlightuserdata(L, WEAK_REF_LOCATION);
+ push_weak_table(L, "v");
+ lua_settable(L, LUA_REGISTRYINDEX);
+}