1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
|
/* LuaEvent - Copyright (C) 2007 Thomas Harning <harningt@gmail.com>
* Licensed as LGPL - See doc/COPYING for details */
#include "luaevent.h"
#include "event_callback.h"
#include <lua.h>
#include <lauxlib.h>
#include <assert.h>
#define EVENT_BASE_MT "EVENT_BASE_MT"
int luaevent_newbase(lua_State* L) {
le_base *base = (le_base*)lua_newuserdata(L, sizeof(le_base));
base->loop_L = NULL; /* No running loop */
base->base = event_init();
luaL_getmetatable(L, EVENT_BASE_MT);
lua_setmetatable(L, -2);
return 1;
}
static int luaevent_base_gc(lua_State* L) {
le_base *base = luaL_checkudata(L, 1, EVENT_BASE_MT);
if(base->base) {
event_base_free(base->base);
base->base = NULL;
}
return 0;
}
int getSocketFd(lua_State* L, int idx) {
int fd;
luaL_checktype(L, idx, LUA_TUSERDATA);
lua_getfield(L, idx, "getfd");
if(lua_isnil(L, -1))
return luaL_error(L, "Socket type missing 'getfd' method");
lua_pushvalue(L, idx);
lua_call(L, 1, 1);
fd = lua_tointeger(L, -1);
lua_pop(L, 1);
return fd;
}
/* sock, event, callback */
static int luaevent_addevent(lua_State* L) {
int fd, event, callbackRef;
le_callback* arg;
le_base *base = luaL_checkudata(L, 1, EVENT_BASE_MT);
fd = getSocketFd(L, 2);
event = luaL_checkinteger(L, 3);
luaL_checktype(L, 4, LUA_TFUNCTION);
lua_pushvalue(L, 4);
callbackRef = luaL_ref(L, LUA_REGISTRYINDEX);
arg = lua_newuserdata(L, sizeof(*arg));
luaL_getmetatable(L, EVENT_CALLBACK_ARG_MT);
lua_setmetatable(L, -2);
arg->base = base;
arg->callbackRef = callbackRef;
/* Setup event... */
event_set(&arg->ev, fd, event | EV_PERSIST, luaevent_callback, arg);
event_base_set(base->base, &arg->ev);
event_add(&arg->ev, NULL);
return 1;
}
static int luaevent_loop(lua_State* L) {
le_base *base = luaL_checkudata(L, 1, EVENT_BASE_MT);
base->loop_L = L;
int ret = event_base_loop(base->base, 0);
lua_pushinteger(L, ret);
return 1;
}
static luaL_Reg base_funcs[] = {
{ "addevent", luaevent_addevent },
{ "loop", luaevent_loop },
{ NULL, NULL }
};
static luaL_Reg funcs[] = {
{ "new", luaevent_newbase },
{ NULL, NULL }
};
typedef struct {
const char* name;
int value;
} namedInteger;
static namedInteger consts[] = {
{"LEAVE", -1},
{"EV_READ", EV_READ},
{"EV_WRITE", EV_WRITE},
{NULL, 0}
};
void setNamedIntegers(lua_State* L, namedInteger* p) {
while(p->name) {
lua_pushinteger(L, p->value);
lua_setfield(L, -2, p->name);
p++;
}
}
/* Verified ok */
int luaopen_luaevent_core(lua_State* L) {
/* Register external items */
event_callback_register(L);
/* Setup metatable */
luaL_newmetatable(L, EVENT_BASE_MT);
lua_newtable(L);
luaL_register(L, NULL, base_funcs);
lua_setfield(L, -2, "__index");
lua_pushcfunction(L, luaevent_base_gc);
lua_setfield(L, -2, "__gc");
lua_pop(L, 1);
luaL_register(L, "luaevent.core", funcs);
setNamedIntegers(L, consts);
return 1;
}
|