103 lines
3.1 KiB
103 lines
3.1 KiB
# --- SDE-COPYRIGHT-NOTE-BEGIN --- |
|
# This copyright note is auto-generated by ./scripts/Create-CopyPatch. |
|
# |
|
# Filename: package/.../tcp_wrappers/0015-tcp_wrappers-7.6-catch-sigchld.patch |
|
# Copyright (C) 2011 The OpenSDE Project |
|
# |
|
# More information can be found in the files COPYING and README. |
|
# |
|
# This patch file is dual-licensed. It is available under the license the |
|
# patched project is licensed under, as long as it is an OpenSource license |
|
# as defined at http://www.opensource.org/ (e.g. BSD, X11) or 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. |
|
# --- SDE-COPYRIGHT-NOTE-END --- |
|
|
|
--- tcp_wrappers_7.6/shell_cmd.c.sigchld 1994-12-28 17:42:44.000000000 +0100 |
|
+++ tcp_wrappers_7.6/shell_cmd.c 2007-06-28 15:42:17.000000000 +0200 |
|
@@ -20,6 +20,11 @@ |
|
#include <stdio.h> |
|
#include <syslog.h> |
|
#include <string.h> |
|
+#include <errno.h> |
|
+#include <unistd.h> |
|
+#include <sys/wait.h> |
|
+#include <sys/stat.h> |
|
+#include <fcntl.h> |
|
|
|
extern void exit(); |
|
|
|
@@ -31,13 +36,42 @@ |
|
|
|
static void do_child(); |
|
|
|
+/* |
|
+ * The sigchld handler. If there is a SIGCHLD caused by a child other than |
|
+ * ours, we set a flag and raise the signal later. |
|
+ */ |
|
+volatile static int foreign_sigchld; |
|
+volatile static int our_child_pid; |
|
+static void sigchld(int sig, siginfo_t *si, void *unused) |
|
+{ |
|
+ if (si && si->si_pid != our_child_pid) |
|
+ foreign_sigchld = 1; |
|
+} |
|
+ |
|
/* shell_cmd - execute shell command */ |
|
|
|
void shell_cmd(command) |
|
char *command; |
|
{ |
|
int child_pid; |
|
- int wait_pid; |
|
+ |
|
+ struct sigaction new_action, old_action; |
|
+ sigset_t new_mask, old_mask, empty_mask; |
|
+ |
|
+ new_action.sa_sigaction = &sigchld; |
|
+ new_action.sa_flags = SA_SIGINFO; |
|
+ sigemptyset(&new_action.sa_mask); |
|
+ sigemptyset(&new_mask); |
|
+ sigemptyset(&empty_mask); |
|
+ sigaddset(&new_mask, SIGCHLD); |
|
+ |
|
+ /* |
|
+ * Set the variables for handler, set the handler and block the signal |
|
+ * until we have the pid. |
|
+ */ |
|
+ foreign_sigchld = 0; our_child_pid = 0; |
|
+ sigprocmask(SIG_BLOCK, &new_mask, &old_mask); |
|
+ sigaction(SIGCHLD, &new_action, &old_action); |
|
|
|
/* |
|
* Most of the work is done within the child process, to minimize the |
|
@@ -49,12 +83,26 @@ |
|
tcpd_warn("cannot fork: %m"); |
|
break; |
|
case 00: /* child */ |
|
+ /* Clear the blocked mask for the child not to be surprised. */ |
|
+ sigprocmask(SIG_SETMASK, &empty_mask, 0); |
|
do_child(command); |
|
/* NOTREACHED */ |
|
default: /* parent */ |
|
- while ((wait_pid = wait((int *) 0)) != -1 && wait_pid != child_pid) |
|
- /* void */ ; |
|
+ our_child_pid = child_pid; |
|
+ sigprocmask(SIG_UNBLOCK, &new_mask, 0); |
|
+ while (waitpid(child_pid, (int *) 0, 0) == -1 && errno == EINTR); |
|
} |
|
+ |
|
+ /* |
|
+ * Revert the signal mask and the SIGCHLD handler. |
|
+ */ |
|
+ sigprocmask(SIG_SETMASK, &old_mask, 0); |
|
+ sigaction(SIGCHLD, &old_action, 0); |
|
+ |
|
+ /* If there was a foreign SIGCHLD, raise it after we have restored the old |
|
+ * mask and handler. */ |
|
+ if (foreign_sigchld) |
|
+ raise(SIGCHLD); |
|
} |
|
|
|
/* do_child - exec command with { stdin, stdout, stderr } to /dev/null */
|
|
|