Compare commits

...

7 Commits

Author SHA1 Message Date
Zbigniew Jędrzejewski-Szmek 8f335169e2
Merge pull request #17190 from poettering/udev-opath
udev: manipulate device nodes with O_PATH
2020-09-29 13:53:41 +02:00
Anita Zhang 21892cc46b
Merge pull request #17186 from poettering/tmpfiles-cleanup-man-fix
man: fix reference to unit file
2020-09-28 23:48:28 -07:00
Lennart Poettering a7fdc6cbd3 udev: apply access mode/ownership to device nodes with O_PATH
Let's open the device node to modify with O_PATH, and then adjust it
only after verifying everything is in order. This fixes a race where the
a device appears, disappears and quickly reappers, while we are still
running the rules for the first appearance: when going by path we'd
possibly adjust half of the old and half of the new node. By O_PATH we
can pin the node while we operate on it, thus removing the race.

Previously, we'd do a superficial racey check if the device node changed
undearneath us, and would propagate EEXIST in that case, failing the
rule set. With this change we'll instead gracefully handle this, exactly
like in the pre-existing case when the device node disappeared in the
meantime.
2020-09-28 18:45:54 +02:00
Lennart Poettering f25bff5eaf fs-util: add new futimens_opath() helper
futimens() that works for O_PATH fds.
2020-09-28 18:45:54 +02:00
Lennart Poettering 9271daeed7 selinux: add apis to set labels/fix labels per fd instead of path 2020-09-28 18:45:54 +02:00
Lennart Poettering b9daaedbb9 udev-util: simplify device_is_renaming() error handling 2020-09-28 18:45:54 +02:00
Lennart Poettering 6457e88902 man: fix reference to unit file
It' "systemd-tmpfiles-clean" not "systemd-tmpfiles-cleanup"

Fixes: #17171
2020-09-28 16:31:07 +02:00
7 changed files with 109 additions and 34 deletions

View File

@ -92,7 +92,7 @@ A+ /path-or-glob/to/append/acls/recursively - - - - POSIX
directories during boot and to do periodic cleanup afterwards. See directories during boot and to do periodic cleanup afterwards. See
<citerefentry><refentrytitle>systemd-tmpfiles</refentrytitle><manvolnum>5</manvolnum></citerefentry> for <citerefentry><refentrytitle>systemd-tmpfiles</refentrytitle><manvolnum>5</manvolnum></citerefentry> for
the description of <filename>systemd-tmpfiles-setup.service</filename>, the description of <filename>systemd-tmpfiles-setup.service</filename>,
<filename>systemd-tmpfiles-cleanup.service</filename>, and associated units.</para> <filename>systemd-tmpfiles-clean.service</filename>, and associated units.</para>
<para>System daemons frequently require private runtime directories below <filename>/run</filename> to <para>System daemons frequently require private runtime directories below <filename>/run</filename> to
store communication sockets and similar. For these, it is better to use store communication sockets and similar. For these, it is better to use

View File

