[PATCH] nbd: Permit nbd-client to set minimum and optimal I/O sizes
From: Richard W.M. Jones
Date: Thu Jun 16 2022 - 07:25:03 EST
Add new netlink attributes to allow the userspace nbd-client to
control the minimum_io_size and optimal_io_size settings.
This is the kernel part of an effort to implement NBD block size
constraints.
Signed-off-by: Richard W.M. Jones <rjones@xxxxxxxxxx>
Link: https://github.com/NetworkBlockDevice/nbd/blob/master/doc/proto.md#block-size-constraints
Link: https://lists.debian.org/nbd/2022/06/msg00022.html
---
drivers/block/nbd.c | 17 ++++++++++++++++-
include/uapi/linux/nbd-netlink.h | 2 ++
2 files changed, 18 insertions(+), 1 deletion(-)
diff --git a/drivers/block/nbd.c b/drivers/block/nbd.c
index 07f3c139a3d7..ff51b5c577b2 100644
--- a/drivers/block/nbd.c
+++ b/drivers/block/nbd.c
@@ -99,6 +99,7 @@ struct nbd_config {
wait_queue_head_t recv_wq;
unsigned int blksize_bits;
loff_t bytesize;
+ u32 minimum_io_size, optimal_io_size;
#if IS_ENABLED(CONFIG_DEBUG_FS)
struct dentry *dbg_dir;
#endif
@@ -338,6 +339,13 @@ static int nbd_set_size(struct nbd_device *nbd, loff_t bytesize,
blk_queue_logical_block_size(nbd->disk->queue, blksize);
blk_queue_physical_block_size(nbd->disk->queue, blksize);
+ if (nbd->config->minimum_io_size)
+ blk_queue_io_min(nbd->disk->queue,
+ nbd->config->minimum_io_size);
+ if (nbd->config->optimal_io_size)
+ blk_queue_io_opt(nbd->disk->queue,
+ nbd->config->optimal_io_size);
+
if (max_part)
set_bit(GD_NEED_PART_SCAN, &nbd->disk->state);
if (!set_capacity_and_notify(nbd->disk, bytesize >> 9))
@@ -1876,6 +1884,8 @@ static const struct nla_policy nbd_attr_policy[NBD_ATTR_MAX + 1] = {
[NBD_ATTR_DEAD_CONN_TIMEOUT] = { .type = NLA_U64 },
[NBD_ATTR_DEVICE_LIST] = { .type = NLA_NESTED},
[NBD_ATTR_BACKEND_IDENTIFIER] = { .type = NLA_STRING},
+ [NBD_ATTR_BLOCK_SIZE_MIN] = { .type = NLA_U32 },
+ [NBD_ATTR_BLOCK_SIZE_OPT] = { .type = NLA_U32 },
};
static const struct nla_policy nbd_sock_policy[NBD_SOCK_MAX + 1] = {
@@ -2031,7 +2041,12 @@ static int nbd_genl_connect(struct sk_buff *skb, struct genl_info *info)
&config->runtime_flags);
}
}
-
+ if (info->attrs[NBD_ATTR_BLOCK_SIZE_MIN])
+ config->minimum_io_size =
+ nla_get_u32(info->attrs[NBD_ATTR_BLOCK_SIZE_MIN]);
+ if (info->attrs[NBD_ATTR_BLOCK_SIZE_OPT])
+ config->optimal_io_size =
+ nla_get_u32(info->attrs[NBD_ATTR_BLOCK_SIZE_OPT]);
if (info->attrs[NBD_ATTR_SOCKETS]) {
struct nlattr *attr;
int rem, fd;
diff --git a/include/uapi/linux/nbd-netlink.h b/include/uapi/linux/nbd-netlink.h
index 2d0b90964227..1d6621487560 100644
--- a/include/uapi/linux/nbd-netlink.h
+++ b/include/uapi/linux/nbd-netlink.h
@@ -36,6 +36,8 @@ enum {
NBD_ATTR_DEAD_CONN_TIMEOUT,
NBD_ATTR_DEVICE_LIST,
NBD_ATTR_BACKEND_IDENTIFIER,
+ NBD_ATTR_BLOCK_SIZE_MIN,
+ NBD_ATTR_BLOCK_SIZE_OPT,
__NBD_ATTR_MAX,
};
#define NBD_ATTR_MAX (__NBD_ATTR_MAX - 1)
--
2.35.1