summaryrefslogtreecommitdiffstats
path: root/src/perl/perl-signals.c
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--src/perl/perl-signals.c744
1 files changed, 744 insertions, 0 deletions
diff --git a/src/perl/perl-signals.c b/src/perl/perl-signals.c
new file mode 100644
index 0000000..683b4c3
--- /dev/null
+++ b/src/perl/perl-signals.c
@@ -0,0 +1,744 @@
+/*
+ perl-signals.c : irssi
+
+ Copyright (C) 1999-2001 Timo Sirainen
+
+ This program is free software; you can redistribute it and/or modify
+ it under the terms of the GNU General Public License as published by
+ the Free Software Foundation; either version 2 of the License, or
+ (at your option) any later version.
+
+ This program is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ GNU General Public License for more details.
+
+ You should have received a copy of the GNU General Public License along
+ with this program; if not, write to the Free Software Foundation, Inc.,
+ 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
+*/
+
+#define NEED_PERL_H
+#define PERL_NO_GET_CONTEXT
+#include "module.h"
+#include <irssi/src/core/commands.h>
+#include <irssi/src/core/modules.h>
+#include <irssi/src/core/servers.h>
+#include <irssi/src/core/signals.h>
+#include <irssi/src/fe-common/core/formats.h>
+
+#include <irssi/src/perl/perl-core.h>
+#include <irssi/src/perl/perl-common.h>
+#include <irssi/src/perl/perl-signals.h>
+
+typedef struct {
+ PERL_SCRIPT_REC *script;
+ int signal_id;
+ char *signal;
+ SV *func;
+} PERL_SIGNAL_REC;
+
+typedef struct {
+ char *signal;
+ char *args[SIGNAL_MAX_ARGUMENTS + 1];
+ int dynamic;
+} PERL_SIGNAL_ARGS_REC;
+
+#include "perl-signals-list.h"
+
+static GHashTable *signals, *signal_stashes;
+static GHashTable *perl_signal_args_hash;
+static GSList *perl_signal_args_partial;
+
+void irssi_add_signal_arg_conv(const char *stash, PERL_BLESS_FUNC func)
+{
+ if (g_hash_table_lookup(signal_stashes, stash) == NULL)
+ g_hash_table_insert(signal_stashes, g_strdup(stash), func);
+}
+
+static PERL_SIGNAL_ARGS_REC *perl_signal_args_find(int signal_id)
+{
+ PERL_SIGNAL_ARGS_REC *rec;
+ GSList *tmp;
+ const char *signame;
+
+ rec = g_hash_table_lookup(perl_signal_args_hash,
+ GINT_TO_POINTER(signal_id));
+ if (rec != NULL) return rec;
+
+ /* try to find by name */
+ signame = signal_get_id_str(signal_id);
+ for (tmp = perl_signal_args_partial; tmp != NULL; tmp = tmp->next) {
+ rec = tmp->data;
+
+ if (strncmp(rec->signal, signame, strlen(rec->signal)) == 0)
+ return rec;
+ }
+
+ return NULL;
+}
+
+void perl_signal_args_to_c(void (*callback)(void *, int, void **), void *cb_arg, int signal_id,
+ SV **args, size_t n_args)
+{
+ union {
+ int v_int;
+ unsigned long v_ulong;
+ GSList *v_gslist;
+ GList *v_glist;
+ GString *v_gstring;
+ } saved_args[SIGNAL_MAX_ARGUMENTS];
+ AV *aargs;
+ void *p[SIGNAL_MAX_ARGUMENTS];
+ PERL_SIGNAL_ARGS_REC *rec;
+ char *arglist[MAX_FORMAT_PARAMS];
+ size_t n;
+
+ if (!(rec = perl_signal_args_find(signal_id))) {
+ const char *name = signal_get_id_str(signal_id);
+ if (!name) {
+ croak("%d is not a known signal id", signal_id);
+ }
+ croak("\"%s\" is not a registered signal", name);
+ }
+
+ for (n = 0; n < SIGNAL_MAX_ARGUMENTS && n < n_args && rec->args[n] != NULL; ++n) {
+ void *c_arg;
+ SV *arg = args[n];
+
+ if (g_strcmp0(rec->args[n], "formatnum_args") == 0 && n >= 3) {
+ const FORMAT_REC *formats;
+ const char *module;
+ int num;
+ int formatnum;
+
+ module = SvPV_nolen(args[n - 2]);
+ formatnum = format_find_tag(module, SvPV_nolen(arg));
+ if (formatnum < 0) { /* format out of bounds */
+ p[n - 2] = NULL;
+ break;
+ }
+
+ formats = g_hash_table_lookup(default_formats, module);
+ arglist[formats[formatnum].params] = NULL;
+
+ p[n++] = GINT_TO_POINTER(formatnum);
+
+ for (num = 0; num < formats[formatnum].params; num++) {
+ if (n + num < n_args)
+ arglist[num] = SvPV_nolen(args[n + num]);
+ else
+ arglist[num] = "";
+ }
+
+ p[n++] = arglist;
+ n_args = n;
+
+ break;
+ } else if (!SvOK(arg)) {
+ c_arg = NULL;
+ } else if (g_strcmp0(rec->args[n], "string") == 0) {
+ c_arg = SvPV_nolen(arg);
+ } else if (g_strcmp0(rec->args[n], "int") == 0) {
+ c_arg = (void *) SvIV(arg);
+ } else if (g_strcmp0(rec->args[n], "ulongptr") == 0) {
+ saved_args[n].v_ulong = SvUV(arg);
+ c_arg = &saved_args[n].v_ulong;
+ } else if (g_strcmp0(rec->args[n], "intptr") == 0) {
+ saved_args[n].v_int = SvIV(SvRV(arg));
+ c_arg = &saved_args[n].v_int;
+ } else if (g_strcmp0(rec->args[n], "gstring") == 0) {
+ char *pv;
+ size_t len;
+
+ pv = SvPV(SvRV(arg), len);
+ c_arg = saved_args[n].v_gstring = g_string_new_len(pv, len);
+ } else if (strncmp(rec->args[n], "glistptr_", 9) == 0) {
+ GList *gl;
+ int is_str;
+ AV *av;
+ SV *t;
+ int count;
+
+ t = SvRV(arg);
+ if (SvTYPE(t) != SVt_PVAV) {
+ croak("Not an ARRAY reference");
+ }
+ av = (AV *) t;
+
+ is_str = g_strcmp0(rec->args[n] + 9, "string") == 0 ||
+ g_strcmp0(rec->args[n] + 9, "char*") == 0; /* deprecated form */
+
+ gl = NULL;
+ count = av_len(av) + 1;
+ while (count-- > 0) {
+ SV **px = av_fetch(av, count, 0);
+ SV *x = px ? *px : NULL;
+ gl = g_list_prepend(gl, x == NULL ?
+ NULL :
+ is_str ? g_strdup(SvPV_nolen(x)) :
+ irssi_ref_object(x));
+ }
+ saved_args[n].v_glist = gl;
+ c_arg = &saved_args[n].v_glist;
+ } else if (strncmp(rec->args[n], "gslist_", 7) == 0) {
+ GSList *gsl;
+ AV *av;
+ SV *t;
+ int count;
+
+ t = SvRV(arg);
+ if (SvTYPE(t) != SVt_PVAV) {
+ croak("Not an ARRAY reference");
+ }
+ av = (AV *) t;
+
+ gsl = NULL;
+ count = av_len(av) + 1;
+ while (count-- > 0) {
+ SV **x = av_fetch(av, count, 0);
+ gsl = g_slist_prepend(gsl, x == NULL ? NULL : irssi_ref_object(*x));
+ }
+ c_arg = saved_args[n].v_gslist = gsl;
+ } else {
+ c_arg = irssi_ref_object(arg);
+ }
+
+ p[n] = c_arg;
+ }
+
+ for (; n < SIGNAL_MAX_ARGUMENTS; ++n) {
+ p[n] = NULL;
+ }
+
+ /* make a copy of the stack now, since the callback might change it */
+ aargs = av_make(n_args, args);
+
+ callback(cb_arg, n_args, p);
+
+ for (n = 0; n < SIGNAL_MAX_ARGUMENTS && n < n_args && rec->args[n] != NULL; ++n) {
+ SV *arg = *av_fetch(aargs, n, 0);
+
+ if (!SvOK(arg)) {
+ continue;
+ }
+
+ if (g_strcmp0(rec->args[n], "intptr") == 0) {
+ SV *t = SvRV(arg);
+ SvIOK_only(t);
+ SvIV_set(t, saved_args[n].v_int);
+ } else if (g_strcmp0(rec->args[n], "gstring") == 0) {
+ GString *str;
+ SV *t;
+
+ str = saved_args[n].v_gstring;
+ t = SvRV(arg);
+ SvPOK_only(t);
+ sv_setpvn(t, str->str, str->len);
+
+ g_string_free(str, TRUE);
+ } else if (strncmp(rec->args[n], "gslist_", 7) == 0) {
+ g_slist_free(saved_args[n].v_gslist);
+ } else if (strncmp(rec->args[n], "glistptr_", 9) == 0) {
+ int is_iobject, is_str;
+ AV *av;
+ GList *gl, *tmp;
+
+ is_iobject = g_strcmp0(rec->args[n] + 9, "iobject") == 0;
+ is_str = g_strcmp0(rec->args[n] + 9, "string") == 0 ||
+ g_strcmp0(rec->args[n] + 9, "char*") == 0; /* deprecated form */
+
+ av = (AV *) SvRV(arg);
+ av_clear(av);
+
+ gl = saved_args[n].v_glist;
+ for (tmp = gl; tmp != NULL; tmp = tmp->next) {
+ av_push(av, is_iobject ?
+ iobject_bless((SERVER_REC *) tmp->data) :
+ is_str ?
+ new_pv(tmp->data) :
+ irssi_bless_plain(rec->args[n] + 9, tmp->data));
+ }
+
+ if (is_str) {
+ g_list_foreach(gl, (GFunc) g_free, NULL);
+ }
+ g_list_free(gl);
+ }
+ }
+ av_undef(aargs);
+}
+
+static void perl_call_signal(PERL_SCRIPT_REC *script, SV *func,
+ int signal_id, gconstpointer *args)
+{
+ dSP;
+
+ PERL_SIGNAL_ARGS_REC *rec;
+ SV *sv, *perlarg, *saved_args[SIGNAL_MAX_ARGUMENTS];
+ AV *av;
+ void *arg;
+ int n;
+
+
+ ENTER;
+ SAVETMPS;
+
+ PUSHMARK(sp);
+
+ /* push signal argument to perl stack */
+ rec = perl_signal_args_find(signal_id);
+
+ memset(saved_args, 0, sizeof(saved_args));
+ for (n = 0; n < SIGNAL_MAX_ARGUMENTS &&
+ rec != NULL && rec->args[n] != NULL; n++) {
+ arg = (void *) args[n];
+
+ if (strncmp(rec->args[n], "glistptr_", 9) == 0) {
+ /* pointer to linked list - push as AV */
+ GList *tmp, **ptr;
+ int is_iobject, is_str;
+
+ is_iobject = g_strcmp0(rec->args[n]+9, "iobject") == 0;
+ is_str = g_strcmp0(rec->args[n] + 9, "string") == 0 ||
+ g_strcmp0(rec->args[n] + 9, "char*") == 0; /* deprecated form */
+ av = newAV();
+
+ ptr = arg;
+ for (tmp = *ptr; tmp != NULL; tmp = tmp->next) {
+ sv = is_iobject ? iobject_bless((SERVER_REC *) tmp->data) :
+ is_str ? new_pv(tmp->data) :
+ irssi_bless_plain(rec->args[n]+9, tmp->data);
+ av_push(av, sv);
+ }
+
+ saved_args[n] = perlarg = newRV_noinc((SV *) av);
+ } else if (g_strcmp0(rec->args[n], "int") == 0)
+ perlarg = newSViv((IV)arg);
+ else if (arg == NULL)
+ perlarg = &PL_sv_undef;
+ else if (g_strcmp0(rec->args[n], "string") == 0)
+ perlarg = new_pv(arg);
+ else if (g_strcmp0(rec->args[n], "ulongptr") == 0)
+ perlarg = newSViv(*(unsigned long *) arg);
+ else if (g_strcmp0(rec->args[n], "intptr") == 0)
+ saved_args[n] = perlarg = newRV_noinc(newSViv(*(int *) arg));
+ else if (g_strcmp0(rec->args[n], "gstring") == 0) {
+ GString *str = arg;
+ saved_args[n] = perlarg = newRV_noinc(newSVpvn(str->str, str->len));
+ } else if (g_strcmp0(rec->args[n], "formatnum_args") == 0 && n >= 3) {
+ const THEME_REC *theme;
+ const MODULE_THEME_REC *rec;
+ const FORMAT_REC *formats;
+ char *const *tmp;
+ int formatnum;
+
+ theme = args[n - 3];
+ if (theme == NULL) /* no theme */
+ continue;
+
+ rec = g_hash_table_lookup(theme->modules, args[n - 2]);
+ if (rec == NULL) /* no module in theme */
+ continue;
+
+ formats = g_hash_table_lookup(default_formats, args[n - 2]);
+ if (formats == NULL) /* no module in default_formats */
+ continue;
+
+ formatnum = GPOINTER_TO_INT(arg);
+ if (formatnum >= rec->count) /* format out of bounds */
+ continue;
+
+ XPUSHs(sv_2mortal(new_pv(formats[formatnum].tag)));
+ for (tmp = args[n + 1]; *tmp != NULL; tmp++) {
+ XPUSHs(sv_2mortal(new_pv(*tmp)));
+ }
+
+ continue;
+ } else if (strncmp(rec->args[n], "gslist_", 7) == 0) {
+ /* linked list - push as AV */
+ GSList *tmp;
+ int is_iobject;
+
+ is_iobject = g_strcmp0(rec->args[n]+7, "iobject") == 0;
+ av = newAV();
+ for (tmp = arg; tmp != NULL; tmp = tmp->next) {
+ sv = is_iobject ? iobject_bless((SERVER_REC *) tmp->data) :
+ irssi_bless_plain(rec->args[n]+7, tmp->data);
+ av_push(av, sv);
+ }
+
+ perlarg = newRV_noinc((SV *) av);
+ } else if (g_strcmp0(rec->args[n], "iobject") == 0) {
+ /* "irssi object" - any struct that has
+ "int type; int chat_type" as it's first
+ variables (server, channel, ..) */
+ perlarg = iobject_bless((SERVER_REC *) arg);
+ } else if (g_strcmp0(rec->args[n], "siobject") == 0) {
+ /* "simple irssi object" - any struct that has
+ int type; as it's first variable (dcc) */
+ perlarg = simple_iobject_bless((SERVER_REC *) arg);
+ } else {
+ PERL_BLESS_FUNC bless_func;
+
+ bless_func = g_hash_table_lookup(signal_stashes, rec->args[n]);
+ if (bless_func != NULL) {
+ void *a1 = NULL;
+ void *a2 = NULL;
+ void *a3 = NULL;
+ if (g_strcmp0(rec->args[n], "Irssi::TextUI::Line") == 0) {
+ /* need to find the corresponding buffer */
+ int j;
+
+ for (j = n - 1; j >= 0; j--) {
+ if (g_strcmp0(rec->args[j],
+ "Irssi::TextUI::TextBufferView") ==
+ 0) {
+ a1 = (void *) args[j];
+ break;
+ } else if (g_strcmp0(rec->args[j],
+ "Irssi::UI::Window") == 0) {
+ a2 = (void *) args[j];
+ break;
+ }
+ }
+ }
+
+ perlarg = bless_func(rec->args[n], a1, a2, a3);
+ } else {
+ /* blessed object */
+ perlarg = plain_bless(arg, rec->args[n]);
+ }
+ }
+ XPUSHs(sv_2mortal(perlarg));
+ }
+
+ PUTBACK;
+ perl_call_sv(func, G_EVAL|G_DISCARD);
+ SPAGAIN;
+
+ if (SvTRUE(ERRSV)) {
+ char *error = g_strdup(SvPV_nolen(ERRSV));
+ perl_signal_remove_script(script);
+ signal_emit("script error", 2, script, error);
+ g_free(error);
+ rec = NULL;
+ }
+
+ /* restore arguments the perl script modified */
+ for (n = 0; n < SIGNAL_MAX_ARGUMENTS &&
+ rec != NULL && rec->args[n] != NULL; n++) {
+ arg = (void *) args[n];
+
+ if (saved_args[n] == NULL)
+ continue;
+
+ if (g_strcmp0(rec->args[n], "intptr") == 0) {
+ int *val = arg;
+ *val = SvIV(SvRV(saved_args[n]));
+ } else if (g_strcmp0(rec->args[n], "gstring") == 0) {
+ SV *os, *ns;
+ GString *str = arg;
+
+ os = sv_2mortal(newSVpvn(str->str, str->len));
+ ns = SvRV(saved_args[n]);
+ if (sv_cmp(os, ns) != 0) {
+ size_t len;
+ char *pv = SvPV(ns, len);
+
+ g_string_truncate(str, 0);
+ g_string_append_len(str, pv, len);
+ }
+ } else if (strncmp(rec->args[n], "glistptr_", 9) == 0) {
+ GList **ret = arg;
+ GList *out = NULL;
+ void *val;
+ int count;
+
+ av = (AV *) SvRV(saved_args[n]);
+ count = av_len(av);
+ while (count-- >= 0) {
+ sv = av_shift(av);
+ if (SvPOKp(sv))
+ val = g_strdup(SvPV_nolen(sv));
+ else
+ val = GINT_TO_POINTER(SvIV(sv));
+
+ out = g_list_append(out, val);
+ }
+
+ if (g_strcmp0(rec->args[n] + 9, "string") == 0 ||
+ g_strcmp0(rec->args[n] + 9, "char*") == 0) /* deprecated form */
+ g_list_foreach(*ret, (GFunc) g_free, NULL);
+ g_list_free(*ret);
+ *ret = out;
+ }
+ }
+
+ FREETMPS;
+ LEAVE;
+}
+
+#if SIGNAL_MAX_ARGUMENTS != 6
+#error SIGNAL_MAX_ARGUMENTS changed - update code
+#endif
+static void sig_func(const void *p1, const void *p2,
+ const void *p3, const void *p4,
+ const void *p5, const void *p6)
+{
+ PERL_SIGNAL_REC *rec;
+ PERL_SCRIPT_REC *script;
+ const void *args[SIGNAL_MAX_ARGUMENTS];
+
+ args[0] = p1; args[1] = p2; args[2] = p3;
+ args[3] = p4; args[4] = p5; args[5] = p6;
+
+ rec = signal_get_user_data();
+ script = rec->script;
+ perl_script_ref(script);
+ perl_call_signal(script, rec->func, signal_get_emitted_id(), args);
+ perl_script_unref(script);
+}
+
+static void perl_signal_add_full_int(const char *signal, SV *func,
+ int priority, int command,
+ const char *category)
+{
+ PERL_SCRIPT_REC *script;
+ PERL_SIGNAL_REC *rec;
+ GSList **siglist;
+ void *signal_idp;
+
+ g_return_if_fail(signal != NULL);
+ g_return_if_fail(func != NULL);
+
+ script = perl_script_find_package(perl_get_package());
+ g_return_if_fail(script != NULL);
+
+ rec = g_new(PERL_SIGNAL_REC, 1);
+ rec->script = script;
+ rec->signal_id = signal_get_uniq_id(signal);
+ rec->signal = g_strdup(signal);
+ rec->func = perl_func_sv_inc(func, perl_get_package());
+
+ if (command || strncmp(signal, "command ", 8) == 0) {
+ /* we used Irssi::signal_add() instead of
+ Irssi::command_bind() - oh well, allow this.. */
+ command_bind_full(MODULE_NAME, priority, signal+8, -1,
+ category, sig_func, rec);
+ } else {
+ signal_add_full_id(MODULE_NAME, priority, rec->signal_id,
+ sig_func, rec);
+ }
+
+ signal_idp = GINT_TO_POINTER(rec->signal_id);
+ siglist = g_hash_table_lookup(signals, signal_idp);
+ if (siglist == NULL) {
+ siglist = g_new0(GSList *, 1);
+ g_hash_table_insert(signals, signal_idp, siglist);
+ }
+
+ *siglist = g_slist_append(*siglist, rec);
+}
+
+void perl_signal_add_full(const char *signal, SV *func, int priority)
+{
+ perl_signal_add_full_int(signal, func, priority, FALSE, NULL);
+}
+
+static void perl_signal_destroy(PERL_SIGNAL_REC *rec)
+{
+ if (strncmp(rec->signal, "command ", 8) == 0)
+ command_unbind_full(rec->signal+8, sig_func, rec);
+ else
+ signal_remove_id(rec->signal_id, sig_func, rec);
+
+ SvREFCNT_dec(rec->func);
+ g_free(rec->signal);
+ g_free(rec);
+}
+
+static void perl_signal_remove_list_one(GSList **siglist, PERL_SIGNAL_REC *rec)
+{
+ *siglist = g_slist_remove(*siglist, rec);
+ if (*siglist == NULL) {
+ g_free(siglist);
+ g_hash_table_remove(signals, GINT_TO_POINTER(rec->signal_id));
+ }
+
+ perl_signal_destroy(rec);
+}
+
+#define sv_func_cmp(f1, f2) \
+ ((SvROK(f1) && SvROK(f2) && SvRV(f1) == SvRV(f2)) || \
+ (SvPOK(f1) && SvPOK(f2) && \
+ g_strcmp0(SvPV_nolen(f1), SvPV_nolen(f2)) == 0))
+
+static void perl_signal_remove_list(GSList **list, SV *func)
+{
+ GSList *tmp;
+
+ for (tmp = *list; tmp != NULL; tmp = tmp->next) {
+ PERL_SIGNAL_REC *rec = tmp->data;
+
+ if (sv_func_cmp(rec->func, func)) {
+ perl_signal_remove_list_one(list, rec);
+ break;
+ }
+ }
+}
+
+void perl_signal_remove(const char *signal, SV *func)
+{
+ GSList **list;
+ void *signal_idp;
+
+ signal_idp = GINT_TO_POINTER(signal_get_uniq_id(signal));
+ list = g_hash_table_lookup(signals, signal_idp);
+
+ if (list != NULL) {
+ func = perl_func_sv_inc(func, perl_get_package());
+ perl_signal_remove_list(list, func);
+ SvREFCNT_dec(func);
+ }
+}
+
+void perl_command_bind_to(const char *cmd, const char *category,
+ SV *func, int priority)
+{
+ char *signal;
+
+ signal = g_strconcat("command ", cmd, NULL);
+ perl_signal_add_full_int(signal, func, priority, TRUE, category);
+ g_free(signal);
+}
+
+void perl_command_runsub(const char *cmd, const char *data,
+ SERVER_REC *server, WI_ITEM_REC *item)
+{
+ command_runsub(cmd, data, server, item);
+}
+
+void perl_command_unbind(const char *cmd, SV *func)
+{
+ char *signal;
+
+ /* perl_signal_remove() calls command_unbind() */
+ signal = g_strconcat("command ", cmd, NULL);
+ perl_signal_remove(signal, func);
+ g_free(signal);
+}
+
+static int signal_destroy_hash(void *key, GSList **list, PERL_SCRIPT_REC *script)
+{
+ GSList *tmp, *next;
+
+ for (tmp = *list; tmp != NULL; tmp = next) {
+ PERL_SIGNAL_REC *rec = tmp->data;
+
+ next = tmp->next;
+ if (script == NULL || rec->script == script) {
+ *list = g_slist_remove(*list, rec);
+ perl_signal_destroy(rec);
+ }
+ }
+
+ if (*list != NULL)
+ return FALSE;
+
+ g_free(list);
+ return TRUE;
+}
+
+/* destroy all signals used by script */
+void perl_signal_remove_script(PERL_SCRIPT_REC *script)
+{
+ g_hash_table_foreach_remove(signals, (GHRFunc) signal_destroy_hash,
+ script);
+}
+
+void perl_signals_start(void)
+{
+ signals = g_hash_table_new(NULL, NULL);
+}
+
+void perl_signals_stop(void)
+{
+ g_hash_table_foreach(signals, (GHFunc) signal_destroy_hash, NULL);
+ g_hash_table_destroy(signals);
+ signals = NULL;
+}
+
+static void register_signal_rec(PERL_SIGNAL_ARGS_REC *rec)
+{
+ if (rec->signal[strlen(rec->signal)-1] == ' ') {
+ perl_signal_args_partial =
+ g_slist_append(perl_signal_args_partial, rec);
+ } else {
+ int signal_id = signal_get_uniq_id(rec->signal);
+ g_hash_table_insert(perl_signal_args_hash,
+ GINT_TO_POINTER(signal_id), rec);
+ }
+}
+
+void perl_signal_register(const char *signal, const char **args)
+{
+ PERL_SIGNAL_ARGS_REC *rec;
+ int i;
+
+ if (perl_signal_args_find(signal_get_uniq_id(signal)) != NULL)
+ return;
+
+ rec = g_new0(PERL_SIGNAL_ARGS_REC, 1);
+ for (i = 0; i < SIGNAL_MAX_ARGUMENTS && args[i] != NULL; i++)
+ rec->args[i] = g_strdup(args[i]);
+ rec->dynamic = TRUE;
+ rec->signal = g_strdup(signal);
+ register_signal_rec(rec);
+}
+
+void perl_signals_init(void)
+{
+ int n;
+
+ signal_stashes = g_hash_table_new((GHashFunc) g_str_hash, (GCompareFunc) g_str_equal);
+ perl_signal_args_hash = g_hash_table_new((GHashFunc) g_direct_hash,
+ (GCompareFunc) g_direct_equal);
+ perl_signal_args_partial = NULL;
+
+ for (n = 0; perl_signal_args[n].signal != NULL; n++)
+ register_signal_rec(&perl_signal_args[n]);
+}
+
+static void signal_args_free(PERL_SIGNAL_ARGS_REC *rec)
+{
+ int i;
+
+ if (!rec->dynamic)
+ return;
+
+ for (i = 0; i < SIGNAL_MAX_ARGUMENTS && rec->args[i] != NULL; i++)
+ g_free(rec->args[i]);
+ g_free(rec->signal);
+ g_free(rec);
+}
+
+static void signal_args_hash_free(void *key, PERL_SIGNAL_ARGS_REC *rec)
+{
+ signal_args_free(rec);
+}
+
+void perl_signals_deinit(void)
+{
+ g_slist_foreach(perl_signal_args_partial,
+ (GFunc) signal_args_free, NULL);
+ g_slist_free(perl_signal_args_partial);
+
+ g_hash_table_foreach(perl_signal_args_hash,
+ (GHFunc) signal_args_hash_free, NULL);
+ g_hash_table_destroy(perl_signal_args_hash);
+
+ g_hash_table_foreach(signal_stashes, (GHFunc) g_free, NULL);
+ g_hash_table_destroy(signal_stashes);
+ signal_stashes = NULL;
+}