Skip to content

Commit a1c7fff

Browse files
edumazetdavem330
authored andcommitted
net: netdev_alloc_skb() use build_skb()
netdev_alloc_skb() is used by networks driver in their RX path to allocate an skb to receive an incoming frame. With recent skb->head_frag infrastructure, it makes sense to change netdev_alloc_skb() to use build_skb() and a frag allocator. This permits a zero copy splice(socket->pipe), and better GRO or TCP coalescing. Signed-off-by: Eric Dumazet <edumazet@google.com> Signed-off-by: David S. Miller <davem@davemloft.net>
1 parent 1de5a71 commit a1c7fff

File tree

1 file changed

+31
-1
lines changed

1 file changed

+31
-1
lines changed

net/core/skbuff.c

Lines changed: 31 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -293,6 +293,12 @@ struct sk_buff *build_skb(void *data, unsigned int frag_size)
293293
}
294294
EXPORT_SYMBOL(build_skb);
295295

296+
struct netdev_alloc_cache {
297+
struct page *page;
298+
unsigned int offset;
299+
};
300+
static DEFINE_PER_CPU(struct netdev_alloc_cache, netdev_alloc_cache);
301+
296302
/**
297303
* __netdev_alloc_skb - allocate an skbuff for rx on a specific device
298304
* @dev: network device to receive on
@@ -310,8 +316,32 @@ struct sk_buff *__netdev_alloc_skb(struct net_device *dev,
310316
unsigned int length, gfp_t gfp_mask)
311317
{
312318
struct sk_buff *skb;
319+
unsigned int fragsz = SKB_DATA_ALIGN(length + NET_SKB_PAD) +
320+
SKB_DATA_ALIGN(sizeof(struct skb_shared_info));
313321

314-
skb = __alloc_skb(length + NET_SKB_PAD, gfp_mask, 0, NUMA_NO_NODE);
322+
if (fragsz <= PAGE_SIZE && !(gfp_mask & __GFP_WAIT)) {
323+
struct netdev_alloc_cache *nc;
324+
void *data = NULL;
325+
326+
nc = &get_cpu_var(netdev_alloc_cache);
327+
if (!nc->page) {
328+
refill: nc->page = alloc_page(gfp_mask);
329+
nc->offset = 0;
330+
}
331+
if (likely(nc->page)) {
332+
if (nc->offset + fragsz > PAGE_SIZE) {
333+
put_page(nc->page);
334+
goto refill;
335+
}
336+
data = page_address(nc->page) + nc->offset;
337+
nc->offset += fragsz;
338+
get_page(nc->page);
339+
}
340+
put_cpu_var(netdev_alloc_cache);
341+
skb = data ? build_skb(data, fragsz) : NULL;
342+
} else {
343+
skb = __alloc_skb(length + NET_SKB_PAD, gfp_mask, 0, NUMA_NO_NODE);
344+
}
315345
if (likely(skb)) {
316346
skb_reserve(skb, NET_SKB_PAD);
317347
skb->dev = dev;

0 commit comments

Comments
 (0)