libselinux mountpoint changing patch.

The Fedora Distribution is looking to standardize kernel subsystem file
systems to be mounted under /sys/fs. They would like us to move /selinux
to /sys/fs/selinux.  This patch changes libselinux in the following
ways:

1.  load_policy will first check if /sys/fs/selinux exists and mount the
selinuxfs at this location, if it does not exists it will fall back to
mounting the file system at /selinux (if it exists).

2.  The init functions of selinux will now check if /sys/fs/selinux is
mounted, if it is and has an SELinuxfs mounted on it, the code will then
check if the selinuxfs is mounted rw, if it is, libselinux will set the
mountpoint, if it is readonly, libselinux will return no mountpoint.  If
/sys/fs/selinux does not exists, the same check will be done for
/selinux and finally for an entry in /proc/mounts.

NOTE:  We added the check for RO, to allow tools like mock to be able to
tell a chroot that SELinux is disabled while enforcing it outside the
chroot.

$ getenforce
Enabled
$ mount --bind /selinux /var/chroot/selinux
$ mount -o remount,ro /var/chroot/selinux
$ chroot /var/chroot
$ getenforce
Disabled

3. In order to make this work, I needed to stop enabled from checking if
/proc/filesystem for entries if selinux_mnt did not exist.  Now enabeled
checks if selinux_mnt has been discovered otherwise it will report
selinux disabled.

Signed-off-by: Eric Paris <eparis@redhat.com>
Acked-by: Dan Walsh <dwalsh@redhat.com>
This commit is contained in:
Daniel J Walsh 2011-05-03 14:06:12 -04:00 committed by Eric Paris
parent 2f921b5832
commit e3cab998b4
4 changed files with 56 additions and 53 deletions

View File

@ -11,10 +11,6 @@
int is_selinux_enabled(void) int is_selinux_enabled(void)
{ {
char *buf=NULL;
FILE *fp;
ssize_t num;
size_t len;
int enabled = 0; int enabled = 0;
security_context_t con; security_context_t con;
@ -32,37 +28,8 @@ int is_selinux_enabled(void)
enabled = 0; enabled = 0;
freecon(con); freecon(con);
} }
return enabled;
} }
/* Drop back to detecting it the long way. */
fp = fopen("/proc/filesystems", "r");
if (!fp)
return -1;
__fsetlocking(fp, FSETLOCKING_BYCALLER);
while ((num = getline(&buf, &len, fp)) != -1) {
if (strstr(buf, "selinuxfs")) {
enabled = 1;
break;
}
}
if (num < 0)
goto out;
/* Since an selinux file system is available, we consider
* selinux enabled. If getcon_raw fails, selinux is still
* enabled. We only consider it disabled if no policy is loaded. */
if (getcon_raw(&con) == 0) {
if (!strcmp(con, "kernel"))
enabled = 0;
freecon(con);
}
out:
free(buf);
fclose(fp);
return enabled; return enabled;
} }

View File

