Commit 86d3ef68 authored by Andrew Morton's avatar Andrew Morton Committed by Linus Torvalds

[PATCH] SELinux: context mount support - LSM/FS

From: James Morris <jmorris@redhat.com>

This series of patches adds support for SELinux 'context mounts', which
allows filesystems to be assigned security context information at mount time.
 For example, some filesystems do not support extended attributes (e.g.  NFS,
vfat), and this feature allows security contexts to be assigned to them on a
per-mountpoint basis.  It is also useful when the existing labeling on a
filesystem is untrusted or unwanted for some reason (e.g.  removable media),
and needs to be overridden with a safe default.

The first patch below consists of infrastructure changes to the kernel:

- A new LSM hook has been added, sb_copy_data, which allows the security
  module to copy security-specific mount data once the superblock has been
  setup by the filesystem.

- The sb_kern_mount hook has been modified to take this security data as a
  parameter, and would typically be used at that point to configure the
  security parameters of the filesystem being mounted.

- Allocation and freeing of the security data has been implemented in the
  core fs code as it is cleaner than trying to do it purely via LSM hooks,
  and should make maintenance easier.  This code will be compiled away if LSM
  is not enabled.
parent 50d8ab92
...@@ -708,6 +708,7 @@ do_kern_mount(const char *fstype, int flags, const char *name, void *data) ...@@ -708,6 +708,7 @@ do_kern_mount(const char *fstype, int flags, const char *name, void *data)
struct super_block *sb = ERR_PTR(-ENOMEM); struct super_block *sb = ERR_PTR(-ENOMEM);
struct vfsmount *mnt; struct vfsmount *mnt;
int error; int error;
char *secdata = NULL;
if (!type) if (!type)
return ERR_PTR(-ENODEV); return ERR_PTR(-ENODEV);
...@@ -715,10 +716,25 @@ do_kern_mount(const char *fstype, int flags, const char *name, void *data) ...@@ -715,10 +716,25 @@ do_kern_mount(const char *fstype, int flags, const char *name, void *data)
mnt = alloc_vfsmnt(name); mnt = alloc_vfsmnt(name);
if (!mnt) if (!mnt)
goto out; goto out;
if (data) {
secdata = alloc_secdata();
if (!secdata) {
sb = ERR_PTR(-ENOMEM);
goto out_mnt;
}
error = security_sb_copy_data(fstype, data, secdata);
if (error) {
sb = ERR_PTR(error);
goto out_free_secdata;
}
}
sb = type->get_sb(type, flags, name, data); sb = type->get_sb(type, flags, name, data);
if (IS_ERR(sb)) if (IS_ERR(sb))
goto out_mnt; goto out_free_secdata;
error = security_sb_kern_mount(sb); error = security_sb_kern_mount(sb, secdata);
if (error) if (error)
goto out_sb; goto out_sb;
mnt->mnt_sb = sb; mnt->mnt_sb = sb;
...@@ -732,6 +748,8 @@ do_kern_mount(const char *fstype, int flags, const char *name, void *data) ...@@ -732,6 +748,8 @@ do_kern_mount(const char *fstype, int flags, const char *name, void *data)
up_write(&sb->s_umount); up_write(&sb->s_umount);
deactivate_super(sb); deactivate_super(sb);
sb = ERR_PTR(error); sb = ERR_PTR(error);
out_free_secdata:
free_secdata(secdata);
out_mnt: out_mnt:
free_vfsmnt(mnt); free_vfsmnt(mnt);
out: out:
......
...@@ -1421,5 +1421,25 @@ static inline ino_t parent_ino(struct dentry *dentry) ...@@ -1421,5 +1421,25 @@ static inline ino_t parent_ino(struct dentry *dentry)
/* kernel/fork.c */ /* kernel/fork.c */
extern int unshare_files(void); extern int unshare_files(void);
#ifdef CONFIG_SECURITY
static inline char *alloc_secdata(void)
{
return (char *)get_zeroed_page(GFP_KERNEL);
}
static inline void free_secdata(void *secdata)
{
free_page((unsigned long)secdata);
}
#else
static inline char *alloc_secdata(void)
{
return (char *)1;
}
static inline void free_secdata(void *secdata)
{ }
#endif /* CONFIG_SECURITY */
#endif /* __KERNEL__ */ #endif /* __KERNEL__ */
#endif /* _LINUX_FS_H */ #endif /* _LINUX_FS_H */
...@@ -171,6 +171,16 @@ struct swap_info_struct; ...@@ -171,6 +171,16 @@ struct swap_info_struct;
* @flags contains the mount flags. * @flags contains the mount flags.
* @data contains the filesystem-specific data. * @data contains the filesystem-specific data.
* Return 0 if permission is granted. * Return 0 if permission is granted.
* @sb_copy_data:
* Allow mount option data to be copied prior to parsing by the filesystem,
* so that the security module can extract security-specific mount
* options cleanly (a filesystem may modify the data e.g. with strsep()).
* This also allows the original mount data to be stripped of security-
* specific options to avoid having to make filesystems aware of them.
* @fstype the type of filesystem being mounted.
* @orig the original mount data copied from userspace.
* @copy copied data which will be passed to the security module.
* Returns 0 if the copy was successful.
* @sb_check_sb: * @sb_check_sb:
* Check permission before the device with superblock @mnt->sb is mounted * Check permission before the device with superblock @mnt->sb is mounted
* on the mount point named by @nd. * on the mount point named by @nd.
...@@ -1024,7 +1034,8 @@ struct security_operations { ...@@ -1024,7 +1034,8 @@ struct security_operations {
int (*sb_alloc_security) (struct super_block * sb); int (*sb_alloc_security) (struct super_block * sb);
void (*sb_free_security) (struct super_block * sb); void (*sb_free_security) (struct super_block * sb);
int (*sb_kern_mount) (struct super_block *sb); int (*sb_copy_data)(const char *fstype, void *orig, void *copy);
int (*sb_kern_mount) (struct super_block *sb, void *data);
int (*sb_statfs) (struct super_block * sb); int (*sb_statfs) (struct super_block * sb);
int (*sb_mount) (char *dev_name, struct nameidata * nd, int (*sb_mount) (char *dev_name, struct nameidata * nd,
char *type, unsigned long flags, void *data); char *type, unsigned long flags, void *data);
...@@ -1308,9 +1319,14 @@ static inline void security_sb_free (struct super_block *sb) ...@@ -1308,9 +1319,14 @@ static inline void security_sb_free (struct super_block *sb)
security_ops->sb_free_security (sb); security_ops->sb_free_security (sb);
} }
static inline int security_sb_kern_mount (struct super_block *sb) static inline int security_sb_copy_data (const char *fstype, void *orig, void *copy)
{ {
return security_ops->sb_kern_mount (sb); return security_ops->sb_copy_data (fstype, orig, copy);
}
static inline int security_sb_kern_mount (struct super_block *sb, void *data)
{
return security_ops->sb_kern_mount (sb, data);
} }
static inline int security_sb_statfs (struct super_block *sb) static inline int security_sb_statfs (struct super_block *sb)
...@@ -1973,7 +1989,12 @@ static inline int security_sb_alloc (struct super_block *sb) ...@@ -1973,7 +1989,12 @@ static inline int security_sb_alloc (struct super_block *sb)
static inline void security_sb_free (struct super_block *sb) static inline void security_sb_free (struct super_block *sb)
{ } { }
static inline int security_sb_kern_mount (struct super_block *sb) static inline int security_sb_copy_data (const char *fstype, void *orig, void *copy)
{
return 0;
}
static inline int security_sb_kern_mount (struct super_block *sb, void *data)
{ {
return 0; return 0;
} }
......
...@@ -194,7 +194,12 @@ static void dummy_sb_free_security (struct super_block *sb) ...@@ -194,7 +194,12 @@ static void dummy_sb_free_security (struct super_block *sb)
return; return;
} }
static int dummy_sb_kern_mount (struct super_block *sb) static int dummy_sb_copy_data (const char *fstype, void *orig, void *copy)
{
return 0;
}
static int dummy_sb_kern_mount (struct super_block *sb, void *data)
{ {
return 0; return 0;
} }
...@@ -877,6 +882,7 @@ void security_fixup_ops (struct security_operations *ops) ...@@ -877,6 +882,7 @@ void security_fixup_ops (struct security_operations *ops)
set_to_dummy_if_null(ops, bprm_secureexec); set_to_dummy_if_null(ops, bprm_secureexec);
set_to_dummy_if_null(ops, sb_alloc_security); set_to_dummy_if_null(ops, sb_alloc_security);
set_to_dummy_if_null(ops, sb_free_security); set_to_dummy_if_null(ops, sb_free_security);
set_to_dummy_if_null(ops, sb_copy_data);
set_to_dummy_if_null(ops, sb_kern_mount); set_to_dummy_if_null(ops, sb_kern_mount);
set_to_dummy_if_null(ops, sb_statfs); set_to_dummy_if_null(ops, sb_statfs);
set_to_dummy_if_null(ops, sb_mount); set_to_dummy_if_null(ops, sb_mount);
......
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