Commit 210a03c9 authored by Christian Brauner's avatar Christian Brauner

fs: claw back a few FMODE_* bits

There's a bunch of flags that are purely based on what the file
operations support while also never being conditionally set or unset.
IOW, they're not subject to change for individual files. Imho, such
flags don't need to live in f_mode they might as well live in the fops
structs itself. And the fops struct already has that lonely
mmap_supported_flags member. We might as well turn that into a generic
fop_flags member and move a few flags from FMODE_* space into FOP_*
space. That gets us four FMODE_* bits back and the ability for new
static flags that are about file ops to not have to live in FMODE_*
space but in their own FOP_* space. It's not the most beautiful thing
ever but it gets the job done. Yes, there'll be an additional pointer
chase but hopefully that won't matter for these flags.

I suspect there's a few more we can move into there and that we can also
redirect a bunch of new flag suggestions that follow this pattern into
the fop_flags field instead of f_mode.

Link: https://lore.kernel.org/r/20240328-gewendet-spargel-aa60a030ef74@braunerReviewed-by: default avatarChristoph Hellwig <hch@lst.de>
Reviewed-by: default avatarJan Kara <jack@suse.cz>
Reviewed-by: default avatarJens Axboe <axboe@kernel.dk>
Signed-off-by: default avatarChristian Brauner <brauner@kernel.org>
parent 68d6f4f3
...@@ -904,7 +904,7 @@ int bdev_open(struct block_device *bdev, blk_mode_t mode, void *holder, ...@@ -904,7 +904,7 @@ int bdev_open(struct block_device *bdev, blk_mode_t mode, void *holder,
disk_unblock_events(disk); disk_unblock_events(disk);
bdev_file->f_flags |= O_LARGEFILE; bdev_file->f_flags |= O_LARGEFILE;
bdev_file->f_mode |= FMODE_BUF_RASYNC | FMODE_CAN_ODIRECT; bdev_file->f_mode |= FMODE_CAN_ODIRECT;
if (bdev_nowait(bdev)) if (bdev_nowait(bdev))
bdev_file->f_mode |= FMODE_NOWAIT; bdev_file->f_mode |= FMODE_NOWAIT;
bdev_file->f_mapping = bdev->bd_inode->i_mapping; bdev_file->f_mapping = bdev->bd_inode->i_mapping;
......
...@@ -863,6 +863,7 @@ const struct file_operations def_blk_fops = { ...@@ -863,6 +863,7 @@ const struct file_operations def_blk_fops = {
.splice_read = filemap_splice_read, .splice_read = filemap_splice_read,
.splice_write = iter_file_splice_write, .splice_write = iter_file_splice_write,
.fallocate = blkdev_fallocate, .fallocate = blkdev_fallocate,
.fop_flags = FOP_BUFFER_RASYNC,
}; };
static __init int blkdev_init(void) static __init int blkdev_init(void)
......
...@@ -377,7 +377,7 @@ static const struct file_operations dax_fops = { ...@@ -377,7 +377,7 @@ static const struct file_operations dax_fops = {
.release = dax_release, .release = dax_release,
.get_unmapped_area = dax_get_unmapped_area, .get_unmapped_area = dax_get_unmapped_area,
.mmap = dax_mmap, .mmap = dax_mmap,
.mmap_supported_flags = MAP_SYNC, .fop_flags = FOP_MMAP_SYNC,
}; };
static void dev_dax_cdev_del(void *cdev) static void dev_dax_cdev_del(void *cdev)
......
...@@ -3719,8 +3719,7 @@ static int btrfs_file_open(struct inode *inode, struct file *filp) ...@@ -3719,8 +3719,7 @@ static int btrfs_file_open(struct inode *inode, struct file *filp)
{ {
int ret; int ret;
filp->f_mode |= FMODE_NOWAIT | FMODE_BUF_RASYNC | FMODE_BUF_WASYNC | filp->f_mode |= FMODE_NOWAIT | FMODE_CAN_ODIRECT;
FMODE_CAN_ODIRECT;
ret = fsverity_file_open(inode, filp); ret = fsverity_file_open(inode, filp);
if (ret) if (ret)
...@@ -3850,6 +3849,7 @@ const struct file_operations btrfs_file_operations = { ...@@ -3850,6 +3849,7 @@ const struct file_operations btrfs_file_operations = {
.compat_ioctl = btrfs_compat_ioctl, .compat_ioctl = btrfs_compat_ioctl,
#endif #endif
.remap_file_range = btrfs_remap_file_range, .remap_file_range = btrfs_remap_file_range,
.fop_flags = FOP_BUFFER_RASYNC | FOP_BUFFER_WASYNC,
}; };
int btrfs_fdatawrite_range(struct inode *inode, loff_t start, loff_t end) int btrfs_fdatawrite_range(struct inode *inode, loff_t start, loff_t end)
......
...@@ -885,8 +885,7 @@ static int ext4_file_open(struct inode *inode, struct file *filp) ...@@ -885,8 +885,7 @@ static int ext4_file_open(struct inode *inode, struct file *filp)
return ret; return ret;
} }
filp->f_mode |= FMODE_NOWAIT | FMODE_BUF_RASYNC | filp->f_mode |= FMODE_NOWAIT;
FMODE_DIO_PARALLEL_WRITE;
return dquot_file_open(inode, filp); return dquot_file_open(inode, filp);
} }
...@@ -938,7 +937,6 @@ const struct file_operations ext4_file_operations = { ...@@ -938,7 +937,6 @@ const struct file_operations ext4_file_operations = {
.compat_ioctl = ext4_compat_ioctl, .compat_ioctl = ext4_compat_ioctl,
#endif #endif
.mmap = ext4_file_mmap, .mmap = ext4_file_mmap,
.mmap_supported_flags = MAP_SYNC,
.open = ext4_file_open, .open = ext4_file_open,
.release = ext4_release_file, .release = ext4_release_file,
.fsync = ext4_sync_file, .fsync = ext4_sync_file,
...@@ -946,6 +944,8 @@ const struct file_operations ext4_file_operations = { ...@@ -946,6 +944,8 @@ const struct file_operations ext4_file_operations = {
.splice_read = ext4_file_splice_read, .splice_read = ext4_file_splice_read,
.splice_write = iter_file_splice_write, .splice_write = iter_file_splice_write,
.fallocate = ext4_fallocate, .fallocate = ext4_fallocate,
.fop_flags = FOP_MMAP_SYNC | FOP_BUFFER_RASYNC |
FOP_DIO_PARALLEL_WRITE,
}; };
const struct inode_operations ext4_file_inode_operations = { const struct inode_operations ext4_file_inode_operations = {
......
...@@ -569,7 +569,7 @@ static int f2fs_file_open(struct inode *inode, struct file *filp) ...@@ -569,7 +569,7 @@ static int f2fs_file_open(struct inode *inode, struct file *filp)
if (err) if (err)
return err; return err;
filp->f_mode |= FMODE_NOWAIT | FMODE_BUF_RASYNC; filp->f_mode |= FMODE_NOWAIT;
filp->f_mode |= FMODE_CAN_ODIRECT; filp->f_mode |= FMODE_CAN_ODIRECT;
return dquot_file_open(inode, filp); return dquot_file_open(inode, filp);
...@@ -5045,4 +5045,5 @@ const struct file_operations f2fs_file_operations = { ...@@ -5045,4 +5045,5 @@ const struct file_operations f2fs_file_operations = {
.splice_read = f2fs_file_splice_read, .splice_read = f2fs_file_splice_read,
.splice_write = iter_file_splice_write, .splice_write = iter_file_splice_write,
.fadvise = f2fs_file_fadvise, .fadvise = f2fs_file_fadvise,
.fop_flags = FOP_BUFFER_RASYNC,
}; };
...@@ -1685,7 +1685,7 @@ int generic_write_checks_count(struct kiocb *iocb, loff_t *count) ...@@ -1685,7 +1685,7 @@ int generic_write_checks_count(struct kiocb *iocb, loff_t *count)
if ((iocb->ki_flags & IOCB_NOWAIT) && if ((iocb->ki_flags & IOCB_NOWAIT) &&
!((iocb->ki_flags & IOCB_DIRECT) || !((iocb->ki_flags & IOCB_DIRECT) ||
(file->f_mode & FMODE_BUF_WASYNC))) (file->f_op->fop_flags & FOP_BUFFER_WASYNC)))
return -EINVAL; return -EINVAL;
return generic_write_check_limits(iocb->ki_filp, iocb->ki_pos, count); return generic_write_check_limits(iocb->ki_filp, iocb->ki_pos, count);
......
...@@ -1230,8 +1230,7 @@ xfs_file_open( ...@@ -1230,8 +1230,7 @@ xfs_file_open(
{ {
if (xfs_is_shutdown(XFS_M(inode->i_sb))) if (xfs_is_shutdown(XFS_M(inode->i_sb)))
return -EIO; return -EIO;
file->f_mode |= FMODE_NOWAIT | FMODE_BUF_RASYNC | FMODE_BUF_WASYNC | file->f_mode |= FMODE_NOWAIT | FMODE_CAN_ODIRECT;
FMODE_DIO_PARALLEL_WRITE | FMODE_CAN_ODIRECT;
return generic_file_open(inode, file); return generic_file_open(inode, file);
} }
...@@ -1490,7 +1489,6 @@ const struct file_operations xfs_file_operations = { ...@@ -1490,7 +1489,6 @@ const struct file_operations xfs_file_operations = {
.compat_ioctl = xfs_file_compat_ioctl, .compat_ioctl = xfs_file_compat_ioctl,
#endif #endif
.mmap = xfs_file_mmap, .mmap = xfs_file_mmap,
.mmap_supported_flags = MAP_SYNC,
.open = xfs_file_open, .open = xfs_file_open,
.release = xfs_file_release, .release = xfs_file_release,
.fsync = xfs_file_fsync, .fsync = xfs_file_fsync,
...@@ -1498,6 +1496,8 @@ const struct file_operations xfs_file_operations = { ...@@ -1498,6 +1496,8 @@ const struct file_operations xfs_file_operations = {
.fallocate = xfs_file_fallocate, .fallocate = xfs_file_fallocate,
.fadvise = xfs_file_fadvise, .fadvise = xfs_file_fadvise,
.remap_file_range = xfs_file_remap_range, .remap_file_range = xfs_file_remap_range,
.fop_flags = FOP_MMAP_SYNC | FOP_BUFFER_RASYNC | FOP_BUFFER_WASYNC |
FOP_DIO_PARALLEL_WRITE,
}; };
const struct file_operations xfs_dir_file_operations = { const struct file_operations xfs_dir_file_operations = {
...@@ -1510,4 +1510,6 @@ const struct file_operations xfs_dir_file_operations = { ...@@ -1510,4 +1510,6 @@ const struct file_operations xfs_dir_file_operations = {
.compat_ioctl = xfs_file_compat_ioctl, .compat_ioctl = xfs_file_compat_ioctl,
#endif #endif
.fsync = xfs_dir_fsync, .fsync = xfs_dir_fsync,
.fop_flags = FOP_MMAP_SYNC | FOP_BUFFER_RASYNC | FOP_BUFFER_WASYNC |
FOP_DIO_PARALLEL_WRITE,
}; };
...@@ -163,9 +163,6 @@ typedef int (dio_iodone_t)(struct kiocb *iocb, loff_t offset, ...@@ -163,9 +163,6 @@ typedef int (dio_iodone_t)(struct kiocb *iocb, loff_t offset,
#define FMODE_NOREUSE ((__force fmode_t)0x800000) #define FMODE_NOREUSE ((__force fmode_t)0x800000)
/* File supports non-exclusive O_DIRECT writes from multiple threads */
#define FMODE_DIO_PARALLEL_WRITE ((__force fmode_t)0x1000000)
/* File is embedded in backing_file object */ /* File is embedded in backing_file object */
#define FMODE_BACKING ((__force fmode_t)0x2000000) #define FMODE_BACKING ((__force fmode_t)0x2000000)
...@@ -181,12 +178,6 @@ typedef int (dio_iodone_t)(struct kiocb *iocb, loff_t offset, ...@@ -181,12 +178,6 @@ typedef int (dio_iodone_t)(struct kiocb *iocb, loff_t offset,
/* File does not contribute to nr_files count */ /* File does not contribute to nr_files count */
#define FMODE_NOACCOUNT ((__force fmode_t)0x20000000) #define FMODE_NOACCOUNT ((__force fmode_t)0x20000000)
/* File supports async buffered reads */
#define FMODE_BUF_RASYNC ((__force fmode_t)0x40000000)
/* File supports async nowait buffered writes */
#define FMODE_BUF_WASYNC ((__force fmode_t)0x80000000)
/* /*
* Attribute flags. These should be or-ed together to figure out what * Attribute flags. These should be or-ed together to figure out what
* has been changed! * has been changed!
...@@ -2001,8 +1992,11 @@ struct iov_iter; ...@@ -2001,8 +1992,11 @@ struct iov_iter;
struct io_uring_cmd; struct io_uring_cmd;
struct offset_ctx; struct offset_ctx;
typedef unsigned int __bitwise fop_flags_t;
struct file_operations { struct file_operations {
struct module *owner; struct module *owner;
fop_flags_t fop_flags;
loff_t (*llseek) (struct file *, loff_t, int); loff_t (*llseek) (struct file *, loff_t, int);
ssize_t (*read) (struct file *, char __user *, size_t, loff_t *); ssize_t (*read) (struct file *, char __user *, size_t, loff_t *);
ssize_t (*write) (struct file *, const char __user *, size_t, loff_t *); ssize_t (*write) (struct file *, const char __user *, size_t, loff_t *);
...@@ -2015,7 +2009,6 @@ struct file_operations { ...@@ -2015,7 +2009,6 @@ struct file_operations {
long (*unlocked_ioctl) (struct file *, unsigned int, unsigned long); long (*unlocked_ioctl) (struct file *, unsigned int, unsigned long);
long (*compat_ioctl) (struct file *, unsigned int, unsigned long); long (*compat_ioctl) (struct file *, unsigned int, unsigned long);
int (*mmap) (struct file *, struct vm_area_struct *); int (*mmap) (struct file *, struct vm_area_struct *);
unsigned long mmap_supported_flags;
int (*open) (struct inode *, struct file *); int (*open) (struct inode *, struct file *);
int (*flush) (struct file *, fl_owner_t id); int (*flush) (struct file *, fl_owner_t id);
int (*release) (struct inode *, struct file *); int (*release) (struct inode *, struct file *);
...@@ -2046,6 +2039,15 @@ struct file_operations { ...@@ -2046,6 +2039,15 @@ struct file_operations {
unsigned int poll_flags); unsigned int poll_flags);
} __randomize_layout; } __randomize_layout;
/* Supports async buffered reads */
#define FOP_BUFFER_RASYNC ((__force fop_flags_t)(1 << 0))
/* Supports async buffered writes */
#define FOP_BUFFER_WASYNC ((__force fop_flags_t)(1 << 1))
/* Supports synchronous page faults for mappings */
#define FOP_MMAP_SYNC ((__force fop_flags_t)(1 << 2))
/* Supports non-exclusive O_DIRECT writes from multiple threads */
#define FOP_DIO_PARALLEL_WRITE ((__force fop_flags_t)(1 << 3))
/* Wrap a directory iterator that needs exclusive inode access */ /* Wrap a directory iterator that needs exclusive inode access */
int wrap_directory_iterator(struct file *, struct dir_context *, int wrap_directory_iterator(struct file *, struct dir_context *,
int (*) (struct file *, struct dir_context *)); int (*) (struct file *, struct dir_context *));
......
...@@ -471,7 +471,7 @@ static void io_prep_async_work(struct io_kiocb *req) ...@@ -471,7 +471,7 @@ static void io_prep_async_work(struct io_kiocb *req)
/* don't serialize this request if the fs doesn't need it */ /* don't serialize this request if the fs doesn't need it */
if (should_hash && (req->file->f_flags & O_DIRECT) && if (should_hash && (req->file->f_flags & O_DIRECT) &&
(req->file->f_mode & FMODE_DIO_PARALLEL_WRITE)) (req->file->f_op->fop_flags & FOP_DIO_PARALLEL_WRITE))
should_hash = false; should_hash = false;
if (should_hash || (ctx->flags & IORING_SETUP_IOPOLL)) if (should_hash || (ctx->flags & IORING_SETUP_IOPOLL))
io_wq_hash_work(&req->work, file_inode(req->file)); io_wq_hash_work(&req->work, file_inode(req->file));
......
...@@ -683,7 +683,8 @@ static bool io_rw_should_retry(struct io_kiocb *req) ...@@ -683,7 +683,8 @@ static bool io_rw_should_retry(struct io_kiocb *req)
* just use poll if we can, and don't attempt if the fs doesn't * just use poll if we can, and don't attempt if the fs doesn't
* support callback based unlocks * support callback based unlocks
*/ */
if (io_file_can_poll(req) || !(req->file->f_mode & FMODE_BUF_RASYNC)) if (io_file_can_poll(req) ||
!(req->file->f_op->fop_flags & FOP_BUFFER_RASYNC))
return false; return false;
wait->wait.func = io_async_buf_func; wait->wait.func = io_async_buf_func;
...@@ -1022,9 +1023,9 @@ int io_write(struct io_kiocb *req, unsigned int issue_flags) ...@@ -1022,9 +1023,9 @@ int io_write(struct io_kiocb *req, unsigned int issue_flags)
if (unlikely(!io_file_supports_nowait(req))) if (unlikely(!io_file_supports_nowait(req)))
goto copy_iov; goto copy_iov;
/* File path supports NOWAIT for non-direct_IO only for block devices. */ /* Check if we can support NOWAIT. */
if (!(kiocb->ki_flags & IOCB_DIRECT) && if (!(kiocb->ki_flags & IOCB_DIRECT) &&
!(kiocb->ki_filp->f_mode & FMODE_BUF_WASYNC) && !(req->file->f_op->fop_flags & FOP_BUFFER_WASYNC) &&
(req->flags & REQ_F_ISREG)) (req->flags & REQ_F_ISREG))
goto copy_iov; goto copy_iov;
......
...@@ -1294,7 +1294,9 @@ unsigned long do_mmap(struct file *file, unsigned long addr, ...@@ -1294,7 +1294,9 @@ unsigned long do_mmap(struct file *file, unsigned long addr,
if (!file_mmap_ok(file, inode, pgoff, len)) if (!file_mmap_ok(file, inode, pgoff, len))
return -EOVERFLOW; return -EOVERFLOW;
flags_mask = LEGACY_MAP_MASK | file->f_op->mmap_supported_flags; flags_mask = LEGACY_MAP_MASK;
if (file->f_op->fop_flags & FOP_MMAP_SYNC)
flags_mask |= MAP_SYNC;
switch (flags & MAP_TYPE) { switch (flags & MAP_TYPE) {
case MAP_SHARED: case MAP_SHARED:
......
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