Blame SOURCES/0002-net-virtio-add-packed-virtqueue-helpers.patch

b91920
From 652a2e3a1ba0db81ae1814e8c3cb989e9e89c4e0 Mon Sep 17 00:00:00 2001
b91920
From: Jens Freimann <jfreimann@redhat.com>
b91920
Date: Mon, 17 Dec 2018 22:31:31 +0100
b91920
Subject: [PATCH 02/18] net/virtio: add packed virtqueue helpers
b91920
b91920
[ upstream commit e9f4feb7e6225f671b59375aff44b9d576121577 ]
b91920
b91920
Add helper functions to set/clear and check descriptor flags.
b91920
b91920
Signed-off-by: Jens Freimann <jfreimann@redhat.com>
b91920
Reviewed-by: Maxime Coquelin <maxime.coquelin@redhat.com>
b91920
(cherry picked from commit e9f4feb7e6225f671b59375aff44b9d576121577)
b91920
Signed-off-by: Jens Freimann <jfreimann@redhat.com>
b91920
---
b91920
 drivers/net/virtio/virtio_pci.h |  6 +++
b91920
 drivers/net/virtio/virtqueue.h  | 72 ++++++++++++++++++++++++++++++++-
b91920
 2 files changed, 76 insertions(+), 2 deletions(-)
b91920
b91920
diff --git a/drivers/net/virtio/virtio_pci.h b/drivers/net/virtio/virtio_pci.h
b91920
index 4c975a531..b22b62dad 100644
b91920
--- a/drivers/net/virtio/virtio_pci.h
b91920
+++ b/drivers/net/virtio/virtio_pci.h
b91920
@@ -315,6 +315,12 @@ vtpci_with_feature(struct virtio_hw *hw, uint64_t bit)
b91920
 	return (hw->guest_features & (1ULL << bit)) != 0;
b91920
 }
b91920
 
b91920
+static inline int
b91920
+vtpci_packed_queue(struct virtio_hw *hw)
b91920
+{
b91920
+	return vtpci_with_feature(hw, VIRTIO_F_RING_PACKED);
b91920
+}
b91920
+
b91920
 /*
b91920
  * Function declaration from virtio_pci.c
b91920
  */
b91920
diff --git a/drivers/net/virtio/virtqueue.h b/drivers/net/virtio/virtqueue.h
b91920
index 1525c7d10..c32812427 100644
b91920
--- a/drivers/net/virtio/virtqueue.h
b91920
+++ b/drivers/net/virtio/virtqueue.h
b91920
@@ -251,6 +251,31 @@ struct virtio_tx_region {
b91920
 			   __attribute__((__aligned__(16)));
b91920
 };
b91920
 
b91920
+static inline int
b91920
+desc_is_used(struct vring_packed_desc *desc, struct virtqueue *vq)
b91920
+{
b91920
+	uint16_t used, avail, flags;
b91920
+
b91920
+	flags = desc->flags;
b91920
+	used = !!(flags & VRING_DESC_F_USED(1));
b91920
+	avail = !!(flags & VRING_DESC_F_AVAIL(1));
b91920
+
b91920
+	return avail == used && used == vq->used_wrap_counter;
b91920
+}
b91920
+
b91920
+
b91920
+static inline void
b91920
+vring_desc_init_packed(struct virtqueue *vq, int n)
b91920
+{
b91920
+	int i;
b91920
+	for (i = 0; i < n - 1; i++) {
b91920
+		vq->ring_packed.desc_packed[i].id = i;
b91920
+		vq->vq_descx[i].next = i + 1;
b91920
+	}
b91920
+	vq->ring_packed.desc_packed[i].id = i;
b91920
+	vq->vq_descx[i].next = VQ_RING_DESC_CHAIN_END;
b91920
+}
b91920
+
b91920
 /* Chain all the descriptors in the ring with an END */
b91920
 static inline void
b91920
 vring_desc_init(struct vring_desc *dp, uint16_t n)
b91920
@@ -262,13 +287,53 @@ vring_desc_init(struct vring_desc *dp, uint16_t n)
b91920
 	dp[i].next = VQ_RING_DESC_CHAIN_END;
b91920
 }
b91920
 
b91920
+/**
b91920
+ * Tell the backend not to interrupt us.
b91920
+ */
b91920
+static inline void
b91920
+virtqueue_disable_intr_packed(struct virtqueue *vq)
b91920
+{
b91920
+	uint16_t *event_flags = &vq->ring_packed.driver_event->desc_event_flags;
b91920
+
b91920
+	*event_flags = RING_EVENT_FLAGS_DISABLE;
b91920
+}
b91920
+
b91920
+
b91920
 /**
b91920
  * Tell the backend not to interrupt us.
b91920
  */
b91920
 static inline void
b91920
 virtqueue_disable_intr(struct virtqueue *vq)
b91920
 {
b91920
-	vq->vq_ring.avail->flags |= VRING_AVAIL_F_NO_INTERRUPT;
b91920
+	if (vtpci_packed_queue(vq->hw))
b91920
+		virtqueue_disable_intr_packed(vq);
b91920
+	else
b91920
+		vq->vq_ring.avail->flags |= VRING_AVAIL_F_NO_INTERRUPT;
b91920
+}
b91920
+
b91920
+/**
b91920
+ * Tell the backend to interrupt. Implementation for packed virtqueues.
b91920
+ */
b91920
+static inline void
b91920
+virtqueue_enable_intr_packed(struct virtqueue *vq)
b91920
+{
b91920
+	uint16_t *event_flags = &vq->ring_packed.driver_event->desc_event_flags;
b91920
+
b91920
+
b91920
+	if (vq->event_flags_shadow == RING_EVENT_FLAGS_DISABLE) {
b91920
+		virtio_wmb();
b91920
+		vq->event_flags_shadow = RING_EVENT_FLAGS_ENABLE;
b91920
+		*event_flags = vq->event_flags_shadow;
b91920
+	}
b91920
+}
b91920
+
b91920
+/**
b91920
+ * Tell the backend to interrupt. Implementation for split virtqueues.
b91920
+ */
b91920
+static inline void
b91920
+virtqueue_enable_intr_split(struct virtqueue *vq)
b91920
+{
b91920
+	vq->vq_ring.avail->flags &= (~VRING_AVAIL_F_NO_INTERRUPT);
b91920
 }
b91920
 
b91920
 /**
b91920
@@ -277,7 +342,10 @@ virtqueue_disable_intr(struct virtqueue *vq)
b91920
 static inline void
b91920
 virtqueue_enable_intr(struct virtqueue *vq)
b91920
 {
b91920
-	vq->vq_ring.avail->flags &= (~VRING_AVAIL_F_NO_INTERRUPT);
b91920
+	if (vtpci_packed_queue(vq->hw))
b91920
+		virtqueue_enable_intr_packed(vq);
b91920
+	else
b91920
+		virtqueue_enable_intr_split(vq);
b91920
 }
b91920
 
b91920
 /**
b91920
-- 
b91920
2.21.0
b91920