nbd: Advertise multi-conn for shared read-only connections
The NBD specification defines NBD_FLAG_CAN_MULTI_CONN, which can be advertised when the server promises cache consistency between simultaneous clients (basically, rules that determine what FUA and flush from one client are able to guarantee for reads from another client). When we don't permit simultaneous clients (such as qemu-nbd without -e), the bit makes no sense; and for writable images, we probably have a lot more work before we can declare that actions from one client are cache-consistent with actions from another. But for read-only images, where flush isn't changing any data, we might as well advertise multi-conn support. What's more, advertisement of the bit makes it easier for clients to determine if 'qemu-nbd -e' was in use, where a second connection will succeed rather than hang until the first client goes away. This patch affects qemu as server in advertising the bit. We may want to consider patches to qemu as client to attempt parallel connections for higher throughput by spreading the load over those connections when a server advertises multi-conn, but for now sticking to one connection per nbd:// BDS is okay. See also: https://bugzilla.redhat.com/1708300 Signed-off-by: Eric Blake <eblake@redhat.com> Message-Id: <20190815185024.7010-1-eblake@redhat.com> [eblake: tweak blockdev-nbd.c to not request shared when writable, fix iotest 233] Reviewed-by: John Snow <jsnow@redhat.com>
This commit is contained in:
parent
eac2f39602
commit
61cc872456
@ -188,7 +188,7 @@ void qmp_nbd_server_add(const char *device, bool has_name, const char *name,
|
|||||||
}
|
}
|
||||||
|
|
||||||
exp = nbd_export_new(bs, 0, len, name, NULL, bitmap,
|
exp = nbd_export_new(bs, 0, len, name, NULL, bitmap,
|
||||||
writable ? 0 : NBD_FLAG_READ_ONLY,
|
writable ? 0 : NBD_FLAG_READ_ONLY, !writable,
|
||||||
NULL, false, on_eject_blk, errp);
|
NULL, false, on_eject_blk, errp);
|
||||||
if (!exp) {
|
if (!exp) {
|
||||||
return;
|
return;
|
||||||
|
@ -53,3 +53,4 @@ the operation of that feature.
|
|||||||
* 2.12: NBD_CMD_BLOCK_STATUS for "base:allocation"
|
* 2.12: NBD_CMD_BLOCK_STATUS for "base:allocation"
|
||||||
* 3.0: NBD_OPT_STARTTLS with TLS Pre-Shared Keys (PSK),
|
* 3.0: NBD_OPT_STARTTLS with TLS Pre-Shared Keys (PSK),
|
||||||
NBD_CMD_BLOCK_STATUS for "qemu:dirty-bitmap:", NBD_CMD_CACHE
|
NBD_CMD_BLOCK_STATUS for "qemu:dirty-bitmap:", NBD_CMD_CACHE
|
||||||
|
* 4.2: NBD_FLAG_CAN_MULTI_CONN for sharable read-only exports
|
||||||
|
@ -326,7 +326,7 @@ typedef struct NBDClient NBDClient;
|
|||||||
|
|
||||||
NBDExport *nbd_export_new(BlockDriverState *bs, uint64_t dev_offset,
|
NBDExport *nbd_export_new(BlockDriverState *bs, uint64_t dev_offset,
|
||||||
uint64_t size, const char *name, const char *desc,
|
uint64_t size, const char *name, const char *desc,
|
||||||
const char *bitmap, uint16_t nbdflags,
|
const char *bitmap, uint16_t nbdflags, bool shared,
|
||||||
void (*close)(NBDExport *), bool writethrough,
|
void (*close)(NBDExport *), bool writethrough,
|
||||||
BlockBackend *on_eject_blk, Error **errp);
|
BlockBackend *on_eject_blk, Error **errp);
|
||||||
void nbd_export_close(NBDExport *exp);
|
void nbd_export_close(NBDExport *exp);
|
||||||
|
@ -1461,7 +1461,7 @@ static void nbd_eject_notifier(Notifier *n, void *data)
|
|||||||
|
|
||||||
NBDExport *nbd_export_new(BlockDriverState *bs, uint64_t dev_offset,
|
NBDExport *nbd_export_new(BlockDriverState *bs, uint64_t dev_offset,
|
||||||
uint64_t size, const char *name, const char *desc,
|
uint64_t size, const char *name, const char *desc,
|
||||||
const char *bitmap, uint16_t nbdflags,
|
const char *bitmap, uint16_t nbdflags, bool shared,
|
||||||
void (*close)(NBDExport *), bool writethrough,
|
void (*close)(NBDExport *), bool writethrough,
|
||||||
BlockBackend *on_eject_blk, Error **errp)
|
BlockBackend *on_eject_blk, Error **errp)
|
||||||
{
|
{
|
||||||
@ -1487,6 +1487,8 @@ NBDExport *nbd_export_new(BlockDriverState *bs, uint64_t dev_offset,
|
|||||||
perm = BLK_PERM_CONSISTENT_READ;
|
perm = BLK_PERM_CONSISTENT_READ;
|
||||||
if ((nbdflags & NBD_FLAG_READ_ONLY) == 0) {
|
if ((nbdflags & NBD_FLAG_READ_ONLY) == 0) {
|
||||||
perm |= BLK_PERM_WRITE;
|
perm |= BLK_PERM_WRITE;
|
||||||
|
} else if (shared) {
|
||||||
|
nbdflags |= NBD_FLAG_CAN_MULTI_CONN;
|
||||||
}
|
}
|
||||||
blk = blk_new(bdrv_get_aio_context(bs), perm,
|
blk = blk_new(bdrv_get_aio_context(bs), perm,
|
||||||
BLK_PERM_CONSISTENT_READ | BLK_PERM_WRITE_UNCHANGED |
|
BLK_PERM_CONSISTENT_READ | BLK_PERM_WRITE_UNCHANGED |
|
||||||
|
@ -1173,7 +1173,7 @@ int main(int argc, char **argv)
|
|||||||
}
|
}
|
||||||
|
|
||||||
export = nbd_export_new(bs, dev_offset, fd_size, export_name,
|
export = nbd_export_new(bs, dev_offset, fd_size, export_name,
|
||||||
export_description, bitmap, nbdflags,
|
export_description, bitmap, nbdflags, shared > 1,
|
||||||
nbd_export_closed, writethrough, NULL,
|
nbd_export_closed, writethrough, NULL,
|
||||||
&error_fatal);
|
&error_fatal);
|
||||||
|
|
||||||
|
@ -40,7 +40,7 @@ exports available: 0
|
|||||||
exports available: 2
|
exports available: 2
|
||||||
export: 'n'
|
export: 'n'
|
||||||
size: 4194304
|
size: 4194304
|
||||||
flags: 0x4ef ( readonly flush fua trim zeroes df cache )
|
flags: 0x5ef ( readonly flush fua trim zeroes df multi cache )
|
||||||
min block: 1
|
min block: 1
|
||||||
opt block: 4096
|
opt block: 4096
|
||||||
max block: 33554432
|
max block: 33554432
|
||||||
|
Loading…
Reference in New Issue
Block a user