| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  | /*
 | 
					
						
							| 
									
										
										
										
											2010-01-15 15:28:14 -08:00
										 |  |  |  |  * Copyright (c) 2008, 2009, 2010 Nicira Networks. | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-06-15 15:11:30 -07:00
										 |  |  |  |  * Licensed under the Apache License, Version 2.0 (the "License"); | 
					
						
							|  |  |  |  |  * you may not use this file except in compliance with the License. | 
					
						
							|  |  |  |  |  * You may obtain a copy of the License at: | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2009-06-15 15:11:30 -07:00
										 |  |  |  |  *     http://www.apache.org/licenses/LICENSE-2.0
 | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Unless required by applicable law or agreed to in writing, software | 
					
						
							|  |  |  |  |  * distributed under the License is distributed on an "AS IS" BASIS, | 
					
						
							|  |  |  |  |  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | 
					
						
							|  |  |  |  |  * See the License for the specific language governing permissions and | 
					
						
							|  |  |  |  |  * limitations under the License. | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  |  */ | 
					
						
							|  |  |  |  | #include <config.h>
 | 
					
						
							|  |  |  |  | #include "fatal-signal.h"
 | 
					
						
							|  |  |  |  | #include <assert.h>
 | 
					
						
							|  |  |  |  | #include <errno.h>
 | 
					
						
							|  |  |  |  | #include <signal.h>
 | 
					
						
							|  |  |  |  | #include <stdbool.h>
 | 
					
						
							|  |  |  |  | #include <stdio.h>
 | 
					
						
							| 
									
										
										
										
											2009-12-08 14:11:22 -08:00
										 |  |  |  | #include <stdint.h>
 | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  |  | #include <string.h>
 | 
					
						
							|  |  |  |  | #include <unistd.h>
 | 
					
						
							| 
									
										
										
										
											2009-12-08 14:11:22 -08:00
										 |  |  |  | #include "poll-loop.h"
 | 
					
						
							| 
									
										
										
										
											2009-09-21 12:38:58 -07:00
										 |  |  |  | #include "shash.h"
 | 
					
						
							| 
									
										
										
										
											2009-12-08 14:11:22 -08:00
										 |  |  |  | #include "socket-util.h"
 | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  | #include "util.h"
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-21 12:37:20 -07:00
										 |  |  |  | #define THIS_MODULE VLM_fatal_signal
 | 
					
						
							|  |  |  |  | #include "vlog.h"
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  | /* Signals to catch. */ | 
					
						
							|  |  |  |  | static const int fatal_signals[] = { SIGTERM, SIGINT, SIGHUP, SIGALRM }; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /* Signals to catch as a sigset_t. */ | 
					
						
							|  |  |  |  | static sigset_t fatal_signal_set; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /* Hooks to call upon catching a signal */ | 
					
						
							|  |  |  |  | struct hook { | 
					
						
							| 
									
										
										
										
											2010-01-15 15:28:14 -08:00
										 |  |  |  |     void (*hook_cb)(void *aux); | 
					
						
							|  |  |  |  |     void (*cancel_cb)(void *aux); | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  |     void *aux; | 
					
						
							|  |  |  |  |     bool run_at_exit; | 
					
						
							|  |  |  |  | }; | 
					
						
							|  |  |  |  | #define MAX_HOOKS 32
 | 
					
						
							|  |  |  |  | static struct hook hooks[MAX_HOOKS]; | 
					
						
							|  |  |  |  | static size_t n_hooks; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-08 14:11:22 -08:00
										 |  |  |  | static int signal_fds[2]; | 
					
						
							|  |  |  |  | static volatile sig_atomic_t stored_sig_nr = SIG_ATOMIC_MAX; | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-08 14:11:22 -08:00
										 |  |  |  | static void fatal_signal_init(void); | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  | static void atexit_handler(void); | 
					
						
							|  |  |  |  | static void call_hooks(int sig_nr); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-08 14:11:22 -08:00
										 |  |  |  | static void | 
					
						
							|  |  |  |  | fatal_signal_init(void) | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     static bool inited = false; | 
					
						
							| 
									
										
										
										
											2009-12-08 14:11:22 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  |     if (!inited) { | 
					
						
							|  |  |  |  |         size_t i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         inited = true; | 
					
						
							| 
									
										
										
										
											2009-12-08 14:11:22 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |         if (pipe(signal_fds)) { | 
					
						
							|  |  |  |  |             ovs_fatal(errno, "could not create pipe"); | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         set_nonblocking(signal_fds[0]); | 
					
						
							|  |  |  |  |         set_nonblocking(signal_fds[1]); | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  |         sigemptyset(&fatal_signal_set); | 
					
						
							|  |  |  |  |         for (i = 0; i < ARRAY_SIZE(fatal_signals); i++) { | 
					
						
							|  |  |  |  |             int sig_nr = fatal_signals[i]; | 
					
						
							|  |  |  |  |             struct sigaction old_sa; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |             sigaddset(&fatal_signal_set, sig_nr); | 
					
						
							|  |  |  |  |             if (sigaction(sig_nr, NULL, &old_sa)) { | 
					
						
							|  |  |  |  |                 ovs_fatal(errno, "sigaction"); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |             if (old_sa.sa_handler == SIG_DFL | 
					
						
							|  |  |  |  |                 && signal(sig_nr, fatal_signal_handler) == SIG_ERR) { | 
					
						
							|  |  |  |  |                 ovs_fatal(errno, "signal"); | 
					
						
							|  |  |  |  |             } | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |         atexit(atexit_handler); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-15 15:28:14 -08:00
										 |  |  |  | /* Registers 'hook_cb' to be called when a process termination signal is
 | 
					
						
							|  |  |  |  |  * raised.  If 'run_at_exit' is true, 'hook_cb' is also called during normal | 
					
						
							|  |  |  |  |  * process termination, e.g. when exit() is called or when main() returns. | 
					
						
							| 
									
										
										
										
											2009-12-08 14:11:22 -08:00
										 |  |  |  |  * | 
					
						
							| 
									
										
										
										
											2010-01-15 15:28:14 -08:00
										 |  |  |  |  * 'hook_cb' is not called immediately from the signal handler but rather the | 
					
						
							| 
									
										
										
										
											2009-12-08 14:11:22 -08:00
										 |  |  |  |  * next time the poll loop iterates, so it is freed from the usual restrictions | 
					
						
							| 
									
										
										
										
											2010-01-15 15:28:14 -08:00
										 |  |  |  |  * on signal handler functions. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * If the current process forks, fatal_signal_fork() may be called to clear the | 
					
						
							|  |  |  |  |  * parent process's fatal signal hooks, so that 'hook_cb' is only called when | 
					
						
							|  |  |  |  |  * the child terminates, not when the parent does.  When fatal_signal_fork() is | 
					
						
							|  |  |  |  |  * called, it calls the 'cancel_cb' function if it is nonnull, passing 'aux', | 
					
						
							|  |  |  |  |  * to notify that the hook has been canceled.  This allows the hook to free | 
					
						
							|  |  |  |  |  * memory, etc. */ | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  | void | 
					
						
							| 
									
										
										
										
											2010-01-15 15:28:14 -08:00
										 |  |  |  | fatal_signal_add_hook(void (*hook_cb)(void *aux), void (*cancel_cb)(void *aux), | 
					
						
							|  |  |  |  |                       void *aux, bool run_at_exit) | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-12-08 14:11:22 -08:00
										 |  |  |  |     fatal_signal_init(); | 
					
						
							| 
									
										
										
										
											2010-01-15 15:28:14 -08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-08 14:11:22 -08:00
										 |  |  |  |     assert(n_hooks < MAX_HOOKS); | 
					
						
							| 
									
										
										
										
											2010-01-15 15:28:14 -08:00
										 |  |  |  |     hooks[n_hooks].hook_cb = hook_cb; | 
					
						
							|  |  |  |  |     hooks[n_hooks].cancel_cb = cancel_cb; | 
					
						
							| 
									
										
										
										
											2009-12-08 14:11:22 -08:00
										 |  |  |  |     hooks[n_hooks].aux = aux; | 
					
						
							|  |  |  |  |     hooks[n_hooks].run_at_exit = run_at_exit; | 
					
						
							|  |  |  |  |     n_hooks++; | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /* Handles fatal signal number 'sig_nr'.
 | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Ordinarily this is the actual signal handler.  When other code needs to | 
					
						
							|  |  |  |  |  * handle one of our signals, however, it can register for that signal and, if | 
					
						
							|  |  |  |  |  * and when necessary, call this function to do fatal signal processing for it | 
					
						
							|  |  |  |  |  * and terminate the process.  Currently only timeval.c does this, for SIGALRM. | 
					
						
							|  |  |  |  |  * (It is not important whether the other code sets up its signal handler | 
					
						
							|  |  |  |  |  * before or after this file, because this file will only set up a signal | 
					
						
							|  |  |  |  |  * handler in the case where the signal has its default handling.)  */ | 
					
						
							|  |  |  |  | void | 
					
						
							|  |  |  |  | fatal_signal_handler(int sig_nr) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-12-08 14:11:22 -08:00
										 |  |  |  |     ignore(write(signal_fds[1], "", 1)); | 
					
						
							|  |  |  |  |     stored_sig_nr = sig_nr; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-13 09:28:13 -07:00
										 |  |  |  | /* Check whether a fatal signal has occurred and, if so, call the fatal signal
 | 
					
						
							|  |  |  |  |  * hooks and exit. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * This function is called automatically by poll_block(), but specialized | 
					
						
							|  |  |  |  |  * programs that may not always call poll_block() on a regular basis should | 
					
						
							|  |  |  |  |  * also call it periodically.  (Therefore, any function with "block" in its | 
					
						
							|  |  |  |  |  * name should call fatal_signal_run() each time it is called, either directly | 
					
						
							|  |  |  |  |  * or through poll_block(), because such functions can only used by specialized | 
					
						
							|  |  |  |  |  * programs that can afford to block outside their main loop around | 
					
						
							|  |  |  |  |  * poll_block().) | 
					
						
							|  |  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-12-08 14:11:22 -08:00
										 |  |  |  | void | 
					
						
							|  |  |  |  | fatal_signal_run(void) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-03-23 15:27:44 -07:00
										 |  |  |  |     int sig_nr; | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-23 15:27:44 -07:00
										 |  |  |  |     fatal_signal_init(); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     sig_nr = stored_sig_nr; | 
					
						
							| 
									
										
										
										
											2009-12-08 14:11:22 -08:00
										 |  |  |  |     if (sig_nr != SIG_ATOMIC_MAX) { | 
					
						
							|  |  |  |  |         call_hooks(sig_nr); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         /* Re-raise the signal with the default handling so that the program
 | 
					
						
							|  |  |  |  |          * termination status reflects that we were killed by this signal */ | 
					
						
							|  |  |  |  |         signal(sig_nr, SIG_DFL); | 
					
						
							|  |  |  |  |         raise(sig_nr); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | void | 
					
						
							|  |  |  |  | fatal_signal_wait(void) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-03-23 15:27:44 -07:00
										 |  |  |  |     fatal_signal_init(); | 
					
						
							| 
									
										
										
										
											2009-12-08 14:11:22 -08:00
										 |  |  |  |     poll_fd_wait(signal_fds[0], POLLIN); | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | atexit_handler(void) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-01-15 15:28:14 -08:00
										 |  |  |  |     call_hooks(0); | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | static void | 
					
						
							|  |  |  |  | call_hooks(int sig_nr) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     static volatile sig_atomic_t recurse = 0; | 
					
						
							|  |  |  |  |     if (!recurse) { | 
					
						
							|  |  |  |  |         size_t i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         recurse = 1; | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |         for (i = 0; i < n_hooks; i++) { | 
					
						
							|  |  |  |  |             struct hook *h = &hooks[i]; | 
					
						
							|  |  |  |  |             if (sig_nr || h->run_at_exit) { | 
					
						
							| 
									
										
										
										
											2010-01-15 15:28:14 -08:00
										 |  |  |  |                 h->hook_cb(h->aux); | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  |             } | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  |  | 
					
						
							| 
									
										
										
										
											2010-01-15 15:28:14 -08:00
										 |  |  |  | /* Files to delete on exit.  (The 'data' member of each node is unused.) */ | 
					
						
							| 
									
										
										
										
											2009-09-21 12:38:58 -07:00
										 |  |  |  | static struct shash files = SHASH_INITIALIZER(&files); | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-15 15:28:14 -08:00
										 |  |  |  | /* Has a hook function been registered with fatal_signal_add_hook() (and not
 | 
					
						
							|  |  |  |  |  * cleared by fatal_signal_fork())? */ | 
					
						
							|  |  |  |  | static bool added_hook; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  | static void unlink_files(void *aux); | 
					
						
							| 
									
										
										
										
											2010-01-15 15:28:14 -08:00
										 |  |  |  | static void cancel_files(void *aux); | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  | static void do_unlink_files(void); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /* Registers 'file' to be unlinked when the program terminates via exit() or a
 | 
					
						
							|  |  |  |  |  * fatal signal. */ | 
					
						
							|  |  |  |  | void | 
					
						
							|  |  |  |  | fatal_signal_add_file_to_unlink(const char *file) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     if (!added_hook) { | 
					
						
							|  |  |  |  |         added_hook = true; | 
					
						
							| 
									
										
										
										
											2010-01-15 15:28:14 -08:00
										 |  |  |  |         fatal_signal_add_hook(unlink_files, cancel_files, NULL, true); | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-30 13:28:22 -07:00
										 |  |  |  |     shash_add_once(&files, file, NULL); | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | /* Unregisters 'file' from being unlinked when the program terminates via
 | 
					
						
							|  |  |  |  |  * exit() or a fatal signal. */ | 
					
						
							|  |  |  |  | void | 
					
						
							|  |  |  |  | fatal_signal_remove_file_to_unlink(const char *file) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-09-21 12:38:58 -07:00
										 |  |  |  |     struct shash_node *node; | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-21 12:38:58 -07:00
										 |  |  |  |     node = shash_find(&files, file); | 
					
						
							|  |  |  |  |     if (node) { | 
					
						
							|  |  |  |  |         shash_delete(&files, node); | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-21 12:37:20 -07:00
										 |  |  |  | /* Like fatal_signal_remove_file_to_unlink(), but also unlinks 'file'.
 | 
					
						
							|  |  |  |  |  * Returns 0 if successful, otherwise a positive errno value. */ | 
					
						
							|  |  |  |  | int | 
					
						
							|  |  |  |  | fatal_signal_unlink_file_now(const char *file) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     int error = unlink(file) ? errno : 0; | 
					
						
							|  |  |  |  |     if (error) { | 
					
						
							|  |  |  |  |         VLOG_WARN("could not unlink \"%s\" (%s)", file, strerror(error)); | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     fatal_signal_remove_file_to_unlink(file); | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     return error; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  | static void | 
					
						
							| 
									
										
										
										
											2010-02-11 10:59:47 -08:00
										 |  |  |  | unlink_files(void *aux OVS_UNUSED) | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     do_unlink_files();  | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-15 15:28:14 -08:00
										 |  |  |  | static void | 
					
						
							| 
									
										
										
										
											2010-02-11 11:11:23 -08:00
										 |  |  |  | cancel_files(void *aux OVS_UNUSED) | 
					
						
							| 
									
										
										
										
											2010-01-15 15:28:14 -08:00
										 |  |  |  | { | 
					
						
							|  |  |  |  |     shash_clear(&files); | 
					
						
							|  |  |  |  |     added_hook = false; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  | static void | 
					
						
							|  |  |  |  | do_unlink_files(void) | 
					
						
							|  |  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-09-21 12:38:58 -07:00
										 |  |  |  |     struct shash_node *node; | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-09-21 12:38:58 -07:00
										 |  |  |  |     SHASH_FOR_EACH (node, &files) { | 
					
						
							|  |  |  |  |         unlink(node->name); | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  |  | 
					
						
							| 
									
										
										
										
											2010-01-15 15:28:14 -08:00
										 |  |  |  | /* Clears all of the fatal signal hooks without executing them.  If any of the
 | 
					
						
							|  |  |  |  |  * hooks passed a 'cancel_cb' function to fatal_signal_add_hook(), then those | 
					
						
							|  |  |  |  |  * functions will be called, allowing them to free resources, etc. | 
					
						
							|  |  |  |  |  * | 
					
						
							|  |  |  |  |  * Following a fork, one of the resulting processes can call this function to | 
					
						
							|  |  |  |  |  * allow it to terminate without calling the hooks registered before calling | 
					
						
							|  |  |  |  |  * this function.  New hooks registered after calling this function will take | 
					
						
							|  |  |  |  |  * effect normally. */ | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  | void | 
					
						
							|  |  |  |  | fatal_signal_fork(void) | 
					
						
							|  |  |  |  | { | 
					
						
							|  |  |  |  |     size_t i; | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-15 15:28:14 -08:00
										 |  |  |  |     for (i = 0; i < n_hooks; i++) { | 
					
						
							|  |  |  |  |         struct hook *h = &hooks[i]; | 
					
						
							|  |  |  |  |         if (h->cancel_cb) { | 
					
						
							|  |  |  |  |             h->cancel_cb(h->aux); | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  |         } | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-01-15 15:28:14 -08:00
										 |  |  |  |     n_hooks = 0; | 
					
						
							| 
									
										
										
										
											2009-12-08 14:11:22 -08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  |     /* Raise any signals that we have already received with the default
 | 
					
						
							|  |  |  |  |      * handler. */ | 
					
						
							|  |  |  |  |     if (stored_sig_nr != SIG_ATOMIC_MAX) { | 
					
						
							|  |  |  |  |         raise(stored_sig_nr); | 
					
						
							| 
									
										
										
										
											2009-07-08 13:19:16 -07:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | } |