Commit cbdf35bc authored by Miklos Szeredi's avatar Miklos Szeredi

vfs: export check_sticky()

It's already duplicated in btrfs and about to be used in overlayfs too.

Move the sticky bit check to an inline helper and call the out-of-line
helper only in the unlikly case of the sticky bit being set.
Signed-off-by: default avatarMiklos Szeredi <mszeredi@suse.cz>
parent c771d683
...@@ -765,23 +765,6 @@ static int create_snapshot(struct btrfs_root *root, struct inode *dir, ...@@ -765,23 +765,6 @@ static int create_snapshot(struct btrfs_root *root, struct inode *dir,
return ret; return ret;
} }
/* copy of check_sticky in fs/namei.c()
* It's inline, so penalty for filesystems that don't use sticky bit is
* minimal.
*/
static inline int btrfs_check_sticky(struct inode *dir, struct inode *inode)
{
kuid_t fsuid = current_fsuid();
if (!(dir->i_mode & S_ISVTX))
return 0;
if (uid_eq(inode->i_uid, fsuid))
return 0;
if (uid_eq(dir->i_uid, fsuid))
return 0;
return !capable(CAP_FOWNER);
}
/* copy of may_delete in fs/namei.c() /* copy of may_delete in fs/namei.c()
* Check whether we can remove a link victim from directory dir, check * Check whether we can remove a link victim from directory dir, check
* whether the type of victim is right. * whether the type of victim is right.
...@@ -817,8 +800,7 @@ static int btrfs_may_delete(struct inode *dir, struct dentry *victim, int isdir) ...@@ -817,8 +800,7 @@ static int btrfs_may_delete(struct inode *dir, struct dentry *victim, int isdir)
return error; return error;
if (IS_APPEND(dir)) if (IS_APPEND(dir))
return -EPERM; return -EPERM;
if (btrfs_check_sticky(dir, victim->d_inode)|| if (check_sticky(dir, victim->d_inode) || IS_APPEND(victim->d_inode) ||
IS_APPEND(victim->d_inode)||
IS_IMMUTABLE(victim->d_inode) || IS_SWAPFILE(victim->d_inode)) IS_IMMUTABLE(victim->d_inode) || IS_SWAPFILE(victim->d_inode))
return -EPERM; return -EPERM;
if (isdir) { if (isdir) {
......
...@@ -2384,22 +2384,17 @@ kern_path_mountpoint(int dfd, const char *name, struct path *path, ...@@ -2384,22 +2384,17 @@ kern_path_mountpoint(int dfd, const char *name, struct path *path,
} }
EXPORT_SYMBOL(kern_path_mountpoint); EXPORT_SYMBOL(kern_path_mountpoint);
/* int __check_sticky(struct inode *dir, struct inode *inode)
* It's inline, so penalty for filesystems that don't use sticky bit is
* minimal.
*/
static inline int check_sticky(struct inode *dir, struct inode *inode)
{ {
kuid_t fsuid = current_fsuid(); kuid_t fsuid = current_fsuid();
if (!(dir->i_mode & S_ISVTX))
return 0;
if (uid_eq(inode->i_uid, fsuid)) if (uid_eq(inode->i_uid, fsuid))
return 0; return 0;
if (uid_eq(dir->i_uid, fsuid)) if (uid_eq(dir->i_uid, fsuid))
return 0; return 0;
return !capable_wrt_inode_uidgid(inode, CAP_FOWNER); return !capable_wrt_inode_uidgid(inode, CAP_FOWNER);
} }
EXPORT_SYMBOL(__check_sticky);
/* /*
* Check whether we can remove a link victim from directory dir, check * Check whether we can remove a link victim from directory dir, check
......
...@@ -2259,6 +2259,7 @@ extern int notify_change(struct dentry *, struct iattr *, struct inode **); ...@@ -2259,6 +2259,7 @@ extern int notify_change(struct dentry *, struct iattr *, struct inode **);
extern int inode_permission(struct inode *, int); extern int inode_permission(struct inode *, int);
extern int __inode_permission(struct inode *, int); extern int __inode_permission(struct inode *, int);
extern int generic_permission(struct inode *, int); extern int generic_permission(struct inode *, int);
extern int __check_sticky(struct inode *dir, struct inode *inode);
static inline bool execute_ok(struct inode *inode) static inline bool execute_ok(struct inode *inode)
{ {
...@@ -2745,6 +2746,14 @@ static inline int is_sxid(umode_t mode) ...@@ -2745,6 +2746,14 @@ static inline int is_sxid(umode_t mode)
return (mode & S_ISUID) || ((mode & S_ISGID) && (mode & S_IXGRP)); return (mode & S_ISUID) || ((mode & S_ISGID) && (mode & S_IXGRP));
} }
static inline int check_sticky(struct inode *dir, struct inode *inode)
{
if (!(dir->i_mode & S_ISVTX))
return 0;
return __check_sticky(dir, inode);
}
static inline void inode_has_no_xattr(struct inode *inode) static inline void inode_has_no_xattr(struct inode *inode)
{ {
if (!is_sxid(inode->i_mode) && (inode->i_sb->s_flags & MS_NOSEC)) if (!is_sxid(inode->i_mode) && (inode->i_sb->s_flags & MS_NOSEC))
......
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