@ -312,6 +312,25 @@ int fchmod_opath(int fd, mode_t m) {
return 0; return 0;
} }
int futimens_opath(int fd, const struct timespec ts[2]) {
char procfs_path[STRLEN("/proc/self/fd/") + DECIMAL_STR_MAX(int)];
/* Similar to fchmod_path() but for futimens() */
xsprintf(procfs_path, "/proc/self/fd/%i", fd);
if (utimensat(AT_FDCWD, procfs_path, ts, 0) < 0) {
if (errno != ENOENT)
return -errno;
if (proc_mounted() == 0)
return -ENOSYS; /* if we have no /proc/, the concept is not implementable */
return -ENOENT;
}
return 0;
}
int stat_warn_permissions(const char *path, const struct stat *st) { int stat_warn_permissions(const char *path, const struct stat *st) {
assert(path); assert(path);
assert(st); assert(st);

View File

@ -38,6 +38,8 @@ int fchmod_and_chown(int fd, mode_t mode, uid_t uid, gid_t gid);
int fchmod_umask(int fd, mode_t mode); int fchmod_umask(int fd, mode_t mode);
int fchmod_opath(int fd, mode_t m); int fchmod_opath(int fd, mode_t m);
int futimens_opath(int fd, const struct timespec ts[2]);
int fd_warn_permissions(const char *path, int fd); int fd_warn_permissions(const char *path, int fd);
int stat_warn_permissions(const char *path, const struct stat *st); int stat_warn_permissions(const char *path, const struct stat *st);

View File

@ -205,14 +205,11 @@ static int mac_selinux_reload(int seqno) {
int mac_selinux_fix_container(const char *path, const char *inside_path, LabelFixFlags flags) { int mac_selinux_fix_container(const char *path, const char *inside_path, LabelFixFlags flags) {
#if HAVE_SELINUX
char procfs_path[STRLEN("/proc/self/fd/") + DECIMAL_STR_MAX(int)];
_cleanup_freecon_ char* fcon = NULL;
_cleanup_close_ int fd = -1;
struct stat st;
int r;
assert(path); assert(path);
assert(inside_path);
#if HAVE_SELINUX
_cleanup_close_ int fd = -1;
/* if mac_selinux_init() wasn't called before we are a NOOP */ /* if mac_selinux_init() wasn't called before we are a NOOP */
if (!label_hnd) if (!label_hnd)
@ -227,6 +224,27 @@ int mac_selinux_fix_container(const char *path, const char *inside_path, LabelFi
return -errno; return -errno;
} }
return mac_selinux_fix_container_fd(fd, path, inside_path, flags);
#endif
return 0;
}
int mac_selinux_fix_container_fd(int fd, const char *path, const char *inside_path, LabelFixFlags flags) {
assert(fd >= 0);
assert(inside_path);
#if HAVE_SELINUX
char procfs_path[STRLEN("/proc/self/fd/") + DECIMAL_STR_MAX(int)];
_cleanup_freecon_ char* fcon = NULL;
struct stat st;
int r;
/* if mac_selinux_init() wasn't called before we are a NOOP */
if (!label_hnd)
return 0;
if (fstat(fd, &st) < 0) if (fstat(fd, &st) < 0)
return -errno; return -errno;
@ -234,12 +252,11 @@ int mac_selinux_fix_container(const char *path, const char *inside_path, LabelFi
mac_selinux_maybe_reload(); mac_selinux_maybe_reload();
if (selabel_lookup_raw(label_hnd, &fcon, inside_path, st.st_mode) < 0) { if (selabel_lookup_raw(label_hnd, &fcon, inside_path, st.st_mode) < 0) {
r = -errno;
/* If there's no label to set, then exit without warning */ /* If there's no label to set, then exit without warning */
if (r == -ENOENT) if (errno == ENOENT)
return 0; return 0;
r = -errno;
goto fail; goto fail;
} }
@ -247,16 +264,16 @@ int mac_selinux_fix_container(const char *path, const char *inside_path, LabelFi
if (setfilecon_raw(procfs_path, fcon) < 0) { if (setfilecon_raw(procfs_path, fcon) < 0) {
_cleanup_freecon_ char *oldcon = NULL; _cleanup_freecon_ char *oldcon = NULL;
r = -errno;
/* If the FS doesn't support labels, then exit without warning */ /* If the FS doesn't support labels, then exit without warning */
if (r == -EOPNOTSUPP) if (ERRNO_IS_NOT_SUPPORTED(errno))
return 0; return 0;
/* It the FS is read-only and we were told to ignore failures caused by that, suppress error */ /* It the FS is read-only and we were told to ignore failures caused by that, suppress error */
if (r == -EROFS && (flags & LABEL_IGNORE_EROFS)) if (errno == EROFS && (flags & LABEL_IGNORE_EROFS))
return 0; return 0;
r = -errno;
/* If the old label is identical to the new one, suppress any kind of error */ /* If the old label is identical to the new one, suppress any kind of error */
if (getfilecon_raw(procfs_path, &oldcon) >= 0 && streq(fcon, oldcon)) if (getfilecon_raw(procfs_path, &oldcon) >= 0 && streq(fcon, oldcon))
return 0; return 0;
@ -267,7 +284,7 @@ int mac_selinux_fix_container(const char *path, const char *inside_path, LabelFi
return 0; return 0;
fail: fail:
return log_enforcing_errno(r, "Unable to fix SELinux security context of %s (%s): %m", path, inside_path); return log_enforcing_errno(r, "Unable to fix SELinux security context of %s (%s): %m", strna(path), strna(inside_path));
#endif #endif
return 0; return 0;
@ -275,11 +292,12 @@ fail:
int mac_selinux_apply(const char *path, const char *label) { int mac_selinux_apply(const char *path, const char *label) {
assert(path);
#if HAVE_SELINUX #if HAVE_SELINUX
if (!mac_selinux_use()) if (!mac_selinux_use())
return 0; return 0;
assert(path);
assert(label); assert(label);
if (setfilecon(path, label) < 0) if (setfilecon(path, label) < 0)
@ -288,6 +306,22 @@ int mac_selinux_apply(const char *path, const char *label) {
return 0; return 0;
} }
int mac_selinux_apply_fd(int fd, const char *path, const char *label) {
assert(fd >= 0);
#if HAVE_SELINUX
if (!mac_selinux_use())
return 0;
assert(label);
if (fsetfilecon(fd, label) < 0)
return log_enforcing_errno(errno, "Failed to set SELinux security context %s on path %s: %m", label, strna(path));
#endif
return 0;
}
int mac_selinux_get_create_label_from_exe(const char *exe, char **label) { int mac_selinux_get_create_label_from_exe(const char *exe, char **label) {
#if HAVE_SELINUX #if HAVE_SELINUX
_cleanup_freecon_ char *mycon = NULL, *fcon = NULL; _cleanup_freecon_ char *mycon = NULL, *fcon = NULL;

View File

@ -28,7 +28,13 @@ static inline int mac_selinux_fix(const char *path, LabelFixFlags flags) {
return mac_selinux_fix_container(path, path, flags); return mac_selinux_fix_container(path, path, flags);
} }
int mac_selinux_fix_container_fd(int fd, const char *path, const char *inside_path, LabelFixFlags flags);
static inline int mac_selinux_fix_fd(int fd, const char *path, LabelFixFlags flags) {
return mac_selinux_fix_container_fd(fd, path, path, flags);
}
int mac_selinux_apply(const char *path, const char *label); int mac_selinux_apply(const char *path, const char *label);
int mac_selinux_apply_fd(int fd, const char *path, const char *label);
int mac_selinux_get_create_label_from_exe(const char *exe, char **label); int mac_selinux_get_create_label_from_exe(const char *exe, char **label);
int mac_selinux_get_our_label(char **label); int mac_selinux_get_our_label(char **label);

View File

@ -288,10 +288,12 @@ int device_is_renaming(sd_device *dev) {
assert(dev); assert(dev);
r = sd_device_get_property_value(dev, "ID_RENAMING", NULL); r = sd_device_get_property_value(dev, "ID_RENAMING", NULL);
if (r < 0 && r != -ENOENT) if (r == -ENOENT)
return false;
if (r < 0)
return r; return r;
return r >= 0; return true;
} }
bool device_for_action(sd_device *dev, DeviceAction action) { bool device_for_action(sd_device *dev, DeviceAction action) {

View File

@ -273,9 +273,10 @@ static int node_permissions_apply(sd_device *dev, bool apply_mac,
mode_t mode, uid_t uid, gid_t gid, mode_t mode, uid_t uid, gid_t gid,
OrderedHashmap *seclabel_list) { OrderedHashmap *seclabel_list) {
const char *devnode, *subsystem, *id_filename = NULL; const char *devnode, *subsystem, *id_filename = NULL;
bool apply_mode, apply_uid, apply_gid;
_cleanup_close_ int node_fd = -1;
struct stat stats; struct stat stats;
dev_t devnum; dev_t devnum;
bool apply_mode, apply_uid, apply_gid;
int r; int r;
assert(dev); assert(dev);
@ -296,16 +297,25 @@ static int node_permissions_apply(sd_device *dev, bool apply_mac,
else else
mode |= S_IFCHR; mode |= S_IFCHR;
if (lstat(devnode, &stats) < 0) { node_fd = open(devnode, O_PATH|O_NOFOLLOW|O_CLOEXEC);
if (errno == ENOENT) if (node_fd < 0) {
return 0; /* this is necessarily racey, so ignore missing the device */ if (errno == ENOENT) {
return log_device_debug_errno(dev, errno, "cannot stat() node %s: %m", devnode); log_device_debug_errno(dev, errno, "Device node %s is missing, skipping handling.", devnode);
return 0; /* This is necessarily racey, so ignore missing the device */
} }
if ((mode != MODE_INVALID && (stats.st_mode & S_IFMT) != (mode & S_IFMT)) || stats.st_rdev != devnum) return log_device_debug_errno(dev, errno, "Cannot open node %s: %m", devnode);
return log_device_debug_errno(dev, SYNTHETIC_ERRNO(EEXIST), }
"Found node '%s' with non-matching devnum %s, skip handling",
if (fstat(node_fd, &stats) < 0)
return log_device_debug_errno(dev, errno, "cannot stat() node %s: %m", devnode);
if ((mode != MODE_INVALID && (stats.st_mode & S_IFMT) != (mode & S_IFMT)) || stats.st_rdev != devnum) {
log_device_debug(dev, "Found node '%s' with non-matching devnum %s, skipping handling.",
devnode, id_filename); devnode, id_filename);
return 0; /* We might process a device that already got replaced by the time we have a look
* at it, handle this gracefully and step away. */
}
apply_mode = mode != MODE_INVALID && (stats.st_mode & 0777) != (mode & 0777); apply_mode = mode != MODE_INVALID && (stats.st_mode & 0777) != (mode & 0777);
apply_uid = uid_is_valid(uid) && stats.st_uid != uid; apply_uid = uid_is_valid(uid) && stats.st_uid != uid;
@ -322,7 +332,7 @@ static int node_permissions_apply(sd_device *dev, bool apply_mac,
gid_is_valid(gid) ? gid : stats.st_gid, gid_is_valid(gid) ? gid : stats.st_gid,
mode != MODE_INVALID ? mode & 0777 : stats.st_mode & 0777); mode != MODE_INVALID ? mode & 0777 : stats.st_mode & 0777);
r = chmod_and_chown(devnode, mode, uid, gid); r = fchmod_and_chown(node_fd, mode, uid, gid);
if (r < 0) if (r < 0)
log_device_full_errno(dev, r == -ENOENT ? LOG_DEBUG : LOG_ERR, r, log_device_full_errno(dev, r == -ENOENT ? LOG_DEBUG : LOG_ERR, r,
"Failed to set owner/mode of %s to uid=" UID_FMT "Failed to set owner/mode of %s to uid=" UID_FMT
@ -345,7 +355,7 @@ static int node_permissions_apply(sd_device *dev, bool apply_mac,
if (streq(name, "selinux")) { if (streq(name, "selinux")) {
selinux = true; selinux = true;
q = mac_selinux_apply(devnode, label); q = mac_selinux_apply_fd(node_fd, devnode, label);
if (q < 0) if (q < 0)
log_device_full_errno(dev, q == -ENOENT ? LOG_DEBUG : LOG_ERR, q, log_device_full_errno(dev, q == -ENOENT ? LOG_DEBUG : LOG_ERR, q,
"SECLABEL: failed to set SELinux label '%s': %m", label); "SECLABEL: failed to set SELinux label '%s': %m", label);
@ -355,7 +365,7 @@ static int node_permissions_apply(sd_device *dev, bool apply_mac,
} else if (streq(name, "smack")) { } else if (streq(name, "smack")) {
smack = true; smack = true;
q = mac_smack_apply(devnode, SMACK_ATTR_ACCESS, label); q = mac_smack_apply_fd(node_fd, SMACK_ATTR_ACCESS, label);
if (q < 0) if (q < 0)
log_device_full_errno(dev, q == -ENOENT ? LOG_DEBUG : LOG_ERR, q, log_device_full_errno(dev, q == -ENOENT ? LOG_DEBUG : LOG_ERR, q,
"SECLABEL: failed to set SMACK label '%s': %m", label); "SECLABEL: failed to set SMACK label '%s': %m", label);
@ -368,13 +378,15 @@ static int node_permissions_apply(sd_device *dev, bool apply_mac,
/* set the defaults */ /* set the defaults */
if (!selinux) if (!selinux)
(void) mac_selinux_fix(devnode, LABEL_IGNORE_ENOENT); (void) mac_selinux_fix_fd(node_fd, devnode, LABEL_IGNORE_ENOENT);
if (!smack) if (!smack)
(void) mac_smack_apply(devnode, SMACK_ATTR_ACCESS, NULL); (void) mac_smack_apply_fd(node_fd, SMACK_ATTR_ACCESS, NULL);
} }
/* always update timestamp when we re-use the node, like on media change events */ /* always update timestamp when we re-use the node, like on media change events */
(void) utimensat(AT_FDCWD, devnode, NULL, 0); r = futimens_opath(node_fd, NULL);
if (r < 0)
log_device_debug_errno(dev, r, "Failed to adjust timestamp of node %s: %m", devnode);
return r; return r;
} }