summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorHenning Brauer <henning@cvs.openbsd.org>2004-01-05 19:10:25 +0000
committerHenning Brauer <henning@cvs.openbsd.org>2004-01-05 19:10:25 +0000
commitb23fd4b94033681d702393178a99a2a173e00950 (patch)
treeda80f62677d9db5e93a4c5089f649f57e3f10d92
parent8633e7a422fd44a7026d84004d5478bd31c55c39 (diff)
correctly handle SIGCHLD.
SIGCHLD does _not_ translate to "a child process went kaboom". waitpid() and check status; if the child exited or terminated log & quit ok claudio@
-rw-r--r--usr.sbin/bgpd/bgpd.c36
1 files changed, 34 insertions, 2 deletions
diff --git a/usr.sbin/bgpd/bgpd.c b/usr.sbin/bgpd/bgpd.c
index 9b35693422d..a60a7a8c3d8 100644
--- a/usr.sbin/bgpd/bgpd.c
+++ b/usr.sbin/bgpd/bgpd.c
@@ -1,4 +1,4 @@
-/* $OpenBSD: bgpd.c,v 1.55 2004/01/05 18:21:51 henning Exp $ */
+/* $OpenBSD: bgpd.c,v 1.56 2004/01/05 19:10:24 henning Exp $ */
/*
* Copyright (c) 2003 Henning Brauer <henning@openbsd.org>
@@ -38,6 +38,7 @@
void sighdlr(int);
void usage(void);
int main(int, char *[]);
+int check_child(pid_t, const char *);
int reconfigure(char *, struct bgpd_config *, struct mrt_config *,
struct peer *);
int dispatch_imsg(struct imsgbuf *, int, struct mrt_config *);
@@ -47,6 +48,7 @@ int rfd = -1;
volatile sig_atomic_t mrtdump = 0;
volatile sig_atomic_t quit = 0;
volatile sig_atomic_t reconfig = 0;
+volatile sig_atomic_t sigchld = 0;
struct imsgbuf ibuf_se;
struct imsgbuf ibuf_rde;
@@ -56,9 +58,11 @@ sighdlr(int sig)
switch (sig) {
case SIGTERM:
case SIGINT:
- case SIGCHLD:
quit = 1;
break;
+ case SIGCHLD:
+ sigchld = 1;
+ break;
case SIGHUP:
reconfig = 1;
break;
@@ -281,6 +285,14 @@ main(int argc, char *argv[])
reconfig = 0;
}
+ if (sigchld) {
+ if (check_child(io_pid, "session engine"))
+ quit = 1;
+ if (check_child(rde_pid, "route decision engine"))
+ quit = 1;
+ sigchld = 0;
+ }
+
if (mrtdump == 1) {
mrt_alrm(&mrtconf, &ibuf_rde);
mrtdump = 0;
@@ -310,6 +322,26 @@ main(int argc, char *argv[])
}
int
+check_child(pid_t pid, const char *pname)
+{
+ int status;
+
+ if (waitpid(pid, &status, WNOHANG) > 0) {
+ if (WIFEXITED(status)) {
+ logit(LOG_CRIT, "Lost child: %s exited", pname);
+ return (1);
+ }
+ if (WIFSIGNALED(status)) {
+ logit(LOG_CRIT, "Lost child: %s terminated; signal %d",
+ pname, WTERMSIG(status));
+ return (1);
+ }
+ }
+
+ return (0);
+}
+
+int
reconfigure(char *conffile, struct bgpd_config *conf, struct mrt_config *mrtc,
struct peer *peer_l)
{