2001-07-29 05:17:53 -04:00
|
|
|
/*
|
|
|
|
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.
|
|
|
|
|
2007-05-08 14:41:10 -04:00
|
|
|
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.
|
2001-07-29 05:17:53 -04:00
|
|
|
*/
|
|
|
|
|
|
|
|
#define NEED_PERL_H
|
2001-01-03 02:34:12 -05:00
|
|
|
#include "module.h"
|
|
|
|
#include "modules.h"
|
|
|
|
#include "signals.h"
|
|
|
|
#include "commands.h"
|
|
|
|
#include "servers.h"
|
|
|
|
|
2001-07-29 05:17:53 -04:00
|
|
|
#include "perl-core.h"
|
2001-01-03 02:34:12 -05:00
|
|
|
#include "perl-common.h"
|
2001-01-04 04:45:46 -05:00
|
|
|
#include "perl-signals.h"
|
2001-01-03 02:34:12 -05:00
|
|
|
|
|
|
|
typedef struct {
|
2001-12-06 12:17:40 -05:00
|
|
|
PERL_SCRIPT_REC *script;
|
2001-01-03 02:34:12 -05:00
|
|
|
int signal_id;
|
|
|
|
char *signal;
|
2001-12-06 12:17:40 -05:00
|
|
|
SV *func;
|
2001-01-03 02:34:12 -05:00
|
|
|
} PERL_SIGNAL_REC;
|
|
|
|
|
|
|
|
typedef struct {
|
|
|
|
char *signal;
|
|
|
|
char *args[7];
|
2002-12-04 17:17:07 -05:00
|
|
|
int dynamic;
|
2001-01-03 02:34:12 -05:00
|
|
|
} PERL_SIGNAL_ARGS_REC;
|
|
|
|
|
|
|
|
#include "perl-signals-list.h"
|
|
|
|
|
2002-04-16 14:11:06 -04:00
|
|
|
static GHashTable *signals;
|
2001-01-03 02:34:12 -05:00
|
|
|
static GHashTable *perl_signal_args_hash;
|
|
|
|
static GSList *perl_signal_args_partial;
|
|
|
|
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2001-12-06 12:17:40 -05:00
|
|
|
static void perl_call_signal(PERL_SCRIPT_REC *script, SV *func,
|
|
|
|
int signal_id, gconstpointer *args)
|
2001-01-03 02:34:12 -05:00
|
|
|
{
|
|
|
|
dSP;
|
|
|
|
|
|
|
|
PERL_SIGNAL_ARGS_REC *rec;
|
2001-06-08 18:27:06 -04:00
|
|
|
SV *sv, *perlarg, *saved_args[SIGNAL_MAX_ARGUMENTS];
|
|
|
|
AV *av;
|
2001-01-03 02:34:12 -05:00
|
|
|
void *arg;
|
|
|
|
int n;
|
|
|
|
|
|
|
|
|
|
|
|
ENTER;
|
|
|
|
SAVETMPS;
|
|
|
|
|
|
|
|
PUSHMARK(sp);
|
|
|
|
|
|
|
|
/* push signal argument to perl stack */
|
|
|
|
rec = perl_signal_args_find(signal_id);
|
|
|
|
|
2001-06-08 18:27:06 -04:00
|
|
|
memset(saved_args, 0, sizeof(saved_args));
|
2001-01-03 02:34:12 -05:00
|
|
|
for (n = 0; n < SIGNAL_MAX_ARGUMENTS &&
|
|
|
|
rec != NULL && rec->args[n] != NULL; n++) {
|
|
|
|
arg = (void *) args[n];
|
|
|
|
|
|
|
|
if (strcmp(rec->args[n], "string") == 0)
|
|
|
|
perlarg = new_pv(arg);
|
|
|
|
else if (strcmp(rec->args[n], "int") == 0)
|
2002-10-27 20:49:14 -05:00
|
|
|
perlarg = newSViv((IV)arg);
|
2001-01-03 02:34:12 -05:00
|
|
|
else if (strcmp(rec->args[n], "ulongptr") == 0)
|
|
|
|
perlarg = newSViv(*(unsigned long *) arg);
|
2001-06-08 18:27:06 -04:00
|
|
|
else if (strcmp(rec->args[n], "intptr") == 0)
|
|
|
|
saved_args[n] = perlarg = newRV_noinc(newSViv(*(int *) arg));
|
|
|
|
else 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 = strcmp(rec->args[n]+9, "iobject") == 0;
|
|
|
|
is_str = strcmp(rec->args[n]+9, "char*") == 0;
|
|
|
|
av = newAV();
|
|
|
|
|
|
|
|
ptr = arg;
|
|
|
|
for (tmp = *ptr; tmp != NULL; tmp = tmp->next) {
|
2001-11-19 14:36:21 -05:00
|
|
|
sv = is_iobject ? iobject_bless((SERVER_REC *) tmp->data) :
|
2001-06-08 18:27:06 -04:00
|
|
|
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 (strncmp(rec->args[n], "gslist_", 7) == 0) {
|
2001-01-03 02:34:12 -05:00
|
|
|
/* linked list - push as AV */
|
|
|
|
GSList *tmp;
|
2001-06-08 18:27:06 -04:00
|
|
|
int is_iobject;
|
2001-01-03 02:34:12 -05:00
|
|
|
|
2001-06-08 18:27:06 -04:00
|
|
|
is_iobject = strcmp(rec->args[n]+7, "iobject") == 0;
|
2001-01-03 02:34:12 -05:00
|
|
|
av = newAV();
|
2001-03-07 07:09:38 -05:00
|
|
|
for (tmp = arg; tmp != NULL; tmp = tmp->next) {
|
2001-11-19 14:36:21 -05:00
|
|
|
sv = is_iobject ? iobject_bless((SERVER_REC *) tmp->data) :
|
2001-03-07 07:09:38 -05:00
|
|
|
irssi_bless_plain(rec->args[n]+7, tmp->data);
|
|
|
|
av_push(av, sv);
|
|
|
|
}
|
|
|
|
|
|
|
|
perlarg = newRV_noinc((SV *) av);
|
2001-01-03 02:34:12 -05:00
|
|
|
} else if (arg == NULL) {
|
|
|
|
/* don't bless NULL arguments */
|
|
|
|
perlarg = newSViv(0);
|
|
|
|
} else if (strcmp(rec->args[n], "iobject") == 0) {
|
|
|
|
/* "irssi object" - any struct that has
|
2001-11-19 14:36:21 -05:00
|
|
|
"int type; int chat_type" as it's first
|
2001-01-03 02:34:12 -05:00
|
|
|
variables (server, channel, ..) */
|
2001-11-19 14:36:21 -05:00
|
|
|
perlarg = iobject_bless((SERVER_REC *) arg);
|
|
|
|
} else if (strcmp(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);
|
2001-01-03 02:34:12 -05:00
|
|
|
} else {
|
|
|
|
/* blessed object */
|
2001-07-29 00:03:58 -04:00
|
|
|
perlarg = plain_bless(arg, rec->args[n]);
|
2001-01-03 02:34:12 -05:00
|
|
|
}
|
2001-03-07 07:09:38 -05:00
|
|
|
XPUSHs(sv_2mortal(perlarg));
|
2001-01-03 02:34:12 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
PUTBACK;
|
2001-12-06 12:17:40 -05:00
|
|
|
perl_call_sv(func, G_EVAL|G_DISCARD);
|
2001-01-03 02:34:12 -05:00
|
|
|
SPAGAIN;
|
|
|
|
|
|
|
|
if (SvTRUE(ERRSV)) {
|
2002-01-10 12:36:41 -05:00
|
|
|
char *error = g_strdup(SvPV(ERRSV, PL_na));
|
|
|
|
signal_emit("script error", 2, script, error);
|
|
|
|
g_free(error);
|
2001-08-08 18:43:00 -04:00
|
|
|
rec = NULL;
|
2001-01-03 02:34:12 -05:00
|
|
|
}
|
|
|
|
|
2001-06-08 18:27:06 -04:00
|
|
|
/* 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 (strcmp(rec->args[n], "intptr") == 0) {
|
|
|
|
int *val = arg;
|
|
|
|
*val = SvIV(SvRV(saved_args[n]));
|
|
|
|
} else if (strncmp(rec->args[n], "glistptr_", 9) == 0) {
|
|
|
|
GList **ret = arg;
|
|
|
|
GList *out = NULL;
|
|
|
|
void *val;
|
|
|
|
STRLEN len;
|
|
|
|
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(sv, len));
|
|
|
|
else
|
|
|
|
val = GINT_TO_POINTER(SvIV(sv));
|
|
|
|
|
|
|
|
out = g_list_append(out, val);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (strcmp(rec->args[n]+9, "char*") == 0)
|
|
|
|
g_list_foreach(*ret, (GFunc) g_free, NULL);
|
|
|
|
g_list_free(*ret);
|
|
|
|
*ret = out;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-01-03 02:34:12 -05:00
|
|
|
PUTBACK;
|
|
|
|
FREETMPS;
|
|
|
|
LEAVE;
|
|
|
|
}
|
|
|
|
|
2002-04-16 14:11:06 -04:00
|
|
|
static void sig_func(const void *p1, const void *p2,
|
|
|
|
const void *p3, const void *p4,
|
|
|
|
const void *p5, const void *p6)
|
2001-01-03 02:34:12 -05:00
|
|
|
{
|
2002-04-16 14:11:06 -04:00
|
|
|
PERL_SIGNAL_REC *rec;
|
|
|
|
const void *args[6];
|
2001-01-03 02:34:12 -05:00
|
|
|
|
2002-04-16 14:11:06 -04:00
|
|
|
args[0] = p1; args[1] = p2; args[2] = p3;
|
|
|
|
args[3] = p4; args[4] = p5; args[5] = p6;
|
2001-01-03 02:34:12 -05:00
|
|
|
|
2002-04-16 14:11:06 -04:00
|
|
|
rec = signal_get_user_data();
|
|
|
|
perl_call_signal(rec->script, rec->func, signal_get_emitted_id(), args);
|
2001-01-03 02:34:12 -05:00
|
|
|
}
|
|
|
|
|
2002-04-16 14:11:06 -04:00
|
|
|
static void perl_signal_add_full_int(const char *signal, SV *func,
|
|
|
|
int priority, int command,
|
|
|
|
const char *category)
|
2001-01-03 02:34:12 -05:00
|
|
|
{
|
2001-12-06 12:17:40 -05:00
|
|
|
PERL_SCRIPT_REC *script;
|
2001-01-03 02:34:12 -05:00
|
|
|
PERL_SIGNAL_REC *rec;
|
|
|
|
GSList **siglist;
|
|
|
|
void *signal_idp;
|
|
|
|
|
|
|
|
g_return_if_fail(signal != NULL);
|
|
|
|
g_return_if_fail(func != NULL);
|
|
|
|
|
2001-12-06 12:17:40 -05:00
|
|
|
script = perl_script_find_package(perl_get_package());
|
|
|
|
g_return_if_fail(script != NULL);
|
|
|
|
|
2001-01-03 02:34:12 -05:00
|
|
|
rec = g_new(PERL_SIGNAL_REC, 1);
|
2001-12-06 12:17:40 -05:00
|
|
|
rec->script = script;
|
2001-01-03 02:34:12 -05:00
|
|
|
rec->signal_id = signal_get_uniq_id(signal);
|
|
|
|
rec->signal = g_strdup(signal);
|
2001-12-06 12:17:40 -05:00
|
|
|
rec->func = perl_func_sv_inc(func, perl_get_package());
|
2001-01-03 02:34:12 -05:00
|
|
|
|
2002-04-16 14:11:06 -04:00
|
|
|
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);
|
|
|
|
}
|
2001-01-03 02:34:12 -05:00
|
|
|
|
2002-04-16 14:11:06 -04:00
|
|
|
signal_idp = GINT_TO_POINTER(rec->signal_id);
|
|
|
|
siglist = g_hash_table_lookup(signals, signal_idp);
|
2001-01-03 02:34:12 -05:00
|
|
|
if (siglist == NULL) {
|
|
|
|
siglist = g_new0(GSList *, 1);
|
2002-04-16 14:11:06 -04:00
|
|
|
g_hash_table_insert(signals, signal_idp, siglist);
|
2001-01-03 02:34:12 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
*siglist = g_slist_append(*siglist, rec);
|
|
|
|
}
|
|
|
|
|
2002-04-16 14:11:06 -04:00
|
|
|
void perl_signal_add_full(const char *signal, SV *func, int priority)
|
2001-02-17 07:23:18 -05:00
|
|
|
{
|
2002-04-16 14:11:06 -04:00
|
|
|
perl_signal_add_full_int(signal, func, priority, FALSE, NULL);
|
2001-02-17 07:23:18 -05:00
|
|
|
}
|
|
|
|
|
2001-01-03 02:34:12 -05:00
|
|
|
static void perl_signal_destroy(PERL_SIGNAL_REC *rec)
|
|
|
|
{
|
|
|
|
if (strncmp(rec->signal, "command ", 8) == 0)
|
2002-04-16 14:11:06 -04:00
|
|
|
command_unbind_full(rec->signal+8, sig_func, rec);
|
|
|
|
else
|
|
|
|
signal_remove_id(rec->signal_id, sig_func, rec);
|
2001-01-03 02:34:12 -05:00
|
|
|
|
2001-12-06 12:17:40 -05:00
|
|
|
SvREFCNT_dec(rec->func);
|
2001-01-03 02:34:12 -05:00
|
|
|
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);
|
2002-04-16 14:11:06 -04:00
|
|
|
g_hash_table_remove(signals, GINT_TO_POINTER(rec->signal_id));
|
2001-01-03 02:34:12 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
perl_signal_destroy(rec);
|
|
|
|
}
|
|
|
|
|
2001-12-06 12:17:40 -05:00
|
|
|
#define sv_func_cmp(f1, f2, len) \
|
|
|
|
(f1 == f2 || (SvPOK(f1) && SvPOK(f2) && \
|
|
|
|
strcmp((char *) SvPV(f1, len), (char *) SvPV(f2, len)) == 0))
|
|
|
|
|
|
|
|
static void perl_signal_remove_list(GSList **list, SV *func)
|
2001-01-03 02:34:12 -05:00
|
|
|
{
|
|
|
|
GSList *tmp;
|
|
|
|
|
|
|
|
for (tmp = *list; tmp != NULL; tmp = tmp->next) {
|
|
|
|
PERL_SIGNAL_REC *rec = tmp->data;
|
|
|
|
|
2001-12-11 11:44:53 -05:00
|
|
|
if (sv_func_cmp(rec->func, func, PL_na)) {
|
2001-01-03 02:34:12 -05:00
|
|
|
perl_signal_remove_list_one(list, rec);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-12-06 12:17:40 -05:00
|
|
|
void perl_signal_remove(const char *signal, SV *func)
|
2001-01-03 02:34:12 -05:00
|
|
|
{
|
|
|
|
GSList **list;
|
|
|
|
void *signal_idp;
|
|
|
|
|
|
|
|
signal_idp = GINT_TO_POINTER(signal_get_uniq_id(signal));
|
2002-04-16 14:11:06 -04:00
|
|
|
list = g_hash_table_lookup(signals, signal_idp);
|
2001-01-03 02:34:12 -05:00
|
|
|
|
2002-04-16 14:11:06 -04:00
|
|
|
if (list != NULL) {
|
|
|
|
func = perl_func_sv_inc(func, perl_get_package());
|
|
|
|
perl_signal_remove_list(list, func);
|
|
|
|
SvREFCNT_dec(func);
|
2001-01-03 02:34:12 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2001-07-16 16:03:36 -04:00
|
|
|
void perl_command_bind_to(const char *cmd, const char *category,
|
2001-12-06 12:17:40 -05:00
|
|
|
SV *func, int priority)
|
2001-01-04 04:45:46 -05:00
|
|
|
{
|
|
|
|
char *signal;
|
|
|
|
|
|
|
|
signal = g_strconcat("command ", cmd, NULL);
|
2002-04-16 14:11:06 -04:00
|
|
|
perl_signal_add_full_int(signal, func, priority, TRUE, category);
|
2001-01-04 04:45:46 -05:00
|
|
|
g_free(signal);
|
|
|
|
}
|
|
|
|
|
2001-05-30 04:58:48 -04:00
|
|
|
void perl_command_runsub(const char *cmd, const char *data,
|
|
|
|
SERVER_REC *server, WI_ITEM_REC *item)
|
|
|
|
{
|
|
|
|
command_runsub(cmd, data, server, item);
|
|
|
|
}
|
|
|
|
|
2001-12-06 12:17:40 -05:00
|
|
|
void perl_command_unbind(const char *cmd, SV *func)
|
2001-01-04 04:45:46 -05:00
|
|
|
{
|
|
|
|
char *signal;
|
|
|
|
|
|
|
|
/* perl_signal_remove() calls command_unbind() */
|
|
|
|
signal = g_strconcat("command ", cmd, NULL);
|
|
|
|
perl_signal_remove(signal, func);
|
|
|
|
g_free(signal);
|
|
|
|
}
|
|
|
|
|
2001-12-06 12:17:40 -05:00
|
|
|
static int signal_destroy_hash(void *key, GSList **list, PERL_SCRIPT_REC *script)
|
2001-01-03 02:34:12 -05:00
|
|
|
{
|
|
|
|
GSList *tmp, *next;
|
|
|
|
|
|
|
|
for (tmp = *list; tmp != NULL; tmp = next) {
|
|
|
|
PERL_SIGNAL_REC *rec = tmp->data;
|
|
|
|
|
|
|
|
next = tmp->next;
|
2001-12-06 12:17:40 -05:00
|
|
|
if (script == NULL || rec->script == script) {
|
|
|
|
*list = g_slist_remove(*list, rec);
|
|
|
|
perl_signal_destroy(rec);
|
2001-01-03 02:34:12 -05:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (*list != NULL)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
g_free(list);
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
2001-12-06 12:17:40 -05:00
|
|
|
/* destroy all signals used by script */
|
|
|
|
void perl_signal_remove_script(PERL_SCRIPT_REC *script)
|
2001-01-03 02:34:12 -05:00
|
|
|
{
|
2002-04-16 14:11:06 -04:00
|
|
|
g_hash_table_foreach_remove(signals, (GHRFunc) signal_destroy_hash,
|
|
|
|
script);
|
2001-01-03 02:34:12 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
void perl_signals_start(void)
|
|
|
|
{
|
2002-04-16 14:11:06 -04:00
|
|
|
signals = g_hash_table_new(NULL, NULL);
|
2001-01-03 02:34:12 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
void perl_signals_stop(void)
|
|
|
|
{
|
2002-04-16 14:11:06 -04:00
|
|
|
g_hash_table_foreach(signals, (GHFunc) signal_destroy_hash, NULL);
|
|
|
|
g_hash_table_destroy(signals);
|
|
|
|
signals = NULL;
|
2001-01-03 02:34:12 -05:00
|
|
|
}
|
|
|
|
|
2002-12-04 17:17:07 -05:00
|
|
|
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 < 6 && args[i] != NULL; i++)
|
|
|
|
rec->args[i] = g_strdup(args[i]);
|
|
|
|
rec->dynamic = TRUE;
|
|
|
|
rec->signal = g_strdup(signal);
|
|
|
|
register_signal_rec(rec);
|
|
|
|
}
|
|
|
|
|
2001-01-03 02:34:12 -05:00
|
|
|
void perl_signals_init(void)
|
|
|
|
{
|
|
|
|
int n;
|
|
|
|
|
|
|
|
perl_signal_args_hash = g_hash_table_new((GHashFunc) g_direct_hash,
|
|
|
|
(GCompareFunc) g_direct_equal);
|
|
|
|
perl_signal_args_partial = NULL;
|
|
|
|
|
2002-12-04 17:17:07 -05:00
|
|
|
for (n = 0; perl_signal_args[n].signal != NULL; n++)
|
|
|
|
register_signal_rec(&perl_signal_args[n]);
|
|
|
|
}
|
2001-01-03 02:34:12 -05:00
|
|
|
|
2002-12-04 17:17:07 -05:00
|
|
|
static void signal_args_free(PERL_SIGNAL_ARGS_REC *rec)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
if (!rec->dynamic)
|
|
|
|
return;
|
|
|
|
|
|
|
|
for (i = 0; i < 6 && 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);
|
2001-01-03 02:34:12 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
void perl_signals_deinit(void)
|
|
|
|
{
|
2002-12-04 17:17:07 -05:00
|
|
|
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);
|
2001-01-03 02:34:12 -05:00
|
|
|
g_hash_table_destroy(perl_signal_args_hash);
|
|
|
|
}
|