mirror of
git://git.kernel.org/pub/scm/linux/kernel/git/torvalds/linux.git
synced 2025-09-04 20:19:47 +08:00
vfs-6.17-rc1.fileattr
-----BEGIN PGP SIGNATURE----- iHUEABYKAB0WIQRAhzRXHqcMeLMyaSiRxhvAZXjcogUCaINCpgAKCRCRxhvAZXjc oqfFAQDcy3rROUF3W34KcSi7rDmaKVSX53d1tUoqH+1zDRpSlwEAriKDNC1ybudp YAnxVzkRHjHs1296WIuwKq5lfhJ60Q4= =geAl -----END PGP SIGNATURE----- Merge tag 'vfs-6.17-rc1.fileattr' of git://git.kernel.org/pub/scm/linux/kernel/git/vfs/vfs Pull fileattr updates from Christian Brauner: "This introduces the new file_getattr() and file_setattr() system calls after lengthy discussions. Both system calls serve as successors and extensible companions to the FS_IOC_FSGETXATTR and FS_IOC_FSSETXATTR system calls which have started to show their age in addition to being named in a way that makes it easy to conflate them with extended attribute related operations. These syscalls allow userspace to set filesystem inode attributes on special files. One of the usage examples is the XFS quota projects. XFS has project quotas which could be attached to a directory. All new inodes in these directories inherit project ID set on parent directory. The project is created from userspace by opening and calling FS_IOC_FSSETXATTR on each inode. This is not possible for special files such as FIFO, SOCK, BLK etc. Therefore, some inodes are left with empty project ID. Those inodes then are not shown in the quota accounting but still exist in the directory. This is not critical but in the case when special files are created in the directory with already existing project quota, these new inodes inherit extended attributes. This creates a mix of special files with and without attributes. Moreover, special files with attributes don't have a possibility to become clear or change the attributes. This, in turn, prevents userspace from re-creating quota project on these existing files. In addition, these new system calls allow the implementation of additional attributes that we couldn't or didn't want to fit into the legacy ioctls anymore" * tag 'vfs-6.17-rc1.fileattr' of git://git.kernel.org/pub/scm/linux/kernel/git/vfs/vfs: fs: tighten a sanity check in file_attr_to_fileattr() tree-wide: s/struct fileattr/struct file_kattr/g fs: introduce file_getattr and file_setattr syscalls fs: prepare for extending file_get/setattr() fs: make vfs_fileattr_[get|set] return -EOPNOTSUPP selinux: implement inode_file_[g|s]etattr hooks lsm: introduce new hooks for setting/getting inode fsxattr fs: split fileattr related helpers into separate file
This commit is contained in:
commit
57fcb7d930
@ -87,8 +87,8 @@ prototypes::
|
||||
int (*tmpfile) (struct mnt_idmap *, struct inode *,
|
||||
struct file *, umode_t);
|
||||
int (*fileattr_set)(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa);
|
||||
int (*fileattr_get)(struct dentry *dentry, struct fileattr *fa);
|
||||
struct dentry *dentry, struct file_kattr *fa);
|
||||
int (*fileattr_get)(struct dentry *dentry, struct file_kattr *fa);
|
||||
struct posix_acl * (*get_acl)(struct mnt_idmap *, struct dentry *, int);
|
||||
struct offset_ctx *(*get_offset_ctx)(struct inode *inode);
|
||||
|
||||
|
@ -515,8 +515,8 @@ As of kernel 2.6.22, the following members are defined:
|
||||
struct posix_acl * (*get_acl)(struct mnt_idmap *, struct dentry *, int);
|
||||
int (*set_acl)(struct mnt_idmap *, struct dentry *, struct posix_acl *, int);
|
||||
int (*fileattr_set)(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa);
|
||||
int (*fileattr_get)(struct dentry *dentry, struct fileattr *fa);
|
||||
struct dentry *dentry, struct file_kattr *fa);
|
||||
int (*fileattr_get)(struct dentry *dentry, struct file_kattr *fa);
|
||||
struct offset_ctx *(*get_offset_ctx)(struct inode *inode);
|
||||
};
|
||||
|
||||
|
@ -507,3 +507,5 @@
|
||||
575 common listxattrat sys_listxattrat
|
||||
576 common removexattrat sys_removexattrat
|
||||
577 common open_tree_attr sys_open_tree_attr
|
||||
578 common file_getattr sys_file_getattr
|
||||
579 common file_setattr sys_file_setattr
|
||||
|
@ -482,3 +482,5 @@
|
||||
465 common listxattrat sys_listxattrat
|
||||
466 common removexattrat sys_removexattrat
|
||||
467 common open_tree_attr sys_open_tree_attr
|
||||
468 common file_getattr sys_file_getattr
|
||||
469 common file_setattr sys_file_setattr
|
||||
|
@ -479,3 +479,5 @@
|
||||
465 common listxattrat sys_listxattrat
|
||||
466 common removexattrat sys_removexattrat
|
||||
467 common open_tree_attr sys_open_tree_attr
|
||||
468 common file_getattr sys_file_getattr
|
||||
469 common file_setattr sys_file_setattr
|
||||
|
@ -467,3 +467,5 @@
|
||||
465 common listxattrat sys_listxattrat
|
||||
466 common removexattrat sys_removexattrat
|
||||
467 common open_tree_attr sys_open_tree_attr
|
||||
468 common file_getattr sys_file_getattr
|
||||
469 common file_setattr sys_file_setattr
|
||||
|
@ -473,3 +473,5 @@
|
||||
465 common listxattrat sys_listxattrat
|
||||
466 common removexattrat sys_removexattrat
|
||||
467 common open_tree_attr sys_open_tree_attr
|
||||
468 common file_getattr sys_file_getattr
|
||||
469 common file_setattr sys_file_setattr
|
||||
|
@ -406,3 +406,5 @@
|
||||
465 n32 listxattrat sys_listxattrat
|
||||
466 n32 removexattrat sys_removexattrat
|
||||
467 n32 open_tree_attr sys_open_tree_attr
|
||||
468 n32 file_getattr sys_file_getattr
|
||||
469 n32 file_setattr sys_file_setattr
|
||||
|
@ -382,3 +382,5 @@
|
||||
465 n64 listxattrat sys_listxattrat
|
||||
466 n64 removexattrat sys_removexattrat
|
||||
467 n64 open_tree_attr sys_open_tree_attr
|
||||
468 n64 file_getattr sys_file_getattr
|
||||
469 n64 file_setattr sys_file_setattr
|
||||
|
@ -455,3 +455,5 @@
|
||||
465 o32 listxattrat sys_listxattrat
|
||||
466 o32 removexattrat sys_removexattrat
|
||||
467 o32 open_tree_attr sys_open_tree_attr
|
||||
468 o32 file_getattr sys_file_getattr
|
||||
469 o32 file_setattr sys_file_setattr
|
||||
|
@ -466,3 +466,5 @@
|
||||
465 common listxattrat sys_listxattrat
|
||||
466 common removexattrat sys_removexattrat
|
||||
467 common open_tree_attr sys_open_tree_attr
|
||||
468 common file_getattr sys_file_getattr
|
||||
469 common file_setattr sys_file_setattr
|
||||
|
@ -558,3 +558,5 @@
|
||||
465 common listxattrat sys_listxattrat
|
||||
466 common removexattrat sys_removexattrat
|
||||
467 common open_tree_attr sys_open_tree_attr
|
||||
468 common file_getattr sys_file_getattr
|
||||
469 common file_setattr sys_file_setattr
|
||||
|
@ -470,3 +470,5 @@
|
||||
465 common listxattrat sys_listxattrat sys_listxattrat
|
||||
466 common removexattrat sys_removexattrat sys_removexattrat
|
||||
467 common open_tree_attr sys_open_tree_attr sys_open_tree_attr
|
||||
468 common file_getattr sys_file_getattr sys_file_getattr
|
||||
469 common file_setattr sys_file_setattr sys_file_setattr
|
||||
|
@ -471,3 +471,5 @@
|
||||
465 common listxattrat sys_listxattrat
|
||||
466 common removexattrat sys_removexattrat
|
||||
467 common open_tree_attr sys_open_tree_attr
|
||||
468 common file_getattr sys_file_getattr
|
||||
469 common file_setattr sys_file_setattr
|
||||
|
@ -513,3 +513,5 @@
|
||||
465 common listxattrat sys_listxattrat
|
||||
466 common removexattrat sys_removexattrat
|
||||
467 common open_tree_attr sys_open_tree_attr
|
||||
468 common file_getattr sys_file_getattr
|
||||
469 common file_setattr sys_file_setattr
|
||||
|
@ -473,3 +473,5 @@
|
||||
465 i386 listxattrat sys_listxattrat
|
||||
466 i386 removexattrat sys_removexattrat
|
||||
467 i386 open_tree_attr sys_open_tree_attr
|
||||
468 i386 file_getattr sys_file_getattr
|
||||
469 i386 file_setattr sys_file_setattr
|
||||
|
@ -391,6 +391,8 @@
|
||||
465 common listxattrat sys_listxattrat
|
||||
466 common removexattrat sys_removexattrat
|
||||
467 common open_tree_attr sys_open_tree_attr
|
||||
468 common file_getattr sys_file_getattr
|
||||
469 common file_setattr sys_file_setattr
|
||||
|
||||
#
|
||||
# Due to a historical design error, certain syscalls are numbered differently
|
||||
|
@ -438,3 +438,5 @@
|
||||
465 common listxattrat sys_listxattrat
|
||||
466 common removexattrat sys_removexattrat
|
||||
467 common open_tree_attr sys_open_tree_attr
|
||||
468 common file_getattr sys_file_getattr
|
||||
469 common file_setattr sys_file_setattr
|
||||
|
@ -15,7 +15,8 @@ obj-y := open.o read_write.o file_table.o super.o \
|
||||
pnode.o splice.o sync.o utimes.o d_path.o \
|
||||
stack.o fs_struct.o statfs.o fs_pin.o nsfs.o \
|
||||
fs_types.o fs_context.o fs_parser.o fsopen.o init.o \
|
||||
kernel_read_file.o mnt_idmapping.o remap_range.o pidfs.o
|
||||
kernel_read_file.o mnt_idmapping.o remap_range.o pidfs.o \
|
||||
file_attr.o
|
||||
|
||||
obj-$(CONFIG_BUFFER_HEAD) += buffer.o mpage.o
|
||||
obj-$(CONFIG_PROC_FS) += proc_namespace.o
|
||||
|
@ -1617,7 +1617,7 @@ static const __maybe_unused unsigned bch_flags_to_xflags[] = {
|
||||
};
|
||||
|
||||
static int bch2_fileattr_get(struct dentry *dentry,
|
||||
struct fileattr *fa)
|
||||
struct file_kattr *fa)
|
||||
{
|
||||
struct bch_inode_info *inode = to_bch_ei(d_inode(dentry));
|
||||
struct bch_fs *c = inode->v.i_sb->s_fs_info;
|
||||
@ -1680,7 +1680,7 @@ static int fssetxattr_inode_update_fn(struct btree_trans *trans,
|
||||
|
||||
static int bch2_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry,
|
||||
struct fileattr *fa)
|
||||
struct file_kattr *fa)
|
||||
{
|
||||
struct bch_inode_info *inode = to_bch_ei(d_inode(dentry));
|
||||
struct bch_fs *c = inode->v.i_sb->s_fs_info;
|
||||
|
@ -245,7 +245,7 @@ static int btrfs_check_ioctl_vol_args2_subvol_name(const struct btrfs_ioctl_vol_
|
||||
* Set flags/xflags from the internal inode flags. The remaining items of
|
||||
* fsxattr are zeroed.
|
||||
*/
|
||||
int btrfs_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
int btrfs_fileattr_get(struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
const struct btrfs_inode *inode = BTRFS_I(d_inode(dentry));
|
||||
|
||||
@ -254,7 +254,7 @@ int btrfs_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
}
|
||||
|
||||
int btrfs_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa)
|
||||
struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
struct btrfs_inode *inode = BTRFS_I(d_inode(dentry));
|
||||
struct btrfs_root *root = inode->root;
|
||||
|
@ -8,7 +8,7 @@
|
||||
struct file;
|
||||
struct dentry;
|
||||
struct mnt_idmap;
|
||||
struct fileattr;
|
||||
struct file_kattr;
|
||||
struct io_uring_cmd;
|
||||
struct btrfs_inode;
|
||||
struct btrfs_fs_info;
|
||||
@ -16,9 +16,9 @@ struct btrfs_ioctl_balance_args;
|
||||
|
||||
long btrfs_ioctl(struct file *file, unsigned int cmd, unsigned long arg);
|
||||
long btrfs_compat_ioctl(struct file *file, unsigned int cmd, unsigned long arg);
|
||||
int btrfs_fileattr_get(struct dentry *dentry, struct fileattr *fa);
|
||||
int btrfs_fileattr_get(struct dentry *dentry, struct file_kattr *fa);
|
||||
int btrfs_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa);
|
||||
struct dentry *dentry, struct file_kattr *fa);
|
||||
int btrfs_ioctl_get_supported_features(void __user *arg);
|
||||
void btrfs_sync_inode_flags_to_i_flags(struct btrfs_inode *inode);
|
||||
void btrfs_update_ioctl_balance_args(struct btrfs_fs_info *fs_info,
|
||||
|
@ -1124,13 +1124,13 @@ out:
|
||||
return rc;
|
||||
}
|
||||
|
||||
static int ecryptfs_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
static int ecryptfs_fileattr_get(struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
return vfs_fileattr_get(ecryptfs_dentry_to_lower(dentry), fa);
|
||||
}
|
||||
|
||||
static int ecryptfs_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa)
|
||||
struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
struct dentry *lower_dentry = ecryptfs_dentry_to_lower(dentry);
|
||||
int rc;
|
||||
|
@ -138,7 +138,7 @@ const struct inode_operations efivarfs_dir_inode_operations = {
|
||||
};
|
||||
|
||||
static int
|
||||
efivarfs_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
efivarfs_fileattr_get(struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
unsigned int i_flags;
|
||||
unsigned int flags = 0;
|
||||
@ -154,7 +154,7 @@ efivarfs_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
|
||||
static int
|
||||
efivarfs_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa)
|
||||
struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
unsigned int i_flags = 0;
|
||||
|
||||
|
@ -750,9 +750,9 @@ extern int ext2_fiemap(struct inode *inode, struct fiemap_extent_info *fieinfo,
|
||||
u64 start, u64 len);
|
||||
|
||||
/* ioctl.c */
|
||||
extern int ext2_fileattr_get(struct dentry *dentry, struct fileattr *fa);
|
||||
extern int ext2_fileattr_get(struct dentry *dentry, struct file_kattr *fa);
|
||||
extern int ext2_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa);
|
||||
struct dentry *dentry, struct file_kattr *fa);
|
||||
extern long ext2_ioctl(struct file *, unsigned int, unsigned long);
|
||||
extern long ext2_compat_ioctl(struct file *, unsigned int, unsigned long);
|
||||
|
||||
|
@ -18,7 +18,7 @@
|
||||
#include <linux/uaccess.h>
|
||||
#include <linux/fileattr.h>
|
||||
|
||||
int ext2_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
int ext2_fileattr_get(struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
struct ext2_inode_info *ei = EXT2_I(d_inode(dentry));
|
||||
|
||||
@ -28,7 +28,7 @@ int ext2_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
}
|
||||
|
||||
int ext2_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa)
|
||||
struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
struct inode *inode = d_inode(dentry);
|
||||
struct ext2_inode_info *ei = EXT2_I(inode);
|
||||
|
@ -3103,8 +3103,8 @@ extern int ext4_ind_remove_space(handle_t *handle, struct inode *inode,
|
||||
extern long ext4_ioctl(struct file *, unsigned int, unsigned long);
|
||||
extern long ext4_compat_ioctl(struct file *, unsigned int, unsigned long);
|
||||
int ext4_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa);
|
||||
int ext4_fileattr_get(struct dentry *dentry, struct fileattr *fa);
|
||||
struct dentry *dentry, struct file_kattr *fa);
|
||||
int ext4_fileattr_get(struct dentry *dentry, struct file_kattr *fa);
|
||||
extern void ext4_reset_inode_seed(struct inode *inode);
|
||||
int ext4_update_overhead(struct super_block *sb, bool force);
|
||||
int ext4_force_shutdown(struct super_block *sb, u32 flags);
|
||||
|
@ -980,7 +980,7 @@ group_add_out:
|
||||
return err;
|
||||
}
|
||||
|
||||
int ext4_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
int ext4_fileattr_get(struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
struct inode *inode = d_inode(dentry);
|
||||
struct ext4_inode_info *ei = EXT4_I(inode);
|
||||
@ -997,7 +997,7 @@ int ext4_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
}
|
||||
|
||||
int ext4_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa)
|
||||
struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
struct inode *inode = d_inode(dentry);
|
||||
u32 flags = fa->flags;
|
||||
|
@ -3615,9 +3615,9 @@ void f2fs_truncate_data_blocks_range(struct dnode_of_data *dn, int count);
|
||||
int f2fs_do_shutdown(struct f2fs_sb_info *sbi, unsigned int flag,
|
||||
bool readonly, bool need_lock);
|
||||
int f2fs_precache_extents(struct inode *inode);
|
||||
int f2fs_fileattr_get(struct dentry *dentry, struct fileattr *fa);
|
||||
int f2fs_fileattr_get(struct dentry *dentry, struct file_kattr *fa);
|
||||
int f2fs_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa);
|
||||
struct dentry *dentry, struct file_kattr *fa);
|
||||
long f2fs_ioctl(struct file *filp, unsigned int cmd, unsigned long arg);
|
||||
long f2fs_compat_ioctl(struct file *file, unsigned int cmd, unsigned long arg);
|
||||
int f2fs_transfer_project_quota(struct inode *inode, kprojid_t kprojid);
|
||||
|
@ -3391,7 +3391,7 @@ static int f2fs_ioc_setproject(struct inode *inode, __u32 projid)
|
||||
}
|
||||
#endif
|
||||
|
||||
int f2fs_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
int f2fs_fileattr_get(struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
struct inode *inode = d_inode(dentry);
|
||||
struct f2fs_inode_info *fi = F2FS_I(inode);
|
||||
@ -3415,7 +3415,7 @@ int f2fs_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
}
|
||||
|
||||
int f2fs_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa)
|
||||
struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
struct inode *inode = d_inode(dentry);
|
||||
u32 fsflags = fa->flags, mask = F2FS_SETTABLE_FS_FL;
|
||||
|
498
fs/file_attr.c
Normal file
498
fs/file_attr.c
Normal file
@ -0,0 +1,498 @@
|
||||
// SPDX-License-Identifier: GPL-2.0
|
||||
#include <linux/fs.h>
|
||||
#include <linux/security.h>
|
||||
#include <linux/fscrypt.h>
|
||||
#include <linux/fileattr.h>
|
||||
#include <linux/export.h>
|
||||
#include <linux/syscalls.h>
|
||||
#include <linux/namei.h>
|
||||
|
||||
#include "internal.h"
|
||||
|
||||
/**
|
||||
* fileattr_fill_xflags - initialize fileattr with xflags
|
||||
* @fa: fileattr pointer
|
||||
* @xflags: FS_XFLAG_* flags
|
||||
*
|
||||
* Set ->fsx_xflags, ->fsx_valid and ->flags (translated xflags). All
|
||||
* other fields are zeroed.
|
||||
*/
|
||||
void fileattr_fill_xflags(struct file_kattr *fa, u32 xflags)
|
||||
{
|
||||
memset(fa, 0, sizeof(*fa));
|
||||
fa->fsx_valid = true;
|
||||
fa->fsx_xflags = xflags;
|
||||
if (fa->fsx_xflags & FS_XFLAG_IMMUTABLE)
|
||||
fa->flags |= FS_IMMUTABLE_FL;
|
||||
if (fa->fsx_xflags & FS_XFLAG_APPEND)
|
||||
fa->flags |= FS_APPEND_FL;
|
||||
if (fa->fsx_xflags & FS_XFLAG_SYNC)
|
||||
fa->flags |= FS_SYNC_FL;
|
||||
if (fa->fsx_xflags & FS_XFLAG_NOATIME)
|
||||
fa->flags |= FS_NOATIME_FL;
|
||||
if (fa->fsx_xflags & FS_XFLAG_NODUMP)
|
||||
fa->flags |= FS_NODUMP_FL;
|
||||
if (fa->fsx_xflags & FS_XFLAG_DAX)
|
||||
fa->flags |= FS_DAX_FL;
|
||||
if (fa->fsx_xflags & FS_XFLAG_PROJINHERIT)
|
||||
fa->flags |= FS_PROJINHERIT_FL;
|
||||
}
|
||||
EXPORT_SYMBOL(fileattr_fill_xflags);
|
||||
|
||||
/**
|
||||
* fileattr_fill_flags - initialize fileattr with flags
|
||||
* @fa: fileattr pointer
|
||||
* @flags: FS_*_FL flags
|
||||
*
|
||||
* Set ->flags, ->flags_valid and ->fsx_xflags (translated flags).
|
||||
* All other fields are zeroed.
|
||||
*/
|
||||
void fileattr_fill_flags(struct file_kattr *fa, u32 flags)
|
||||
{
|
||||
memset(fa, 0, sizeof(*fa));
|
||||
fa->flags_valid = true;
|
||||
fa->flags = flags;
|
||||
if (fa->flags & FS_SYNC_FL)
|
||||
fa->fsx_xflags |= FS_XFLAG_SYNC;
|
||||
if (fa->flags & FS_IMMUTABLE_FL)
|
||||
fa->fsx_xflags |= FS_XFLAG_IMMUTABLE;
|
||||
if (fa->flags & FS_APPEND_FL)
|
||||
fa->fsx_xflags |= FS_XFLAG_APPEND;
|
||||
if (fa->flags & FS_NODUMP_FL)
|
||||
fa->fsx_xflags |= FS_XFLAG_NODUMP;
|
||||
if (fa->flags & FS_NOATIME_FL)
|
||||
fa->fsx_xflags |= FS_XFLAG_NOATIME;
|
||||
if (fa->flags & FS_DAX_FL)
|
||||
fa->fsx_xflags |= FS_XFLAG_DAX;
|
||||
if (fa->flags & FS_PROJINHERIT_FL)
|
||||
fa->fsx_xflags |= FS_XFLAG_PROJINHERIT;
|
||||
}
|
||||
EXPORT_SYMBOL(fileattr_fill_flags);
|
||||
|
||||
/**
|
||||
* vfs_fileattr_get - retrieve miscellaneous file attributes
|
||||
* @dentry: the object to retrieve from
|
||||
* @fa: fileattr pointer
|
||||
*
|
||||
* Call i_op->fileattr_get() callback, if exists.
|
||||
*
|
||||
* Return: 0 on success, or a negative error on failure.
|
||||
*/
|
||||
int vfs_fileattr_get(struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
struct inode *inode = d_inode(dentry);
|
||||
int error;
|
||||
|
||||
if (!inode->i_op->fileattr_get)
|
||||
return -EOPNOTSUPP;
|
||||
|
||||
error = security_inode_file_getattr(dentry, fa);
|
||||
if (error)
|
||||
return error;
|
||||
|
||||
return inode->i_op->fileattr_get(dentry, fa);
|
||||
}
|
||||
EXPORT_SYMBOL(vfs_fileattr_get);
|
||||
|
||||
static void fileattr_to_file_attr(const struct file_kattr *fa,
|
||||
struct file_attr *fattr)
|
||||
{
|
||||
__u32 mask = FS_XFLAGS_MASK;
|
||||
|
||||
memset(fattr, 0, sizeof(struct file_attr));
|
||||
fattr->fa_xflags = fa->fsx_xflags & mask;
|
||||
fattr->fa_extsize = fa->fsx_extsize;
|
||||
fattr->fa_nextents = fa->fsx_nextents;
|
||||
fattr->fa_projid = fa->fsx_projid;
|
||||
fattr->fa_cowextsize = fa->fsx_cowextsize;
|
||||
}
|
||||
|
||||
/**
|
||||
* copy_fsxattr_to_user - copy fsxattr to userspace.
|
||||
* @fa: fileattr pointer
|
||||
* @ufa: fsxattr user pointer
|
||||
*
|
||||
* Return: 0 on success, or -EFAULT on failure.
|
||||
*/
|
||||
int copy_fsxattr_to_user(const struct file_kattr *fa, struct fsxattr __user *ufa)
|
||||
{
|
||||
struct fsxattr xfa;
|
||||
__u32 mask = FS_XFLAGS_MASK;
|
||||
|
||||
memset(&xfa, 0, sizeof(xfa));
|
||||
xfa.fsx_xflags = fa->fsx_xflags & mask;
|
||||
xfa.fsx_extsize = fa->fsx_extsize;
|
||||
xfa.fsx_nextents = fa->fsx_nextents;
|
||||
xfa.fsx_projid = fa->fsx_projid;
|
||||
xfa.fsx_cowextsize = fa->fsx_cowextsize;
|
||||
|
||||
if (copy_to_user(ufa, &xfa, sizeof(xfa)))
|
||||
return -EFAULT;
|
||||
|
||||
return 0;
|
||||
}
|
||||
EXPORT_SYMBOL(copy_fsxattr_to_user);
|
||||
|
||||
static int file_attr_to_fileattr(const struct file_attr *fattr,
|
||||
struct file_kattr *fa)
|
||||
{
|
||||
__u64 mask = FS_XFLAGS_MASK;
|
||||
|
||||
if (fattr->fa_xflags & ~mask)
|
||||
return -EINVAL;
|
||||
|
||||
fileattr_fill_xflags(fa, fattr->fa_xflags);
|
||||
fa->fsx_xflags &= ~FS_XFLAG_RDONLY_MASK;
|
||||
fa->fsx_extsize = fattr->fa_extsize;
|
||||
fa->fsx_projid = fattr->fa_projid;
|
||||
fa->fsx_cowextsize = fattr->fa_cowextsize;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
static int copy_fsxattr_from_user(struct file_kattr *fa,
|
||||
struct fsxattr __user *ufa)
|
||||
{
|
||||
struct fsxattr xfa;
|
||||
__u32 mask = FS_XFLAGS_MASK;
|
||||
|
||||
if (copy_from_user(&xfa, ufa, sizeof(xfa)))
|
||||
return -EFAULT;
|
||||
|
||||
if (xfa.fsx_xflags & ~mask)
|
||||
return -EOPNOTSUPP;
|
||||
|
||||
fileattr_fill_xflags(fa, xfa.fsx_xflags);
|
||||
fa->fsx_xflags &= ~FS_XFLAG_RDONLY_MASK;
|
||||
fa->fsx_extsize = xfa.fsx_extsize;
|
||||
fa->fsx_nextents = xfa.fsx_nextents;
|
||||
fa->fsx_projid = xfa.fsx_projid;
|
||||
fa->fsx_cowextsize = xfa.fsx_cowextsize;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* Generic function to check FS_IOC_FSSETXATTR/FS_IOC_SETFLAGS values and reject
|
||||
* any invalid configurations.
|
||||
*
|
||||
* Note: must be called with inode lock held.
|
||||
*/
|
||||
static int fileattr_set_prepare(struct inode *inode,
|
||||
const struct file_kattr *old_ma,
|
||||
struct file_kattr *fa)
|
||||
{
|
||||
int err;
|
||||
|
||||
/*
|
||||
* The IMMUTABLE and APPEND_ONLY flags can only be changed by
|
||||
* the relevant capability.
|
||||
*/
|
||||
if ((fa->flags ^ old_ma->flags) & (FS_APPEND_FL | FS_IMMUTABLE_FL) &&
|
||||
!capable(CAP_LINUX_IMMUTABLE))
|
||||
return -EPERM;
|
||||
|
||||
err = fscrypt_prepare_setflags(inode, old_ma->flags, fa->flags);
|
||||
if (err)
|
||||
return err;
|
||||
|
||||
/*
|
||||
* Project Quota ID state is only allowed to change from within the init
|
||||
* namespace. Enforce that restriction only if we are trying to change
|
||||
* the quota ID state. Everything else is allowed in user namespaces.
|
||||
*/
|
||||
if (current_user_ns() != &init_user_ns) {
|
||||
if (old_ma->fsx_projid != fa->fsx_projid)
|
||||
return -EINVAL;
|
||||
if ((old_ma->fsx_xflags ^ fa->fsx_xflags) &
|
||||
FS_XFLAG_PROJINHERIT)
|
||||
return -EINVAL;
|
||||
} else {
|
||||
/*
|
||||
* Caller is allowed to change the project ID. If it is being
|
||||
* changed, make sure that the new value is valid.
|
||||
*/
|
||||
if (old_ma->fsx_projid != fa->fsx_projid &&
|
||||
!projid_valid(make_kprojid(&init_user_ns, fa->fsx_projid)))
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
/* Check extent size hints. */
|
||||
if ((fa->fsx_xflags & FS_XFLAG_EXTSIZE) && !S_ISREG(inode->i_mode))
|
||||
return -EINVAL;
|
||||
|
||||
if ((fa->fsx_xflags & FS_XFLAG_EXTSZINHERIT) &&
|
||||
!S_ISDIR(inode->i_mode))
|
||||
return -EINVAL;
|
||||
|
||||
if ((fa->fsx_xflags & FS_XFLAG_COWEXTSIZE) &&
|
||||
!S_ISREG(inode->i_mode) && !S_ISDIR(inode->i_mode))
|
||||
return -EINVAL;
|
||||
|
||||
/*
|
||||
* It is only valid to set the DAX flag on regular files and
|
||||
* directories on filesystems.
|
||||
*/
|
||||
if ((fa->fsx_xflags & FS_XFLAG_DAX) &&
|
||||
!(S_ISREG(inode->i_mode) || S_ISDIR(inode->i_mode)))
|
||||
return -EINVAL;
|
||||
|
||||
/* Extent size hints of zero turn off the flags. */
|
||||
if (fa->fsx_extsize == 0)
|
||||
fa->fsx_xflags &= ~(FS_XFLAG_EXTSIZE | FS_XFLAG_EXTSZINHERIT);
|
||||
if (fa->fsx_cowextsize == 0)
|
||||
fa->fsx_xflags &= ~FS_XFLAG_COWEXTSIZE;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* vfs_fileattr_set - change miscellaneous file attributes
|
||||
* @idmap: idmap of the mount
|
||||
* @dentry: the object to change
|
||||
* @fa: fileattr pointer
|
||||
*
|
||||
* After verifying permissions, call i_op->fileattr_set() callback, if
|
||||
* exists.
|
||||
*
|
||||
* Verifying attributes involves retrieving current attributes with
|
||||
* i_op->fileattr_get(), this also allows initializing attributes that have
|
||||
* not been set by the caller to current values. Inode lock is held
|
||||
* thoughout to prevent racing with another instance.
|
||||
*
|
||||
* Return: 0 on success, or a negative error on failure.
|
||||
*/
|
||||
int vfs_fileattr_set(struct mnt_idmap *idmap, struct dentry *dentry,
|
||||
struct file_kattr *fa)
|
||||
{
|
||||
struct inode *inode = d_inode(dentry);
|
||||
struct file_kattr old_ma = {};
|
||||
int err;
|
||||
|
||||
if (!inode->i_op->fileattr_set)
|
||||
return -EOPNOTSUPP;
|
||||
|
||||
if (!inode_owner_or_capable(idmap, inode))
|
||||
return -EPERM;
|
||||
|
||||
inode_lock(inode);
|
||||
err = vfs_fileattr_get(dentry, &old_ma);
|
||||
if (!err) {
|
||||
/* initialize missing bits from old_ma */
|
||||
if (fa->flags_valid) {
|
||||
fa->fsx_xflags |= old_ma.fsx_xflags & ~FS_XFLAG_COMMON;
|
||||
fa->fsx_extsize = old_ma.fsx_extsize;
|
||||
fa->fsx_nextents = old_ma.fsx_nextents;
|
||||
fa->fsx_projid = old_ma.fsx_projid;
|
||||
fa->fsx_cowextsize = old_ma.fsx_cowextsize;
|
||||
} else {
|
||||
fa->flags |= old_ma.flags & ~FS_COMMON_FL;
|
||||
}
|
||||
|
||||
err = fileattr_set_prepare(inode, &old_ma, fa);
|
||||
if (err)
|
||||
goto out;
|
||||
err = security_inode_file_setattr(dentry, fa);
|
||||
if (err)
|
||||
goto out;
|
||||
err = inode->i_op->fileattr_set(idmap, dentry, fa);
|
||||
if (err)
|
||||
goto out;
|
||||
}
|
||||
|
||||
out:
|
||||
inode_unlock(inode);
|
||||
return err;
|
||||
}
|
||||
EXPORT_SYMBOL(vfs_fileattr_set);
|
||||
|
||||
int ioctl_getflags(struct file *file, unsigned int __user *argp)
|
||||
{
|
||||
struct file_kattr fa = { .flags_valid = true }; /* hint only */
|
||||
int err;
|
||||
|
||||
err = vfs_fileattr_get(file->f_path.dentry, &fa);
|
||||
if (err == -EOPNOTSUPP)
|
||||
err = -ENOIOCTLCMD;
|
||||
if (!err)
|
||||
err = put_user(fa.flags, argp);
|
||||
return err;
|
||||
}
|
||||
EXPORT_SYMBOL(ioctl_getflags);
|
||||
|
||||
int ioctl_setflags(struct file *file, unsigned int __user *argp)
|
||||
{
|
||||
struct mnt_idmap *idmap = file_mnt_idmap(file);
|
||||
struct dentry *dentry = file->f_path.dentry;
|
||||
struct file_kattr fa;
|
||||
unsigned int flags;
|
||||
int err;
|
||||
|
||||
err = get_user(flags, argp);
|
||||
if (!err) {
|
||||
err = mnt_want_write_file(file);
|
||||
if (!err) {
|
||||
fileattr_fill_flags(&fa, flags);
|
||||
err = vfs_fileattr_set(idmap, dentry, &fa);
|
||||
mnt_drop_write_file(file);
|
||||
if (err == -EOPNOTSUPP)
|
||||
err = -ENOIOCTLCMD;
|
||||
}
|
||||
}
|
||||
return err;
|
||||
}
|
||||
EXPORT_SYMBOL(ioctl_setflags);
|
||||
|
||||
int ioctl_fsgetxattr(struct file *file, void __user *argp)
|
||||
{
|
||||
struct file_kattr fa = { .fsx_valid = true }; /* hint only */
|
||||
int err;
|
||||
|
||||
err = vfs_fileattr_get(file->f_path.dentry, &fa);
|
||||
if (err == -EOPNOTSUPP)
|
||||
err = -ENOIOCTLCMD;
|
||||
if (!err)
|
||||
err = copy_fsxattr_to_user(&fa, argp);
|
||||
|
||||
return err;
|
||||
}
|
||||
EXPORT_SYMBOL(ioctl_fsgetxattr);
|
||||
|
||||
int ioctl_fssetxattr(struct file *file, void __user *argp)
|
||||
{
|
||||
struct mnt_idmap *idmap = file_mnt_idmap(file);
|
||||
struct dentry *dentry = file->f_path.dentry;
|
||||
struct file_kattr fa;
|
||||
int err;
|
||||
|
||||
err = copy_fsxattr_from_user(&fa, argp);
|
||||
if (!err) {
|
||||
err = mnt_want_write_file(file);
|
||||
if (!err) {
|
||||
err = vfs_fileattr_set(idmap, dentry, &fa);
|
||||
mnt_drop_write_file(file);
|
||||
if (err == -EOPNOTSUPP)
|
||||
err = -ENOIOCTLCMD;
|
||||
}
|
||||
}
|
||||
return err;
|
||||
}
|
||||
EXPORT_SYMBOL(ioctl_fssetxattr);
|
||||
|
||||
SYSCALL_DEFINE5(file_getattr, int, dfd, const char __user *, filename,
|
||||
struct file_attr __user *, ufattr, size_t, usize,
|
||||
unsigned int, at_flags)
|
||||
{
|
||||
struct path filepath __free(path_put) = {};
|
||||
struct filename *name __free(putname) = NULL;
|
||||
unsigned int lookup_flags = 0;
|
||||
struct file_attr fattr;
|
||||
struct file_kattr fa;
|
||||
int error;
|
||||
|
||||
BUILD_BUG_ON(sizeof(struct file_attr) < FILE_ATTR_SIZE_VER0);
|
||||
BUILD_BUG_ON(sizeof(struct file_attr) != FILE_ATTR_SIZE_LATEST);
|
||||
|
||||
if ((at_flags & ~(AT_SYMLINK_NOFOLLOW | AT_EMPTY_PATH)) != 0)
|
||||
return -EINVAL;
|
||||
|
||||
if (!(at_flags & AT_SYMLINK_NOFOLLOW))
|
||||
lookup_flags |= LOOKUP_FOLLOW;
|
||||
|
||||
if (usize > PAGE_SIZE)
|
||||
return -E2BIG;
|
||||
|
||||
if (usize < FILE_ATTR_SIZE_VER0)
|
||||
return -EINVAL;
|
||||
|
||||
name = getname_maybe_null(filename, at_flags);
|
||||
if (IS_ERR(name))
|
||||
return PTR_ERR(name);
|
||||
|
||||
if (!name && dfd >= 0) {
|
||||
CLASS(fd, f)(dfd);
|
||||
if (fd_empty(f))
|
||||
return -EBADF;
|
||||
|
||||
filepath = fd_file(f)->f_path;
|
||||
path_get(&filepath);
|
||||
} else {
|
||||
error = filename_lookup(dfd, name, lookup_flags, &filepath,
|
||||
NULL);
|
||||
if (error)
|
||||
return error;
|
||||
}
|
||||
|
||||
error = vfs_fileattr_get(filepath.dentry, &fa);
|
||||
if (error)
|
||||
return error;
|
||||
|
||||
fileattr_to_file_attr(&fa, &fattr);
|
||||
error = copy_struct_to_user(ufattr, usize, &fattr,
|
||||
sizeof(struct file_attr), NULL);
|
||||
|
||||
return error;
|
||||
}
|
||||
|
||||
SYSCALL_DEFINE5(file_setattr, int, dfd, const char __user *, filename,
|
||||
struct file_attr __user *, ufattr, size_t, usize,
|
||||
unsigned int, at_flags)
|
||||
{
|
||||
struct path filepath __free(path_put) = {};
|
||||
struct filename *name __free(putname) = NULL;
|
||||
unsigned int lookup_flags = 0;
|
||||
struct file_attr fattr;
|
||||
struct file_kattr fa;
|
||||
int error;
|
||||
|
||||
BUILD_BUG_ON(sizeof(struct file_attr) < FILE_ATTR_SIZE_VER0);
|
||||
BUILD_BUG_ON(sizeof(struct file_attr) != FILE_ATTR_SIZE_LATEST);
|
||||
|
||||
if ((at_flags & ~(AT_SYMLINK_NOFOLLOW | AT_EMPTY_PATH)) != 0)
|
||||
return -EINVAL;
|
||||
|
||||
if (!(at_flags & AT_SYMLINK_NOFOLLOW))
|
||||
lookup_flags |= LOOKUP_FOLLOW;
|
||||
|
||||
if (usize > PAGE_SIZE)
|
||||
return -E2BIG;
|
||||
|
||||
if (usize < FILE_ATTR_SIZE_VER0)
|
||||
return -EINVAL;
|
||||
|
||||
error = copy_struct_from_user(&fattr, sizeof(struct file_attr), ufattr,
|
||||
usize);
|
||||
if (error)
|
||||
return error;
|
||||
|
||||
error = file_attr_to_fileattr(&fattr, &fa);
|
||||
if (error)
|
||||
return error;
|
||||
|
||||
name = getname_maybe_null(filename, at_flags);
|
||||
if (IS_ERR(name))
|
||||
return PTR_ERR(name);
|
||||
|
||||
if (!name && dfd >= 0) {
|
||||
CLASS(fd, f)(dfd);
|
||||
if (fd_empty(f))
|
||||
return -EBADF;
|
||||
|
||||
filepath = fd_file(f)->f_path;
|
||||
path_get(&filepath);
|
||||
} else {
|
||||
error = filename_lookup(dfd, name, lookup_flags, &filepath,
|
||||
NULL);
|
||||
if (error)
|
||||
return error;
|
||||
}
|
||||
|
||||
error = mnt_want_write(filepath.mnt);
|
||||
if (!error) {
|
||||
error = vfs_fileattr_set(mnt_idmap(filepath.mnt),
|
||||
filepath.dentry, &fa);
|
||||
mnt_drop_write(filepath.mnt);
|
||||
}
|
||||
|
||||
return error;
|
||||
}
|
@ -1479,9 +1479,9 @@ void fuse_dax_cancel_work(struct fuse_conn *fc);
|
||||
long fuse_file_ioctl(struct file *file, unsigned int cmd, unsigned long arg);
|
||||
long fuse_file_compat_ioctl(struct file *file, unsigned int cmd,
|
||||
unsigned long arg);
|
||||
int fuse_fileattr_get(struct dentry *dentry, struct fileattr *fa);
|
||||
int fuse_fileattr_get(struct dentry *dentry, struct file_kattr *fa);
|
||||
int fuse_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa);
|
||||
struct dentry *dentry, struct file_kattr *fa);
|
||||
|
||||
/* iomode.c */
|
||||
int fuse_file_cached_io_open(struct inode *inode, struct fuse_file *ff);
|
||||
|
@ -502,7 +502,7 @@ static void fuse_priv_ioctl_cleanup(struct inode *inode, struct fuse_file *ff)
|
||||
fuse_file_release(inode, ff, O_RDONLY, NULL, S_ISDIR(inode->i_mode));
|
||||
}
|
||||
|
||||
int fuse_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
int fuse_fileattr_get(struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
struct inode *inode = d_inode(dentry);
|
||||
struct fuse_file *ff;
|
||||
@ -536,11 +536,13 @@ int fuse_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
cleanup:
|
||||
fuse_priv_ioctl_cleanup(inode, ff);
|
||||
|
||||
if (err == -ENOTTY)
|
||||
err = -EOPNOTSUPP;
|
||||
return err;
|
||||
}
|
||||
|
||||
int fuse_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa)
|
||||
struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
struct inode *inode = d_inode(dentry);
|
||||
struct fuse_file *ff;
|
||||
@ -572,5 +574,7 @@ int fuse_fileattr_set(struct mnt_idmap *idmap,
|
||||
cleanup:
|
||||
fuse_priv_ioctl_cleanup(inode, ff);
|
||||
|
||||
if (err == -ENOTTY)
|
||||
err = -EOPNOTSUPP;
|
||||
return err;
|
||||
}
|
||||
|
@ -155,7 +155,7 @@ static inline u32 gfs2_gfsflags_to_fsflags(struct inode *inode, u32 gfsflags)
|
||||
return fsflags;
|
||||
}
|
||||
|
||||
int gfs2_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
int gfs2_fileattr_get(struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
struct inode *inode = d_inode(dentry);
|
||||
struct gfs2_inode *ip = GFS2_I(inode);
|
||||
@ -276,7 +276,7 @@ out:
|
||||
}
|
||||
|
||||
int gfs2_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa)
|
||||
struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
struct inode *inode = d_inode(dentry);
|
||||
u32 fsflags = fa->flags, gfsflags = 0;
|
||||
|
@ -107,9 +107,9 @@ loff_t gfs2_seek_hole(struct file *file, loff_t offset);
|
||||
extern const struct file_operations gfs2_file_fops_nolock;
|
||||
extern const struct file_operations gfs2_dir_fops_nolock;
|
||||
|
||||
int gfs2_fileattr_get(struct dentry *dentry, struct fileattr *fa);
|
||||
int gfs2_fileattr_get(struct dentry *dentry, struct file_kattr *fa);
|
||||
int gfs2_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa);
|
||||
struct dentry *dentry, struct file_kattr *fa);
|
||||
void gfs2_set_inode_flags(struct inode *inode);
|
||||
|
||||
#ifdef CONFIG_GFS2_FS_LOCKING_DLM
|
||||
|
@ -491,9 +491,9 @@ int hfsplus_getattr(struct mnt_idmap *idmap, const struct path *path,
|
||||
unsigned int query_flags);
|
||||
int hfsplus_file_fsync(struct file *file, loff_t start, loff_t end,
|
||||
int datasync);
|
||||
int hfsplus_fileattr_get(struct dentry *dentry, struct fileattr *fa);
|
||||
int hfsplus_fileattr_get(struct dentry *dentry, struct file_kattr *fa);
|
||||
int hfsplus_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa);
|
||||
struct dentry *dentry, struct file_kattr *fa);
|
||||
|
||||
/* ioctl.c */
|
||||
long hfsplus_ioctl(struct file *filp, unsigned int cmd, unsigned long arg);
|
||||
|
@ -656,7 +656,7 @@ out:
|
||||
return res;
|
||||
}
|
||||
|
||||
int hfsplus_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
int hfsplus_fileattr_get(struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
struct inode *inode = d_inode(dentry);
|
||||
struct hfsplus_inode_info *hip = HFSPLUS_I(inode);
|
||||
@ -675,7 +675,7 @@ int hfsplus_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
}
|
||||
|
||||
int hfsplus_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa)
|
||||
struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
struct inode *inode = d_inode(dentry);
|
||||
struct hfsplus_inode_info *hip = HFSPLUS_I(inode);
|
||||
|
309
fs/ioctl.c
309
fs/ioctl.c
@ -453,315 +453,6 @@ out:
|
||||
return ret;
|
||||
}
|
||||
|
||||
/**
|
||||
* fileattr_fill_xflags - initialize fileattr with xflags
|
||||
* @fa: fileattr pointer
|
||||
* @xflags: FS_XFLAG_* flags
|
||||
*
|
||||
* Set ->fsx_xflags, ->fsx_valid and ->flags (translated xflags). All
|
||||
* other fields are zeroed.
|
||||
*/
|
||||
void fileattr_fill_xflags(struct fileattr *fa, u32 xflags)
|
||||
{
|
||||
memset(fa, 0, sizeof(*fa));
|
||||
fa->fsx_valid = true;
|
||||
fa->fsx_xflags = xflags;
|
||||
if (fa->fsx_xflags & FS_XFLAG_IMMUTABLE)
|
||||
fa->flags |= FS_IMMUTABLE_FL;
|
||||
if (fa->fsx_xflags & FS_XFLAG_APPEND)
|
||||
fa->flags |= FS_APPEND_FL;
|
||||
if (fa->fsx_xflags & FS_XFLAG_SYNC)
|
||||
fa->flags |= FS_SYNC_FL;
|
||||
if (fa->fsx_xflags & FS_XFLAG_NOATIME)
|
||||
fa->flags |= FS_NOATIME_FL;
|
||||
if (fa->fsx_xflags & FS_XFLAG_NODUMP)
|
||||
fa->flags |= FS_NODUMP_FL;
|
||||
if (fa->fsx_xflags & FS_XFLAG_DAX)
|
||||
fa->flags |= FS_DAX_FL;
|
||||
if (fa->fsx_xflags & FS_XFLAG_PROJINHERIT)
|
||||
fa->flags |= FS_PROJINHERIT_FL;
|
||||
}
|
||||
EXPORT_SYMBOL(fileattr_fill_xflags);
|
||||
|
||||
/**
|
||||
* fileattr_fill_flags - initialize fileattr with flags
|
||||
* @fa: fileattr pointer
|
||||
* @flags: FS_*_FL flags
|
||||
*
|
||||
* Set ->flags, ->flags_valid and ->fsx_xflags (translated flags).
|
||||
* All other fields are zeroed.
|
||||
*/
|
||||
void fileattr_fill_flags(struct fileattr *fa, u32 flags)
|
||||
{
|
||||
memset(fa, 0, sizeof(*fa));
|
||||
fa->flags_valid = true;
|
||||
fa->flags = flags;
|
||||
if (fa->flags & FS_SYNC_FL)
|
||||
fa->fsx_xflags |= FS_XFLAG_SYNC;
|
||||
if (fa->flags & FS_IMMUTABLE_FL)
|
||||
fa->fsx_xflags |= FS_XFLAG_IMMUTABLE;
|
||||
if (fa->flags & FS_APPEND_FL)
|
||||
fa->fsx_xflags |= FS_XFLAG_APPEND;
|
||||
if (fa->flags & FS_NODUMP_FL)
|
||||
fa->fsx_xflags |= FS_XFLAG_NODUMP;
|
||||
if (fa->flags & FS_NOATIME_FL)
|
||||
fa->fsx_xflags |= FS_XFLAG_NOATIME;
|
||||
if (fa->flags & FS_DAX_FL)
|
||||
fa->fsx_xflags |= FS_XFLAG_DAX;
|
||||
if (fa->flags & FS_PROJINHERIT_FL)
|
||||
fa->fsx_xflags |= FS_XFLAG_PROJINHERIT;
|
||||
}
|
||||
EXPORT_SYMBOL(fileattr_fill_flags);
|
||||
|
||||
/**
|
||||
* vfs_fileattr_get - retrieve miscellaneous file attributes
|
||||
* @dentry: the object to retrieve from
|
||||
* @fa: fileattr pointer
|
||||
*
|
||||
* Call i_op->fileattr_get() callback, if exists.
|
||||
*
|
||||
* Return: 0 on success, or a negative error on failure.
|
||||
*/
|
||||
int vfs_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
{
|
||||
struct inode *inode = d_inode(dentry);
|
||||
|
||||
if (!inode->i_op->fileattr_get)
|
||||
return -ENOIOCTLCMD;
|
||||
|
||||
return inode->i_op->fileattr_get(dentry, fa);
|
||||
}
|
||||
EXPORT_SYMBOL(vfs_fileattr_get);
|
||||
|
||||
/**
|
||||
* copy_fsxattr_to_user - copy fsxattr to userspace.
|
||||
* @fa: fileattr pointer
|
||||
* @ufa: fsxattr user pointer
|
||||
*
|
||||
* Return: 0 on success, or -EFAULT on failure.
|
||||
*/
|
||||
int copy_fsxattr_to_user(const struct fileattr *fa, struct fsxattr __user *ufa)
|
||||
{
|
||||
struct fsxattr xfa;
|
||||
|
||||
memset(&xfa, 0, sizeof(xfa));
|
||||
xfa.fsx_xflags = fa->fsx_xflags;
|
||||
xfa.fsx_extsize = fa->fsx_extsize;
|
||||
xfa.fsx_nextents = fa->fsx_nextents;
|
||||
xfa.fsx_projid = fa->fsx_projid;
|
||||
xfa.fsx_cowextsize = fa->fsx_cowextsize;
|
||||
|
||||
if (copy_to_user(ufa, &xfa, sizeof(xfa)))
|
||||
return -EFAULT;
|
||||
|
||||
return 0;
|
||||
}
|
||||
EXPORT_SYMBOL(copy_fsxattr_to_user);
|
||||
|
||||
static int copy_fsxattr_from_user(struct fileattr *fa,
|
||||
struct fsxattr __user *ufa)
|
||||
{
|
||||
struct fsxattr xfa;
|
||||
|
||||
if (copy_from_user(&xfa, ufa, sizeof(xfa)))
|
||||
return -EFAULT;
|
||||
|
||||
fileattr_fill_xflags(fa, xfa.fsx_xflags);
|
||||
fa->fsx_extsize = xfa.fsx_extsize;
|
||||
fa->fsx_nextents = xfa.fsx_nextents;
|
||||
fa->fsx_projid = xfa.fsx_projid;
|
||||
fa->fsx_cowextsize = xfa.fsx_cowextsize;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/*
|
||||
* Generic function to check FS_IOC_FSSETXATTR/FS_IOC_SETFLAGS values and reject
|
||||
* any invalid configurations.
|
||||
*
|
||||
* Note: must be called with inode lock held.
|
||||
*/
|
||||
static int fileattr_set_prepare(struct inode *inode,
|
||||
const struct fileattr *old_ma,
|
||||
struct fileattr *fa)
|
||||
{
|
||||
int err;
|
||||
|
||||
/*
|
||||
* The IMMUTABLE and APPEND_ONLY flags can only be changed by
|
||||
* the relevant capability.
|
||||
*/
|
||||
if ((fa->flags ^ old_ma->flags) & (FS_APPEND_FL | FS_IMMUTABLE_FL) &&
|
||||
!capable(CAP_LINUX_IMMUTABLE))
|
||||
return -EPERM;
|
||||
|
||||
err = fscrypt_prepare_setflags(inode, old_ma->flags, fa->flags);
|
||||
if (err)
|
||||
return err;
|
||||
|
||||
/*
|
||||
* Project Quota ID state is only allowed to change from within the init
|
||||
* namespace. Enforce that restriction only if we are trying to change
|
||||
* the quota ID state. Everything else is allowed in user namespaces.
|
||||
*/
|
||||
if (current_user_ns() != &init_user_ns) {
|
||||
if (old_ma->fsx_projid != fa->fsx_projid)
|
||||
return -EINVAL;
|
||||
if ((old_ma->fsx_xflags ^ fa->fsx_xflags) &
|
||||
FS_XFLAG_PROJINHERIT)
|
||||
return -EINVAL;
|
||||
} else {
|
||||
/*
|
||||
* Caller is allowed to change the project ID. If it is being
|
||||
* changed, make sure that the new value is valid.
|
||||
*/
|
||||
if (old_ma->fsx_projid != fa->fsx_projid &&
|
||||
!projid_valid(make_kprojid(&init_user_ns, fa->fsx_projid)))
|
||||
return -EINVAL;
|
||||
}
|
||||
|
||||
/* Check extent size hints. */
|
||||
if ((fa->fsx_xflags & FS_XFLAG_EXTSIZE) && !S_ISREG(inode->i_mode))
|
||||
return -EINVAL;
|
||||
|
||||
if ((fa->fsx_xflags & FS_XFLAG_EXTSZINHERIT) &&
|
||||
!S_ISDIR(inode->i_mode))
|
||||
return -EINVAL;
|
||||
|
||||
if ((fa->fsx_xflags & FS_XFLAG_COWEXTSIZE) &&
|
||||
!S_ISREG(inode->i_mode) && !S_ISDIR(inode->i_mode))
|
||||
return -EINVAL;
|
||||
|
||||
/*
|
||||
* It is only valid to set the DAX flag on regular files and
|
||||
* directories on filesystems.
|
||||
*/
|
||||
if ((fa->fsx_xflags & FS_XFLAG_DAX) &&
|
||||
!(S_ISREG(inode->i_mode) || S_ISDIR(inode->i_mode)))
|
||||
return -EINVAL;
|
||||
|
||||
/* Extent size hints of zero turn off the flags. */
|
||||
if (fa->fsx_extsize == 0)
|
||||
fa->fsx_xflags &= ~(FS_XFLAG_EXTSIZE | FS_XFLAG_EXTSZINHERIT);
|
||||
if (fa->fsx_cowextsize == 0)
|
||||
fa->fsx_xflags &= ~FS_XFLAG_COWEXTSIZE;
|
||||
|
||||
return 0;
|
||||
}
|
||||
|
||||
/**
|
||||
* vfs_fileattr_set - change miscellaneous file attributes
|
||||
* @idmap: idmap of the mount
|
||||
* @dentry: the object to change
|
||||
* @fa: fileattr pointer
|
||||
*
|
||||
* After verifying permissions, call i_op->fileattr_set() callback, if
|
||||
* exists.
|
||||
*
|
||||
* Verifying attributes involves retrieving current attributes with
|
||||
* i_op->fileattr_get(), this also allows initializing attributes that have
|
||||
* not been set by the caller to current values. Inode lock is held
|
||||
* thoughout to prevent racing with another instance.
|
||||
*
|
||||
* Return: 0 on success, or a negative error on failure.
|
||||
*/
|
||||
int vfs_fileattr_set(struct mnt_idmap *idmap, struct dentry *dentry,
|
||||
struct fileattr *fa)
|
||||
{
|
||||
struct inode *inode = d_inode(dentry);
|
||||
struct fileattr old_ma = {};
|
||||
int err;
|
||||
|
||||
if (!inode->i_op->fileattr_set)
|
||||
return -ENOIOCTLCMD;
|
||||
|
||||
if (!inode_owner_or_capable(idmap, inode))
|
||||
return -EPERM;
|
||||
|
||||
inode_lock(inode);
|
||||
err = vfs_fileattr_get(dentry, &old_ma);
|
||||
if (!err) {
|
||||
/* initialize missing bits from old_ma */
|
||||
if (fa->flags_valid) {
|
||||
fa->fsx_xflags |= old_ma.fsx_xflags & ~FS_XFLAG_COMMON;
|
||||
fa->fsx_extsize = old_ma.fsx_extsize;
|
||||
fa->fsx_nextents = old_ma.fsx_nextents;
|
||||
fa->fsx_projid = old_ma.fsx_projid;
|
||||
fa->fsx_cowextsize = old_ma.fsx_cowextsize;
|
||||
} else {
|
||||
fa->flags |= old_ma.flags & ~FS_COMMON_FL;
|
||||
}
|
||||
err = fileattr_set_prepare(inode, &old_ma, fa);
|
||||
if (!err)
|
||||
err = inode->i_op->fileattr_set(idmap, dentry, fa);
|
||||
}
|
||||
inode_unlock(inode);
|
||||
|
||||
return err;
|
||||
}
|
||||
EXPORT_SYMBOL(vfs_fileattr_set);
|
||||
|
||||
static int ioctl_getflags(struct file *file, unsigned int __user *argp)
|
||||
{
|
||||
struct fileattr fa = { .flags_valid = true }; /* hint only */
|
||||
int err;
|
||||
|
||||
err = vfs_fileattr_get(file->f_path.dentry, &fa);
|
||||
if (!err)
|
||||
err = put_user(fa.flags, argp);
|
||||
return err;
|
||||
}
|
||||
|
||||
static int ioctl_setflags(struct file *file, unsigned int __user *argp)
|
||||
{
|
||||
struct mnt_idmap *idmap = file_mnt_idmap(file);
|
||||
struct dentry *dentry = file->f_path.dentry;
|
||||
struct fileattr fa;
|
||||
unsigned int flags;
|
||||
int err;
|
||||
|
||||
err = get_user(flags, argp);
|
||||
if (!err) {
|
||||
err = mnt_want_write_file(file);
|
||||
if (!err) {
|
||||
fileattr_fill_flags(&fa, flags);
|
||||
err = vfs_fileattr_set(idmap, dentry, &fa);
|
||||
mnt_drop_write_file(file);
|
||||
}
|
||||
}
|
||||
return err;
|
||||
}
|
||||
|
||||
static int ioctl_fsgetxattr(struct file *file, void __user *argp)
|
||||
{
|
||||
struct fileattr fa = { .fsx_valid = true }; /* hint only */
|
||||
int err;
|
||||
|
||||
err = vfs_fileattr_get(file->f_path.dentry, &fa);
|
||||
if (!err)
|
||||
err = copy_fsxattr_to_user(&fa, argp);
|
||||
|
||||
return err;
|
||||
}
|
||||
|
||||
static int ioctl_fssetxattr(struct file *file, void __user *argp)
|
||||
{
|
||||
struct mnt_idmap *idmap = file_mnt_idmap(file);
|
||||
struct dentry *dentry = file->f_path.dentry;
|
||||
struct fileattr fa;
|
||||
int err;
|
||||
|
||||
err = copy_fsxattr_from_user(&fa, argp);
|
||||
if (!err) {
|
||||
err = mnt_want_write_file(file);
|
||||
if (!err) {
|
||||
err = vfs_fileattr_set(idmap, dentry, &fa);
|
||||
mnt_drop_write_file(file);
|
||||
}
|
||||
}
|
||||
return err;
|
||||
}
|
||||
|
||||
static int ioctl_getfsuuid(struct file *file, void __user *argp)
|
||||
{
|
||||
struct super_block *sb = file_inode(file)->i_sb;
|
||||
|
@ -57,7 +57,7 @@ static long jfs_map_ext2(unsigned long flags, int from)
|
||||
return mapped;
|
||||
}
|
||||
|
||||
int jfs_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
int jfs_fileattr_get(struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
struct jfs_inode_info *jfs_inode = JFS_IP(d_inode(dentry));
|
||||
unsigned int flags = jfs_inode->mode2 & JFS_FL_USER_VISIBLE;
|
||||
@ -71,7 +71,7 @@ int jfs_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
}
|
||||
|
||||
int jfs_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa)
|
||||
struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
struct inode *inode = d_inode(dentry);
|
||||
struct jfs_inode_info *jfs_inode = JFS_IP(inode);
|
||||
|
@ -9,9 +9,9 @@ struct fid;
|
||||
|
||||
extern struct inode *ialloc(struct inode *, umode_t);
|
||||
extern int jfs_fsync(struct file *, loff_t, loff_t, int);
|
||||
extern int jfs_fileattr_get(struct dentry *dentry, struct fileattr *fa);
|
||||
extern int jfs_fileattr_get(struct dentry *dentry, struct file_kattr *fa);
|
||||
extern int jfs_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa);
|
||||
struct dentry *dentry, struct file_kattr *fa);
|
||||
extern long jfs_ioctl(struct file *, unsigned int, unsigned long);
|
||||
extern struct inode *jfs_iget(struct super_block *, unsigned long);
|
||||
extern int jfs_commit_inode(struct inode *, int);
|
||||
|
@ -118,7 +118,7 @@ static int nilfs_ioctl_wrap_copy(struct the_nilfs *nilfs,
|
||||
*
|
||||
* Return: always 0 as success.
|
||||
*/
|
||||
int nilfs_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
int nilfs_fileattr_get(struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
struct inode *inode = d_inode(dentry);
|
||||
|
||||
@ -136,7 +136,7 @@ int nilfs_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
* Return: 0 on success, or a negative error code on failure.
|
||||
*/
|
||||
int nilfs_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa)
|
||||
struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
struct inode *inode = d_inode(dentry);
|
||||
struct nilfs_transaction_info ti;
|
||||
|
@ -268,9 +268,9 @@ int nilfs_set_link(struct inode *dir, struct nilfs_dir_entry *de,
|
||||
extern int nilfs_sync_file(struct file *, loff_t, loff_t, int);
|
||||
|
||||
/* ioctl.c */
|
||||
int nilfs_fileattr_get(struct dentry *dentry, struct fileattr *m);
|
||||
int nilfs_fileattr_get(struct dentry *dentry, struct file_kattr *m);
|
||||
int nilfs_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa);
|
||||
struct dentry *dentry, struct file_kattr *fa);
|
||||
long nilfs_ioctl(struct file *, unsigned int, unsigned long);
|
||||
long nilfs_compat_ioctl(struct file *file, unsigned int cmd, unsigned long arg);
|
||||
int nilfs_ioctl_prepare_clean_segments(struct the_nilfs *, struct nilfs_argv *,
|
||||
|
@ -62,7 +62,7 @@ static inline int o2info_coherent(struct ocfs2_info_request *req)
|
||||
return (!(req->ir_flags & OCFS2_INFO_FL_NON_COHERENT));
|
||||
}
|
||||
|
||||
int ocfs2_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
int ocfs2_fileattr_get(struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
struct inode *inode = d_inode(dentry);
|
||||
unsigned int flags;
|
||||
@ -83,7 +83,7 @@ int ocfs2_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
}
|
||||
|
||||
int ocfs2_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa)
|
||||
struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
struct inode *inode = d_inode(dentry);
|
||||
unsigned int flags = fa->flags;
|
||||
|
@ -11,9 +11,9 @@
|
||||
#ifndef OCFS2_IOCTL_PROTO_H
|
||||
#define OCFS2_IOCTL_PROTO_H
|
||||
|
||||
int ocfs2_fileattr_get(struct dentry *dentry, struct fileattr *fa);
|
||||
int ocfs2_fileattr_get(struct dentry *dentry, struct file_kattr *fa);
|
||||
int ocfs2_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa);
|
||||
struct dentry *dentry, struct file_kattr *fa);
|
||||
long ocfs2_ioctl(struct file *filp, unsigned int cmd, unsigned long arg);
|
||||
long ocfs2_compat_ioctl(struct file *file, unsigned cmd, unsigned long arg);
|
||||
|
||||
|
@ -889,7 +889,7 @@ int orangefs_update_time(struct inode *inode, int flags)
|
||||
return __orangefs_setattr(inode, &iattr);
|
||||
}
|
||||
|
||||
static int orangefs_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
static int orangefs_fileattr_get(struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
u64 val = 0;
|
||||
int ret;
|
||||
@ -910,7 +910,7 @@ static int orangefs_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
}
|
||||
|
||||
static int orangefs_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa)
|
||||
struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
u64 val = 0;
|
||||
|
||||
|
@ -171,14 +171,14 @@ out:
|
||||
static int ovl_copy_fileattr(struct inode *inode, const struct path *old,
|
||||
const struct path *new)
|
||||
{
|
||||
struct fileattr oldfa = { .flags_valid = true };
|
||||
struct fileattr newfa = { .flags_valid = true };
|
||||
struct file_kattr oldfa = { .flags_valid = true };
|
||||
struct file_kattr newfa = { .flags_valid = true };
|
||||
int err;
|
||||
|
||||
err = ovl_real_fileattr_get(old, &oldfa);
|
||||
if (err) {
|
||||
/* Ntfs-3g returns -EINVAL for "no fileattr support" */
|
||||
if (err == -ENOTTY || err == -EINVAL)
|
||||
if (err == -EOPNOTSUPP || err == -EINVAL)
|
||||
return 0;
|
||||
pr_warn("failed to retrieve lower fileattr (%pd2, err=%i)\n",
|
||||
old->dentry, err);
|
||||
|
@ -610,7 +610,7 @@ static int ovl_fiemap(struct inode *inode, struct fiemap_extent_info *fieinfo,
|
||||
* Introducing security_inode_fileattr_get/set() hooks would solve this issue
|
||||
* properly.
|
||||
*/
|
||||
static int ovl_security_fileattr(const struct path *realpath, struct fileattr *fa,
|
||||
static int ovl_security_fileattr(const struct path *realpath, struct file_kattr *fa,
|
||||
bool set)
|
||||
{
|
||||
struct file *file;
|
||||
@ -637,7 +637,7 @@ static int ovl_security_fileattr(const struct path *realpath, struct fileattr *f
|
||||
return err;
|
||||
}
|
||||
|
||||
int ovl_real_fileattr_set(const struct path *realpath, struct fileattr *fa)
|
||||
int ovl_real_fileattr_set(const struct path *realpath, struct file_kattr *fa)
|
||||
{
|
||||
int err;
|
||||
|
||||
@ -649,7 +649,7 @@ int ovl_real_fileattr_set(const struct path *realpath, struct fileattr *fa)
|
||||
}
|
||||
|
||||
int ovl_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa)
|
||||
struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
struct inode *inode = d_inode(dentry);
|
||||
struct path upperpath;
|
||||
@ -697,7 +697,7 @@ out:
|
||||
}
|
||||
|
||||
/* Convert inode protection flags to fileattr flags */
|
||||
static void ovl_fileattr_prot_flags(struct inode *inode, struct fileattr *fa)
|
||||
static void ovl_fileattr_prot_flags(struct inode *inode, struct file_kattr *fa)
|
||||
{
|
||||
BUILD_BUG_ON(OVL_PROT_FS_FLAGS_MASK & ~FS_COMMON_FL);
|
||||
BUILD_BUG_ON(OVL_PROT_FSX_FLAGS_MASK & ~FS_XFLAG_COMMON);
|
||||
@ -712,7 +712,7 @@ static void ovl_fileattr_prot_flags(struct inode *inode, struct fileattr *fa)
|
||||
}
|
||||
}
|
||||
|
||||
int ovl_real_fileattr_get(const struct path *realpath, struct fileattr *fa)
|
||||
int ovl_real_fileattr_get(const struct path *realpath, struct file_kattr *fa)
|
||||
{
|
||||
int err;
|
||||
|
||||
@ -720,13 +720,10 @@ int ovl_real_fileattr_get(const struct path *realpath, struct fileattr *fa)
|
||||
if (err)
|
||||
return err;
|
||||
|
||||
err = vfs_fileattr_get(realpath->dentry, fa);
|
||||
if (err == -ENOIOCTLCMD)
|
||||
err = -ENOTTY;
|
||||
return err;
|
||||
return vfs_fileattr_get(realpath->dentry, fa);
|
||||
}
|
||||
|
||||
int ovl_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
int ovl_fileattr_get(struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
struct inode *inode = d_inode(dentry);
|
||||
struct path realpath;
|
||||
|
@ -838,7 +838,7 @@ void ovl_copyattr(struct inode *to);
|
||||
|
||||
void ovl_check_protattr(struct inode *inode, struct dentry *upper);
|
||||
int ovl_set_protattr(struct inode *inode, struct dentry *upper,
|
||||
struct fileattr *fa);
|
||||
struct file_kattr *fa);
|
||||
|
||||
static inline void ovl_copyflags(struct inode *from, struct inode *to)
|
||||
{
|
||||
@ -870,11 +870,11 @@ struct dentry *ovl_create_temp(struct ovl_fs *ofs, struct dentry *workdir,
|
||||
|
||||
/* file.c */
|
||||
extern const struct file_operations ovl_file_operations;
|
||||
int ovl_real_fileattr_get(const struct path *realpath, struct fileattr *fa);
|
||||
int ovl_real_fileattr_set(const struct path *realpath, struct fileattr *fa);
|
||||
int ovl_fileattr_get(struct dentry *dentry, struct fileattr *fa);
|
||||
int ovl_real_fileattr_get(const struct path *realpath, struct file_kattr *fa);
|
||||
int ovl_real_fileattr_set(const struct path *realpath, struct file_kattr *fa);
|
||||
int ovl_fileattr_get(struct dentry *dentry, struct file_kattr *fa);
|
||||
int ovl_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa);
|
||||
struct dentry *dentry, struct file_kattr *fa);
|
||||
struct ovl_file;
|
||||
struct ovl_file *ovl_file_alloc(struct file *realfile);
|
||||
void ovl_file_free(struct ovl_file *of);
|
||||
|
@ -966,7 +966,7 @@ void ovl_check_protattr(struct inode *inode, struct dentry *upper)
|
||||
}
|
||||
|
||||
int ovl_set_protattr(struct inode *inode, struct dentry *upper,
|
||||
struct fileattr *fa)
|
||||
struct file_kattr *fa)
|
||||
{
|
||||
struct ovl_fs *ofs = OVL_FS(inode->i_sb);
|
||||
char buf[OVL_PROTATTR_MAX];
|
||||
|
@ -130,7 +130,7 @@ static int setflags(struct inode *inode, int flags)
|
||||
return err;
|
||||
}
|
||||
|
||||
int ubifs_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
int ubifs_fileattr_get(struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
struct inode *inode = d_inode(dentry);
|
||||
int flags = ubifs2ioctl(ubifs_inode(inode)->flags);
|
||||
@ -145,7 +145,7 @@ int ubifs_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
}
|
||||
|
||||
int ubifs_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa)
|
||||
struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
struct inode *inode = d_inode(dentry);
|
||||
int flags = fa->flags;
|
||||
|
@ -2073,9 +2073,9 @@ int ubifs_recover_size(struct ubifs_info *c, bool in_place);
|
||||
void ubifs_destroy_size_tree(struct ubifs_info *c);
|
||||
|
||||
/* ioctl.c */
|
||||
int ubifs_fileattr_get(struct dentry *dentry, struct fileattr *fa);
|
||||
int ubifs_fileattr_get(struct dentry *dentry, struct file_kattr *fa);
|
||||
int ubifs_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa);
|
||||
struct dentry *dentry, struct file_kattr *fa);
|
||||
long ubifs_ioctl(struct file *file, unsigned int cmd, unsigned long arg);
|
||||
void ubifs_set_inode_flags(struct inode *inode);
|
||||
#ifdef CONFIG_COMPAT
|
||||
|
@ -444,7 +444,7 @@ static void
|
||||
xfs_fill_fsxattr(
|
||||
struct xfs_inode *ip,
|
||||
int whichfork,
|
||||
struct fileattr *fa)
|
||||
struct file_kattr *fa)
|
||||
{
|
||||
struct xfs_mount *mp = ip->i_mount;
|
||||
struct xfs_ifork *ifp = xfs_ifork_ptr(ip, whichfork);
|
||||
@ -496,7 +496,7 @@ xfs_ioc_fsgetxattra(
|
||||
xfs_inode_t *ip,
|
||||
void __user *arg)
|
||||
{
|
||||
struct fileattr fa;
|
||||
struct file_kattr fa;
|
||||
|
||||
xfs_ilock(ip, XFS_ILOCK_SHARED);
|
||||
xfs_fill_fsxattr(ip, XFS_ATTR_FORK, &fa);
|
||||
@ -508,7 +508,7 @@ xfs_ioc_fsgetxattra(
|
||||
int
|
||||
xfs_fileattr_get(
|
||||
struct dentry *dentry,
|
||||
struct fileattr *fa)
|
||||
struct file_kattr *fa)
|
||||
{
|
||||
struct xfs_inode *ip = XFS_I(d_inode(dentry));
|
||||
|
||||
@ -526,7 +526,7 @@ static int
|
||||
xfs_ioctl_setattr_xflags(
|
||||
struct xfs_trans *tp,
|
||||
struct xfs_inode *ip,
|
||||
struct fileattr *fa)
|
||||
struct file_kattr *fa)
|
||||
{
|
||||
struct xfs_mount *mp = ip->i_mount;
|
||||
bool rtflag = (fa->fsx_xflags & FS_XFLAG_REALTIME);
|
||||
@ -582,7 +582,7 @@ xfs_ioctl_setattr_xflags(
|
||||
static void
|
||||
xfs_ioctl_setattr_prepare_dax(
|
||||
struct xfs_inode *ip,
|
||||
struct fileattr *fa)
|
||||
struct file_kattr *fa)
|
||||
{
|
||||
struct xfs_mount *mp = ip->i_mount;
|
||||
struct inode *inode = VFS_I(ip);
|
||||
@ -642,7 +642,7 @@ out_error:
|
||||
static int
|
||||
xfs_ioctl_setattr_check_extsize(
|
||||
struct xfs_inode *ip,
|
||||
struct fileattr *fa)
|
||||
struct file_kattr *fa)
|
||||
{
|
||||
struct xfs_mount *mp = ip->i_mount;
|
||||
xfs_failaddr_t failaddr;
|
||||
@ -684,7 +684,7 @@ xfs_ioctl_setattr_check_extsize(
|
||||
static int
|
||||
xfs_ioctl_setattr_check_cowextsize(
|
||||
struct xfs_inode *ip,
|
||||
struct fileattr *fa)
|
||||
struct file_kattr *fa)
|
||||
{
|
||||
struct xfs_mount *mp = ip->i_mount;
|
||||
xfs_failaddr_t failaddr;
|
||||
@ -709,7 +709,7 @@ xfs_ioctl_setattr_check_cowextsize(
|
||||
static int
|
||||
xfs_ioctl_setattr_check_projid(
|
||||
struct xfs_inode *ip,
|
||||
struct fileattr *fa)
|
||||
struct file_kattr *fa)
|
||||
{
|
||||
if (!fa->fsx_valid)
|
||||
return 0;
|
||||
@ -725,7 +725,7 @@ int
|
||||
xfs_fileattr_set(
|
||||
struct mnt_idmap *idmap,
|
||||
struct dentry *dentry,
|
||||
struct fileattr *fa)
|
||||
struct file_kattr *fa)
|
||||
{
|
||||
struct xfs_inode *ip = XFS_I(d_inode(dentry));
|
||||
struct xfs_mount *mp = ip->i_mount;
|
||||
|
@ -17,13 +17,13 @@ xfs_ioc_swapext(
|
||||
extern int
|
||||
xfs_fileattr_get(
|
||||
struct dentry *dentry,
|
||||
struct fileattr *fa);
|
||||
struct file_kattr *fa);
|
||||
|
||||
extern int
|
||||
xfs_fileattr_set(
|
||||
struct mnt_idmap *idmap,
|
||||
struct dentry *dentry,
|
||||
struct fileattr *fa);
|
||||
struct file_kattr *fa);
|
||||
|
||||
extern long
|
||||
xfs_file_ioctl(
|
||||
|
@ -14,13 +14,33 @@
|
||||
FS_XFLAG_NODUMP | FS_XFLAG_NOATIME | FS_XFLAG_DAX | \
|
||||
FS_XFLAG_PROJINHERIT)
|
||||
|
||||
/* Read-only inode flags */
|
||||
#define FS_XFLAG_RDONLY_MASK \
|
||||
(FS_XFLAG_PREALLOC | FS_XFLAG_HASATTR)
|
||||
|
||||
/* Flags to indicate valid value of fsx_ fields */
|
||||
#define FS_XFLAG_VALUES_MASK \
|
||||
(FS_XFLAG_EXTSIZE | FS_XFLAG_COWEXTSIZE)
|
||||
|
||||
/* Flags for directories */
|
||||
#define FS_XFLAG_DIRONLY_MASK \
|
||||
(FS_XFLAG_RTINHERIT | FS_XFLAG_NOSYMLINKS | FS_XFLAG_EXTSZINHERIT)
|
||||
|
||||
/* Misc settable flags */
|
||||
#define FS_XFLAG_MISC_MASK \
|
||||
(FS_XFLAG_REALTIME | FS_XFLAG_NODEFRAG | FS_XFLAG_FILESTREAM)
|
||||
|
||||
#define FS_XFLAGS_MASK \
|
||||
(FS_XFLAG_COMMON | FS_XFLAG_RDONLY_MASK | FS_XFLAG_VALUES_MASK | \
|
||||
FS_XFLAG_DIRONLY_MASK | FS_XFLAG_MISC_MASK)
|
||||
|
||||
/*
|
||||
* Merged interface for miscellaneous file attributes. 'flags' originates from
|
||||
* ext* and 'fsx_flags' from xfs. There's some overlap between the two, which
|
||||
* is handled by the VFS helpers, so filesystems are free to implement just one
|
||||
* or both of these sub-interfaces.
|
||||
*/
|
||||
struct fileattr {
|
||||
struct file_kattr {
|
||||
u32 flags; /* flags (FS_IOC_GETFLAGS/FS_IOC_SETFLAGS) */
|
||||
/* struct fsxattr: */
|
||||
u32 fsx_xflags; /* xflags field value (get/set) */
|
||||
@ -33,10 +53,10 @@ struct fileattr {
|
||||
bool fsx_valid:1;
|
||||
};
|
||||
|
||||
int copy_fsxattr_to_user(const struct fileattr *fa, struct fsxattr __user *ufa);
|
||||
int copy_fsxattr_to_user(const struct file_kattr *fa, struct fsxattr __user *ufa);
|
||||
|
||||
void fileattr_fill_xflags(struct fileattr *fa, u32 xflags);
|
||||
void fileattr_fill_flags(struct fileattr *fa, u32 flags);
|
||||
void fileattr_fill_xflags(struct file_kattr *fa, u32 xflags);
|
||||
void fileattr_fill_flags(struct file_kattr *fa, u32 flags);
|
||||
|
||||
/**
|
||||
* fileattr_has_fsx - check for extended flags/attributes
|
||||
@ -45,15 +65,19 @@ void fileattr_fill_flags(struct fileattr *fa, u32 flags);
|
||||
* Return: true if any attributes are present that are not represented in
|
||||
* ->flags.
|
||||
*/
|
||||
static inline bool fileattr_has_fsx(const struct fileattr *fa)
|
||||
static inline bool fileattr_has_fsx(const struct file_kattr *fa)
|
||||
{
|
||||
return fa->fsx_valid &&
|
||||
((fa->fsx_xflags & ~FS_XFLAG_COMMON) || fa->fsx_extsize != 0 ||
|
||||
fa->fsx_projid != 0 || fa->fsx_cowextsize != 0);
|
||||
}
|
||||
|
||||
int vfs_fileattr_get(struct dentry *dentry, struct fileattr *fa);
|
||||
int vfs_fileattr_get(struct dentry *dentry, struct file_kattr *fa);
|
||||
int vfs_fileattr_set(struct mnt_idmap *idmap, struct dentry *dentry,
|
||||
struct fileattr *fa);
|
||||
struct file_kattr *fa);
|
||||
int ioctl_getflags(struct file *file, unsigned int __user *argp);
|
||||
int ioctl_setflags(struct file *file, unsigned int __user *argp);
|
||||
int ioctl_fsgetxattr(struct file *file, void __user *argp);
|
||||
int ioctl_fssetxattr(struct file *file, void __user *argp);
|
||||
|
||||
#endif /* _LINUX_FILEATTR_H */
|
||||
|
@ -80,7 +80,7 @@ struct fsnotify_mark_connector;
|
||||
struct fsnotify_sb_info;
|
||||
struct fs_context;
|
||||
struct fs_parameter_spec;
|
||||
struct fileattr;
|
||||
struct file_kattr;
|
||||
struct iomap_ops;
|
||||
|
||||
extern void __init inode_init(void);
|
||||
@ -2257,8 +2257,8 @@ struct inode_operations {
|
||||
int (*set_acl)(struct mnt_idmap *, struct dentry *,
|
||||
struct posix_acl *, int);
|
||||
int (*fileattr_set)(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa);
|
||||
int (*fileattr_get)(struct dentry *dentry, struct fileattr *fa);
|
||||
struct dentry *dentry, struct file_kattr *fa);
|
||||
int (*fileattr_get)(struct dentry *dentry, struct file_kattr *fa);
|
||||
struct offset_ctx *(*get_offset_ctx)(struct inode *inode);
|
||||
} ____cacheline_aligned;
|
||||
|
||||
|
@ -157,6 +157,8 @@ LSM_HOOK(int, 0, inode_removexattr, struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, const char *name)
|
||||
LSM_HOOK(void, LSM_RET_VOID, inode_post_removexattr, struct dentry *dentry,
|
||||
const char *name)
|
||||
LSM_HOOK(int, 0, inode_file_setattr, struct dentry *dentry, struct file_kattr *fa)
|
||||
LSM_HOOK(int, 0, inode_file_getattr, struct dentry *dentry, struct file_kattr *fa)
|
||||
LSM_HOOK(int, 0, inode_set_acl, struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, const char *acl_name, struct posix_acl *kacl)
|
||||
LSM_HOOK(void, LSM_RET_VOID, inode_post_set_acl, struct dentry *dentry,
|
||||
|
@ -451,6 +451,10 @@ int security_inode_listxattr(struct dentry *dentry);
|
||||
int security_inode_removexattr(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, const char *name);
|
||||
void security_inode_post_removexattr(struct dentry *dentry, const char *name);
|
||||
int security_inode_file_setattr(struct dentry *dentry,
|
||||
struct file_kattr *fa);
|
||||
int security_inode_file_getattr(struct dentry *dentry,
|
||||
struct file_kattr *fa);
|
||||
int security_inode_need_killpriv(struct dentry *dentry);
|
||||
int security_inode_killpriv(struct mnt_idmap *idmap, struct dentry *dentry);
|
||||
int security_inode_getsecurity(struct mnt_idmap *idmap,
|
||||
@ -1052,6 +1056,18 @@ static inline void security_inode_post_removexattr(struct dentry *dentry,
|
||||
const char *name)
|
||||
{ }
|
||||
|
||||
static inline int security_inode_file_setattr(struct dentry *dentry,
|
||||
struct file_kattr *fa)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
static inline int security_inode_file_getattr(struct dentry *dentry,
|
||||
struct file_kattr *fa)
|
||||
{
|
||||
return 0;
|
||||
}
|
||||
|
||||
static inline int security_inode_need_killpriv(struct dentry *dentry)
|
||||
{
|
||||
return cap_inode_need_killpriv(dentry);
|
||||
|
@ -78,6 +78,7 @@ struct cachestat;
|
||||
struct statmount;
|
||||
struct mnt_id_req;
|
||||
struct xattr_args;
|
||||
struct file_attr;
|
||||
|
||||
#include <linux/types.h>
|
||||
#include <linux/aio_abi.h>
|
||||
@ -371,6 +372,12 @@ asmlinkage long sys_removexattrat(int dfd, const char __user *path,
|
||||
asmlinkage long sys_lremovexattr(const char __user *path,
|
||||
const char __user *name);
|
||||
asmlinkage long sys_fremovexattr(int fd, const char __user *name);
|
||||
asmlinkage long sys_file_getattr(int dfd, const char __user *filename,
|
||||
struct file_attr __user *attr, size_t usize,
|
||||
unsigned int at_flags);
|
||||
asmlinkage long sys_file_setattr(int dfd, const char __user *filename,
|
||||
struct file_attr __user *attr, size_t usize,
|
||||
unsigned int at_flags);
|
||||
asmlinkage long sys_getcwd(char __user *buf, unsigned long size);
|
||||
asmlinkage long sys_eventfd2(unsigned int count, int flags);
|
||||
asmlinkage long sys_epoll_create1(int flags);
|
||||
|
@ -852,8 +852,14 @@ __SYSCALL(__NR_removexattrat, sys_removexattrat)
|
||||
#define __NR_open_tree_attr 467
|
||||
__SYSCALL(__NR_open_tree_attr, sys_open_tree_attr)
|
||||
|
||||
/* fs/inode.c */
|
||||
#define __NR_file_getattr 468
|
||||
__SYSCALL(__NR_file_getattr, sys_file_getattr)
|
||||
#define __NR_file_setattr 469
|
||||
__SYSCALL(__NR_file_setattr, sys_file_setattr)
|
||||
|
||||
#undef __NR_syscalls
|
||||
#define __NR_syscalls 468
|
||||
#define __NR_syscalls 470
|
||||
|
||||
/*
|
||||
* 32 bit systems traditionally used different
|
||||
|
@ -216,6 +216,24 @@ struct fsxattr {
|
||||
unsigned char fsx_pad[8];
|
||||
};
|
||||
|
||||
/*
|
||||
* Variable size structure for file_[sg]et_attr().
|
||||
*
|
||||
* Note. This is alternative to the structure 'struct file_kattr'/'struct fsxattr'.
|
||||
* As this structure is passed to/from userspace with its size, this can
|
||||
* be versioned based on the size.
|
||||
*/
|
||||
struct file_attr {
|
||||
__u64 fa_xflags; /* xflags field value (get/set) */
|
||||
__u32 fa_extsize; /* extsize field value (get/set)*/
|
||||
__u32 fa_nextents; /* nextents field value (get) */
|
||||
__u32 fa_projid; /* project identifier (get/set) */
|
||||
__u32 fa_cowextsize; /* CoW extsize field value (get/set) */
|
||||
};
|
||||
|
||||
#define FILE_ATTR_SIZE_VER0 24
|
||||
#define FILE_ATTR_SIZE_LATEST FILE_ATTR_SIZE_VER0
|
||||
|
||||
/*
|
||||
* Flags for the fsx_xflags field
|
||||
*/
|
||||
|
@ -4187,7 +4187,7 @@ static const char *shmem_get_link(struct dentry *dentry, struct inode *inode,
|
||||
|
||||
#ifdef CONFIG_TMPFS_XATTR
|
||||
|
||||
static int shmem_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
static int shmem_fileattr_get(struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
struct shmem_inode_info *info = SHMEM_I(d_inode(dentry));
|
||||
|
||||
@ -4197,7 +4197,7 @@ static int shmem_fileattr_get(struct dentry *dentry, struct fileattr *fa)
|
||||
}
|
||||
|
||||
static int shmem_fileattr_set(struct mnt_idmap *idmap,
|
||||
struct dentry *dentry, struct fileattr *fa)
|
||||
struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
struct inode *inode = d_inode(dentry);
|
||||
struct shmem_inode_info *info = SHMEM_I(inode);
|
||||
|
@ -408,3 +408,5 @@
|
||||
465 common listxattrat sys_listxattrat
|
||||
466 common removexattrat sys_removexattrat
|
||||
467 common open_tree_attr sys_open_tree_attr
|
||||
468 common file_getattr sys_file_getattr
|
||||
469 common file_setattr sys_file_setattr
|
||||
|
@ -2622,6 +2622,36 @@ void security_inode_post_removexattr(struct dentry *dentry, const char *name)
|
||||
call_void_hook(inode_post_removexattr, dentry, name);
|
||||
}
|
||||
|
||||
/**
|
||||
* security_inode_file_setattr() - check if setting fsxattr is allowed
|
||||
* @dentry: file to set filesystem extended attributes on
|
||||
* @fa: extended attributes to set on the inode
|
||||
*
|
||||
* Called when file_setattr() syscall or FS_IOC_FSSETXATTR ioctl() is called on
|
||||
* inode
|
||||
*
|
||||
* Return: Returns 0 if permission is granted.
|
||||
*/
|
||||
int security_inode_file_setattr(struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
return call_int_hook(inode_file_setattr, dentry, fa);
|
||||
}
|
||||
|
||||
/**
|
||||
* security_inode_file_getattr() - check if retrieving fsxattr is allowed
|
||||
* @dentry: file to retrieve filesystem extended attributes from
|
||||
* @fa: extended attributes to get
|
||||
*
|
||||
* Called when file_getattr() syscall or FS_IOC_FSGETXATTR ioctl() is called on
|
||||
* inode
|
||||
*
|
||||
* Return: Returns 0 if permission is granted.
|
||||
*/
|
||||
int security_inode_file_getattr(struct dentry *dentry, struct file_kattr *fa)
|
||||
{
|
||||
return call_int_hook(inode_file_getattr, dentry, fa);
|
||||
}
|
||||
|
||||
/**
|
||||
* security_inode_need_killpriv() - Check if security_inode_killpriv() required
|
||||
* @dentry: associated dentry
|
||||
|
@ -3480,6 +3480,18 @@ static int selinux_inode_removexattr(struct mnt_idmap *idmap,
|
||||
return -EACCES;
|
||||
}
|
||||
|
||||
static int selinux_inode_file_setattr(struct dentry *dentry,
|
||||
struct file_kattr *fa)
|
||||
{
|
||||
return dentry_has_perm(current_cred(), dentry, FILE__SETATTR);
|
||||
}
|
||||
|
||||
static int selinux_inode_file_getattr(struct dentry *dentry,
|
||||
struct file_kattr *fa)
|
||||
{
|
||||
return dentry_has_perm(current_cred(), dentry, FILE__GETATTR);
|
||||
}
|
||||
|
||||
static int selinux_path_notify(const struct path *path, u64 mask,
|
||||
unsigned int obj_type)
|
||||
{
|
||||
@ -7350,6 +7362,8 @@ static struct security_hook_list selinux_hooks[] __ro_after_init = {
|
||||
LSM_HOOK_INIT(inode_getxattr, selinux_inode_getxattr),
|
||||
LSM_HOOK_INIT(inode_listxattr, selinux_inode_listxattr),
|
||||
LSM_HOOK_INIT(inode_removexattr, selinux_inode_removexattr),
|
||||
LSM_HOOK_INIT(inode_file_getattr, selinux_inode_file_getattr),
|
||||
LSM_HOOK_INIT(inode_file_setattr, selinux_inode_file_setattr),
|
||||
LSM_HOOK_INIT(inode_set_acl, selinux_inode_set_acl),
|
||||
LSM_HOOK_INIT(inode_get_acl, selinux_inode_get_acl),
|
||||
LSM_HOOK_INIT(inode_remove_acl, selinux_inode_remove_acl),
|
||||
|
Loading…
Reference in New Issue
Block a user