[RFC PATCH 2/2] virtio-net: support net sysfs to fine-tune dim profile

From: Heng Qi
Date: Thu Mar 14 2024 - 09:10:15 EST


Virtio-net has different types of back-end device
implementations. In order to effectively optimize
the dim library's gains for different device
implementations, let's use the interface provided
by net-sysfs to fine-tune the profile list.

Signed-off-by: Heng Qi <hengqi@xxxxxxxxxxxxxxxxx>
---
drivers/net/virtio_net.c | 64 +++++++++++++++++++++++++++++++++++++++++++++++-
1 file changed, 63 insertions(+), 1 deletion(-)

diff --git a/drivers/net/virtio_net.c b/drivers/net/virtio_net.c
index e709d44..7fae737 100644
--- a/drivers/net/virtio_net.c
+++ b/drivers/net/virtio_net.c
@@ -57,6 +57,16 @@

#define VIRTNET_DRIVER_VERSION "1.0.0"

+/* This is copied from NET_DIM_RX_EQE_PROFILES in DIM library */
+#define VIRTNET_DIM_RX_PKTS 256
+static struct dim_cq_moder rx_itr_conf[] = {
+ {.usec = 1, .pkts = VIRTNET_DIM_RX_PKTS,},
+ {.usec = 8, .pkts = VIRTNET_DIM_RX_PKTS,},
+ {.usec = 64, .pkts = VIRTNET_DIM_RX_PKTS,},
+ {.usec = 128, .pkts = VIRTNET_DIM_RX_PKTS,},
+ {.usec = 256, .pkts = VIRTNET_DIM_RX_PKTS,}
+};
+
static const unsigned long guest_offloads[] = {
VIRTIO_NET_F_GUEST_TSO4,
VIRTIO_NET_F_GUEST_TSO6,
@@ -3584,7 +3594,10 @@ static void virtnet_rx_dim_work(struct work_struct *work)
if (!rq->dim_enabled)
continue;

- update_moder = net_dim_get_rx_moderation(dim->mode, dim->profile_ix);
+ if (dim->profile_ix >= ARRAY_SIZE(rx_itr_conf))
+ dim->profile_ix = ARRAY_SIZE(rx_itr_conf) - 1;
+
+ update_moder = rx_itr_conf[dim->profile_ix];
if (update_moder.usec != rq->intr_coal.max_usecs ||
update_moder.pkts != rq->intr_coal.max_packets) {
err = virtnet_send_rx_ctrl_coal_vq_cmd(vi, qnum,
@@ -4170,6 +4183,53 @@ static void virtnet_tx_timeout(struct net_device *dev, unsigned int txqueue)
jiffies_to_usecs(jiffies - READ_ONCE(txq->trans_start)));
}

+static int virtnet_dim_moder_valid(struct net_device *dev, struct dim_profs_list *list)
+{
+ struct virtnet_info *vi = netdev_priv(dev);
+
+ if (!virtio_has_feature(vi->vdev, VIRTIO_NET_F_VQ_NOTF_COAL))
+ return -EOPNOTSUPP;
+
+ if (!list || list->direction != DIM_RX_DIRECTION ||
+ list->num != NET_DIM_PARAMS_NUM_PROFILES ||
+ list->mode != DIM_CQ_PERIOD_MODE_START_FROM_EQE) {
+ return -EINVAL;
+ }
+
+ return 0;
+}
+
+static int virtnet_dim_moder_get(struct net_device *dev, struct dim_profs_list *list)
+{
+ int ret;
+
+ ret = virtnet_dim_moder_valid(dev, list);
+ if (ret)
+ return ret;
+
+ memcpy(list->profs, rx_itr_conf, sizeof(*list->profs) * list->num);
+
+ return 0;
+}
+
+static int virtnet_dim_moder_set(struct net_device *dev, struct dim_profs_list *list)
+{
+ int i, ret;
+
+ ret = virtnet_dim_moder_valid(dev, list);
+ if (ret)
+ return ret;
+
+ for (i = 0; i < list->num; i++) {
+ rx_itr_conf[i].usec = list->profs[i].usec;
+ rx_itr_conf[i].pkts = list->profs[i].pkts;
+ if (list->profs[i].comps)
+ return -EINVAL;
+ }
+
+ return 0;
+}
+
static const struct net_device_ops virtnet_netdev = {
.ndo_open = virtnet_open,
.ndo_stop = virtnet_close,
@@ -4186,6 +4246,8 @@ static void virtnet_tx_timeout(struct net_device *dev, unsigned int txqueue)
.ndo_get_phys_port_name = virtnet_get_phys_port_name,
.ndo_set_features = virtnet_set_features,
.ndo_tx_timeout = virtnet_tx_timeout,
+ .ndo_dim_moder_get = virtnet_dim_moder_get,
+ .ndo_dim_moder_set = virtnet_dim_moder_set,
};

static void virtnet_config_changed_work(struct work_struct *work)
--
1.8.3.1