Skip to content

Commit 680557c

Browse files
committed
virtio_net: rework mergeable buffer handling
Use the new _ctx virtio API to maintain true length for each buffer. Signed-off-by: Michael S. Tsirkin <[email protected]>
1 parent d45b897 commit 680557c

File tree

1 file changed

+43
-46
lines changed

1 file changed

+43
-46
lines changed

drivers/net/virtio_net.c

Lines changed: 43 additions & 46 deletions
Original file line numberDiff line numberDiff line change
@@ -250,24 +250,6 @@ static void skb_xmit_done(struct virtqueue *vq)
250250
netif_wake_subqueue(vi->dev, vq2txq(vq));
251251
}
252252

253-
static unsigned int mergeable_ctx_to_buf_truesize(unsigned long mrg_ctx)
254-
{
255-
unsigned int truesize = mrg_ctx & (MERGEABLE_BUFFER_ALIGN - 1);
256-
return (truesize + 1) * MERGEABLE_BUFFER_ALIGN;
257-
}
258-
259-
static void *mergeable_ctx_to_buf_address(unsigned long mrg_ctx)
260-
{
261-
return (void *)(mrg_ctx & -MERGEABLE_BUFFER_ALIGN);
262-
263-
}
264-
265-
static unsigned long mergeable_buf_to_ctx(void *buf, unsigned int truesize)
266-
{
267-
unsigned int size = truesize / MERGEABLE_BUFFER_ALIGN;
268-
return (unsigned long)buf | (size - 1);
269-
}
270-
271253
/* Called from bottom half context */
272254
static struct sk_buff *page_to_skb(struct virtnet_info *vi,
273255
struct receive_queue *rq,
@@ -511,15 +493,13 @@ static struct page *xdp_linearize_page(struct receive_queue *rq,
511493

512494
while (--*num_buf) {
513495
unsigned int buflen;
514-
unsigned long ctx;
515496
void *buf;
516497
int off;
517498

518-
ctx = (unsigned long)virtqueue_get_buf(rq->vq, &buflen);
519-
if (unlikely(!ctx))
499+
buf = virtqueue_get_buf(rq->vq, &buflen);
500+
if (unlikely(!buf))
520501
goto err_buf;
521502

522-
buf = mergeable_ctx_to_buf_address(ctx);
523503
p = virt_to_head_page(buf);
524504
off = buf - page_address(p);
525505

@@ -548,10 +528,10 @@ static struct page *xdp_linearize_page(struct receive_queue *rq,
548528
static struct sk_buff *receive_mergeable(struct net_device *dev,
549529
struct virtnet_info *vi,
550530
struct receive_queue *rq,
551-
unsigned long ctx,
531+
void *buf,
532+
void *ctx,
552533
unsigned int len)
553534
{
554-
void *buf = mergeable_ctx_to_buf_address(ctx);
555535
struct virtio_net_hdr_mrg_rxbuf *hdr = buf;
556536
u16 num_buf = virtio16_to_cpu(vi->vdev, hdr->num_buffers);
557537
struct page *page = virt_to_head_page(buf);
@@ -639,7 +619,13 @@ static struct sk_buff *receive_mergeable(struct net_device *dev,
639619
}
640620
rcu_read_unlock();
641621

642-
truesize = max(len, mergeable_ctx_to_buf_truesize(ctx));
622+
if (unlikely(len > (unsigned long)ctx)) {
623+
pr_debug("%s: rx error: len %u exceeds truesize 0x%lu\n",
624+
dev->name, len, (unsigned long)ctx);
625+
dev->stats.rx_length_errors++;
626+
goto err_skb;
627+
}
628+
truesize = (unsigned long)ctx;
643629
head_skb = page_to_skb(vi, rq, page, offset, len, truesize);
644630
curr_skb = head_skb;
645631

@@ -648,7 +634,7 @@ static struct sk_buff *receive_mergeable(struct net_device *dev,
648634
while (--num_buf) {
649635
int num_skb_frags;
650636

651-
ctx = (unsigned long)virtqueue_get_buf(rq->vq, &len);
637+
buf = virtqueue_get_buf_ctx(rq->vq, &len, &ctx);
652638
if (unlikely(!ctx)) {
653639
pr_debug("%s: rx error: %d buffers out of %d missing\n",
654640
dev->name, num_buf,
@@ -658,8 +644,14 @@ static struct sk_buff *receive_mergeable(struct net_device *dev,
658644
goto err_buf;
659645
}
660646

661-
buf = mergeable_ctx_to_buf_address(ctx);
662647
page = virt_to_head_page(buf);
648+
if (unlikely(len > (unsigned long)ctx)) {
649+
pr_debug("%s: rx error: len %u exceeds truesize 0x%lu\n",
650+
dev->name, len, (unsigned long)ctx);
651+
dev->stats.rx_length_errors++;
652+
goto err_skb;
653+
}
654+
truesize = (unsigned long)ctx;
663655

664656
num_skb_frags = skb_shinfo(curr_skb)->nr_frags;
665657
if (unlikely(num_skb_frags == MAX_SKB_FRAGS)) {
@@ -675,7 +667,6 @@ static struct sk_buff *receive_mergeable(struct net_device *dev,
675667
head_skb->truesize += nskb->truesize;
676668
num_skb_frags = 0;
677669
}
678-
truesize = max(len, mergeable_ctx_to_buf_truesize(ctx));
679670
if (curr_skb != head_skb) {
680671
head_skb->data_len += len;
681672
head_skb->len += len;
@@ -700,14 +691,14 @@ static struct sk_buff *receive_mergeable(struct net_device *dev,
700691
err_skb:
701692
put_page(page);
702693
while (--num_buf) {
703-
ctx = (unsigned long)virtqueue_get_buf(rq->vq, &len);
704-
if (unlikely(!ctx)) {
694+
buf = virtqueue_get_buf(rq->vq, &len);
695+
if (unlikely(!buf)) {
705696
pr_debug("%s: rx error: %d buffers missing\n",
706697
dev->name, num_buf);
707698
dev->stats.rx_length_errors++;
708699
break;
709700
}
710-
page = virt_to_head_page(mergeable_ctx_to_buf_address(ctx));
701+
page = virt_to_head_page(buf);
711702
put_page(page);
712703
}
713704
err_buf:
@@ -718,7 +709,7 @@ static struct sk_buff *receive_mergeable(struct net_device *dev,
718709
}
719710

720711
static int receive_buf(struct virtnet_info *vi, struct receive_queue *rq,
721-
void *buf, unsigned int len)
712+
void *buf, unsigned int len, void **ctx)
722713
{
723714
struct net_device *dev = vi->dev;
724715
struct sk_buff *skb;
@@ -729,9 +720,7 @@ static int receive_buf(struct virtnet_info *vi, struct receive_queue *rq,
729720
pr_debug("%s: short packet %i\n", dev->name, len);
730721
dev->stats.rx_length_errors++;
731722
if (vi->mergeable_rx_bufs) {
732-
unsigned long ctx = (unsigned long)buf;
733-
void *base = mergeable_ctx_to_buf_address(ctx);
734-
put_page(virt_to_head_page(base));
723+
put_page(virt_to_head_page(buf));
735724
} else if (vi->big_packets) {
736725
give_pages(rq, buf);
737726
} else {
@@ -741,7 +730,7 @@ static int receive_buf(struct virtnet_info *vi, struct receive_queue *rq,
741730
}
742731

743732
if (vi->mergeable_rx_bufs)
744-
skb = receive_mergeable(dev, vi, rq, (unsigned long)buf, len);
733+
skb = receive_mergeable(dev, vi, rq, buf, ctx, len);
745734
else if (vi->big_packets)
746735
skb = receive_big(dev, vi, rq, buf, len);
747736
else
@@ -869,7 +858,7 @@ static int add_recvbuf_mergeable(struct virtnet_info *vi,
869858
struct page_frag *alloc_frag = &rq->alloc_frag;
870859
unsigned int headroom = virtnet_get_headroom(vi);
871860
char *buf;
872-
unsigned long ctx;
861+
void *ctx;
873862
int err;
874863
unsigned int len, hole;
875864

@@ -879,7 +868,7 @@ static int add_recvbuf_mergeable(struct virtnet_info *vi,
879868

880869
buf = (char *)page_address(alloc_frag->page) + alloc_frag->offset;
881870
buf += headroom; /* advance address leaving hole at front of pkt */
882-
ctx = mergeable_buf_to_ctx(buf, len);
871+
ctx = (void *)(unsigned long)len;
883872
get_page(alloc_frag->page);
884873
alloc_frag->offset += len + headroom;
885874
hole = alloc_frag->size - alloc_frag->offset;
@@ -894,7 +883,7 @@ static int add_recvbuf_mergeable(struct virtnet_info *vi,
894883
}
895884

896885
sg_init_one(rq->sg, buf, len);
897-
err = virtqueue_add_inbuf(rq->vq, rq->sg, 1, (void *)ctx, gfp);
886+
err = virtqueue_add_inbuf_ctx(rq->vq, rq->sg, 1, buf, ctx, gfp);
898887
if (err < 0)
899888
put_page(virt_to_head_page(buf));
900889

@@ -988,10 +977,20 @@ static int virtnet_receive(struct receive_queue *rq, int budget)
988977
void *buf;
989978
struct virtnet_stats *stats = this_cpu_ptr(vi->stats);
990979

991-
while (received < budget &&
992-
(buf = virtqueue_get_buf(rq->vq, &len)) != NULL) {
993-
bytes += receive_buf(vi, rq, buf, len);
994-
received++;
980+
if (vi->mergeable_rx_bufs) {
981+
void *ctx;
982+
983+
while (received < budget &&
984+
(buf = virtqueue_get_buf_ctx(rq->vq, &len, &ctx))) {
985+
bytes += receive_buf(vi, rq, buf, len, ctx);
986+
received++;
987+
}
988+
} else {
989+
while (received < budget &&
990+
(buf = virtqueue_get_buf(rq->vq, &len)) != NULL) {
991+
bytes += receive_buf(vi, rq, buf, len, NULL);
992+
received++;
993+
}
995994
}
996995

997996
if (rq->vq->num_free > virtqueue_get_vring_size(rq->vq) / 2) {
@@ -2014,9 +2013,7 @@ static void free_unused_bufs(struct virtnet_info *vi)
20142013

20152014
while ((buf = virtqueue_detach_unused_buf(vq)) != NULL) {
20162015
if (vi->mergeable_rx_bufs) {
2017-
unsigned long ctx = (unsigned long)buf;
2018-
void *base = mergeable_ctx_to_buf_address(ctx);
2019-
put_page(virt_to_head_page(base));
2016+
put_page(virt_to_head_page(buf));
20202017
} else if (vi->big_packets) {
20212018
give_pages(&vi->rq[i], buf);
20222019
} else {

0 commit comments

Comments
 (0)