Signed-off-by: Alberto Faria <afa...@redhat.com> --- block/export/vhost-user-blk-server.c | 1 + contrib/vhost-user-blk/vhost-user-blk.c | 16 +++++-- hw/block/vhost-user-blk.c | 3 ++ hw/core/machine.c | 1 + tests/qtest/vhost-user-blk-test.c | 56 +++++++++++++++++++++++++ 5 files changed, 74 insertions(+), 3 deletions(-)
diff --git a/block/export/vhost-user-blk-server.c b/block/export/vhost-user-blk-server.c index d9d2014d9b7..fad9560b219 100644 --- a/block/export/vhost-user-blk-server.c +++ b/block/export/vhost-user-blk-server.c @@ -125,6 +125,7 @@ static uint64_t vu_blk_get_features(VuDev *dev) 1ull << VIRTIO_BLK_F_FLUSH | 1ull << VIRTIO_BLK_F_DISCARD | 1ull << VIRTIO_BLK_F_WRITE_ZEROES | + 1ull << VIRTIO_BLK_F_OUT_FUA | 1ull << VIRTIO_BLK_F_CONFIG_WCE | 1ull << VIRTIO_BLK_F_MQ | 1ull << VIRTIO_F_VERSION_1 | diff --git a/contrib/vhost-user-blk/vhost-user-blk.c b/contrib/vhost-user-blk/vhost-user-blk.c index 6cc18a1c04f..8d732bbe468 100644 --- a/contrib/vhost-user-blk/vhost-user-blk.c +++ b/contrib/vhost-user-blk/vhost-user-blk.c @@ -154,7 +154,7 @@ vub_readv(VubReq *req, struct iovec *iov, uint32_t iovcnt) } static ssize_t -vub_writev(VubReq *req, struct iovec *iov, uint32_t iovcnt) +vub_writev(VubReq *req, struct iovec *iov, uint32_t iovcnt, int flags) { VubDev *vdev_blk = req->vdev_blk; ssize_t rc; @@ -271,12 +271,19 @@ static int vub_virtio_process_req(VubDev *vdev_blk, type = le32_to_cpu(req->out->type); switch (type & ~VIRTIO_BLK_T_BARRIER) { case VIRTIO_BLK_T_IN: - case VIRTIO_BLK_T_OUT: { + case VIRTIO_BLK_T_OUT: + case VIRTIO_BLK_T_OUT_FUA: { ssize_t ret = 0; bool is_write = type & VIRTIO_BLK_T_OUT; + int flags = 0; req->sector_num = le64_to_cpu(req->out->sector); if (is_write) { - ret = vub_writev(req, &elem->out_sg[1], out_num); + #ifdef RWF_SYNC + if (type == VIRTIO_BLK_T_OUT_FUA) { + flags |= RWF_SYNC; + } + #endif + ret = vub_writev(req, &elem->out_sg[1], out_num, flags); } else { ret = vub_readv(req, &elem->in_sg[0], in_num); } @@ -379,6 +386,9 @@ vub_get_features(VuDev *dev) 1ull << VIRTIO_BLK_F_DISCARD | 1ull << VIRTIO_BLK_F_WRITE_ZEROES | #endif + #ifdef RWF_SYNC + 1ull << VIRTIO_BLK_F_OUT_FUA | + #endif 1ull << VIRTIO_BLK_F_CONFIG_WCE; if (vdev_blk->enable_ro) { diff --git a/hw/block/vhost-user-blk.c b/hw/block/vhost-user-blk.c index 4bb5ed299e7..988be625969 100644 --- a/hw/block/vhost-user-blk.c +++ b/hw/block/vhost-user-blk.c @@ -44,6 +44,7 @@ static const int user_feature_bits[] = { VIRTIO_BLK_F_CONFIG_WCE, VIRTIO_BLK_F_DISCARD, VIRTIO_BLK_F_WRITE_ZEROES, + VIRTIO_BLK_F_OUT_FUA, VIRTIO_F_VERSION_1, VIRTIO_RING_F_INDIRECT_DESC, VIRTIO_RING_F_EVENT_IDX, @@ -581,6 +582,8 @@ static const Property vhost_user_blk_properties[] = { VIRTIO_BLK_F_DISCARD, true), DEFINE_PROP_BIT64("write-zeroes", VHostUserBlk, parent_obj.host_features, VIRTIO_BLK_F_WRITE_ZEROES, true), + DEFINE_PROP_BIT64("fua-write", VHostUserBlk, parent_obj.host_features, + VIRTIO_BLK_F_OUT_FUA, true), }; static void vhost_user_blk_class_init(ObjectClass *klass, const void *data) diff --git a/hw/core/machine.c b/hw/core/machine.c index 8439b094904..bcf4f3423cb 100644 --- a/hw/core/machine.c +++ b/hw/core/machine.c @@ -39,6 +39,7 @@ GlobalProperty hw_compat_10_0[] = { { "virtio-blk-device", "fua-write", "false" }, + { "vhost-user-blk", "fua-write", "false" }, }; const size_t hw_compat_10_0_len = G_N_ELEMENTS(hw_compat_10_0); diff --git a/tests/qtest/vhost-user-blk-test.c b/tests/qtest/vhost-user-blk-test.c index ea90d41232e..4c68702e0b0 100644 --- a/tests/qtest/vhost-user-blk-test.c +++ b/tests/qtest/vhost-user-blk-test.c @@ -72,6 +72,7 @@ static uint64_t virtio_blk_request(QGuestAllocator *alloc, QVirtioDevice *d, switch (req->type) { case VIRTIO_BLK_T_IN: case VIRTIO_BLK_T_OUT: + case VIRTIO_BLK_T_OUT_FUA: g_assert_cmpuint(data_size % 512, ==, 0); break; case VIRTIO_BLK_T_DISCARD: @@ -389,6 +390,61 @@ static QVirtQueue *test_basic(QVirtioDevice *dev, QGuestAllocator *alloc) VIRTIO_BLK_T_DISCARD); } + if (features & (1u << VIRTIO_BLK_F_OUT_FUA)) { + /* FUA write and read with 3 descriptor layout */ + /* FUA write request */ + req.type = VIRTIO_BLK_T_OUT_FUA; + req.ioprio = 1; + req.sector = 0; + req.data = g_malloc0(512); + strcpy(req.data, "test"); + + req_addr = virtio_blk_request(alloc, dev, &req, 512); + + g_free(req.data); + + free_head = qvirtqueue_add(qts, vq, req_addr, 16, false, true); + qvirtqueue_add(qts, vq, req_addr + 16, 512, false, true); + qvirtqueue_add(qts, vq, req_addr + 528, 1, true, false); + + qvirtqueue_kick(qts, dev, vq, free_head); + + qvirtio_wait_used_elem(qts, dev, vq, free_head, NULL, + QVIRTIO_BLK_TIMEOUT_US); + status = readb(req_addr + 528); + g_assert_cmpint(status, ==, 0); + + guest_free(alloc, req_addr); + + /* Read request */ + req.type = VIRTIO_BLK_T_IN; + req.ioprio = 1; + req.sector = 0; + req.data = g_malloc0(512); + + req_addr = virtio_blk_request(alloc, dev, &req, 512); + + g_free(req.data); + + free_head = qvirtqueue_add(qts, vq, req_addr, 16, false, true); + qvirtqueue_add(qts, vq, req_addr + 16, 512, true, true); + qvirtqueue_add(qts, vq, req_addr + 528, 1, true, false); + + qvirtqueue_kick(qts, dev, vq, free_head); + + qvirtio_wait_used_elem(qts, dev, vq, free_head, NULL, + QVIRTIO_BLK_TIMEOUT_US); + status = readb(req_addr + 528); + g_assert_cmpint(status, ==, 0); + + data = g_malloc0(512); + qtest_memread(qts, req_addr + 16, data, 512); + g_assert_cmpstr(data, ==, "test"); + g_free(data); + + guest_free(alloc, req_addr); + } + if (features & (1u << VIRTIO_F_ANY_LAYOUT)) { /* Write and read with 2 descriptor layout */ /* Write request */ -- 2.49.0