From: David Miller <davem@davemloft.net>
To: eric.dumazet@gmail.com
Cc: netdev@vger.kernel.org
Subject: Re: [PATCH net-next-2.6] net: pskb_expand_head() optimization
Date: Mon, 06 Sep 2010 19:20:03 -0700 (PDT) [thread overview]
Message-ID: <20100906.192003.232914158.davem@davemloft.net> (raw)
In-Reply-To: <20100903.064633.27806839.davem@davemloft.net>
From: David Miller <davem@davemloft.net>
Date: Fri, 03 Sep 2010 06:46:33 -0700 (PDT)
> There are only two operations that make a double-linked list currently
> impossible. This pskb_expand_head() thing, and pskb_copy().
>
> They cause the situation where the list is shared between multiple SKB
> data shared areas.
>
> And because of this a doubly-linked list like list_head won't work at
> all.
>
> For the optimized case of pskb_expand_head() you discovered we can avoid
> this sharing. And at this point I imagine that for the remaining cases
> we can simply copy the full SKBs in the frag list to avoid this list
> sharing.
Eric, this goes on top of your patch and demonstrates the idea.
Please review if you have a chance:
diff --git a/net/core/skbuff.c b/net/core/skbuff.c
index 2d1bc76..aeb56af 100644
--- a/net/core/skbuff.c
+++ b/net/core/skbuff.c
@@ -327,6 +327,32 @@ static void skb_clone_fraglist(struct sk_buff *skb)
skb_get(list);
}
+static struct sk_buff *skb_copy_fraglist(struct sk_buff *parent,
+ gfp_t gfp_mask)
+{
+ struct sk_buff *first_skb = NULL;
+ struct sk_buff *prev_skb = NULL;
+ struct sk_buff *skb;
+
+ skb_walk_frags(parent, skb) {
+ struct sk_buff *nskb = pskb_copy(skb, gfp_mask);
+
+ if (!nskb)
+ goto fail;
+ if (!first_skb)
+ first_skb = skb;
+ else
+ prev_skb->next = skb;
+ prev_skb = skb;
+ }
+
+ return first_skb;
+
+fail:
+ skb_drop_list(&first_skb);
+ return NULL;
+}
+
static void skb_release_data(struct sk_buff *skb)
{
if (!skb->cloned ||
@@ -812,17 +838,22 @@ int pskb_expand_head(struct sk_buff *skb, int nhead, int ntail,
fastpath = atomic_read(&skb_shinfo(skb)->dataref) == delta;
}
- if (fastpath) {
- kfree(skb->head);
- } else {
+ if (!fastpath) {
+ if (skb_has_frag_list(skb)) {
+ struct sk_buff *new_list;
+
+ new_list = skb_copy_fraglist(skb, gfp_mask);
+ if (!new_list)
+ goto free_data;
+ skb_shinfo(skb)->frag_list = new_list;
+ }
for (i = 0; i < skb_shinfo(skb)->nr_frags; i++)
get_page(skb_shinfo(skb)->frags[i].page);
- if (skb_has_frag_list(skb))
- skb_clone_fraglist(skb);
-
- skb_release_data(skb);
}
+
+ kfree(skb->head);
+
off = (data + nhead) - skb->head;
skb->head = data;
@@ -848,6 +879,8 @@ int pskb_expand_head(struct sk_buff *skb, int nhead, int ntail,
atomic_set(&skb_shinfo(skb)->dataref, 1);
return 0;
+free_data:
+ kfree(data);
nodata:
return -ENOMEM;
}
next prev parent reply other threads:[~2010-09-07 2:19 UTC|newest]
Thread overview: 33+ messages / expand[flat|nested] mbox.gz Atom feed top
2010-09-03 3:43 [PATCH] net: Frag list lost on head expansion David Miller
2010-09-03 5:48 ` Eric Dumazet
2010-09-03 6:19 ` Eric Dumazet
2010-09-03 9:09 ` [PATCH net-next-2.6] net: pskb_expand_head() optimization Eric Dumazet
2010-09-03 13:46 ` David Miller
2010-09-07 2:20 ` David Miller [this message]
2010-09-07 5:02 ` Eric Dumazet
2010-09-07 5:05 ` David Miller
2010-09-07 9:16 ` Jarek Poplawski
2010-09-07 9:37 ` Eric Dumazet
2010-09-10 19:54 ` David Miller
2010-09-11 12:31 ` Jarek Poplawski
2010-09-12 3:30 ` David Miller
2010-09-12 10:45 ` Jarek Poplawski
2010-09-12 10:58 ` Jarek Poplawski
2010-09-12 15:58 ` David Miller
2010-09-12 16:13 ` David Miller
2010-09-12 20:57 ` Jarek Poplawski
2010-09-12 22:08 ` David Miller
2010-09-13 7:49 ` Jarek Poplawski
2010-09-12 19:55 ` Ben Pfaff
2010-09-12 20:24 ` David Miller
2010-09-12 20:45 ` Jarek Poplawski
2010-09-20 0:17 ` David Miller
2010-09-20 7:21 ` Jarek Poplawski
2010-09-20 9:02 ` Eric Dumazet
2010-09-20 9:14 ` Jarek Poplawski
2010-09-20 12:12 ` Jarek Poplawski
2010-09-20 12:40 ` Eric Dumazet
2010-09-20 16:59 ` David Miller
2010-09-07 1:25 ` David Miller
-- strict thread matches above, loose matches on Subject: below --
2010-07-22 19:12 [PATCH] Fix corruption of skb csum field in pskb_expand_head() of net/core/skbuff.c Andrea Shepard
2010-07-23 5:09 ` [PATCH net-next-2.6] net: pskb_expand_head() optimization Eric Dumazet
2010-07-25 4:06 ` David Miller
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=20100906.192003.232914158.davem@davemloft.net \
--to=davem@davemloft.net \
--cc=eric.dumazet@gmail.com \
--cc=netdev@vger.kernel.org \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is an external index of several public inboxes,
see mirroring instructions on how to clone and mirror
all data and code used by this external index.