This ensures that once a process unshares its selinux namespace, it can no longer act on the parent namespace's selinuxfs instance, irrespective of policy. This is a safety measure so that even if an otherwise unconfined process unshares its selinux namespace, it won't be able to subsequently affect the enforcing mode or policy of the parent. This also helps avoid common mistakes like failing to create a mount namespace and mount a new selinuxfs instance in order to act on one's own selinux namespace after unsharing. Signed-off-by: Stephen Smalley <sds@xxxxxxxxxxxxx> --- security/selinux/selinuxfs.c | 31 +++++++++++++++++++++++++++++++ 1 file changed, 31 insertions(+) diff --git a/security/selinux/selinuxfs.c b/security/selinux/selinuxfs.c index 48afdc3a4aa4..1ba4d874fc86 100644 --- a/security/selinux/selinuxfs.c +++ b/security/selinux/selinuxfs.c @@ -143,6 +143,9 @@ static ssize_t sel_write_enforce(struct file *file, const char __user *buf, ssize_t length; int old_value, new_value; + if (ns != current_selinux_ns) + return -EPERM; + if (count >= PAGE_SIZE) return -ENOMEM; @@ -284,6 +287,9 @@ static ssize_t sel_write_disable(struct file *file, const char __user *buf, int new_value; int enforcing; + if (fsi->ns != current_selinux_ns) + return -EPERM; + if (count >= PAGE_SIZE) return -ENOMEM; @@ -337,6 +343,9 @@ static ssize_t sel_write_unshare(struct file *file, const char __user *buf, bool set; int rc; + if (ns != current_selinux_ns) + return -EPERM; + if (count >= PAGE_SIZE) return -ENOMEM; @@ -601,6 +610,9 @@ static ssize_t sel_write_load(struct file *file, const char __user *buf, ssize_t length; void *data = NULL; + if (fsi->ns != current_selinux_ns) + return -EPERM; + mutex_lock(&fsi->mutex); length = avc_has_perm(current_selinux_ns, @@ -706,6 +718,9 @@ static ssize_t sel_write_checkreqprot(struct file *file, const char __user *buf, ssize_t length; unsigned int new_value; + if (fsi->ns != current_selinux_ns) + return -EPERM; + length = avc_has_perm(current_selinux_ns, current_sid(), SECINITSID_SECURITY, SECCLASS_SECURITY, SECURITY__SETCHECKREQPROT, @@ -752,6 +767,9 @@ static ssize_t sel_write_validatetrans(struct file *file, u16 tclass; int rc; + if (ns != current_selinux_ns) + return -EPERM; + rc = avc_has_perm(current_selinux_ns, current_sid(), SECINITSID_SECURITY, SECCLASS_SECURITY, SECURITY__VALIDATE_TRANS, NULL); @@ -839,10 +857,14 @@ static ssize_t (*const write_op[])(struct file *, char *, size_t) = { static ssize_t selinux_transaction_write(struct file *file, const char __user *buf, size_t size, loff_t *pos) { + struct selinux_fs_info *fsi = file_inode(file)->i_sb->s_fs_info; ino_t ino = file_inode(file)->i_ino; char *data; ssize_t rv; + if (fsi->ns != current_selinux_ns) + return -EPERM; + if (ino >= ARRAY_SIZE(write_op) || !write_op[ino]) return -EINVAL; @@ -1278,6 +1300,9 @@ static ssize_t sel_write_bool(struct file *filep, const char __user *buf, unsigned index = file_inode(filep)->i_ino & SEL_INO_MASK; const char *name = filep->f_path.dentry->d_name.name; + if (fsi->ns != current_selinux_ns) + return -EPERM; + if (count >= PAGE_SIZE) return -ENOMEM; @@ -1334,6 +1359,9 @@ static ssize_t sel_commit_bools_write(struct file *filep, ssize_t length; int new_value; + if (fsi->ns != current_selinux_ns) + return -EPERM; + if (count >= PAGE_SIZE) return -ENOMEM; @@ -1498,6 +1526,9 @@ static ssize_t sel_write_avc_cache_threshold(struct file *file, ssize_t ret; unsigned int new_value; + if (ns != current_selinux_ns) + return -EPERM; + ret = avc_has_perm(current_selinux_ns, current_sid(), SECINITSID_SECURITY, SECCLASS_SECURITY, SECURITY__SETSECPARAM, -- 2.21.0