mirror of
https://github.com/systemd/systemd.git
synced 2025-01-25 10:04:04 +03:00
core: fall back to bind-mounts for PrivateDevices= execution environments
In environments where CAP_MKNOD is not available or inside user namespaces it is still desirable to enable services to use PrivateDevices= . So fall back to using bind-mounts on EPERM.
This commit is contained in:
parent
1238250f1d
commit
1649861744
@ -578,7 +578,7 @@ static void drop_outside_root(const char *root_directory, MountEntry *m, unsigne
|
||||
*n = t - m;
|
||||
}
|
||||
|
||||
static int clone_device_node(const char *d, const char *temporary_mount) {
|
||||
static int clone_device_node(const char *d, const char *temporary_mount, bool *make_devnode) {
|
||||
const char *dn;
|
||||
struct stat st;
|
||||
int r;
|
||||
@ -598,12 +598,36 @@ static int clone_device_node(const char *d, const char *temporary_mount) {
|
||||
|
||||
dn = strjoina(temporary_mount, d);
|
||||
|
||||
if (*make_devnode) {
|
||||
mac_selinux_create_file_prepare(d, st.st_mode);
|
||||
r = mknod(dn, st.st_mode, st.st_rdev);
|
||||
mac_selinux_create_file_clear();
|
||||
if (r < 0)
|
||||
|
||||
if (r == 0)
|
||||
return 1;
|
||||
if (errno != EPERM)
|
||||
return log_debug_errno(errno, "mknod failed for %s: %m", d);
|
||||
|
||||
*make_devnode = false;
|
||||
}
|
||||
|
||||
/* We're about to fallback to bind-mounting the device
|
||||
* node. So create a dummy bind-mount target. */
|
||||
mac_selinux_create_file_prepare(d, 0);
|
||||
r = mknod(dn, S_IFREG, 0);
|
||||
mac_selinux_create_file_clear();
|
||||
|
||||
if (r < 0 && errno != EEXIST)
|
||||
return log_debug_errno(errno, "mknod fallback failed for %s: %m", d);
|
||||
|
||||
/* Fallback to bind-mounting:
|
||||
* The assumption here is that all used device nodes carry standard
|
||||
* properties. Specifically, the devices nodes we bind-mount should
|
||||
* either be owned by root:root or root:tty (e.g. /dev/tty, /dev/ptmx)
|
||||
* and should not carry ACLs. */
|
||||
if (mount(d, dn, NULL, MS_BIND, NULL) < 0)
|
||||
return log_debug_errno(errno, "mount failed for %s: %m", d);
|
||||
|
||||
return 1;
|
||||
}
|
||||
|
||||
@ -618,6 +642,7 @@ static int mount_private_dev(MountEntry *m) {
|
||||
|
||||
char temporary_mount[] = "/tmp/namespace-dev-XXXXXX";
|
||||
const char *d, *dev = NULL, *devpts = NULL, *devshm = NULL, *devhugepages = NULL, *devmqueue = NULL, *devlog = NULL, *devptmx = NULL;
|
||||
bool can_mknod = true;
|
||||
_cleanup_umask_ mode_t u;
|
||||
int r;
|
||||
|
||||
@ -658,7 +683,7 @@ static int mount_private_dev(MountEntry *m) {
|
||||
goto fail;
|
||||
}
|
||||
} else {
|
||||
r = clone_device_node("/dev/ptmx", temporary_mount);
|
||||
r = clone_device_node("/dev/ptmx", temporary_mount, &can_mknod);
|
||||
if (r < 0)
|
||||
goto fail;
|
||||
if (r == 0) {
|
||||
@ -687,7 +712,7 @@ static int mount_private_dev(MountEntry *m) {
|
||||
(void) symlink("/run/systemd/journal/dev-log", devlog);
|
||||
|
||||
NULSTR_FOREACH(d, devnodes) {
|
||||
r = clone_device_node(d, temporary_mount);
|
||||
r = clone_device_node(d, temporary_mount, &can_mknod);
|
||||
if (r < 0)
|
||||
goto fail;
|
||||
}
|
||||
|
Loading…
x
Reference in New Issue
Block a user