diff options
author | Johan Hedberg <johan.hedberg@nokia.com> | 2007-05-07 08:53:52 +0000 |
---|---|---|
committer | Johan Hedberg <johan.hedberg@nokia.com> | 2007-05-07 08:53:52 +0000 |
commit | 9306a904c561b8d6552c0de2584ee875d2034751 (patch) | |
tree | 9c57e0b103db3cda088cb3b0994344e699ad56b6 /hcid | |
parent | 060a1f8b64b41815c040050f2431741c6a49032d (diff) |
Use a pipe to inform the parent process of child exits during init_device() and configure_device()
Diffstat (limited to 'hcid')
-rw-r--r-- | hcid/main.c | 46 |
1 files changed, 44 insertions, 2 deletions
diff --git a/hcid/main.c b/hcid/main.c index f7b1bfd0..7cbf679a 100644 --- a/hcid/main.c +++ b/hcid/main.c @@ -37,6 +37,8 @@ #include <sys/stat.h> #include <sys/ioctl.h> #include <sys/socket.h> +#include <sys/types.h> +#include <sys/wait.h> #include <bluetooth/bluetooth.h> #include <bluetooth/hci.h> @@ -57,6 +59,7 @@ struct hcid_opts hcid; struct device_opts default_device; struct device_opts *parser_device; static struct device_list *device_list = NULL; +static int child_pipe[2]; static inline void init_device_defaults(struct device_opts *device_opts) { @@ -275,6 +278,32 @@ static char *expand_name(char *dst, int size, char *str, int dev_id) return dst; } +static gboolean child_exit(GIOChannel *io, GIOCondition cond, void *user_data) +{ + int status, fd = g_io_channel_unix_get_fd(io); + pid_t child_pid; + + if (read(fd, &child_pid, sizeof(child_pid)) != sizeof(child_pid)) { + error("child_exit: unable to read child pid from pipe"); + return TRUE; + } + + if (waitpid(child_pid, &status, 0) != child_pid) + error("waitpid(%d) failed", child_pid); + else + debug("child %d exited", child_pid); + + return TRUE; +} + +static void at_child_exit(void) +{ + pid_t pid = getpid(); + + if (write(child_pipe[1], &pid, sizeof(pid)) != sizeof(pid)) + error("unable to write to child pipe"); +} + static void configure_device(int dev_id) { struct device_opts *device_opts; @@ -313,6 +342,7 @@ static void configure_device(int dev_id) /* Do configuration in the separate process */ switch (fork()) { case 0: + atexit(at_child_exit); break; case -1: error("Fork failed. Can't init device hci%d: %s (%d)", @@ -437,6 +467,7 @@ static void init_device(int dev_id) /* Do initialization in the separate process */ switch (fork()) { case 0: + atexit(at_child_exit); break; case -1: error("Fork failed. Can't init device hci%d: %s (%d)", @@ -652,7 +683,7 @@ int main(int argc, char *argv[]) struct sockaddr_hci addr; struct hci_filter flt; struct sigaction sa; - GIOChannel *ctl_io; + GIOChannel *ctl_io, *child_io; int opt, daemonize = 1, debug = 0, sdp = 0, experimental = 0; /* Default HCId settings */ @@ -720,7 +751,6 @@ int main(int argc, char *argv[]) sigaction(SIGUSR2, &sa, NULL); sa.sa_handler = SIG_IGN; - sigaction(SIGCHLD, &sa, NULL); sigaction(SIGPIPE, &sa, NULL); if (debug) { @@ -756,6 +786,18 @@ int main(int argc, char *argv[]) if (read_config(hcid.config_file) < 0) error("Config load failed"); + if (pipe(child_pipe) < 0) { + error("pipe(): %s (%d)", strerror(errno), errno); + exit(1); + } + + child_io = g_io_channel_unix_new(child_pipe[0]); + g_io_channel_set_close_on_unref(child_io, TRUE); + g_io_add_watch(child_io, + G_IO_IN | G_IO_ERR | G_IO_HUP | G_IO_NVAL, + child_exit, NULL); + g_io_channel_unref(child_io); + init_devices(); if (hcid_dbus_init() < 0) { |