diff options
Diffstat (limited to 'net/core/skbuff.c')
| -rw-r--r-- | net/core/skbuff.c | 32 | 
1 files changed, 25 insertions, 7 deletions
| diff --git a/net/core/skbuff.c b/net/core/skbuff.c index d1967dab9cc6..41ec02242ea7 100644 --- a/net/core/skbuff.c +++ b/net/core/skbuff.c @@ -280,13 +280,14 @@ nodata:  EXPORT_SYMBOL(__alloc_skb);  /** - * build_skb - build a network buffer + * __build_skb - build a network buffer   * @data: data buffer provided by caller - * @frag_size: size of fragment, or 0 if head was kmalloced + * @frag_size: size of data, or 0 if head was kmalloced   *   * Allocate a new &sk_buff. Caller provides space holding head and   * skb_shared_info. @data must have been allocated by kmalloc() only if - * @frag_size is 0, otherwise data should come from the page allocator. + * @frag_size is 0, otherwise data should come from the page allocator + *  or vmalloc()   * The return is the new skb buffer.   * On a failure the return is %NULL, and @data is not freed.   * Notes : @@ -297,7 +298,7 @@ EXPORT_SYMBOL(__alloc_skb);   *  before giving packet to stack.   *  RX rings only contains data buffers, not full skbs.   */ -struct sk_buff *build_skb(void *data, unsigned int frag_size) +struct sk_buff *__build_skb(void *data, unsigned int frag_size)  {  	struct skb_shared_info *shinfo;  	struct sk_buff *skb; @@ -311,7 +312,6 @@ struct sk_buff *build_skb(void *data, unsigned int frag_size)  	memset(skb, 0, offsetof(struct sk_buff, tail));  	skb->truesize = SKB_TRUESIZE(size); -	skb->head_frag = frag_size != 0;  	atomic_set(&skb->users, 1);  	skb->head = data;  	skb->data = data; @@ -328,6 +328,23 @@ struct sk_buff *build_skb(void *data, unsigned int frag_size)  	return skb;  } + +/* build_skb() is wrapper over __build_skb(), that specifically + * takes care of skb->head and skb->pfmemalloc + * This means that if @frag_size is not zero, then @data must be backed + * by a page fragment, not kmalloc() or vmalloc() + */ +struct sk_buff *build_skb(void *data, unsigned int frag_size) +{ +	struct sk_buff *skb = __build_skb(data, frag_size); + +	if (skb && frag_size) { +		skb->head_frag = 1; +		if (virt_to_head_page(data)->pfmemalloc) +			skb->pfmemalloc = 1; +	} +	return skb; +}  EXPORT_SYMBOL(build_skb);  struct netdev_alloc_cache { @@ -348,7 +365,8 @@ static struct page *__page_frag_refill(struct netdev_alloc_cache *nc,  	gfp_t gfp = gfp_mask;  	if (order) { -		gfp_mask |= __GFP_COMP | __GFP_NOWARN | __GFP_NORETRY; +		gfp_mask |= __GFP_COMP | __GFP_NOWARN | __GFP_NORETRY | +			    __GFP_NOMEMALLOC;  		page = alloc_pages_node(NUMA_NO_NODE, gfp_mask, order);  		nc->frag.size = PAGE_SIZE << (page ? order : 0);  	} @@ -4380,7 +4398,7 @@ struct sk_buff *alloc_skb_with_frags(unsigned long header_len,  		while (order) {  			if (npages >= 1 << order) { -				page = alloc_pages(gfp_mask | +				page = alloc_pages((gfp_mask & ~__GFP_WAIT) |  						   __GFP_COMP |  						   __GFP_NOWARN |  						   __GFP_NORETRY, |