4 Copyright (C) 1999-2001 Timo Sirainen
6 This program is free software; you can redistribute it and/or modify
7 it under the terms of the GNU General Public License as published by
8 the Free Software Foundation; either version 2 of the License, or
9 (at your option) any later version.
11 This program is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 GNU General Public License for more details.
16 You should have received a copy of the GNU General Public License along
17 with this program; if not, write to the Free Software Foundation, Inc.,
18 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
28 #include "perl-core.h"
29 #include "perl-common.h"
30 #include "perl-signals.h"
33 PERL_SCRIPT_REC *script;
43 } PERL_SIGNAL_ARGS_REC;
45 #include "perl-signals-list.h"
47 static GHashTable *signals;
48 static GHashTable *perl_signal_args_hash;
49 static GSList *perl_signal_args_partial;
51 static PERL_SIGNAL_ARGS_REC *perl_signal_args_find(int signal_id)
53 PERL_SIGNAL_ARGS_REC *rec;
57 rec = g_hash_table_lookup(perl_signal_args_hash,
58 GINT_TO_POINTER(signal_id));
59 if (rec != NULL) return rec;
61 /* try to find by name */
62 signame = signal_get_id_str(signal_id);
63 for (tmp = perl_signal_args_partial; tmp != NULL; tmp = tmp->next) {
66 if (strncmp(rec->signal, signame, strlen(rec->signal)) == 0)
73 void perl_signal_args_to_c(
74 void (*callback)(void *, void **), void *cb_arg,
75 int signal_id, SV **args, size_t n_args)
79 unsigned long v_ulong;
82 } saved_args[SIGNAL_MAX_ARGUMENTS];
83 void *p[SIGNAL_MAX_ARGUMENTS];
84 PERL_SIGNAL_ARGS_REC *rec;
87 if (!(rec = perl_signal_args_find(signal_id))) {
88 const char *name = signal_get_id_str(signal_id);
90 croak("%d is not a known signal id", signal_id);
92 croak("\"%s\" is not a registered signal", name);
95 for (n = 0; n < SIGNAL_MAX_ARGUMENTS && n < n_args && rec->args[n] != NULL; ++n) {
101 } else if (strcmp(rec->args[n], "string") == 0) {
102 c_arg = SvPV_nolen(arg);
103 } else if (strcmp(rec->args[n], "int") == 0) {
104 c_arg = (void *)SvIV(arg);
105 } else if (strcmp(rec->args[n], "ulongptr") == 0) {
106 saved_args[n].v_ulong = SvUV(arg);
107 c_arg = &saved_args[n].v_ulong;
108 } else if (strcmp(rec->args[n], "intptr") == 0) {
109 saved_args[n].v_int = SvIV(SvRV(arg));
110 c_arg = &saved_args[n].v_int;
111 } else if (strncmp(rec->args[n], "glistptr_", 9) == 0) {
119 if (SvTYPE(t) != SVt_PVAV) {
120 croak("Not an ARRAY reference");
124 is_str = strcmp(rec->args[n]+9, "char*") == 0;
127 count = av_len(av) + 1;
128 while (count-- > 0) {
129 SV **px = av_fetch(av, count, 0);
130 SV *x = px ? *px : NULL;
134 is_str ? g_strdup(SvPV_nolen(x)) :
138 saved_args[n].v_glist = gl;
139 c_arg = &saved_args[n].v_glist;
140 } else if (strncmp(rec->args[n], "gslist_", 7) == 0) {
147 if (SvTYPE(t) != SVt_PVAV) {
148 croak("Not an ARRAY reference");
153 count = av_len(av) + 1;
154 while (count-- > 0) {
155 SV **x = av_fetch(av, count, 0);
156 gsl = g_slist_prepend(
162 c_arg = saved_args[n].v_gslist = gsl;
164 c_arg = irssi_ref_object(arg);
170 for (; n < SIGNAL_MAX_ARGUMENTS; ++n) {
176 for (n = 0; n < SIGNAL_MAX_ARGUMENTS && n < n_args && rec->args[n] != NULL; ++n) {
183 if (strcmp(rec->args[n], "intptr") == 0) {
186 SvIV_set(t, saved_args[n].v_int);
187 } else if (strncmp(rec->args[n], "gslist_", 7) == 0) {
188 g_slist_free(saved_args[n].v_gslist);
189 } else if (strncmp(rec->args[n], "glistptr_", 9) == 0) {
190 int is_iobject, is_str;
194 is_iobject = strcmp(rec->args[n]+9, "iobject") == 0;
195 is_str = strcmp(rec->args[n]+9, "char*") == 0;
197 av = (AV *)SvRV(arg);
200 gl = saved_args[n].v_glist;
201 for (tmp = gl; tmp != NULL; tmp = tmp->next) {
203 is_iobject ? iobject_bless((SERVER_REC *)tmp->data) :
204 is_str ? new_pv(tmp->data) :
205 irssi_bless_plain(rec->args[n]+9, tmp->data)
210 g_list_foreach(gl, (GFunc)g_free, NULL);
217 static void perl_call_signal(PERL_SCRIPT_REC *script, SV *func,
218 int signal_id, gconstpointer *args)
222 PERL_SIGNAL_ARGS_REC *rec;
223 SV *sv, *perlarg, *saved_args[SIGNAL_MAX_ARGUMENTS];
234 /* push signal argument to perl stack */
235 rec = perl_signal_args_find(signal_id);
237 memset(saved_args, 0, sizeof(saved_args));
238 for (n = 0; n < SIGNAL_MAX_ARGUMENTS &&
239 rec != NULL && rec->args[n] != NULL; n++) {
240 arg = (void *) args[n];
242 if (strncmp(rec->args[n], "glistptr_", 9) == 0) {
243 /* pointer to linked list - push as AV */
245 int is_iobject, is_str;
247 is_iobject = strcmp(rec->args[n]+9, "iobject") == 0;
248 is_str = strcmp(rec->args[n]+9, "char*") == 0;
252 for (tmp = *ptr; tmp != NULL; tmp = tmp->next) {
253 sv = is_iobject ? iobject_bless((SERVER_REC *) tmp->data) :
254 is_str ? new_pv(tmp->data) :
255 irssi_bless_plain(rec->args[n]+9, tmp->data);
259 saved_args[n] = perlarg = newRV_noinc((SV *) av);
260 } else if (strcmp(rec->args[n], "int") == 0)
261 perlarg = newSViv((IV)arg);
262 else if (arg == NULL)
263 perlarg = &PL_sv_undef;
264 else if (strcmp(rec->args[n], "string") == 0)
265 perlarg = new_pv(arg);
266 else if (strcmp(rec->args[n], "ulongptr") == 0)
267 perlarg = newSViv(*(unsigned long *) arg);
268 else if (strcmp(rec->args[n], "intptr") == 0)
269 saved_args[n] = perlarg = newRV_noinc(newSViv(*(int *) arg));
270 else if (strncmp(rec->args[n], "gslist_", 7) == 0) {
271 /* linked list - push as AV */
275 is_iobject = strcmp(rec->args[n]+7, "iobject") == 0;
277 for (tmp = arg; tmp != NULL; tmp = tmp->next) {
278 sv = is_iobject ? iobject_bless((SERVER_REC *) tmp->data) :
279 irssi_bless_plain(rec->args[n]+7, tmp->data);
283 perlarg = newRV_noinc((SV *) av);
284 } else if (strcmp(rec->args[n], "iobject") == 0) {
285 /* "irssi object" - any struct that has
286 "int type; int chat_type" as it's first
287 variables (server, channel, ..) */
288 perlarg = iobject_bless((SERVER_REC *) arg);
289 } else if (strcmp(rec->args[n], "siobject") == 0) {
290 /* "simple irssi object" - any struct that has
291 int type; as it's first variable (dcc) */
292 perlarg = simple_iobject_bless((SERVER_REC *) arg);
295 perlarg = plain_bless(arg, rec->args[n]);
297 XPUSHs(sv_2mortal(perlarg));
301 perl_call_sv(func, G_EVAL|G_DISCARD);
305 char *error = g_strdup(SvPV(ERRSV, PL_na));
306 signal_emit("script error", 2, script, error);
311 /* restore arguments the perl script modified */
312 for (n = 0; n < SIGNAL_MAX_ARGUMENTS &&
313 rec != NULL && rec->args[n] != NULL; n++) {
314 arg = (void *) args[n];
316 if (saved_args[n] == NULL)
319 if (strcmp(rec->args[n], "intptr") == 0) {
321 *val = SvIV(SvRV(saved_args[n]));
322 } else if (strncmp(rec->args[n], "glistptr_", 9) == 0) {
329 av = (AV *) SvRV(saved_args[n]);
331 while (count-- >= 0) {
334 val = g_strdup(SvPV(sv, len));
336 val = GINT_TO_POINTER(SvIV(sv));
338 out = g_list_append(out, val);
341 if (strcmp(rec->args[n]+9, "char*") == 0)
342 g_list_foreach(*ret, (GFunc) g_free, NULL);
353 static void sig_func(const void *p1, const void *p2,
354 const void *p3, const void *p4,
355 const void *p5, const void *p6)
357 PERL_SIGNAL_REC *rec;
360 args[0] = p1; args[1] = p2; args[2] = p3;
361 args[3] = p4; args[4] = p5; args[5] = p6;
363 rec = signal_get_user_data();
364 perl_call_signal(rec->script, rec->func, signal_get_emitted_id(), args);
367 static void perl_signal_add_full_int(const char *signal, SV *func,
368 int priority, int command,
369 const char *category)
371 PERL_SCRIPT_REC *script;
372 PERL_SIGNAL_REC *rec;
376 g_return_if_fail(signal != NULL);
377 g_return_if_fail(func != NULL);
379 script = perl_script_find_package(perl_get_package());
380 g_return_if_fail(script != NULL);
382 rec = g_new(PERL_SIGNAL_REC, 1);
383 rec->script = script;
384 rec->signal_id = signal_get_uniq_id(signal);
385 rec->signal = g_strdup(signal);
386 rec->func = perl_func_sv_inc(func, perl_get_package());
388 if (command || strncmp(signal, "command ", 8) == 0) {
389 /* we used Irssi::signal_add() instead of
390 Irssi::command_bind() - oh well, allow this.. */
391 command_bind_full(MODULE_NAME, priority, signal+8, -1,
392 category, sig_func, rec);
394 signal_add_full_id(MODULE_NAME, priority, rec->signal_id,
398 signal_idp = GINT_TO_POINTER(rec->signal_id);
399 siglist = g_hash_table_lookup(signals, signal_idp);
400 if (siglist == NULL) {
401 siglist = g_new0(GSList *, 1);
402 g_hash_table_insert(signals, signal_idp, siglist);
405 *siglist = g_slist_append(*siglist, rec);
408 void perl_signal_add_full(const char *signal, SV *func, int priority)
410 perl_signal_add_full_int(signal, func, priority, FALSE, NULL);
413 static void perl_signal_destroy(PERL_SIGNAL_REC *rec)
415 if (strncmp(rec->signal, "command ", 8) == 0)
416 command_unbind_full(rec->signal+8, sig_func, rec);
418 signal_remove_id(rec->signal_id, sig_func, rec);
420 SvREFCNT_dec(rec->func);
425 static void perl_signal_remove_list_one(GSList **siglist, PERL_SIGNAL_REC *rec)
427 *siglist = g_slist_remove(*siglist, rec);
428 if (*siglist == NULL) {
430 g_hash_table_remove(signals, GINT_TO_POINTER(rec->signal_id));
433 perl_signal_destroy(rec);
436 #define sv_func_cmp(f1, f2) \
437 (f1 == f2 || (SvPOK(f1) && SvPOK(f2) && \
438 strcmp((char *) SvPV_nolen(f1), (char *) SvPV_nolen(f2)) == 0))
440 static void perl_signal_remove_list(GSList **list, SV *func)
444 for (tmp = *list; tmp != NULL; tmp = tmp->next) {
445 PERL_SIGNAL_REC *rec = tmp->data;
447 if (sv_func_cmp(rec->func, func)) {
448 perl_signal_remove_list_one(list, rec);
454 void perl_signal_remove(const char *signal, SV *func)
459 signal_idp = GINT_TO_POINTER(signal_get_uniq_id(signal));
460 list = g_hash_table_lookup(signals, signal_idp);
463 func = perl_func_sv_inc(func, perl_get_package());
464 perl_signal_remove_list(list, func);
469 void perl_command_bind_to(const char *cmd, const char *category,
470 SV *func, int priority)
474 signal = g_strconcat("command ", cmd, NULL);
475 perl_signal_add_full_int(signal, func, priority, TRUE, category);
479 void perl_command_runsub(const char *cmd, const char *data,
480 SERVER_REC *server, WI_ITEM_REC *item)
482 command_runsub(cmd, data, server, item);
485 void perl_command_unbind(const char *cmd, SV *func)
489 /* perl_signal_remove() calls command_unbind() */
490 signal = g_strconcat("command ", cmd, NULL);
491 perl_signal_remove(signal, func);
495 static int signal_destroy_hash(void *key, GSList **list, PERL_SCRIPT_REC *script)
499 for (tmp = *list; tmp != NULL; tmp = next) {
500 PERL_SIGNAL_REC *rec = tmp->data;
503 if (script == NULL || rec->script == script) {
504 *list = g_slist_remove(*list, rec);
505 perl_signal_destroy(rec);
516 /* destroy all signals used by script */
517 void perl_signal_remove_script(PERL_SCRIPT_REC *script)
519 g_hash_table_foreach_remove(signals, (GHRFunc) signal_destroy_hash,
523 void perl_signals_start(void)
525 signals = g_hash_table_new(NULL, NULL);
528 void perl_signals_stop(void)
530 g_hash_table_foreach(signals, (GHFunc) signal_destroy_hash, NULL);
531 g_hash_table_destroy(signals);
535 static void register_signal_rec(PERL_SIGNAL_ARGS_REC *rec)
537 if (rec->signal[strlen(rec->signal)-1] == ' ') {
538 perl_signal_args_partial =
539 g_slist_append(perl_signal_args_partial, rec);
541 int signal_id = signal_get_uniq_id(rec->signal);
542 g_hash_table_insert(perl_signal_args_hash,
543 GINT_TO_POINTER(signal_id), rec);
547 void perl_signal_register(const char *signal, const char **args)
549 PERL_SIGNAL_ARGS_REC *rec;
552 if (perl_signal_args_find(signal_get_uniq_id(signal)) != NULL)
555 rec = g_new0(PERL_SIGNAL_ARGS_REC, 1);
556 for (i = 0; i < 6 && args[i] != NULL; i++)
557 rec->args[i] = g_strdup(args[i]);
559 rec->signal = g_strdup(signal);
560 register_signal_rec(rec);
563 void perl_signals_init(void)
567 perl_signal_args_hash = g_hash_table_new((GHashFunc) g_direct_hash,
568 (GCompareFunc) g_direct_equal);
569 perl_signal_args_partial = NULL;
571 for (n = 0; perl_signal_args[n].signal != NULL; n++)
572 register_signal_rec(&perl_signal_args[n]);
575 static void signal_args_free(PERL_SIGNAL_ARGS_REC *rec)
582 for (i = 0; i < 6 && rec->args[i] != NULL; i++)
583 g_free(rec->args[i]);
588 static void signal_args_hash_free(void *key, PERL_SIGNAL_ARGS_REC *rec)
590 signal_args_free(rec);
593 void perl_signals_deinit(void)
595 g_slist_foreach(perl_signal_args_partial,
596 (GFunc) signal_args_free, NULL);
597 g_slist_free(perl_signal_args_partial);
599 g_hash_table_foreach(perl_signal_args_hash,
600 (GHFunc) signal_args_hash_free, NULL);
601 g_hash_table_destroy(perl_signal_args_hash);