Software /
code /
prosody
Comparison
util-src/lsignal.c @ 988:8cb9a0d59e78
util.lsignal: Add lsignal library for catching POSIX signals
author | Matthew Wild <mwild1@gmail.com> |
---|---|
date | Sun, 12 Apr 2009 02:53:50 +0100 |
child | 1841:0dc1daa94e7f |
comparison
equal
deleted
inserted
replaced
987:d55ec60331e7 | 988:8cb9a0d59e78 |
---|---|
1 /* | |
2 * lsignal.h -- Signal Handler Library for Lua | |
3 * | |
4 * Version: 1.000 | |
5 * | |
6 * Copyright (C) 2007 Patrick J. Donnelly (batrick@unm.edu) | |
7 * | |
8 * This software is distributed under the same license as Lua 5.0: | |
9 * | |
10 * Permission is hereby granted, free of charge, to any person obtaining a | |
11 * copy of this software and associated documentation files (the "Software"), | |
12 * to deal in the Software without restriction, including without limitation | |
13 * the rights to use, copy, modify, merge, publish, distribute, sublicense, | |
14 * and/or sell copies of the Software, and to permit persons to whom the | |
15 * Software is furnished to do so, subject to the following conditions: | |
16 * | |
17 * The above copyright notice and this permission notice shall be included | |
18 * in all copies or substantial portions of the Software. | |
19 * | |
20 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS | |
21 * OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, | |
22 * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL | |
23 * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR | |
24 * OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, | |
25 * ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR | |
26 * OTHER DEALINGS IN THE SOFTWARE. | |
27 */ | |
28 | |
29 #include <signal.h> | |
30 | |
31 #include "lua.h" | |
32 #include "lauxlib.h" | |
33 | |
34 #ifndef lsig | |
35 | |
36 #define lsig | |
37 | |
38 struct lua_signal | |
39 { | |
40 char *name; /* name of the signal */ | |
41 int sig; /* the signal */ | |
42 }; | |
43 | |
44 #endif | |
45 | |
46 #define LUA_SIGNAL "lua_signal" | |
47 | |
48 static const struct lua_signal lua_signals[] = { | |
49 /* ANSI C signals */ | |
50 #ifdef SIGABRT | |
51 {"SIGABRT", SIGABRT}, | |
52 #endif | |
53 #ifdef SIGFPE | |
54 {"SIGFPE", SIGFPE}, | |
55 #endif | |
56 #ifdef SIGILL | |
57 {"SIGILL", SIGILL}, | |
58 #endif | |
59 #ifdef SIGINT | |
60 {"SIGINT", SIGINT}, | |
61 #endif | |
62 #ifdef SIGSEGV | |
63 {"SIGSEGV", SIGSEGV}, | |
64 #endif | |
65 #ifdef SIGTERM | |
66 {"SIGTERM", SIGTERM}, | |
67 #endif | |
68 /* posix signals */ | |
69 #ifdef SIGHUP | |
70 {"SIGHUP", SIGHUP}, | |
71 #endif | |
72 #ifdef SIGQUIT | |
73 {"SIGQUIT", SIGQUIT}, | |
74 #endif | |
75 #ifdef SIGTRAP | |
76 {"SIGTRAP", SIGTRAP}, | |
77 #endif | |
78 #ifdef SIGKILL | |
79 {"SIGKILL", SIGKILL}, | |
80 #endif | |
81 #ifdef SIGUSR1 | |
82 {"SIGUSR1", SIGUSR1}, | |
83 #endif | |
84 #ifdef SIGUSR2 | |
85 {"SIGUSR2", SIGUSR2}, | |
86 #endif | |
87 #ifdef SIGPIPE | |
88 {"SIGPIPE", SIGPIPE}, | |
89 #endif | |
90 #ifdef SIGALRM | |
91 {"SIGALRM", SIGALRM}, | |
92 #endif | |
93 #ifdef SIGCHLD | |
94 {"SIGCHLD", SIGCHLD}, | |
95 #endif | |
96 #ifdef SIGCONT | |
97 {"SIGCONT", SIGCONT}, | |
98 #endif | |
99 #ifdef SIGSTOP | |
100 {"SIGSTOP", SIGSTOP}, | |
101 #endif | |
102 #ifdef SIGTTIN | |
103 {"SIGTTIN", SIGTTIN}, | |
104 #endif | |
105 #ifdef SIGTTOU | |
106 {"SIGTTOU", SIGTTOU}, | |
107 #endif | |
108 /* some BSD signals */ | |
109 #ifdef SIGIOT | |
110 {"SIGIOT", SIGIOT}, | |
111 #endif | |
112 #ifdef SIGBUS | |
113 {"SIGBUS", SIGBUS}, | |
114 #endif | |
115 #ifdef SIGCLD | |
116 {"SIGCLD", SIGCLD}, | |
117 #endif | |
118 #ifdef SIGURG | |
119 {"SIGURG", SIGURG}, | |
120 #endif | |
121 #ifdef SIGXCPU | |
122 {"SIGXCPU", SIGXCPU}, | |
123 #endif | |
124 #ifdef SIGXFSZ | |
125 {"SIGXFSZ", SIGXFSZ}, | |
126 #endif | |
127 #ifdef SIGVTALRM | |
128 {"SIGVTALRM", SIGVTALRM}, | |
129 #endif | |
130 #ifdef SIGPROF | |
131 {"SIGPROF", SIGPROF}, | |
132 #endif | |
133 #ifdef SIGWINCH | |
134 {"SIGWINCH", SIGWINCH}, | |
135 #endif | |
136 #ifdef SIGPOLL | |
137 {"SIGPOLL", SIGPOLL}, | |
138 #endif | |
139 #ifdef SIGIO | |
140 {"SIGIO", SIGIO}, | |
141 #endif | |
142 /* add odd signals */ | |
143 #ifdef SIGSTKFLT | |
144 {"SIGSTKFLT", SIGSTKFLT}, /* stack fault */ | |
145 #endif | |
146 #ifdef SIGSYS | |
147 {"SIGSYS", SIGSYS}, | |
148 #endif | |
149 {NULL, 0} | |
150 }; | |
151 | |
152 static int Nsig = 0; | |
153 static lua_State *Lsig = NULL; | |
154 static lua_Hook Hsig = NULL; | |
155 static int Hmask = 0; | |
156 static int Hcount = 0; | |
157 | |
158 static void sighook(lua_State *L, lua_Debug *ar) | |
159 { | |
160 lua_pushstring(L, LUA_SIGNAL); | |
161 lua_gettable(L, LUA_REGISTRYINDEX); | |
162 lua_pushnumber(L, Nsig); | |
163 lua_gettable(L, -2); | |
164 | |
165 lua_call(L, 0, 0); | |
166 | |
167 /* set the old hook */ | |
168 lua_sethook(L, Hsig, Hmask, Hcount); | |
169 } | |
170 | |
171 static void handle(int sig) | |
172 { | |
173 Hsig = lua_gethook(Lsig); | |
174 Hmask = lua_gethookmask(Lsig); | |
175 Hcount = lua_gethookcount(Lsig); | |
176 Nsig = sig; | |
177 | |
178 lua_sethook(Lsig, sighook, LUA_MASKCALL | LUA_MASKRET | LUA_MASKCOUNT, 1); | |
179 /* | |
180 switch (sig) | |
181 { | |
182 case SIGABRT: ; | |
183 case SIGFPE: ; | |
184 case SIGILL: ; | |
185 case SIGINT: ; | |
186 case SIGSEGV: ; | |
187 case SIGTERM: ; | |
188 } */ | |
189 } | |
190 | |
191 /* | |
192 * l_signal == signal(signal [, func [, chook]]) | |
193 * | |
194 * signal = signal number or string | |
195 * func = Lua function to call | |
196 * chook = catch within C functions | |
197 * if caught, Lua function _must_ | |
198 * exit, as the stack is most likely | |
199 * in an unstable state. | |
200 */ | |
201 | |
202 static int l_signal(lua_State *L) | |
203 { | |
204 int args = lua_gettop(L); | |
205 int t, sig; /* type, signal */ | |
206 | |
207 /* get type of signal */ | |
208 luaL_checkany(L, 1); | |
209 t = lua_type(L, 1); | |
210 if (t == LUA_TNUMBER) | |
211 sig = (int) lua_tonumber(L, 1); | |
212 else if (t == LUA_TSTRING) | |
213 { | |
214 lua_pushstring(L, LUA_SIGNAL); | |
215 lua_gettable(L, LUA_REGISTRYINDEX); | |
216 lua_pushvalue(L, 1); | |
217 lua_gettable(L, -2); | |
218 if (!lua_isnumber(L, -1)) | |
219 luaL_error(L, "invalid signal string"); | |
220 sig = (int) lua_tonumber(L, -1); | |
221 lua_pop(L, 1); /* get rid of number we pushed */ | |
222 } else | |
223 luaL_checknumber(L, 1); /* will always error, with good error msg */ | |
224 | |
225 /* set handler */ | |
226 if (args == 1 || lua_isnil(L, 2)) /* clear handler */ | |
227 { | |
228 lua_pushstring(L, LUA_SIGNAL); | |
229 lua_gettable(L, LUA_REGISTRYINDEX); | |
230 lua_pushnumber(L, sig); | |
231 lua_gettable(L, -2); /* return old handler */ | |
232 lua_pushnumber(L, sig); | |
233 lua_pushnil(L); | |
234 lua_settable(L, -4); | |
235 lua_remove(L, -2); /* remove LUA_SIGNAL table */ | |
236 signal(sig, SIG_DFL); | |
237 } else | |
238 { | |
239 luaL_checktype(L, 2, LUA_TFUNCTION); | |
240 | |
241 lua_pushstring(L, LUA_SIGNAL); | |
242 lua_gettable(L, LUA_REGISTRYINDEX); | |
243 | |
244 lua_pushnumber(L, sig); | |
245 lua_pushvalue(L, 2); | |
246 lua_settable(L, -3); | |
247 | |
248 /* Set the state for the handler */ | |
249 Lsig = L; | |
250 | |
251 if (lua_toboolean(L, 3)) /* c hook? */ | |
252 { | |
253 if (signal(sig, handle) == SIG_ERR) | |
254 lua_pushboolean(L, 0); | |
255 else | |
256 lua_pushboolean(L, 1); | |
257 } else /* lua_hook */ | |
258 { | |
259 if (signal(sig, handle) == SIG_ERR) | |
260 lua_pushboolean(L, 0); | |
261 else | |
262 lua_pushboolean(L, 1); | |
263 } | |
264 } | |
265 return 1; | |
266 } | |
267 | |
268 /* | |
269 * l_raise == raise(signal) | |
270 * | |
271 * signal = signal number or string | |
272 */ | |
273 | |
274 static int l_raise(lua_State *L) | |
275 { | |
276 /* int args = lua_gettop(L); */ | |
277 int t = 0; /* type */ | |
278 lua_Number ret; | |
279 | |
280 luaL_checkany(L, 1); | |
281 | |
282 t = lua_type(L, 1); | |
283 if (t == LUA_TNUMBER) | |
284 { | |
285 ret = (lua_Number) raise((int) lua_tonumber(L, 1)); | |
286 lua_pushnumber(L, ret); | |
287 } else if (t == LUA_TSTRING) | |
288 { | |
289 lua_pushstring(L, LUA_SIGNAL); | |
290 lua_gettable(L, LUA_REGISTRYINDEX); | |
291 lua_pushvalue(L, 1); | |
292 lua_gettable(L, -2); | |
293 if (!lua_isnumber(L, -1)) | |
294 luaL_error(L, "invalid signal string"); | |
295 ret = (lua_Number) raise((int) lua_tonumber(L, -1)); | |
296 lua_pop(L, 1); /* get rid of number we pushed */ | |
297 lua_pushnumber(L, ret); | |
298 } else | |
299 luaL_checknumber(L, 1); /* will always error, with good error msg */ | |
300 | |
301 return 1; | |
302 } | |
303 | |
304 #ifdef _POSIX_SOURCE | |
305 | |
306 /* define some posix only functions */ | |
307 | |
308 /* | |
309 * l_kill == kill(pid, signal) | |
310 * | |
311 * pid = process id | |
312 * signal = signal number or string | |
313 */ | |
314 | |
315 static int l_kill(lua_State *L) | |
316 { | |
317 int t; /* type */ | |
318 lua_Number ret; /* return value */ | |
319 | |
320 luaL_checknumber(L, 1); /* must be int for pid */ | |
321 luaL_checkany(L, 2); /* check for a second arg */ | |
322 | |
323 t = lua_type(L, 2); | |
324 if (t == LUA_TNUMBER) | |
325 { | |
326 ret = (lua_Number) kill((int) lua_tonumber(L, 1), | |
327 (int) lua_tonumber(L, 2)); | |
328 lua_pushnumber(L, ret); | |
329 } else if (t == LUA_TSTRING) | |
330 { | |
331 lua_pushstring(L, LUA_SIGNAL); | |
332 lua_gettable(L, LUA_REGISTRYINDEX); | |
333 lua_pushvalue(L, 2); | |
334 lua_gettable(L, -2); | |
335 if (!lua_isnumber(L, -1)) | |
336 luaL_error(L, "invalid signal string"); | |
337 ret = (lua_Number) kill((int) lua_tonumber(L, 1), | |
338 (int) lua_tonumber(L, -1)); | |
339 lua_pop(L, 1); /* get rid of number we pushed */ | |
340 lua_pushnumber(L, ret); | |
341 } else | |
342 luaL_checknumber(L, 2); /* will always error, with good error msg */ | |
343 return 1; | |
344 } | |
345 | |
346 #endif | |
347 | |
348 static const struct luaL_Reg lsignal_lib[] = { | |
349 {"signal", l_signal}, | |
350 {"raise", l_raise}, | |
351 #ifdef _POSIX_SOURCE | |
352 {"kill", l_kill}, | |
353 #endif | |
354 {NULL, NULL} | |
355 }; | |
356 | |
357 int luaopen_util_signal(lua_State *L) | |
358 { | |
359 int i = 0; | |
360 | |
361 /* add the library */ | |
362 luaL_register(L, "signal", lsignal_lib); | |
363 | |
364 /* push lua_signals table into the registry */ | |
365 /* put the signals inside the library table too, | |
366 * they are only a reference */ | |
367 lua_pushstring(L, LUA_SIGNAL); | |
368 lua_createtable(L, 0, 0); | |
369 | |
370 while (lua_signals[i].name != NULL) | |
371 { | |
372 /* registry table */ | |
373 lua_pushstring(L, lua_signals[i].name); | |
374 lua_pushnumber(L, lua_signals[i].sig); | |
375 lua_settable(L, -3); | |
376 /* signal table */ | |
377 lua_pushstring(L, lua_signals[i].name); | |
378 lua_pushnumber(L, lua_signals[i].sig); | |
379 lua_settable(L, -5); | |
380 i++; | |
381 } | |
382 | |
383 /* add newtable to the registry */ | |
384 lua_settable(L, LUA_REGISTRYINDEX); | |
385 | |
386 return 1; | |
387 } |