From 4ef31082884e1c8df8887a8f53c428a894d53fb7 Mon Sep 17 00:00:00 2001 From: Lennart Poettering Date: Fri, 29 Jul 2011 01:48:18 +0200 Subject: [PATCH] selinux: mount /sys, /proc, /dev before we load the SELinux policy --- src/label.c | 8 ++++---- src/mount-setup.c | 40 ++++++++++++++++++++++++++++++++++------ src/mount-setup.h | 2 ++ src/selinux-setup.c | 6 +++++- 4 files changed, 45 insertions(+), 11 deletions(-) diff --git a/src/label.c b/src/label.c index b34ee2d3dc..a5994622c7 100644 --- a/src/label.c +++ b/src/label.c @@ -34,12 +34,12 @@ static struct selabel_handle *label_hnd = NULL; static inline bool use_selinux(void) { - static int use_selinux_ind = -1; + static int use_selinux_cached = -1; - if (use_selinux_ind < 0) - use_selinux_ind = is_selinux_enabled() > 0; + if (use_selinux_cached < 0) + use_selinux_cached = is_selinux_enabled() > 0; - return use_selinux_ind; + return use_selinux_cached; } #endif diff --git a/src/mount-setup.c b/src/mount-setup.c index 290698939e..ec4184de58 100644 --- a/src/mount-setup.c +++ b/src/mount-setup.c @@ -48,6 +48,10 @@ typedef struct MountPoint { bool fatal; } MountPoint; +/* The first three entries we might need before SELinux is up. The + * other ones we can delay until SELinux is loaded. */ +#define N_EARLY_MOUNT 3 + static const MountPoint mount_table[] = { { "proc", "/proc", "proc", NULL, MS_NOSUID|MS_NOEXEC|MS_NODEV, true }, { "sysfs", "/sys", "sysfs", NULL, MS_NOSUID|MS_NOEXEC|MS_NODEV, true }, @@ -91,13 +95,14 @@ bool mount_point_ignore(const char *path) { return false; } -static int mount_one(const MountPoint *p) { +static int mount_one(const MountPoint *p, bool relabel) { int r; assert(p); /* Relabel first, just in case */ - label_fix(p->where, true); + if (relabel) + label_fix(p->where, true); if ((r = path_is_mount_point(p->where)) < 0) return r; @@ -125,11 +130,31 @@ static int mount_one(const MountPoint *p) { } /* Relabel again, since we now mounted something fresh here */ - label_fix(p->where, false); + if (relabel) + label_fix(p->where, false); return 0; } +int mount_setup_early(void) { + unsigned i; + int r = 0; + + assert_cc(N_EARLY_MOUNT <= ELEMENTSOF(mount_table)); + + /* Do a minimal mount of /proc and friends to enable the most + * basic stuff, such as SELinux */ + for (i = 0; i < N_EARLY_MOUNT; i ++) { + int j; + + j = mount_one(mount_table + i, false); + if (r == 0) + r = j; + } + + return r; +} + static int mount_cgroup_controllers(void) { int r; FILE *f; @@ -179,7 +204,7 @@ static int mount_cgroup_controllers(void) { p.flags = MS_NOSUID|MS_NOEXEC|MS_NODEV; p.fatal = false; - r = mount_one(&p); + r = mount_one(&p, true); free(controller); free(where); @@ -239,9 +264,12 @@ int mount_setup(bool loaded_policy) { unsigned i; const char *j, *k; - for (i = 0; i < ELEMENTSOF(mount_table); i ++) - if ((r = mount_one(mount_table+i)) < 0) + for (i = 0; i < ELEMENTSOF(mount_table); i ++) { + r = mount_one(mount_table + i, true); + + if (r < 0) return r; + } /* Nodes in devtmpfs and /run need to be manually updated for * the appropriate labels, after mounting. The other virtual diff --git a/src/mount-setup.h b/src/mount-setup.h index f05ee31740..9311cacc44 100644 --- a/src/mount-setup.h +++ b/src/mount-setup.h @@ -24,6 +24,8 @@ #include +int mount_setup_early(void); + int mount_setup(bool loaded_policy); bool mount_point_is_api(const char *path); diff --git a/src/selinux-setup.c b/src/selinux-setup.c index fdc316048a..dc101b13ba 100644 --- a/src/selinux-setup.c +++ b/src/selinux-setup.c @@ -30,6 +30,7 @@ #endif #include "selinux-setup.h" +#include "mount-setup.h" #include "macro.h" #include "util.h" #include "log.h" @@ -45,6 +46,9 @@ int selinux_setup(bool *loaded_policy) { assert(loaded_policy); + /* Make sure getcon() works, which needs /proc and /sys */ + mount_setup_early(); + /* Already initialized by somebody else? */ r = getcon_raw(&con); if (r == 0) { @@ -71,7 +75,7 @@ int selinux_setup(bool *loaded_policy) { /* Transition to the new context */ r = label_get_create_label_from_exe(SYSTEMD_BINARY_PATH, &label); - if (r < 0) { + if (r < 0 || label == NULL) { log_open(); log_error("Failed to compute init label, ignoring."); } else {