Merge remote-tracking branch 'origin/master' into credentials
[oweals/gnunet.git] / src / util / signal.c
1 /*
2      This file is part of GNUnet.
3      Copyright (C) 2001, 2002, 2006 GNUnet e.V.
4
5      GNUnet is free software; you can redistribute it and/or modify
6      it under the terms of the GNU General Public License as published
7      by the Free Software Foundation; either version 3, or (at your
8      option) any later version.
9
10      GNUnet is distributed in the hope that it will be useful, but
11      WITHOUT ANY WARRANTY; without even the implied warranty of
12      MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
13      General Public License for more details.
14
15      You should have received a copy of the GNU General Public License
16      along with GNUnet; see the file COPYING.  If not, write to the
17      Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
18      Boston, MA 02110-1301, USA.
19 */
20
21 /**
22  * @file util/signal.c
23  * @brief code for installing and uninstalling signal handlers
24  * @author Christian Grothoff
25  */
26
27 #include "platform.h"
28 #include "gnunet_util_lib.h"
29
30 #define LOG(kind,...) GNUNET_log_from (kind, "util-signal", __VA_ARGS__)
31
32
33 struct GNUNET_SIGNAL_Context
34 {
35
36   struct GNUNET_SIGNAL_Context *next;
37
38   struct GNUNET_SIGNAL_Context *prev;
39
40   int sig;
41
42   GNUNET_SIGNAL_Handler method;
43
44 #ifndef MINGW
45   struct sigaction oldsig;
46 #endif
47 };
48
49 static struct GNUNET_SIGNAL_Context *sc_head;
50
51 static struct GNUNET_SIGNAL_Context *sc_tail;
52
53
54 #ifdef WINDOWS
55 GNUNET_SIGNAL_Handler w32_sigchld_handler = NULL;
56 #endif
57
58 struct GNUNET_SIGNAL_Context *
59 GNUNET_SIGNAL_handler_install (int signum, GNUNET_SIGNAL_Handler handler)
60 {
61   struct GNUNET_SIGNAL_Context *ret;
62
63 #ifndef MINGW
64   struct sigaction sig;
65 #endif
66
67   ret = GNUNET_new (struct GNUNET_SIGNAL_Context);
68   ret->sig = signum;
69   ret->method = handler;
70 #ifndef MINGW
71   memset (&sig, 0, sizeof (sig));
72   sig.sa_handler = (void *) handler;
73   sigemptyset (&sig.sa_mask);
74 #ifdef SA_INTERRUPT
75   sig.sa_flags = SA_INTERRUPT;  /* SunOS */
76 #else
77   sig.sa_flags = SA_RESTART;
78 #endif
79   sigaction (signum, &sig, &ret->oldsig);
80 #else
81   if (signum == GNUNET_SIGCHLD)
82     w32_sigchld_handler = handler;
83   else
84   {
85     __p_sig_fn_t sigret = signal (signum, (__p_sig_fn_t) handler);
86
87     if (sigret == SIG_ERR)
88     {
89       LOG (GNUNET_ERROR_TYPE_WARNING, _("signal (%d, %p) returned %d.\n"),
90            signum, handler, sigret);
91     }
92   }
93 #endif
94   GNUNET_CONTAINER_DLL_insert_tail (sc_head, sc_tail, ret);
95   return ret;
96 }
97
98 void
99 GNUNET_SIGNAL_handler_uninstall (struct GNUNET_SIGNAL_Context *ctx)
100 {
101 #ifndef MINGW
102   struct sigaction sig;
103
104   sigemptyset (&sig.sa_mask);
105   sigaction (ctx->sig, &ctx->oldsig, &sig);
106 #endif
107   GNUNET_CONTAINER_DLL_remove (sc_head, sc_tail, ctx);
108   GNUNET_free (ctx);
109 }
110
111
112 /**
113  * Raise the given signal by calling the installed signal handlers.  This will
114  * not use the @em raise() system call but only calls the handlers registered
115  * through GNUNET_SIGNAL_handler_install().
116  *
117  * @param sig the signal to raise
118  */
119 void
120 GNUNET_SIGNAL_raise (const int sig)
121 {
122   struct GNUNET_SIGNAL_Context *ctx;
123
124   for (ctx = sc_head; NULL != ctx; ctx = ctx->next)
125   {
126     if (sig != ctx->sig)
127       continue;
128     if (NULL == ctx->method)
129       continue;
130     ctx->method ();
131   }
132 }