Placing this patch after "[PATCH 3/4] virtio-net: added USO support" may interfer with "git bisect" on a host that does not support USO as virtio-net can advertise USO support where it's not supported.

I suggest to combine this patch with the earlier patch aforementioned. It will make the entire patch big so I think it's also better to extract the change for tap into another patch.

On 2023/07/20 0:21, Yuri Benditovich wrote:
From: Andrew Melnychenko <and...@daynix.com>

Added tap uso check with stubs for non-Linux systems.

Signed-off-by: Yuri Benditovich <yuri.benditov...@daynix.com>
Signed-off-by: Andrew Melnychenko <and...@daynix.com>
---
  hw/net/virtio-net.c | 15 +++++++++++++++
  include/net/net.h   |  3 +++
  net/net.c           |  9 +++++++++
  net/tap-bsd.c       |  5 +++++
  net/tap-linux.c     | 12 ++++++++++++
  net/tap-solaris.c   |  5 +++++
  net/tap-stub.c      |  5 +++++
  net/tap.c           | 12 ++++++++++++
  net/tap_int.h       |  1 +
  9 files changed, 67 insertions(+)

diff --git a/hw/net/virtio-net.c b/hw/net/virtio-net.c
index e76cad923b..d950d3a77f 100644
--- a/hw/net/virtio-net.c
+++ b/hw/net/virtio-net.c
@@ -659,6 +659,15 @@ static int peer_has_ufo(VirtIONet *n)
      return n->has_ufo;
  }
