mirror of
https://github.com/raspberrypi/linux.git
synced 2025-12-06 01:49:46 +00:00
virtio-net: fix incorrect flags recording in big mode
The purpose of commit703eec1b24("virtio_net: fixing XDP for fully checksummed packets handling") is to record the flags in advance, as their value may be overwritten in the XDP case. However, the flags recorded under big mode are incorrect, because in big mode, the passed buf does not point to the rx buffer, but rather to the page of the submitted buffer. This commit fixes this issue. For the small mode, the commitc11a49d58a("virtio_net: Fix mismatched buf address when unmapping for small packets") fixed it. Tested-by: Alyssa Ross <hi@alyssa.is> Fixes:703eec1b24("virtio_net: fixing XDP for fully checksummed packets handling") Signed-off-by: Xuan Zhuo <xuanzhuo@linux.alibaba.com> Acked-by: Jason Wang <jasowang@redhat.com> Acked-by: Michael S. Tsirkin <mst@redhat.com> Link: https://patch.msgid.link/20251111090828.23186-1-xuanzhuo@linux.alibaba.com Signed-off-by: Paolo Abeni <pabeni@redhat.com>
This commit is contained in:
@@ -2631,22 +2631,28 @@ static void receive_buf(struct virtnet_info *vi, struct receive_queue *rq,
|
||||
return;
|
||||
}
|
||||
|
||||
/* 1. Save the flags early, as the XDP program might overwrite them.
|
||||
/* About the flags below:
|
||||
* 1. Save the flags early, as the XDP program might overwrite them.
|
||||
* These flags ensure packets marked as VIRTIO_NET_HDR_F_DATA_VALID
|
||||
* stay valid after XDP processing.
|
||||
* 2. XDP doesn't work with partially checksummed packets (refer to
|
||||
* virtnet_xdp_set()), so packets marked as
|
||||
* VIRTIO_NET_HDR_F_NEEDS_CSUM get dropped during XDP processing.
|
||||
*/
|
||||
flags = ((struct virtio_net_common_hdr *)buf)->hdr.flags;
|
||||
|
||||
if (vi->mergeable_rx_bufs)
|
||||
if (vi->mergeable_rx_bufs) {
|
||||
flags = ((struct virtio_net_common_hdr *)buf)->hdr.flags;
|
||||
skb = receive_mergeable(dev, vi, rq, buf, ctx, len, xdp_xmit,
|
||||
stats);
|
||||
else if (vi->big_packets)
|
||||
} else if (vi->big_packets) {
|
||||
void *p = page_address((struct page *)buf);
|
||||
|
||||
flags = ((struct virtio_net_common_hdr *)p)->hdr.flags;
|
||||
skb = receive_big(dev, vi, rq, buf, len, stats);
|
||||
else
|
||||
} else {
|
||||
flags = ((struct virtio_net_common_hdr *)buf)->hdr.flags;
|
||||
skb = receive_small(dev, vi, rq, buf, ctx, len, xdp_xmit, stats);
|
||||
}
|
||||
|
||||
if (unlikely(!skb))
|
||||
return;
|
||||
|
||||
Reference in New Issue
Block a user