diff options
author | Eric Biggers <ebiggers@google.com> | 2022-08-26 23:58:45 -0700 |
---|---|---|
committer | Eric Biggers <ebiggers@google.com> | 2022-09-11 19:47:12 -0500 |
commit | 2d985f8c6b91b5007a16e640bb9c038c5fb2839b (patch) | |
tree | 0fc609386fbba4cabc8c4c0a99005cc7cb53807d /block/bdev.c | |
parent | 825cf206ed510c4a1758bef8957e2b039253e2e3 (diff) | |
download | lwn-2d985f8c6b91b5007a16e640bb9c038c5fb2839b.tar.gz lwn-2d985f8c6b91b5007a16e640bb9c038c5fb2839b.zip |
vfs: support STATX_DIOALIGN on block devices
Add support for STATX_DIOALIGN to block devices, so that direct I/O
alignment restrictions are exposed to userspace in a generic way.
Note that this breaks the tradition of stat operating only on the block
device node, not the block device itself. However, it was felt that
doing this is preferable, in order to make the interface useful and
avoid needing separate interfaces for regular files and block devices.
Signed-off-by: Eric Biggers <ebiggers@google.com>
Reviewed-by: Christoph Hellwig <hch@lst.de>
Reviewed-by: Christian Brauner (Microsoft) <brauner@kernel.org>
Link: https://lore.kernel.org/r/20220827065851.135710-3-ebiggers@kernel.org
Diffstat (limited to 'block/bdev.c')
-rw-r--r-- | block/bdev.c | 23 |
1 files changed, 23 insertions, 0 deletions
diff --git a/block/bdev.c b/block/bdev.c index ce05175e71ce..d699ecdb3260 100644 --- a/block/bdev.c +++ b/block/bdev.c @@ -26,6 +26,7 @@ #include <linux/namei.h> #include <linux/part_stat.h> #include <linux/uaccess.h> +#include <linux/stat.h> #include "../fs/internal.h" #include "blk.h" @@ -1069,3 +1070,25 @@ void sync_bdevs(bool wait) spin_unlock(&blockdev_superblock->s_inode_list_lock); iput(old_inode); } + +/* + * Handle STATX_DIOALIGN for block devices. + * + * Note that the inode passed to this is the inode of a block device node file, + * not the block device's internal inode. Therefore it is *not* valid to use + * I_BDEV() here; the block device has to be looked up by i_rdev instead. + */ +void bdev_statx_dioalign(struct inode *inode, struct kstat *stat) +{ + struct block_device *bdev; + + bdev = blkdev_get_no_open(inode->i_rdev); + if (!bdev) + return; + + stat->dio_mem_align = bdev_dma_alignment(bdev) + 1; + stat->dio_offset_align = bdev_logical_block_size(bdev); + stat->result_mask |= STATX_DIOALIGN; + + blkdev_put_no_open(bdev); +} |