@ -7,6 +7,7 @@
#include <stdio.h> #include <stdio.h>
#include <stdio_ext.h> #include <stdio_ext.h>
#include <dlfcn.h> #include <dlfcn.h>
#include <sys/statvfs.h>
#include <sys/vfs.h> #include <sys/vfs.h>
#include <stdint.h> #include <stdint.h>
#include <limits.h> #include <limits.h>
@ -20,12 +21,41 @@ char *selinux_mnt = NULL;
int selinux_page_size = 0; int selinux_page_size = 0;
int obj_class_compat = 1; int obj_class_compat = 1;
/* Verify the mount point for selinux file system has a selinuxfs.
If the file system:
* Exist,
* Is mounted with an selinux file system,
* The file system is read/write
* then set this as the default file system.
*/
static int verify_selinuxmnt(char *mnt)
{
struct statfs sfbuf;
int rc;
do {
rc = statfs(mnt, &sfbuf);
} while (rc < 0 && errno == EINTR);
if (rc == 0) {
if ((uint32_t)sfbuf.f_type == (uint32_t)SELINUX_MAGIC) {
struct statvfs vfsbuf;
rc = statvfs(mnt, &vfsbuf);
if (rc == 0) {
if (!(vfsbuf.f_flag & ST_RDONLY)) {
set_selinuxmnt(mnt);
}
return 0;
}
}
}
return -1;
}
static void init_selinuxmnt(void) static void init_selinuxmnt(void)
{ {
char *buf=NULL, *p; char *buf=NULL, *p;
FILE *fp=NULL; FILE *fp=NULL;
struct statfs sfbuf;
int rc;
size_t len; size_t len;
ssize_t num; ssize_t num;
int exists = 0; int exists = 0;
@ -33,17 +63,9 @@ static void init_selinuxmnt(void)
if (selinux_mnt) if (selinux_mnt)
return; return;
/* We check to see if the preferred mount point for selinux file if (verify_selinuxmnt(SELINUXMNT) == 0) return;
* system has a selinuxfs. */
do { if (verify_selinuxmnt(OLDSELINUXMNT) == 0) return;
rc = statfs(SELINUXMNT, &sfbuf);
} while (rc < 0 && errno == EINTR);
if (rc == 0) {
if ((uint32_t)sfbuf.f_type == (uint32_t)SELINUX_MAGIC) {
selinux_mnt = strdup(SELINUXMNT);
return;
}
}
/* Drop back to detecting it the long way. */ /* Drop back to detecting it the long way. */
fp = fopen("/proc/filesystems", "r"); fp = fopen("/proc/filesystems", "r");
@ -52,7 +74,7 @@ static void init_selinuxmnt(void)
__fsetlocking(fp, FSETLOCKING_BYCALLER); __fsetlocking(fp, FSETLOCKING_BYCALLER);
while ((num = getline(&buf, &len, fp)) != -1) { while ((num = getline(&buf, &len, fp)) != -1) {
if (strstr(buf, "selinuxfs")) { if (strstr(buf, SELINUXFS)) {
exists = 1; exists = 1;
break; break;
} }
@ -79,7 +101,7 @@ static void init_selinuxmnt(void)
tmp = strchr(p, ' '); tmp = strchr(p, ' ');
if (!tmp) if (!tmp)
goto out; goto out;
if (!strncmp(tmp + 1, "selinuxfs ", 10)) { if (!strncmp(tmp + 1, SELINUXFS" ", strlen(SELINUXFS)+1)) {
*tmp = '\0'; *tmp = '\0';
break; break;
} }
@ -87,7 +109,7 @@ static void init_selinuxmnt(void)
/* If we found something, dup it */ /* If we found something, dup it */
if (num > 0) if (num > 0)
selinux_mnt = strdup(p); verify_selinuxmnt(p);
out: out:
free(buf); free(buf);

View File

@ -369,7 +369,17 @@ int selinux_init_load_policy(int *enforce)
* Check for the existence of SELinux via selinuxfs, and * Check for the existence of SELinux via selinuxfs, and
* mount it if present for use in the calls below. * mount it if present for use in the calls below.
*/ */
if (mount("selinuxfs", SELINUXMNT, "selinuxfs", 0, 0) < 0 && errno != EBUSY) { char *mntpoint = NULL;
if (mount(SELINUXFS, SELINUXMNT, SELINUXFS, 0, 0) == 0 || errno == EBUSY) {
mntpoint = SELINUXMNT;
} else {
/* check old mountpoint */
if (mount(SELINUXFS, OLDSELINUXMNT, SELINUXFS, 0, 0) == 0 || errno == EBUSY) {
mntpoint = OLDSELINUXMNT;
}
}
if (! mntpoint ) {
if (errno == ENODEV) { if (errno == ENODEV) {
/* /*
* SELinux was disabled in the kernel, either * SELinux was disabled in the kernel, either
@ -385,7 +395,7 @@ int selinux_init_load_policy(int *enforce)
goto noload; goto noload;
} }
set_selinuxmnt(SELINUXMNT); set_selinuxmnt(mntpoint);
/* /*
* Note: The following code depends on having selinuxfs * Note: The following code depends on having selinuxfs
@ -397,7 +407,7 @@ int selinux_init_load_policy(int *enforce)
rc = security_disable(); rc = security_disable();
if (rc == 0) { if (rc == 0) {
/* Successfully disabled, so umount selinuxfs too. */ /* Successfully disabled, so umount selinuxfs too. */
umount(SELINUXMNT); umount(selinux_mnt);
fini_selinuxmnt(); fini_selinuxmnt();
} }
/* /*

View File

@ -9,11 +9,15 @@
/* Initial length guess for getting contexts. */ /* Initial length guess for getting contexts. */
#define INITCONTEXTLEN 255 #define INITCONTEXTLEN 255
/* selinux file system type */
#define SELINUXFS "selinuxfs"
/* selinuxfs magic number */ /* selinuxfs magic number */
#define SELINUX_MAGIC 0xf97cff8c #define SELINUX_MAGIC 0xf97cff8c
/* Preferred selinux mount location */ /* Preferred selinux mount location */
#define SELINUXMNT "/selinux" #define SELINUXMNT "/sys/fs/selinux"
#define OLDSELINUXMNT "/selinux"
/* selinuxfs mount point */ /* selinuxfs mount point */
extern char *selinux_mnt; extern char *selinux_mnt;