Compare commits
No commits in common. "ea9f36ac83e1b90e46719f03e4d4629ddda11393" and "0ce80921099fd113eb94e72249375555022079ac" have entirely different histories.
ea9f36ac83
...
0ce8092109
5
NEWS
5
NEWS
|
@ -100,11 +100,6 @@ CHANGES WITH 246 in spe:
|
||||||
can now be suspended or resumed either using new systemctl verbs,
|
can now be suspended or resumed either using new systemctl verbs,
|
||||||
freeze and thaw respectively, or via D-Bus.
|
freeze and thaw respectively, or via D-Bus.
|
||||||
|
|
||||||
* systemd-udevd gained new configuration option timeout_signal= as well
|
|
||||||
as coresponding kernel command line option udev.timeout_signal.
|
|
||||||
The option can be used to configure the UNIX signal that the main
|
|
||||||
daemon sends to the worker processes on timeout.
|
|
||||||
|
|
||||||
* A new sd-path.h API has been added to libsystemd. It provides a
|
* A new sd-path.h API has been added to libsystemd. It provides a
|
||||||
simple API for retrieving various search paths and primary
|
simple API for retrieving various search paths and primary
|
||||||
directories for various resources.
|
directories for various resources.
|
||||||
|
|
|
@ -271,9 +271,6 @@
|
||||||
<term><varname>rd.udev.exec_delay=</varname></term>
|
<term><varname>rd.udev.exec_delay=</varname></term>
|
||||||
<term><varname>udev.event_timeout=</varname></term>
|
<term><varname>udev.event_timeout=</varname></term>
|
||||||
<term><varname>rd.udev.event_timeout=</varname></term>
|
<term><varname>rd.udev.event_timeout=</varname></term>
|
||||||
<term><varname>udev.timeout_signal=</varname></term>
|
|
||||||
<term><varname>rd.udev.timeout_signal=</varname></term>
|
|
||||||
|
|
||||||
<term><varname>net.ifnames=</varname></term>
|
<term><varname>net.ifnames=</varname></term>
|
||||||
<term><varname>net.naming-scheme=</varname></term>
|
<term><varname>net.naming-scheme=</varname></term>
|
||||||
|
|
||||||
|
|
|
@ -105,21 +105,6 @@
|
||||||
</listitem>
|
</listitem>
|
||||||
</varlistentry>
|
</varlistentry>
|
||||||
|
|
||||||
<varlistentry>
|
|
||||||
<term><option>-s</option></term>
|
|
||||||
<term><option>--timeout-signal=</option></term>
|
|
||||||
<listitem>
|
|
||||||
<para>Set the signal which <filename>systemd-udevd</filename> will send to
|
|
||||||
forked off processes after reaching event timeout. The setting can be overriden
|
|
||||||
at boot time with the kernel command line option
|
|
||||||
<varname>udev.timeout_signal=</varname>. Setting to <constant>SIGABRT</constant>
|
|
||||||
may be helpful in order to debug worker timeouts. Defaults to
|
|
||||||
<constant>SIGKILL</constant>. Note that setting the option on the command line
|
|
||||||
overrides the setting from the configuration file.
|
|
||||||
</para>
|
|
||||||
</listitem>
|
|
||||||
</varlistentry>
|
|
||||||
|
|
||||||
<varlistentry>
|
<varlistentry>
|
||||||
<term><option>-N=</option></term>
|
<term><option>-N=</option></term>
|
||||||
<term><option>--resolve-names=</option></term>
|
<term><option>--resolve-names=</option></term>
|
||||||
|
@ -175,15 +160,6 @@
|
||||||
terminated due to kernel drivers taking too long to initialize.</para>
|
terminated due to kernel drivers taking too long to initialize.</para>
|
||||||
</listitem>
|
</listitem>
|
||||||
</varlistentry>
|
</varlistentry>
|
||||||
<varlistentry>
|
|
||||||
<term><varname>udev.timeout_signal=</varname></term>
|
|
||||||
<term><varname>rd.udev.timeout_signal=</varname></term>
|
|
||||||
<listitem>
|
|
||||||
<para>Specifies a signal that <filename>systemd-udevd</filename> will send to
|
|
||||||
workers on timeout. Note that kernel command line option overrides both the
|
|
||||||
setting in the configuration file and the one on the program command line.</para>
|
|
||||||
</listitem>
|
|
||||||
</varlistentry>
|
|
||||||
<varlistentry>
|
<varlistentry>
|
||||||
<term><varname>net.ifnames=</varname></term>
|
<term><varname>net.ifnames=</varname></term>
|
||||||
<listitem>
|
<listitem>
|
||||||
|
|
|
@ -97,16 +97,6 @@
|
||||||
<para>This is the same as the <option>--resolve-names=</option> option.</para>
|
<para>This is the same as the <option>--resolve-names=</option> option.</para>
|
||||||
</listitem>
|
</listitem>
|
||||||
</varlistentry>
|
</varlistentry>
|
||||||
|
|
||||||
<varlistentry>
|
|
||||||
<term><varname>timeout_signal=</varname></term>
|
|
||||||
|
|
||||||
<listitem>
|
|
||||||
<para>Specifies a signal that <filename>systemd-udevd</filename> will send on worker
|
|
||||||
timeouts. Note that both workers and spawned processes will be killed using this
|
|
||||||
signal. Defaults to <option>SIGKILL</option>.</para>
|
|
||||||
</listitem>
|
|
||||||
</varlistentry>
|
|
||||||
</variablelist>
|
</variablelist>
|
||||||
|
|
||||||
<para>
|
<para>
|
||||||
|
|
|
@ -7,7 +7,6 @@
|
||||||
#include "env-file.h"
|
#include "env-file.h"
|
||||||
#include "log.h"
|
#include "log.h"
|
||||||
#include "parse-util.h"
|
#include "parse-util.h"
|
||||||
#include "signal-util.h"
|
|
||||||
#include "string-table.h"
|
#include "string-table.h"
|
||||||
#include "string-util.h"
|
#include "string-util.h"
|
||||||
#include "udev-util.h"
|
#include "udev-util.h"
|
||||||
|
@ -24,10 +23,9 @@ int udev_parse_config_full(
|
||||||
unsigned *ret_children_max,
|
unsigned *ret_children_max,
|
||||||
usec_t *ret_exec_delay_usec,
|
usec_t *ret_exec_delay_usec,
|
||||||
usec_t *ret_event_timeout_usec,
|
usec_t *ret_event_timeout_usec,
|
||||||
ResolveNameTiming *ret_resolve_name_timing,
|
ResolveNameTiming *ret_resolve_name_timing) {
|
||||||
int *ret_timeout_signal) {
|
|
||||||
|
|
||||||
_cleanup_free_ char *log_val = NULL, *children_max = NULL, *exec_delay = NULL, *event_timeout = NULL, *resolve_names = NULL, *timeout_signal = NULL;
|
_cleanup_free_ char *log_val = NULL, *children_max = NULL, *exec_delay = NULL, *event_timeout = NULL, *resolve_names = NULL;
|
||||||
int r;
|
int r;
|
||||||
|
|
||||||
r = parse_env_file(NULL, "/etc/udev/udev.conf",
|
r = parse_env_file(NULL, "/etc/udev/udev.conf",
|
||||||
|
@ -35,8 +33,7 @@ int udev_parse_config_full(
|
||||||
"children_max", &children_max,
|
"children_max", &children_max,
|
||||||
"exec_delay", &exec_delay,
|
"exec_delay", &exec_delay,
|
||||||
"event_timeout", &event_timeout,
|
"event_timeout", &event_timeout,
|
||||||
"resolve_names", &resolve_names,
|
"resolve_names", &resolve_names);
|
||||||
"timeout_signal", &timeout_signal);
|
|
||||||
if (r == -ENOENT)
|
if (r == -ENOENT)
|
||||||
return 0;
|
return 0;
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
|
@ -68,21 +65,21 @@ int udev_parse_config_full(
|
||||||
r = safe_atou(children_max, ret_children_max);
|
r = safe_atou(children_max, ret_children_max);
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
log_syntax(NULL, LOG_WARNING, "/etc/udev/udev.conf", 0, r,
|
log_syntax(NULL, LOG_WARNING, "/etc/udev/udev.conf", 0, r,
|
||||||
"failed to parse children_max=%s, ignoring: %m", children_max);
|
"failed to set parse children_max=%s, ignoring: %m", children_max);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ret_exec_delay_usec && exec_delay) {
|
if (ret_exec_delay_usec && exec_delay) {
|
||||||
r = parse_sec(exec_delay, ret_exec_delay_usec);
|
r = parse_sec(exec_delay, ret_exec_delay_usec);
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
log_syntax(NULL, LOG_WARNING, "/etc/udev/udev.conf", 0, r,
|
log_syntax(NULL, LOG_WARNING, "/etc/udev/udev.conf", 0, r,
|
||||||
"failed to parse exec_delay=%s, ignoring: %m", exec_delay);
|
"failed to set parse exec_delay=%s, ignoring: %m", exec_delay);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ret_event_timeout_usec && event_timeout) {
|
if (ret_event_timeout_usec && event_timeout) {
|
||||||
r = parse_sec(event_timeout, ret_event_timeout_usec);
|
r = parse_sec(event_timeout, ret_event_timeout_usec);
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
log_syntax(NULL, LOG_WARNING, "/etc/udev/udev.conf", 0, r,
|
log_syntax(NULL, LOG_WARNING, "/etc/udev/udev.conf", 0, r,
|
||||||
"failed to parse event_timeout=%s, ignoring: %m", event_timeout);
|
"failed to set parse event_timeout=%s, ignoring: %m", event_timeout);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ret_resolve_name_timing && resolve_names) {
|
if (ret_resolve_name_timing && resolve_names) {
|
||||||
|
@ -91,20 +88,11 @@ int udev_parse_config_full(
|
||||||
t = resolve_name_timing_from_string(resolve_names);
|
t = resolve_name_timing_from_string(resolve_names);
|
||||||
if (t < 0)
|
if (t < 0)
|
||||||
log_syntax(NULL, LOG_WARNING, "/etc/udev/udev.conf", 0, r,
|
log_syntax(NULL, LOG_WARNING, "/etc/udev/udev.conf", 0, r,
|
||||||
"failed to parse resolve_names=%s, ignoring.", resolve_names);
|
"failed to set parse resolve_names=%s, ignoring.", resolve_names);
|
||||||
else
|
else
|
||||||
*ret_resolve_name_timing = t;
|
*ret_resolve_name_timing = t;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (ret_timeout_signal && timeout_signal) {
|
|
||||||
r = signal_from_string(timeout_signal);
|
|
||||||
if (r < 0)
|
|
||||||
log_syntax(NULL, LOG_WARNING, "/etc/udev/udev.conf", 0, r,
|
|
||||||
"failed to parse timeout_signal=%s, ignoring: %m", timeout_signal);
|
|
||||||
else
|
|
||||||
*ret_timeout_signal = r;
|
|
||||||
}
|
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -21,11 +21,10 @@ int udev_parse_config_full(
|
||||||
unsigned *ret_children_max,
|
unsigned *ret_children_max,
|
||||||
usec_t *ret_exec_delay_usec,
|
usec_t *ret_exec_delay_usec,
|
||||||
usec_t *ret_event_timeout_usec,
|
usec_t *ret_event_timeout_usec,
|
||||||
ResolveNameTiming *ret_resolve_name_timing,
|
ResolveNameTiming *ret_resolve_name_timing);
|
||||||
int *ret_timeout_signal);
|
|
||||||
|
|
||||||
static inline int udev_parse_config(void) {
|
static inline int udev_parse_config(void) {
|
||||||
return udev_parse_config_full(NULL, NULL, NULL, NULL, NULL);
|
return udev_parse_config_full(NULL, NULL, NULL, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
int device_wait_for_initialization(sd_device *device, const char *subsystem, usec_t timeout, sd_device **ret);
|
int device_wait_for_initialization(sd_device *device, const char *subsystem, usec_t timeout, sd_device **ret);
|
||||||
|
|
|
@ -122,8 +122,8 @@ static int run(int argc, char *argv[]) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
udev_event_execute_rules(event, 3 * USEC_PER_SEC, SIGKILL, NULL, rules);
|
udev_event_execute_rules(event, 3 * USEC_PER_SEC, NULL, rules);
|
||||||
udev_event_execute_run(event, 3 * USEC_PER_SEC, SIGKILL);
|
udev_event_execute_run(event, 3 * USEC_PER_SEC);
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
|
@ -41,7 +41,6 @@ typedef struct Spawn {
|
||||||
pid_t pid;
|
pid_t pid;
|
||||||
usec_t timeout_warn_usec;
|
usec_t timeout_warn_usec;
|
||||||
usec_t timeout_usec;
|
usec_t timeout_usec;
|
||||||
int timeout_signal;
|
|
||||||
usec_t event_birth_usec;
|
usec_t event_birth_usec;
|
||||||
bool accept_failure;
|
bool accept_failure;
|
||||||
int fd_stdout;
|
int fd_stdout;
|
||||||
|
@ -597,7 +596,7 @@ static int on_spawn_timeout(sd_event_source *s, uint64_t usec, void *userdata) {
|
||||||
|
|
||||||
assert(spawn);
|
assert(spawn);
|
||||||
|
|
||||||
kill_and_sigcont(spawn->pid, spawn->timeout_signal);
|
kill_and_sigcont(spawn->pid, SIGKILL);
|
||||||
|
|
||||||
log_device_error(spawn->device, "Spawned process '%s' ["PID_FMT"] timed out after %s, killing",
|
log_device_error(spawn->device, "Spawned process '%s' ["PID_FMT"] timed out after %s, killing",
|
||||||
spawn->cmd, spawn->pid,
|
spawn->cmd, spawn->pid,
|
||||||
|
@ -718,7 +717,6 @@ static int spawn_wait(Spawn *spawn) {
|
||||||
|
|
||||||
int udev_event_spawn(UdevEvent *event,
|
int udev_event_spawn(UdevEvent *event,
|
||||||
usec_t timeout_usec,
|
usec_t timeout_usec,
|
||||||
int timeout_signal,
|
|
||||||
bool accept_failure,
|
bool accept_failure,
|
||||||
const char *cmd,
|
const char *cmd,
|
||||||
char *result, size_t ressize) {
|
char *result, size_t ressize) {
|
||||||
|
@ -795,7 +793,6 @@ int udev_event_spawn(UdevEvent *event,
|
||||||
.accept_failure = accept_failure,
|
.accept_failure = accept_failure,
|
||||||
.timeout_warn_usec = udev_warn_timeout(timeout_usec),
|
.timeout_warn_usec = udev_warn_timeout(timeout_usec),
|
||||||
.timeout_usec = timeout_usec,
|
.timeout_usec = timeout_usec,
|
||||||
.timeout_signal = timeout_signal,
|
|
||||||
.event_birth_usec = event->birth_usec,
|
.event_birth_usec = event->birth_usec,
|
||||||
.fd_stdout = outpipe[READ_END],
|
.fd_stdout = outpipe[READ_END],
|
||||||
.fd_stderr = errpipe[READ_END],
|
.fd_stderr = errpipe[READ_END],
|
||||||
|
@ -899,7 +896,6 @@ static int update_devnode(UdevEvent *event) {
|
||||||
static void event_execute_rules_on_remove(
|
static void event_execute_rules_on_remove(
|
||||||
UdevEvent *event,
|
UdevEvent *event,
|
||||||
usec_t timeout_usec,
|
usec_t timeout_usec,
|
||||||
int timeout_signal,
|
|
||||||
Hashmap *properties_list,
|
Hashmap *properties_list,
|
||||||
UdevRules *rules) {
|
UdevRules *rules) {
|
||||||
|
|
||||||
|
@ -921,7 +917,7 @@ static void event_execute_rules_on_remove(
|
||||||
if (sd_device_get_devnum(dev, NULL) >= 0)
|
if (sd_device_get_devnum(dev, NULL) >= 0)
|
||||||
(void) udev_watch_end(dev);
|
(void) udev_watch_end(dev);
|
||||||
|
|
||||||
(void) udev_rules_apply_to_event(rules, event, timeout_usec, timeout_signal, properties_list);
|
(void) udev_rules_apply_to_event(rules, event, timeout_usec, properties_list);
|
||||||
|
|
||||||
if (sd_device_get_devnum(dev, NULL) >= 0)
|
if (sd_device_get_devnum(dev, NULL) >= 0)
|
||||||
(void) udev_node_remove(dev);
|
(void) udev_node_remove(dev);
|
||||||
|
@ -948,7 +944,6 @@ static int udev_event_on_move(UdevEvent *event) {
|
||||||
|
|
||||||
int udev_event_execute_rules(UdevEvent *event,
|
int udev_event_execute_rules(UdevEvent *event,
|
||||||
usec_t timeout_usec,
|
usec_t timeout_usec,
|
||||||
int timeout_signal,
|
|
||||||
Hashmap *properties_list,
|
Hashmap *properties_list,
|
||||||
UdevRules *rules) {
|
UdevRules *rules) {
|
||||||
const char *subsystem;
|
const char *subsystem;
|
||||||
|
@ -970,7 +965,7 @@ int udev_event_execute_rules(UdevEvent *event,
|
||||||
return log_device_error_errno(dev, r, "Failed to get ACTION: %m");
|
return log_device_error_errno(dev, r, "Failed to get ACTION: %m");
|
||||||
|
|
||||||
if (action == DEVICE_ACTION_REMOVE) {
|
if (action == DEVICE_ACTION_REMOVE) {
|
||||||
event_execute_rules_on_remove(event, timeout_usec, timeout_signal, properties_list, rules);
|
event_execute_rules_on_remove(event, timeout_usec, properties_list, rules);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -988,7 +983,7 @@ int udev_event_execute_rules(UdevEvent *event,
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
||||||
r = udev_rules_apply_to_event(rules, event, timeout_usec, timeout_signal, properties_list);
|
r = udev_rules_apply_to_event(rules, event, timeout_usec, properties_list);
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return log_device_debug_errno(dev, r, "Failed to apply udev rules: %m");
|
return log_device_debug_errno(dev, r, "Failed to apply udev rules: %m");
|
||||||
|
|
||||||
|
@ -1021,7 +1016,7 @@ int udev_event_execute_rules(UdevEvent *event,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
void udev_event_execute_run(UdevEvent *event, usec_t timeout_usec, int timeout_signal) {
|
void udev_event_execute_run(UdevEvent *event, usec_t timeout_usec) {
|
||||||
const char *command;
|
const char *command;
|
||||||
void *val;
|
void *val;
|
||||||
Iterator i;
|
Iterator i;
|
||||||
|
@ -1045,8 +1040,7 @@ void udev_event_execute_run(UdevEvent *event, usec_t timeout_usec, int timeout_s
|
||||||
}
|
}
|
||||||
|
|
||||||
log_device_debug(event->dev, "Running command \"%s\"", command);
|
log_device_debug(event->dev, "Running command \"%s\"", command);
|
||||||
|
r = udev_event_spawn(event, timeout_usec, false, command, NULL, 0);
|
||||||
r = udev_event_spawn(event, timeout_usec, timeout_signal, false, command, NULL, 0);
|
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
log_device_warning_errno(event->dev, r, "Failed to execute '%s', ignoring: %m", command);
|
log_device_warning_errno(event->dev, r, "Failed to execute '%s', ignoring: %m", command);
|
||||||
else if (r > 0) /* returned value is positive when program fails */
|
else if (r > 0) /* returned value is positive when program fails */
|
||||||
|
|
|
@ -54,15 +54,13 @@ ssize_t udev_event_apply_format(UdevEvent *event,
|
||||||
int udev_check_format(const char *value, size_t *offset, const char **hint);
|
int udev_check_format(const char *value, size_t *offset, const char **hint);
|
||||||
int udev_event_spawn(UdevEvent *event,
|
int udev_event_spawn(UdevEvent *event,
|
||||||
usec_t timeout_usec,
|
usec_t timeout_usec,
|
||||||
int timeout_signal,
|
|
||||||
bool accept_failure,
|
bool accept_failure,
|
||||||
const char *cmd, char *result, size_t ressize);
|
const char *cmd, char *result, size_t ressize);
|
||||||
int udev_event_execute_rules(UdevEvent *event,
|
int udev_event_execute_rules(UdevEvent *event,
|
||||||
usec_t timeout_usec,
|
usec_t timeout_usec,
|
||||||
int timeout_signal,
|
|
||||||
Hashmap *properties_list,
|
Hashmap *properties_list,
|
||||||
UdevRules *rules);
|
UdevRules *rules);
|
||||||
void udev_event_execute_run(UdevEvent *event, usec_t timeout_usec, int timeout_signal);
|
void udev_event_execute_run(UdevEvent *event, usec_t timeout_usec);
|
||||||
|
|
||||||
static inline usec_t udev_warn_timeout(usec_t timeout_usec) {
|
static inline usec_t udev_warn_timeout(usec_t timeout_usec) {
|
||||||
return DIV_ROUND_UP(timeout_usec, 3);
|
return DIV_ROUND_UP(timeout_usec, 3);
|
||||||
|
|
|
@ -1520,7 +1520,6 @@ static int udev_rule_apply_token_to_event(
|
||||||
sd_device *dev,
|
sd_device *dev,
|
||||||
UdevEvent *event,
|
UdevEvent *event,
|
||||||
usec_t timeout_usec,
|
usec_t timeout_usec,
|
||||||
int timeout_signal,
|
|
||||||
Hashmap *properties_list) {
|
Hashmap *properties_list) {
|
||||||
|
|
||||||
UdevRuleToken *token;
|
UdevRuleToken *token;
|
||||||
|
@ -1663,7 +1662,7 @@ static int udev_rule_apply_token_to_event(
|
||||||
(void) udev_event_apply_format(event, token->value, buf, sizeof(buf), false);
|
(void) udev_event_apply_format(event, token->value, buf, sizeof(buf), false);
|
||||||
log_rule_debug(dev, rules, "Running PROGRAM '%s'", buf);
|
log_rule_debug(dev, rules, "Running PROGRAM '%s'", buf);
|
||||||
|
|
||||||
r = udev_event_spawn(event, timeout_usec, timeout_signal, true, buf, result, sizeof(result));
|
r = udev_event_spawn(event, timeout_usec, true, buf, result, sizeof(result));
|
||||||
if (r != 0) {
|
if (r != 0) {
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
log_rule_warning_errno(dev, rules, r, "Failed to execute '%s', ignoring: %m", buf);
|
log_rule_warning_errno(dev, rules, r, "Failed to execute '%s', ignoring: %m", buf);
|
||||||
|
@ -1733,7 +1732,7 @@ static int udev_rule_apply_token_to_event(
|
||||||
(void) udev_event_apply_format(event, token->value, buf, sizeof(buf), false);
|
(void) udev_event_apply_format(event, token->value, buf, sizeof(buf), false);
|
||||||
log_rule_debug(dev, rules, "Importing properties from results of '%s'", buf);
|
log_rule_debug(dev, rules, "Importing properties from results of '%s'", buf);
|
||||||
|
|
||||||
r = udev_event_spawn(event, timeout_usec, timeout_signal, true, buf, result, sizeof result);
|
r = udev_event_spawn(event, timeout_usec, true, buf, result, sizeof result);
|
||||||
if (r != 0) {
|
if (r != 0) {
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
log_rule_warning_errno(dev, rules, r, "Failed to execute '%s', ignoring: %m", buf);
|
log_rule_warning_errno(dev, rules, r, "Failed to execute '%s', ignoring: %m", buf);
|
||||||
|
@ -2163,8 +2162,7 @@ static bool token_is_for_parents(UdevRuleToken *token) {
|
||||||
|
|
||||||
static int udev_rule_apply_parent_token_to_event(
|
static int udev_rule_apply_parent_token_to_event(
|
||||||
UdevRules *rules,
|
UdevRules *rules,
|
||||||
UdevEvent *event,
|
UdevEvent *event) {
|
||||||
int timeout_signal) {
|
|
||||||
|
|
||||||
UdevRuleLine *line;
|
UdevRuleLine *line;
|
||||||
UdevRuleToken *head;
|
UdevRuleToken *head;
|
||||||
|
@ -2177,7 +2175,7 @@ static int udev_rule_apply_parent_token_to_event(
|
||||||
LIST_FOREACH(tokens, line->current_token, head) {
|
LIST_FOREACH(tokens, line->current_token, head) {
|
||||||
if (!token_is_for_parents(line->current_token))
|
if (!token_is_for_parents(line->current_token))
|
||||||
return true; /* All parent tokens match. */
|
return true; /* All parent tokens match. */
|
||||||
r = udev_rule_apply_token_to_event(rules, event->dev_parent, event, 0, timeout_signal, NULL);
|
r = udev_rule_apply_token_to_event(rules, event->dev_parent, event, 0, NULL);
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
if (r == 0)
|
if (r == 0)
|
||||||
|
@ -2198,7 +2196,6 @@ static int udev_rule_apply_line_to_event(
|
||||||
UdevRules *rules,
|
UdevRules *rules,
|
||||||
UdevEvent *event,
|
UdevEvent *event,
|
||||||
usec_t timeout_usec,
|
usec_t timeout_usec,
|
||||||
int timeout_signal,
|
|
||||||
Hashmap *properties_list,
|
Hashmap *properties_list,
|
||||||
UdevRuleLine **next_line) {
|
UdevRuleLine **next_line) {
|
||||||
|
|
||||||
|
@ -2232,7 +2229,7 @@ static int udev_rule_apply_line_to_event(
|
||||||
if (parents_done)
|
if (parents_done)
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
r = udev_rule_apply_parent_token_to_event(rules, event, timeout_signal);
|
r = udev_rule_apply_parent_token_to_event(rules, event);
|
||||||
if (r <= 0)
|
if (r <= 0)
|
||||||
return r;
|
return r;
|
||||||
|
|
||||||
|
@ -2240,7 +2237,7 @@ static int udev_rule_apply_line_to_event(
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
r = udev_rule_apply_token_to_event(rules, event->dev, event, timeout_usec, timeout_signal, properties_list);
|
r = udev_rule_apply_token_to_event(rules, event->dev, event, timeout_usec, properties_list);
|
||||||
if (r <= 0)
|
if (r <= 0)
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
@ -2255,7 +2252,6 @@ int udev_rules_apply_to_event(
|
||||||
UdevRules *rules,
|
UdevRules *rules,
|
||||||
UdevEvent *event,
|
UdevEvent *event,
|
||||||
usec_t timeout_usec,
|
usec_t timeout_usec,
|
||||||
int timeout_signal,
|
|
||||||
Hashmap *properties_list) {
|
Hashmap *properties_list) {
|
||||||
|
|
||||||
UdevRuleFile *file;
|
UdevRuleFile *file;
|
||||||
|
@ -2268,7 +2264,7 @@ int udev_rules_apply_to_event(
|
||||||
LIST_FOREACH(rule_files, file, rules->rule_files) {
|
LIST_FOREACH(rule_files, file, rules->rule_files) {
|
||||||
rules->current_file = file;
|
rules->current_file = file;
|
||||||
LIST_FOREACH_SAFE(rule_lines, file->current_line, next_line, file->rule_lines) {
|
LIST_FOREACH_SAFE(rule_lines, file->current_line, next_line, file->rule_lines) {
|
||||||
r = udev_rule_apply_line_to_event(rules, event, timeout_usec, timeout_signal, properties_list, &next_line);
|
r = udev_rule_apply_line_to_event(rules, event, timeout_usec, properties_list, &next_line);
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
}
|
}
|
||||||
|
|
|
@ -23,6 +23,5 @@ DEFINE_TRIVIAL_CLEANUP_FUNC(UdevRules*, udev_rules_free);
|
||||||
bool udev_rules_check_timestamp(UdevRules *rules);
|
bool udev_rules_check_timestamp(UdevRules *rules);
|
||||||
int udev_rules_apply_to_event(UdevRules *rules, UdevEvent *event,
|
int udev_rules_apply_to_event(UdevRules *rules, UdevEvent *event,
|
||||||
usec_t timeout_usec,
|
usec_t timeout_usec,
|
||||||
int timeout_signal,
|
|
||||||
Hashmap *properties_list);
|
Hashmap *properties_list);
|
||||||
int udev_rules_apply_static_dev_perms(UdevRules *rules);
|
int udev_rules_apply_static_dev_perms(UdevRules *rules);
|
||||||
|
|
|
@ -7,5 +7,4 @@
|
||||||
#children_max=
|
#children_max=
|
||||||
#exec_delay=
|
#exec_delay=
|
||||||
#event_timeout=180
|
#event_timeout=180
|
||||||
#timeout_signal=SIGKILL
|
|
||||||
#resolve_names=early
|
#resolve_names=early
|
||||||
|
|
|
@ -143,7 +143,7 @@ int test_main(int argc, char *argv[], void *userdata) {
|
||||||
assert_se(sigfillset(&mask) >= 0);
|
assert_se(sigfillset(&mask) >= 0);
|
||||||
assert_se(sigprocmask(SIG_SETMASK, &mask, &sigmask_orig) >= 0);
|
assert_se(sigprocmask(SIG_SETMASK, &mask, &sigmask_orig) >= 0);
|
||||||
|
|
||||||
udev_event_execute_rules(event, 60 * USEC_PER_SEC, SIGKILL, NULL, rules);
|
udev_event_execute_rules(event, 60 * USEC_PER_SEC, NULL, rules);
|
||||||
|
|
||||||
FOREACH_DEVICE_PROPERTY(dev, key, value)
|
FOREACH_DEVICE_PROPERTY(dev, key, value)
|
||||||
printf("%s=%s\n", key, value);
|
printf("%s=%s\n", key, value);
|
||||||
|
|
|
@ -75,7 +75,6 @@ static ResolveNameTiming arg_resolve_name_timing = RESOLVE_NAME_EARLY;
|
||||||
static unsigned arg_children_max = 0;
|
static unsigned arg_children_max = 0;
|
||||||
static usec_t arg_exec_delay_usec = 0;
|
static usec_t arg_exec_delay_usec = 0;
|
||||||
static usec_t arg_event_timeout_usec = 180 * USEC_PER_SEC;
|
static usec_t arg_event_timeout_usec = 180 * USEC_PER_SEC;
|
||||||
static int arg_timeout_signal = SIGKILL;
|
|
||||||
|
|
||||||
typedef struct Manager {
|
typedef struct Manager {
|
||||||
sd_event *event;
|
sd_event *event;
|
||||||
|
@ -229,7 +228,7 @@ static int on_event_timeout(sd_event_source *s, uint64_t usec, void *userdata) {
|
||||||
assert(event);
|
assert(event);
|
||||||
assert(event->worker);
|
assert(event->worker);
|
||||||
|
|
||||||
kill_and_sigcont(event->worker->pid, arg_timeout_signal);
|
kill_and_sigcont(event->worker->pid, SIGKILL);
|
||||||
event->worker->state = WORKER_KILLED;
|
event->worker->state = WORKER_KILLED;
|
||||||
|
|
||||||
log_device_error(event->dev, "Worker ["PID_FMT"] processing SEQNUM=%"PRIu64" killed", event->worker->pid, event->seqnum);
|
log_device_error(event->dev, "Worker ["PID_FMT"] processing SEQNUM=%"PRIu64" killed", event->worker->pid, event->seqnum);
|
||||||
|
@ -413,11 +412,11 @@ static int worker_process_device(Manager *manager, sd_device *dev) {
|
||||||
return r;
|
return r;
|
||||||
|
|
||||||
/* apply rules, create node, symlinks */
|
/* apply rules, create node, symlinks */
|
||||||
r = udev_event_execute_rules(udev_event, arg_event_timeout_usec, arg_timeout_signal, manager->properties, manager->rules);
|
r = udev_event_execute_rules(udev_event, arg_event_timeout_usec, manager->properties, manager->rules);
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
return r;
|
return r;
|
||||||
|
|
||||||
udev_event_execute_run(udev_event, arg_event_timeout_usec, arg_timeout_signal);
|
udev_event_execute_run(udev_event, arg_event_timeout_usec);
|
||||||
|
|
||||||
if (!manager->rtnl)
|
if (!manager->rtnl)
|
||||||
/* in case rtnl was initialized */
|
/* in case rtnl was initialized */
|
||||||
|
@ -1456,13 +1455,6 @@ static int parse_proc_cmdline_item(const char *key, const char *value, void *dat
|
||||||
|
|
||||||
r = parse_sec(value, &arg_exec_delay_usec);
|
r = parse_sec(value, &arg_exec_delay_usec);
|
||||||
|
|
||||||
} else if (proc_cmdline_key_streq(key, "udev.timeout_signal")) {
|
|
||||||
if (proc_cmdline_value_missing(key, value))
|
|
||||||
return 0;
|
|
||||||
|
|
||||||
r = signal_from_string(value);
|
|
||||||
if (r > 0)
|
|
||||||
arg_timeout_signal = r;
|
|
||||||
} else if (startswith(key, "udev."))
|
} else if (startswith(key, "udev."))
|
||||||
log_warning("Unknown udev kernel command line option \"%s\", ignoring", key);
|
log_warning("Unknown udev kernel command line option \"%s\", ignoring", key);
|
||||||
|
|
||||||
|
@ -1500,10 +1492,6 @@ static int help(void) {
|
||||||
}
|
}
|
||||||
|
|
||||||
static int parse_argv(int argc, char *argv[]) {
|
static int parse_argv(int argc, char *argv[]) {
|
||||||
enum {
|
|
||||||
ARG_TIMEOUT_SIGNAL,
|
|
||||||
};
|
|
||||||
|
|
||||||
static const struct option options[] = {
|
static const struct option options[] = {
|
||||||
{ "daemon", no_argument, NULL, 'd' },
|
{ "daemon", no_argument, NULL, 'd' },
|
||||||
{ "debug", no_argument, NULL, 'D' },
|
{ "debug", no_argument, NULL, 'D' },
|
||||||
|
@ -1513,7 +1501,6 @@ static int parse_argv(int argc, char *argv[]) {
|
||||||
{ "resolve-names", required_argument, NULL, 'N' },
|
{ "resolve-names", required_argument, NULL, 'N' },
|
||||||
{ "help", no_argument, NULL, 'h' },
|
{ "help", no_argument, NULL, 'h' },
|
||||||
{ "version", no_argument, NULL, 'V' },
|
{ "version", no_argument, NULL, 'V' },
|
||||||
{ "timeout-signal", required_argument, NULL, ARG_TIMEOUT_SIGNAL },
|
|
||||||
{}
|
{}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -1538,14 +1525,6 @@ static int parse_argv(int argc, char *argv[]) {
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
log_warning_errno(r, "Failed to parse --exec-delay= value '%s', ignoring: %m", optarg);
|
log_warning_errno(r, "Failed to parse --exec-delay= value '%s', ignoring: %m", optarg);
|
||||||
break;
|
break;
|
||||||
case ARG_TIMEOUT_SIGNAL:
|
|
||||||
r = signal_from_string(optarg);
|
|
||||||
if (r <= 0)
|
|
||||||
log_warning_errno(r, "Failed to parse --timeout-signal= value '%s', ignoring: %m", optarg);
|
|
||||||
else
|
|
||||||
arg_timeout_signal = r;
|
|
||||||
|
|
||||||
break;
|
|
||||||
case 't':
|
case 't':
|
||||||
r = parse_sec(optarg, &arg_event_timeout_usec);
|
r = parse_sec(optarg, &arg_event_timeout_usec);
|
||||||
if (r < 0)
|
if (r < 0)
|
||||||
|
@ -1743,7 +1722,7 @@ int run_udevd(int argc, char *argv[]) {
|
||||||
|
|
||||||
log_set_target(LOG_TARGET_AUTO);
|
log_set_target(LOG_TARGET_AUTO);
|
||||||
log_open();
|
log_open();
|
||||||
udev_parse_config_full(&arg_children_max, &arg_exec_delay_usec, &arg_event_timeout_usec, &arg_resolve_name_timing, &arg_timeout_signal);
|
udev_parse_config_full(&arg_children_max, &arg_exec_delay_usec, &arg_event_timeout_usec, &arg_resolve_name_timing);
|
||||||
log_parse_environment();
|
log_parse_environment();
|
||||||
log_open(); /* Done again to update after reading configuration. */
|
log_open(); /* Done again to update after reading configuration. */
|
||||||
|
|
||||||
|
|
|
@ -1 +0,0 @@
|
||||||
../TEST-01-BASIC/Makefile
|
|
|
@ -1,8 +0,0 @@
|
||||||
#!/usr/bin/env bash
|
|
||||||
set -e
|
|
||||||
|
|
||||||
TEST_DESCRIPTION="test udev's event-timeout and timeout-signal options"
|
|
||||||
TEST_NO_NSPAWN=1
|
|
||||||
. $TEST_BASE_DIR/test-functions
|
|
||||||
|
|
||||||
do_test "$@" 49
|
|
|
@ -1,6 +0,0 @@
|
||||||
[Unit]
|
|
||||||
Description=TEST-49-UDEV-EVENT-TIMEOUT
|
|
||||||
|
|
||||||
[Service]
|
|
||||||
ExecStart=/usr/lib/systemd/tests/testdata/units/%N.sh
|
|
||||||
Type=oneshot
|
|
|
@ -1,47 +0,0 @@
|
||||||
#!/usr/bin/env bash
|
|
||||||
|
|
||||||
set -ex
|
|
||||||
|
|
||||||
test_rule="/run/udev/rules.d/49-test.rules"
|
|
||||||
|
|
||||||
setup() {
|
|
||||||
mkdir -p "${test_rule%/*}"
|
|
||||||
cp -f /etc/udev/udev.conf /etc/udev/udev.conf.bckp
|
|
||||||
echo 'KERNEL=="lo", SUBSYSTEM=="net", PROGRAM=="/bin/sleep 60"' > "${test_rule}"
|
|
||||||
echo "event_timeout=30" >> /etc/udev/udev.conf
|
|
||||||
echo "timeout_signal=SIGABRT" >> /etc/udev/udev.conf
|
|
||||||
|
|
||||||
systemctl restart systemd-udevd.service
|
|
||||||
}
|
|
||||||
|
|
||||||
teardown() {
|
|
||||||
set +e
|
|
||||||
|
|
||||||
mv -f /etc/udev/udev.conf.bckp /etc/udev/udev.conf
|
|
||||||
rm -f "$test_rule"
|
|
||||||
systemctl restart systemd-udevd.service
|
|
||||||
}
|
|
||||||
|
|
||||||
run_test() {
|
|
||||||
since="$(date +%T)"
|
|
||||||
|
|
||||||
echo add > /sys/class/net/lo/uevent
|
|
||||||
|
|
||||||
for n in {1..20}; do
|
|
||||||
sleep 5
|
|
||||||
if coredumpctl --since "$since" --no-legend --no-pager | grep /bin/udevadm ; then
|
|
||||||
return 0
|
|
||||||
fi
|
|
||||||
done
|
|
||||||
|
|
||||||
return 1
|
|
||||||
}
|
|
||||||
|
|
||||||
trap teardown EXIT
|
|
||||||
|
|
||||||
setup
|
|
||||||
run_test
|
|
||||||
|
|
||||||
echo OK > /testok
|
|
||||||
|
|
||||||
exit 0
|
|
Loading…
Reference in New Issue