aboutsummaryrefslogtreecommitdiffstats
path: root/net/core
diff options
context:
space:
mode:
authorEric Dumazet <edumazet@google.com>2012-04-30 04:10:34 -0400
committerDavid S. Miller <davem@davemloft.net>2012-04-30 21:35:49 -0400
commitd7e8883cfcf4851afe74fb380cc62b7fa9cf66ba (patch)
tree1b513349c4f1c7edf7991de2016b3a2ed1083deb /net/core
parent8d4057a938481351dc690fbe23e8c72af08d5890 (diff)
net: make GRO aware of skb->head_frag
GRO can check if skb to be merged has its skb->head mapped to a page fragment, instead of a kmalloc() area. We 'upgrade' skb->head as a fragment in itself This avoids the frag_list fallback, and permits to build true GRO skb (one sk_buff and up to 16 fragments), using less memory. This reduces number of cache misses when user makes its copy, since a single sk_buff is fetched. This is a followup of patch "net: allow skb->head to be a page fragment" Signed-off-by: Eric Dumazet <edumazet@google.com> Cc: Ilpo Järvinen <ilpo.jarvinen@helsinki.fi> Cc: Herbert Xu <herbert@gondor.apana.org.au> Cc: Maciej Żenczykowski <maze@google.com> Cc: Neal Cardwell <ncardwell@google.com> Cc: Tom Herbert <therbert@google.com> Cc: Jeff Kirsher <jeffrey.t.kirsher@intel.com> Cc: Ben Hutchings <bhutchings@solarflare.com> Cc: Matt Carlson <mcarlson@broadcom.com> Cc: Michael Chan <mchan@broadcom.com> Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/core')
-rw-r--r--net/core/dev.c5
-rw-r--r--net/core/skbuff.c27
2 files changed, 30 insertions, 2 deletions
diff --git a/net/core/dev.c b/net/core/dev.c
index 501f3cc703dd..a2be59fe6ab8 100644
--- a/net/core/dev.c
+++ b/net/core/dev.c
@@ -3546,7 +3546,10 @@ gro_result_t napi_skb_finish(gro_result_t ret, struct sk_buff *skb)
3546 break; 3546 break;
3547 3547
3548 case GRO_MERGED_FREE: 3548 case GRO_MERGED_FREE:
3549 consume_skb(skb); 3549 if (NAPI_GRO_CB(skb)->free == NAPI_GRO_FREE_STOLEN_HEAD)
3550 kmem_cache_free(skbuff_head_cache, skb);
3551 else
3552 __kfree_skb(skb);
3550 break; 3553 break;
3551 3554
3552 case GRO_HELD: 3555 case GRO_HELD:
diff --git a/net/core/skbuff.c b/net/core/skbuff.c
index effa75d0e318..09cc38651b2f 100644
--- a/net/core/skbuff.c
+++ b/net/core/skbuff.c
@@ -69,7 +69,7 @@
69#include <trace/events/skb.h> 69#include <trace/events/skb.h>
70#include <linux/highmem.h> 70#include <linux/highmem.h>
71 71
72static struct kmem_cache *skbuff_head_cache __read_mostly; 72struct kmem_cache *skbuff_head_cache __read_mostly;
73static struct kmem_cache *skbuff_fclone_cache __read_mostly; 73static struct kmem_cache *skbuff_fclone_cache __read_mostly;
74 74
75static void sock_pipe_buf_release(struct pipe_inode_info *pipe, 75static void sock_pipe_buf_release(struct pipe_inode_info *pipe,
@@ -2901,6 +2901,31 @@ int skb_gro_receive(struct sk_buff **head, struct sk_buff *skb)
2901 2901
2902 NAPI_GRO_CB(skb)->free = 1; 2902 NAPI_GRO_CB(skb)->free = 1;
2903 goto done; 2903 goto done;
2904 } else if (skb->head_frag) {
2905 int nr_frags = pinfo->nr_frags;
2906 skb_frag_t *frag = pinfo->frags + nr_frags;
2907 struct page *page = virt_to_head_page(skb->head);
2908 unsigned int first_size = headlen - offset;
2909 unsigned int first_offset;
2910
2911 if (nr_frags + 1 + skbinfo->nr_frags > MAX_SKB_FRAGS)
2912 return -E2BIG;
2913
2914 first_offset = skb->data -
2915 (unsigned char *)page_address(page) +
2916 offset;
2917
2918 pinfo->nr_frags = nr_frags + 1 + skbinfo->nr_frags;
2919
2920 frag->page.p = page;
2921 frag->page_offset = first_offset;
2922 skb_frag_size_set(frag, first_size);
2923
2924 memcpy(frag + 1, skbinfo->frags, sizeof(*frag) * skbinfo->nr_frags);
2925 /* We dont need to clear skbinfo->nr_frags here */
2926
2927 NAPI_GRO_CB(skb)->free = NAPI_GRO_FREE_STOLEN_HEAD;
2928 goto done;
2904 } else if (skb_gro_len(p) != pinfo->gso_size) 2929 } else if (skb_gro_len(p) != pinfo->gso_size)
2905 return -E2BIG; 2930 return -E2BIG;
2906 2931