summaryrefslogtreecommitdiff
path: root/udev
diff options
context:
space:
mode:
authorJeremy Kerr <jk@ozlabs.org>2009-09-05 17:48:23 +1000
committerKay Sievers <kay.sievers@vrfy.org>2009-09-07 12:13:50 +0200
commitd412a685736e3b3350b555f4d7d8ebfc80aa54c9 (patch)
tree6e49304c9c86a1db4711f1df8fcc4d6d4ca0344a /udev
parent9e6273c713c5a4b4317decca33d2fba830562ff2 (diff)
util_run_program: restore signal mask before executing event RUN commands
External programs triggered by events (via RUN=) will inherit udev's signal mask, which is set to block all but SIGALRM. For most utilities, this is OK, but if we start daemons from RUN=, we run into trouble (especially as SIGCHLD is blocked). This change saves the original sigmask when udev starts, and restores it just before we exec() the external command. Signed-off-by: Jeremy Kerr <jk@ozlabs.org>
Diffstat (limited to 'udev')
-rw-r--r--udev/test-udev.c2
-rw-r--r--udev/udev-event.c4
-rw-r--r--udev/udev-rules.c4
-rw-r--r--udev/udev.h3
-rw-r--r--udev/udevd.c6
5 files changed, 11 insertions, 8 deletions
diff --git a/udev/test-udev.c b/udev/test-udev.c
index c6b8bf573b..0806fbf9ce 100644
--- a/udev/test-udev.c
+++ b/udev/test-udev.c
@@ -103,7 +103,7 @@ int main(int argc, char *argv[])
alarm(udev_device_get_event_timeout(dev));
if (err == 0 && !event->ignore_device && udev_get_run(udev))
- udev_event_execute_run(event);
+ udev_event_execute_run(event, NULL);
udev_event_unref(event);
udev_device_unref(dev);
diff --git a/udev/udev-event.c b/udev/udev-event.c
index 8183793e5f..7b4e4ac5c3 100644
--- a/udev/udev-event.c
+++ b/udev/udev-event.c
@@ -722,7 +722,7 @@ exit:
return err;
}
-int udev_event_execute_run(struct udev_event *event)
+int udev_event_execute_run(struct udev_event *event, const sigset_t *sigmask)
{
struct udev_list_entry *list_entry;
int err = 0;
@@ -745,7 +745,7 @@ int udev_event_execute_run(struct udev_event *event)
udev_event_apply_format(event, cmd, program, sizeof(program));
envp = udev_device_get_properties_envp(event->dev);
- if (util_run_program(event->udev, program, envp, NULL, 0, NULL) != 0) {
+ if (util_run_program(event->udev, program, envp, NULL, 0, NULL, sigmask) != 0) {
if (udev_list_entry_get_flag(list_entry))
err = -1;
}
diff --git a/udev/udev-rules.c b/udev/udev-rules.c
index a92446403f..20ec2706bd 100644
--- a/udev/udev-rules.c
+++ b/udev/udev-rules.c
@@ -749,7 +749,7 @@ static int import_program_into_properties(struct udev_device *dev, const char *p
char *line;
envp = udev_device_get_properties_envp(dev);
- if (util_run_program(udev, program, envp, result, sizeof(result), &reslen) != 0)
+ if (util_run_program(udev, program, envp, result, sizeof(result), &reslen, NULL) != 0)
return -1;
line = result;
@@ -2206,7 +2206,7 @@ int udev_rules_apply_to_event(struct udev_rules *rules, struct udev_event *event
program,
&rules->buf[rule->rule.filename_off],
rule->rule.filename_line);
- if (util_run_program(event->udev, program, envp, result, sizeof(result), NULL) != 0) {
+ if (util_run_program(event->udev, program, envp, result, sizeof(result), NULL, NULL) != 0) {
if (cur->key.op != OP_NOMATCH)
goto nomatch;
} else {
diff --git a/udev/udev.h b/udev/udev.h
index 391276f6e4..8f0a3014b6 100644
--- a/udev/udev.h
+++ b/udev/udev.h
@@ -21,6 +21,7 @@
#include <sys/types.h>
#include <sys/param.h>
+#include <signal.h>
#include "libudev.h"
#include "libudev-private.h"
@@ -67,7 +68,7 @@ int udev_rules_apply_to_event(struct udev_rules *rules, struct udev_event *event
struct udev_event *udev_event_new(struct udev_device *dev);
void udev_event_unref(struct udev_event *event);
int udev_event_execute_rules(struct udev_event *event, struct udev_rules *rules);
-int udev_event_execute_run(struct udev_event *event);
+int udev_event_execute_run(struct udev_event *event, const sigset_t *sigset);
size_t udev_event_apply_format(struct udev_event *event, const char *src, char *dest, size_t size);
int udev_event_apply_subsys_kernel(struct udev_event *event, const char *string,
char *result, size_t maxsize, int read_value);
diff --git a/udev/udevd.c b/udev/udevd.c
index 54064cbfcf..2eb914a3f3 100644
--- a/udev/udevd.c
+++ b/udev/udevd.c
@@ -79,6 +79,7 @@ static bool stop_exec_queue;
static bool reload_config;
static int max_childs;
static int childs;
+static sigset_t orig_sigmask;
static struct udev_list_node event_list;
static struct udev_list_node worker_list;
static bool udev_exit;
@@ -292,7 +293,8 @@ static void worker_new(struct event *event)
/* execute RUN= */
if (err == 0 && !udev_event->ignore_device && udev_get_run(udev_event->udev))
- failed = udev_event_execute_run(udev_event);
+ failed = udev_event_execute_run(udev_event,
+ &orig_sigmask);
/* reset alarm */
alarm(0);
@@ -926,7 +928,7 @@ int main(int argc, char *argv[])
/* block and listen to all signals on signalfd */
sigfillset(&mask);
- sigprocmask(SIG_SETMASK, &mask, NULL);
+ sigprocmask(SIG_SETMASK, &mask, &orig_sigmask);
pfd[FD_SIGNAL].fd = signalfd(-1, &mask, 0);
if (pfd[FD_SIGNAL].fd < 0) {
fprintf(stderr, "error getting signalfd\n");