1 1.3 martin /* $NetBSD: raise_default_signal.c,v 1.3 2008/04/28 20:23:03 martin Exp $ */ 2 1.1 lukem 3 1.1 lukem /*- 4 1.1 lukem * Copyright (c) 2007 The NetBSD Foundation, Inc. 5 1.1 lukem * All rights reserved. 6 1.1 lukem * 7 1.1 lukem * This code is derived from software contributed to The NetBSD Foundation 8 1.1 lukem * by Luke Mewburn. 9 1.1 lukem * 10 1.1 lukem * Redistribution and use in source and binary forms, with or without 11 1.1 lukem * modification, are permitted provided that the following conditions 12 1.1 lukem * are met: 13 1.1 lukem * 1. Redistributions of source code must retain the above copyright 14 1.1 lukem * notice, this list of conditions and the following disclaimer. 15 1.1 lukem * 2. Redistributions in binary form must reproduce the above copyright 16 1.1 lukem * notice, this list of conditions and the following disclaimer in the 17 1.1 lukem * documentation and/or other materials provided with the distribution. 18 1.1 lukem * 19 1.1 lukem * THIS SOFTWARE IS PROVIDED BY THE NETBSD FOUNDATION, INC. AND CONTRIBUTORS 20 1.1 lukem * ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED 21 1.1 lukem * TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR 22 1.1 lukem * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE FOUNDATION OR CONTRIBUTORS 23 1.1 lukem * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR 24 1.1 lukem * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF 25 1.1 lukem * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS 26 1.1 lukem * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN 27 1.1 lukem * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) 28 1.1 lukem * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE 29 1.1 lukem * POSSIBILITY OF SUCH DAMAGE. 30 1.1 lukem */ 31 1.1 lukem 32 1.2 lukem #if HAVE_NBTOOL_CONFIG_H 33 1.2 lukem #include "nbtool_config.h" 34 1.2 lukem #endif 35 1.2 lukem 36 1.1 lukem #include <sys/cdefs.h> 37 1.1 lukem #if defined(LIBC_SCCS) && !defined(lint) 38 1.3 martin __RCSID("$NetBSD: raise_default_signal.c,v 1.3 2008/04/28 20:23:03 martin Exp $"); 39 1.1 lukem #endif 40 1.1 lukem 41 1.1 lukem #include <errno.h> 42 1.1 lukem #include <signal.h> 43 1.1 lukem #include <stdio.h> 44 1.1 lukem #include <string.h> 45 1.1 lukem #include <util.h> 46 1.1 lukem 47 1.2 lukem #if ! HAVE_RAISE_DEFAULT_SIGNAL 48 1.1 lukem /* 49 1.1 lukem * raise_default_signal sig 50 1.1 lukem * Raise the default signal handler for sig, by 51 1.1 lukem * - block all signals 52 1.1 lukem * - set the signal handler to SIG_DFL 53 1.1 lukem * - raise the signal 54 1.1 lukem * - unblock the signal to deliver it 55 1.1 lukem * 56 1.1 lukem * The original signal mask and signal handler is restored on exit 57 1.1 lukem * (whether successful or not). 58 1.1 lukem * 59 1.1 lukem * Returns 0 on success, or -1 on failure with errno set to 60 1.1 lukem * on of the values for sigemptyset(), sigaddset(), sigprocmask(), 61 1.1 lukem * sigaction(), or raise(). 62 1.1 lukem */ 63 1.1 lukem int 64 1.1 lukem raise_default_signal(int sig) 65 1.1 lukem { 66 1.1 lukem struct sigaction origact, act; 67 1.1 lukem sigset_t origmask, fullmask, mask; 68 1.1 lukem int retval, oerrno; 69 1.1 lukem 70 1.1 lukem retval = -1; 71 1.1 lukem 72 1.1 lukem /* Setup data structures */ 73 1.1 lukem /* XXX memset(3) isn't async-safe according to signal(7) */ 74 1.1 lukem (void)memset(&act, 0, sizeof(act)); 75 1.1 lukem act.sa_handler = SIG_DFL; 76 1.1 lukem act.sa_flags = 0; 77 1.1 lukem if ((sigemptyset(&act.sa_mask) == -1) || 78 1.1 lukem (sigfillset(&fullmask) == -1) || 79 1.1 lukem (sigemptyset(&mask) == -1) || 80 1.1 lukem (sigaddset(&mask, sig) == -1)) 81 1.1 lukem goto restore_none; 82 1.1 lukem 83 1.1 lukem /* Block all signals */ 84 1.1 lukem if (sigprocmask(SIG_BLOCK, &fullmask, &origmask) == -1) 85 1.1 lukem goto restore_none; 86 1.1 lukem /* (use 'goto restore_mask' to restore state) */ 87 1.1 lukem 88 1.1 lukem /* Enable the SIG_DFL handler */ 89 1.1 lukem if (sigaction(sig, &act, &origact) == -1) 90 1.1 lukem goto restore_mask; 91 1.1 lukem /* (use 'goto restore_act' to restore state) */ 92 1.1 lukem 93 1.1 lukem /* Raise the signal, and unblock the signal to deliver it */ 94 1.1 lukem if ((raise(sig) == -1) || 95 1.1 lukem (sigprocmask(SIG_UNBLOCK, &mask, NULL) == -1)) 96 1.1 lukem goto restore_act; 97 1.1 lukem 98 1.1 lukem /* Flag successful raise() */ 99 1.1 lukem retval = 0; 100 1.1 lukem 101 1.1 lukem /* Restore the original handler */ 102 1.1 lukem restore_act: 103 1.1 lukem oerrno = errno; 104 1.1 lukem (void)sigaction(sig, &origact, NULL); 105 1.1 lukem errno = oerrno; 106 1.1 lukem 107 1.1 lukem /* Restore the original mask */ 108 1.1 lukem restore_mask: 109 1.1 lukem oerrno = errno; 110 1.1 lukem (void)sigprocmask(SIG_SETMASK, &origmask, NULL); 111 1.1 lukem errno = oerrno; 112 1.1 lukem 113 1.1 lukem restore_none: 114 1.1 lukem return retval; 115 1.1 lukem } 116 1.2 lukem 117 1.2 lukem #endif /* ! HAVE_RAISE_DEFAULT_SIGNAL */ 118