+static int peer_has_uso(VirtIONet *n)
+{
+    if (!peer_has_vnet_hdr(n)) {
+        return 0;
+    }
+
+    return qemu_has_uso(qemu_get_queue(n->nic)->peer);
+}
+
  static void virtio_net_set_mrg_rx_bufs(VirtIONet *n, int mergeable_rx_bufs,
                                         int version_1, int hash_report)
  {
@@ -808,6 +817,12 @@ static uint64_t virtio_net_get_features(VirtIODevice 
*vdev, uint64_t features,
          virtio_clear_feature(&features, VIRTIO_NET_F_HOST_UFO);
      }
+ if (!peer_has_uso(n)) {
+        virtio_clear_feature(&features, VIRTIO_NET_F_HOST_USO);
+        virtio_clear_feature(&features, VIRTIO_NET_F_GUEST_USO4);
+        virtio_clear_feature(&features, VIRTIO_NET_F_GUEST_USO6);
+    }
+
      if (!get_vhost_net(nc->peer)) {
          return features;
      }
diff --git a/include/net/net.h b/include/net/net.h
index b5ccfbbffb..330d285930 100644
--- a/include/net/net.h
+++ b/include/net/net.h
@@ -54,6 +54,7 @@ typedef void (LinkStatusChanged)(NetClientState *);
  typedef void (NetClientDestructor)(NetClientState *);
  typedef RxFilterInfo *(QueryRxFilter)(NetClientState *);
  typedef bool (HasUfo)(NetClientState *);
+typedef bool (HasUso)(NetClientState *);
  typedef bool (HasVnetHdr)(NetClientState *);
  typedef bool (HasVnetHdrLen)(NetClientState *, int);
  typedef bool (GetUsingVnetHdr)(NetClientState *);
@@ -84,6 +85,7 @@ typedef struct NetClientInfo {
      QueryRxFilter *query_rx_filter;
      NetPoll *poll;
      HasUfo *has_ufo;
+    HasUso *has_uso;
      HasVnetHdr *has_vnet_hdr;
      HasVnetHdrLen *has_vnet_hdr_len;
      GetUsingVnetHdr *get_using_vnet_hdr;
@@ -187,6 +189,7 @@ void qemu_set_info_str(NetClientState *nc,
                         const char *fmt, ...) G_GNUC_PRINTF(2, 3);
  void qemu_format_nic_info_str(NetClientState *nc, uint8_t macaddr[6]);
  bool qemu_has_ufo(NetClientState *nc);
+bool qemu_has_uso(NetClientState *nc);
  bool qemu_has_vnet_hdr(NetClientState *nc);
  bool qemu_has_vnet_hdr_len(NetClientState *nc, int len);
  bool qemu_get_using_vnet_hdr(NetClientState *nc);
diff --git a/net/net.c b/net/net.c
index 543e6dec43..b110e61f66 100644
--- a/net/net.c
+++ b/net/net.c
@@ -495,6 +495,15 @@ bool qemu_has_ufo(NetClientState *nc)
      return nc->info->has_ufo(nc);
  }
+bool qemu_has_uso(NetClientState *nc)
+{
+    if (!nc || !nc->info->has_uso) {
+        return false;
+    }
+
+    return nc->info->has_uso(nc);
+}
+
  bool qemu_has_vnet_hdr(NetClientState *nc)
  {
      if (!nc || !nc->info->has_vnet_hdr) {
diff --git a/net/tap-bsd.c b/net/tap-bsd.c
index abd16a2ad2..274ea7bd2c 100644
--- a/net/tap-bsd.c
+++ b/net/tap-bsd.c
@@ -212,6 +212,11 @@ int tap_probe_has_ufo(int fd)
      return 0;
  }
+int tap_probe_has_uso(int fd)
+{
+    return 0;
+}
+
  int tap_probe_vnet_hdr_len(int fd, int len)
  {
      return 0;
diff --git a/net/tap-linux.c b/net/tap-linux.c
index 30fcca1bc2..c7e514ecb0 100644
--- a/net/tap-linux.c
+++ b/net/tap-linux.c
@@ -173,6 +173,18 @@ int tap_probe_has_ufo(int fd)
      return 1;
  }
+int tap_probe_has_uso(int fd)
+{
+    unsigned offload;
+
+    offload = TUN_F_CSUM | TUN_F_USO4 | TUN_F_USO6;
+
+    if (ioctl(fd, TUNSETOFFLOAD, offload) < 0) {
+        return 0;
+    }
+    return 1;
+}
+
  /* Verify that we can assign given length */
  int tap_probe_vnet_hdr_len(int fd, int len)
  {
diff --git a/net/tap-solaris.c b/net/tap-solaris.c
index a617a10e5c..08b13af512 100644
--- a/net/tap-solaris.c
+++ b/net/tap-solaris.c
@@ -216,6 +216,11 @@ int tap_probe_has_ufo(int fd)
      return 0;
  }
+int tap_probe_has_uso(int fd)
+{
+    return 0;
+}
+
  int tap_probe_vnet_hdr_len(int fd, int len)
  {
      return 0;
diff --git a/net/tap-stub.c b/net/tap-stub.c
index ac8dfc03b4..4b24f61e3a 100644
--- a/net/tap-stub.c
+++ b/net/tap-stub.c
@@ -47,6 +47,11 @@ int tap_probe_has_ufo(int fd)
      return 0;
  }
+int tap_probe_has_uso(int fd)
+{
+    return 0;
+}
+
  int tap_probe_vnet_hdr_len(int fd, int len)
  {
      return 0;
diff --git a/net/tap.c b/net/tap.c
index 14ea4ef26f..bcea8d03f9 100644
--- a/net/tap.c
+++ b/net/tap.c
@@ -57,6 +57,7 @@ typedef struct TAPState {
      bool write_poll;
      bool using_vnet_hdr;
      bool has_ufo;
+    bool has_uso;
      bool enabled;
      VHostNetState *vhost_net;
      unsigned host_vnet_hdr_len;
@@ -237,6 +238,15 @@ static bool tap_has_ufo(NetClientState *nc)
      return s->has_ufo;
  }
+static bool tap_has_uso(NetClientState *nc)
+{
+    TAPState *s = DO_UPCAST(TAPState, nc, nc);
+
+    assert(nc->info->type == NET_CLIENT_DRIVER_TAP);
+
+    return s->has_uso;
+}
+
  static bool tap_has_vnet_hdr(NetClientState *nc)
  {
      TAPState *s = DO_UPCAST(TAPState, nc, nc);
@@ -384,6 +394,7 @@ static NetClientInfo net_tap_info = {
      .poll = tap_poll,
      .cleanup = tap_cleanup,
      .has_ufo = tap_has_ufo,
+    .has_uso = tap_has_uso,
      .has_vnet_hdr = tap_has_vnet_hdr,
      .has_vnet_hdr_len = tap_has_vnet_hdr_len,
      .get_using_vnet_hdr = tap_get_using_vnet_hdr,
@@ -413,6 +424,7 @@ static TAPState *net_tap_fd_init(NetClientState *peer,
      s->host_vnet_hdr_len = vnet_hdr ? sizeof(struct virtio_net_hdr) : 0;
      s->using_vnet_hdr = false;
      s->has_ufo = tap_probe_has_ufo(s->fd);
+    s->has_uso = tap_probe_has_uso(s->fd);
      s->enabled = true;
      tap_set_offload(&s->nc, 0, 0, 0, 0, 0, 0, 0);
      /*
diff --git a/net/tap_int.h b/net/tap_int.h
index d8861d81ba..9a2175655b 100644
--- a/net/tap_int.h
+++ b/net/tap_int.h
@@ -37,6 +37,7 @@ void tap_set_sndbuf(int fd, const NetdevTapOptions *tap, 
Error **errp);
  int tap_probe_vnet_hdr(int fd, Error **errp);
  int tap_probe_vnet_hdr_len(int fd, int len);
  int tap_probe_has_ufo(int fd);
+int tap_probe_has_uso(int fd);
  void tap_fd_set_offload(int fd, int csum, int tso4, int tso6, int ecn, int 
ufo,
                          int uso4, int uso6);
  void tap_fd_set_vnet_hdr_len(int fd, int len);

Reply via email to