Commit 90f8572b authored by Eric W. Biederman's avatar Eric W. Biederman

vfs: Commit to never having exectuables on proc and sysfs.

Today proc and sysfs do not contain any executable files.  Several
applications today mount proc or sysfs without noexec and nosuid and
then depend on there being no exectuables files on proc or sysfs.
Having any executable files show on proc or sysfs would cause
a user space visible regression, and most likely security problems.

Therefore commit to never allowing executables on proc and sysfs by
adding a new flag to mark them as filesystems without executables and
enforce that flag.

Test the flag where MNT_NOEXEC is tested today, so that the only user
visible effect will be that exectuables will be treated as if the
execute bit is cleared.

The filesystems proc and sysfs do not currently incoporate any
executable files so this does not result in any user visible effects.

This makes it unnecessary to vet changes to proc and sysfs tightly for
adding exectuable files or changes to chattr that would modify
existing files, as no matter what the individual file say they will
not be treated as exectuable files by the vfs.

Not having to vet changes to closely is important as without this we
are only one proc_create call (or another goof up in the
implementation of notify_change) from having problematic executables
on proc.  Those mistakes are all too easy to make and would create
a situation where there are security issues or the assumptions of
some program having to be broken (and cause userspace regressions).
Signed-off-by: default avatar"Eric W. Biederman" <ebiederm@xmission.com>
parent d770e558
...@@ -98,6 +98,12 @@ static inline void put_binfmt(struct linux_binfmt * fmt) ...@@ -98,6 +98,12 @@ static inline void put_binfmt(struct linux_binfmt * fmt)
module_put(fmt->module); module_put(fmt->module);
} }
bool path_noexec(const struct path *path)
{
return (path->mnt->mnt_flags & MNT_NOEXEC) ||
(path->mnt->mnt_sb->s_iflags & SB_I_NOEXEC);
}
#ifdef CONFIG_USELIB #ifdef CONFIG_USELIB
/* /*
* Note that a shared library must be both readable and executable due to * Note that a shared library must be both readable and executable due to
...@@ -132,7 +138,7 @@ SYSCALL_DEFINE1(uselib, const char __user *, library) ...@@ -132,7 +138,7 @@ SYSCALL_DEFINE1(uselib, const char __user *, library)
goto exit; goto exit;
error = -EACCES; error = -EACCES;
if (file->f_path.mnt->mnt_flags & MNT_NOEXEC) if (path_noexec(&file->f_path))
goto exit; goto exit;
fsnotify_open(file); fsnotify_open(file);
...@@ -777,7 +783,7 @@ static struct file *do_open_execat(int fd, struct filename *name, int flags) ...@@ -777,7 +783,7 @@ static struct file *do_open_execat(int fd, struct filename *name, int flags)
if (!S_ISREG(file_inode(file)->i_mode)) if (!S_ISREG(file_inode(file)->i_mode))
goto exit; goto exit;
if (file->f_path.mnt->mnt_flags & MNT_NOEXEC) if (path_noexec(&file->f_path))
goto exit; goto exit;
err = deny_write_access(file); err = deny_write_access(file);
......
...@@ -377,7 +377,7 @@ SYSCALL_DEFINE3(faccessat, int, dfd, const char __user *, filename, int, mode) ...@@ -377,7 +377,7 @@ SYSCALL_DEFINE3(faccessat, int, dfd, const char __user *, filename, int, mode)
* with the "noexec" flag. * with the "noexec" flag.
*/ */
res = -EACCES; res = -EACCES;
if (path.mnt->mnt_flags & MNT_NOEXEC) if (path_noexec(&path))
goto out_path_release; goto out_path_release;
} }
......
...@@ -134,6 +134,8 @@ static struct dentry *proc_mount(struct file_system_type *fs_type, ...@@ -134,6 +134,8 @@ static struct dentry *proc_mount(struct file_system_type *fs_type,
} }
sb->s_flags |= MS_ACTIVE; sb->s_flags |= MS_ACTIVE;
/* User space would break if executables appear on proc */
sb->s_iflags |= SB_I_NOEXEC;
} }
return dget(sb->s_root); return dget(sb->s_root);
......
...@@ -40,6 +40,10 @@ static struct dentry *sysfs_mount(struct file_system_type *fs_type, ...@@ -40,6 +40,10 @@ static struct dentry *sysfs_mount(struct file_system_type *fs_type,
SYSFS_MAGIC, &new_sb, ns); SYSFS_MAGIC, &new_sb, ns);
if (IS_ERR(root) || !new_sb) if (IS_ERR(root) || !new_sb)
kobj_ns_drop(KOBJ_NS_TYPE_NET, ns); kobj_ns_drop(KOBJ_NS_TYPE_NET, ns);
else if (new_sb)
/* Userspace would break if executables appear on sysfs */
root->d_sb->s_iflags |= SB_I_NOEXEC;
return root; return root;
} }
......
...@@ -1244,6 +1244,7 @@ struct mm_struct; ...@@ -1244,6 +1244,7 @@ struct mm_struct;
/* sb->s_iflags */ /* sb->s_iflags */
#define SB_I_CGROUPWB 0x00000001 /* cgroup-aware writeback enabled */ #define SB_I_CGROUPWB 0x00000001 /* cgroup-aware writeback enabled */
#define SB_I_NOEXEC 0x00000002 /* Ignore executables on this fs */
/* Possible states of 'frozen' field */ /* Possible states of 'frozen' field */
enum { enum {
...@@ -3030,4 +3031,6 @@ static inline bool dir_relax(struct inode *inode) ...@@ -3030,4 +3031,6 @@ static inline bool dir_relax(struct inode *inode)
return !IS_DEADDIR(inode); return !IS_DEADDIR(inode);
} }
extern bool path_noexec(const struct path *path);
#endif /* _LINUX_FS_H */ #endif /* _LINUX_FS_H */
...@@ -1668,8 +1668,7 @@ static int prctl_set_mm_exe_file(struct mm_struct *mm, unsigned int fd) ...@@ -1668,8 +1668,7 @@ static int prctl_set_mm_exe_file(struct mm_struct *mm, unsigned int fd)
* overall picture. * overall picture.
*/ */
err = -EACCES; err = -EACCES;
if (!S_ISREG(inode->i_mode) || if (!S_ISREG(inode->i_mode) || path_noexec(&exe.file->f_path))
exe.file->f_path.mnt->mnt_flags & MNT_NOEXEC)
goto exit; goto exit;
err = inode_permission(inode, MAY_EXEC); err = inode_permission(inode, MAY_EXEC);
......
...@@ -1268,7 +1268,7 @@ unsigned long do_mmap_pgoff(struct file *file, unsigned long addr, ...@@ -1268,7 +1268,7 @@ unsigned long do_mmap_pgoff(struct file *file, unsigned long addr,
* mounted, in which case we dont add PROT_EXEC.) * mounted, in which case we dont add PROT_EXEC.)
*/ */
if ((prot & PROT_READ) && (current->personality & READ_IMPLIES_EXEC)) if ((prot & PROT_READ) && (current->personality & READ_IMPLIES_EXEC))
if (!(file && (file->f_path.mnt->mnt_flags & MNT_NOEXEC))) if (!(file && path_noexec(&file->f_path)))
prot |= PROT_EXEC; prot |= PROT_EXEC;
if (!(flags & MAP_FIXED)) if (!(flags & MAP_FIXED))
...@@ -1337,7 +1337,7 @@ unsigned long do_mmap_pgoff(struct file *file, unsigned long addr, ...@@ -1337,7 +1337,7 @@ unsigned long do_mmap_pgoff(struct file *file, unsigned long addr,
case MAP_PRIVATE: case MAP_PRIVATE:
if (!(file->f_mode & FMODE_READ)) if (!(file->f_mode & FMODE_READ))
return -EACCES; return -EACCES;
if (file->f_path.mnt->mnt_flags & MNT_NOEXEC) { if (path_noexec(&file->f_path)) {
if (vm_flags & VM_EXEC) if (vm_flags & VM_EXEC)
return -EPERM; return -EPERM;
vm_flags &= ~VM_MAYEXEC; vm_flags &= ~VM_MAYEXEC;
......
...@@ -1035,7 +1035,7 @@ static int validate_mmap_request(struct file *file, ...@@ -1035,7 +1035,7 @@ static int validate_mmap_request(struct file *file,
/* handle executable mappings and implied executable /* handle executable mappings and implied executable
* mappings */ * mappings */
if (file->f_path.mnt->mnt_flags & MNT_NOEXEC) { if (path_noexec(&file->f_path)) {
if (prot & PROT_EXEC) if (prot & PROT_EXEC)
return -EPERM; return -EPERM;
} else if ((prot & PROT_READ) && !(prot & PROT_EXEC)) { } else if ((prot & PROT_READ) && !(prot & PROT_EXEC)) {
......
...@@ -776,7 +776,7 @@ static inline unsigned long mmap_prot(struct file *file, unsigned long prot) ...@@ -776,7 +776,7 @@ static inline unsigned long mmap_prot(struct file *file, unsigned long prot)
* ditto if it's not on noexec mount, except that on !MMU we need * ditto if it's not on noexec mount, except that on !MMU we need
* NOMMU_MAP_EXEC (== VM_MAYEXEC) in this case * NOMMU_MAP_EXEC (== VM_MAYEXEC) in this case
*/ */
if (!(file->f_path.mnt->mnt_flags & MNT_NOEXEC)) { if (!path_noexec(&file->f_path)) {
#ifndef CONFIG_MMU #ifndef CONFIG_MMU
if (file->f_op->mmap_capabilities) { if (file->f_op->mmap_capabilities) {
unsigned caps = file->f_op->mmap_capabilities(file); unsigned caps = file->f_op->mmap_capabilities(file);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment