1
/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 8 -*-
3
* Copyright (C) 2006 Red Hat, Inc.
4
* Copyright (C) 2007 William Jon McCann <mccann@jhu.edu>
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
17
* along with this program; if not, write to the Free Software
18
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
38
#include <glib/gi18n.h>
39
#include <glib/gstdio.h>
40
#include <glib-object.h>
42
#include "gdm-signal-handler.h"
44
#define GDM_SIGNAL_HANDLER_GET_PRIVATE(o) (G_TYPE_INSTANCE_GET_PRIVATE ((o), GDM_TYPE_SIGNAL_HANDLER, GdmSignalHandlerPrivate))
48
GdmSignalHandlerFunc func;
53
struct GdmSignalHandlerPrivate
56
GHashTable *id_lookup;
57
GHashTable *action_lookup;
59
GDestroyNotify fatal_func;
63
static void gdm_signal_handler_class_init (GdmSignalHandlerClass *klass);
64
static void gdm_signal_handler_init (GdmSignalHandler *signal_handler);
65
static void gdm_signal_handler_finalize (GObject *object);
67
static gpointer signal_handler_object = NULL;
68
static int signal_pipes[2];
69
static int signals_blocked = 0;
70
static sigset_t signals_block_mask;
71
static sigset_t signals_oldmask;
73
G_DEFINE_TYPE (GdmSignalHandler, gdm_signal_handler, G_TYPE_OBJECT)
76
block_signals_push (void)
80
if (signals_blocked == 1) {
82
sigemptyset (&signals_block_mask);
83
sigfillset (&signals_block_mask);
84
sigprocmask (SIG_BLOCK, &signals_block_mask, &signals_oldmask);
89
block_signals_pop (void)
93
if (signals_blocked == 0) {
95
sigprocmask (SIG_SETMASK, &signals_oldmask, NULL);
100
signal_io_watch (GIOChannel *ioc,
101
GIOCondition condition,
102
GdmSignalHandler *handler)
109
block_signals_push ();
111
g_io_channel_read_chars (ioc, buf, sizeof (buf), &bytes_read, NULL);
115
for (i = 0; i < bytes_read; i++) {
120
signum = (gint32)buf[i];
122
g_debug ("GdmSignalHandler: handling signal %d", signum);
123
handlers = g_hash_table_lookup (handler->priv->lookup, GINT_TO_POINTER (signum));
125
g_debug ("GdmSignalHandler: Found %u callbacks", g_slist_length (handlers));
126
for (l = handlers; l != NULL; l = l->next) {
130
data = g_hash_table_lookup (handler->priv->id_lookup, l->data);
132
if (data->func != NULL) {
133
g_debug ("GdmSignalHandler: running %d handler: %p", signum, data->func);
134
res = data->func (signum, data->data);
143
block_signals_pop ();
146
if (handler->priv->fatal_func != NULL) {
147
g_debug ("GdmSignalHandler: Caught termination signal - calling fatal func");
148
handler->priv->fatal_func (handler->priv->fatal_data);
150
g_debug ("GdmSignalHandler: Caught termination signal - exiting");
157
g_debug ("GdmSignalHandler: Done handling signals");
163
fallback_get_backtrace (void)
165
#ifdef HAVE_EXECINFO_H
171
size = backtrace (frames, G_N_ELEMENTS (frames));
172
if ((strings = backtrace_symbols (frames, size))) {
173
syslog (LOG_CRIT, "******************* START ********************************");
174
for (i = 0; i < size; i++) {
175
syslog (LOG_CRIT, "Frame %zd: %s", i, strings[i]);
178
syslog (LOG_CRIT, "******************* END **********************************");
182
g_warning ("GDM crashed, but symbols couldn't be retrieved.");
187
crashlogger_get_backtrace (void)
189
gboolean success = FALSE;
194
/* Wait for the child to finish */
196
if (waitpid (pid, &estatus, 0) != -1) {
197
/* Only succeed if the crashlogger succeeded */
198
if (WIFEXITED (estatus) && (WEXITSTATUS (estatus) == 0)) {
202
} else if (pid == 0) {
204
execl (LIBEXECDIR "/gdm-crash-logger",
205
LIBEXECDIR "/gdm-crash-logger", NULL);
213
gdm_signal_handler_backtrace (void)
216
gboolean fallback = TRUE;
218
/* Try to use gdb via gdm-crash-logger if it exists, since
219
* we get much better information out of it. Otherwise
220
* fall back to execinfo.
222
if (g_stat (LIBEXECDIR "/gdm-crash-logger", &s) == 0) {
223
fallback = crashlogger_get_backtrace () ? FALSE : TRUE;
227
fallback_get_backtrace ();
232
signal_handler (int signo)
234
static int in_fatal = 0;
236
guchar signo_byte = signo;
251
gdm_signal_handler_backtrace ();
256
/* let the fatal signals interrupt us */
258
gdm_signal_handler_backtrace ();
259
ignore = write (signal_pipes [1], &signo_byte, 1);
263
ignore = write (signal_pipes [1], &signo_byte, 1);
269
catch_signal (GdmSignalHandler *handler,
272
struct sigaction action;
273
struct sigaction *old_action;
275
g_debug ("GdmSignalHandler: Registering for %d signals", signal_number);
277
action.sa_handler = signal_handler;
278
sigemptyset (&action.sa_mask);
281
old_action = g_new0 (struct sigaction, 1);
283
sigaction (signal_number, &action, old_action);
285
g_hash_table_insert (handler->priv->action_lookup,
286
GINT_TO_POINTER (signal_number),
291
uncatch_signal (GdmSignalHandler *handler,
294
struct sigaction *old_action;
296
g_debug ("GdmSignalHandler: Unregistering for %d signals", signal_number);
298
old_action = g_hash_table_lookup (handler->priv->action_lookup,
299
GINT_TO_POINTER (signal_number));
300
g_hash_table_remove (handler->priv->action_lookup,
301
GINT_TO_POINTER (signal_number));
303
sigaction (signal_number, old_action, NULL);
309
gdm_signal_handler_add (GdmSignalHandler *handler,
311
GdmSignalHandlerFunc callback,
317
g_return_val_if_fail (GDM_IS_SIGNAL_HANDLER (handler), 0);
319
cdata = g_new0 (CallbackData, 1);
320
cdata->signal_number = signal_number;
321
cdata->func = callback;
323
cdata->id = handler->priv->next_id++;
325
g_debug ("GdmSignalHandler: Adding handler %u: signum=%d %p", cdata->id, cdata->signal_number, cdata->func);
327
if (g_hash_table_lookup (handler->priv->action_lookup, GINT_TO_POINTER (signal_number)) == NULL) {
328
catch_signal (handler, signal_number);
331
/* ID lookup owns the CallbackData */
332
g_hash_table_insert (handler->priv->id_lookup, GUINT_TO_POINTER (cdata->id), cdata);
334
list = g_hash_table_lookup (handler->priv->lookup, GINT_TO_POINTER (signal_number));
335
list = g_slist_prepend (list, GUINT_TO_POINTER (cdata->id));
337
g_hash_table_insert (handler->priv->lookup, GINT_TO_POINTER (signal_number), list);
343
gdm_signal_handler_add_fatal (GdmSignalHandler *handler)
345
g_return_if_fail (GDM_IS_SIGNAL_HANDLER (handler));
347
gdm_signal_handler_add (handler, SIGILL, NULL, NULL);
348
gdm_signal_handler_add (handler, SIGBUS, NULL, NULL);
349
gdm_signal_handler_add (handler, SIGSEGV, NULL, NULL);
350
gdm_signal_handler_add (handler, SIGABRT, NULL, NULL);
351
gdm_signal_handler_add (handler, SIGTRAP, NULL, NULL);
355
callback_data_free (CallbackData *d)
361
gdm_signal_handler_remove_and_free_data (GdmSignalHandler *handler,
366
g_return_if_fail (GDM_IS_SIGNAL_HANDLER (handler));
368
list = g_hash_table_lookup (handler->priv->lookup, GINT_TO_POINTER (cdata->signal_number));
369
list = g_slist_remove_all (list, GUINT_TO_POINTER (cdata->id));
371
uncatch_signal (handler, cdata->signal_number);
374
g_debug ("GdmSignalHandler: Removing handler %u: signum=%d %p", cdata->signal_number, cdata->id, cdata->func);
375
/* put changed list back in */
376
g_hash_table_insert (handler->priv->lookup, GINT_TO_POINTER (cdata->signal_number), list);
378
g_hash_table_remove (handler->priv->id_lookup, GUINT_TO_POINTER (cdata->id));
382
gdm_signal_handler_remove (GdmSignalHandler *handler,
387
g_return_if_fail (GDM_IS_SIGNAL_HANDLER (handler));
389
found = g_hash_table_lookup (handler->priv->id_lookup, GUINT_TO_POINTER (id));
391
gdm_signal_handler_remove_and_free_data (handler, found);
396
static CallbackData *
397
find_callback_data_by_func (GdmSignalHandler *handler,
399
GdmSignalHandlerFunc callback,
408
list = g_hash_table_lookup (handler->priv->lookup, GINT_TO_POINTER (signal_number));
410
for (l = list; l != NULL; l = l->next) {
414
id = GPOINTER_TO_UINT (l->data);
416
d = g_hash_table_lookup (handler->priv->id_lookup, GUINT_TO_POINTER (id));
418
&& d->func == callback
419
&& d->data == data) {
429
gdm_signal_handler_remove_func (GdmSignalHandler *handler,
431
GdmSignalHandlerFunc callback,
436
g_return_if_fail (GDM_IS_SIGNAL_HANDLER (handler));
438
found = find_callback_data_by_func (handler, signal_number, callback, data);
441
gdm_signal_handler_remove_and_free_data (handler, found);
445
/* FIXME: once all handlers are removed deregister signum handler */
449
gdm_signal_handler_class_init (GdmSignalHandlerClass *klass)
451
GObjectClass *object_class = G_OBJECT_CLASS (klass);
453
object_class->finalize = gdm_signal_handler_finalize;
455
g_type_class_add_private (klass, sizeof (GdmSignalHandlerPrivate));
459
signal_list_free (GSList *list)
465
gdm_signal_handler_set_fatal_func (GdmSignalHandler *handler,
469
g_return_if_fail (GDM_IS_SIGNAL_HANDLER (handler));
471
handler->priv->fatal_func = func;
472
handler->priv->fatal_data = user_data;
476
gdm_signal_handler_init (GdmSignalHandler *handler)
480
handler->priv = GDM_SIGNAL_HANDLER_GET_PRIVATE (handler);
482
handler->priv->next_id = 1;
484
handler->priv->lookup = g_hash_table_new (NULL, NULL);
485
handler->priv->id_lookup = g_hash_table_new (NULL, NULL);
486
handler->priv->action_lookup = g_hash_table_new (NULL, NULL);
488
if (pipe (signal_pipes) == -1) {
489
g_error ("Could not create pipe() for signal handling");
491
fcntl(signal_pipes[0], F_SETFD, FD_CLOEXEC);
492
fcntl(signal_pipes[1], F_SETFD, FD_CLOEXEC);
494
ioc = g_io_channel_unix_new (signal_pipes[0]);
495
g_io_channel_set_flags (ioc, G_IO_FLAG_NONBLOCK, NULL);
496
g_io_add_watch (ioc, G_IO_IN, (GIOFunc)signal_io_watch, handler);
497
g_io_channel_set_close_on_unref (ioc, TRUE);
498
g_io_channel_unref (ioc);
502
gdm_signal_handler_finalize (GObject *object)
504
GdmSignalHandler *handler;
507
g_return_if_fail (object != NULL);
508
g_return_if_fail (GDM_IS_SIGNAL_HANDLER (object));
510
handler = GDM_SIGNAL_HANDLER (object);
512
g_debug ("GdmSignalHandler: Finalizing signal handler");
514
g_return_if_fail (handler->priv != NULL);
515
for (l = g_hash_table_get_values (handler->priv->lookup);
516
l != NULL; l = l->next) {
517
signal_list_free ((GSList *) l->data);
519
g_hash_table_destroy (handler->priv->lookup);
520
for (l = g_hash_table_get_values (handler->priv->id_lookup);
521
l != NULL; l = l->next) {
522
callback_data_free ((CallbackData *) l->data);
524
g_hash_table_destroy (handler->priv->id_lookup);
525
for (l = g_hash_table_get_values (handler->priv->action_lookup);
526
l != NULL; l = l->next) {
529
g_hash_table_destroy (handler->priv->action_lookup);
531
close (signal_pipes [0]);
532
close (signal_pipes [1]);
534
G_OBJECT_CLASS (gdm_signal_handler_parent_class)->finalize (object);
538
gdm_signal_handler_new (void)
540
if (signal_handler_object != NULL) {
541
g_object_ref (signal_handler_object);
543
signal_handler_object = g_object_new (GDM_TYPE_SIGNAL_HANDLER, NULL);
544
g_object_add_weak_pointer (signal_handler_object,
545
(gpointer *) &signal_handler_object);
548
return GDM_SIGNAL_HANDLER (signal_handler_object);