lists.openwall.net   lists  /  announce  owl-users  owl-dev  john-users  john-dev  passwdqc-users  yescrypt  popa3d-users  /  oss-security  kernel-hardening  musl  sabotage  tlsify  passwords  /  crypt-dev  xvendor  /  Bugtraq  Full-Disclosure  linux-kernel  linux-netdev  linux-ext4  linux-hardening  linux-cve-announce  PHC 
Open Source and information security mailing list archives
 
Hash Suite: Windows password security audit tool. GUI, reports in PDF.
[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <e2bf192391a86358f5c7502980268f17682bb328.camel@gmail.com>
Date:   Wed, 21 Sep 2022 11:11:57 -0700
From:   Alexander H Duyck <alexander.duyck@...il.com>
To:     Paolo Abeni <pabeni@...hat.com>, netdev@...r.kernel.org
Cc:     Eric Dumazet <edumazet@...gle.com>,
        "David S. Miller" <davem@...emloft.net>,
        Jakub Kicinski <kuba@...nel.org>
Subject: Re: [PATCH net-next] net: skb: introduce and use a single page frag
 cache

On Wed, 2022-09-21 at 18:41 +0200, Paolo Abeni wrote:
> After commit 3226b158e67c ("net: avoid 32 x truesize under-estimation
> for tiny skbs") we are observing 10-20% regressions in performance
> tests with small packets. The perf trace points to high pressure on
> the slab allocator.
> 
> This change tries to improve the allocation schema for small packets
> using an idea originally suggested by Eric: a new per CPU page frag is
> introduced and used in __napi_alloc_skb to cope with small allocation
> requests.
> 
> To ensure that the above does not lead to excessive truesize
> underestimation, the frag size for small allocation is inflated to 1K
> and all the above is restricted to build with 4K page size.
> 
> Note that we need to update accordingly the run-time check introduced
> with commit fd9ea57f4e95 ("net: add napi_get_frags_check() helper").
> 
> Alex suggested a smart page refcount schema to reduce the number
> of atomic operations and deal properly with pfmemalloc pages.
> 
> Under small packet UDP flood, I measure a 15% peak tput increases.
> 
> Suggested-by: Eric Dumazet <eric.dumazet@...il.com>
> Suggested-by: Alexander H Duyck <alexander.duyck@...il.com>
> Signed-off-by: Paolo Abeni <pabeni@...hat.com>
> ---
> @Eric, @Alex please let me know if you are comfortable with the
> attribution
> ---
>  include/linux/netdevice.h |   1 +
>  net/core/dev.c            |  17 ------
>  net/core/skbuff.c         | 115 +++++++++++++++++++++++++++++++++++++-
>  3 files changed, 113 insertions(+), 20 deletions(-)
> 
> diff --git a/include/linux/netdevice.h b/include/linux/netdevice.h
> index 9f42fc871c3b..a1938560192a 100644
> --- a/include/linux/netdevice.h
> +++ b/include/linux/netdevice.h
> @@ -3822,6 +3822,7 @@ void netif_receive_skb_list(struct list_head *head);
>  gro_result_t napi_gro_receive(struct napi_struct *napi, struct sk_buff *skb);
>  void napi_gro_flush(struct napi_struct *napi, bool flush_old);
>  struct sk_buff *napi_get_frags(struct napi_struct *napi);
> +void napi_get_frags_check(struct napi_struct *napi);
>  gro_result_t napi_gro_frags(struct napi_struct *napi);
>  struct packet_offload *gro_find_receive_by_type(__be16 type);
>  struct packet_offload *gro_find_complete_by_type(__be16 type);
> diff --git a/net/core/dev.c b/net/core/dev.c
> index d66c73c1c734..fa53830d0683 100644
> --- a/net/core/dev.c
> +++ b/net/core/dev.c
> @@ -6358,23 +6358,6 @@ int dev_set_threaded(struct net_device *dev, bool threaded)
>  }
>  EXPORT_SYMBOL(dev_set_threaded);
>  
> -/* Double check that napi_get_frags() allocates skbs with
> - * skb->head being backed by slab, not a page fragment.
> - * This is to make sure bug fixed in 3226b158e67c
> - * ("net: avoid 32 x truesize under-estimation for tiny skbs")
> - * does not accidentally come back.
> - */
> -static void napi_get_frags_check(struct napi_struct *napi)
> -{
> -	struct sk_buff *skb;
> -
> -	local_bh_disable();
> -	skb = napi_get_frags(napi);
> -	WARN_ON_ONCE(skb && skb->head_frag);
> -	napi_free_frags(napi);
> -	local_bh_enable();
> -}
> -
>  void netif_napi_add_weight(struct net_device *dev, struct napi_struct *napi,
>  			   int (*poll)(struct napi_struct *, int), int weight)
>  {
> diff --git a/net/core/skbuff.c b/net/core/skbuff.c
> index f1b8b20fc20b..2be11b487df1 100644
> --- a/net/core/skbuff.c
> +++ b/net/core/skbuff.c
> @@ -134,8 +134,73 @@ static void skb_under_panic(struct sk_buff *skb, unsigned int sz, void *addr)
>  #define NAPI_SKB_CACHE_BULK	16
>  #define NAPI_SKB_CACHE_HALF	(NAPI_SKB_CACHE_SIZE / 2)
>  
> +/* the compiler doesn't like 'SKB_TRUESIZE(GRO_MAX_HEAD) > 512', but we
> + * can imply such condition checking the double word and MAX_HEADER size
> + */
> +#if PAGE_SIZE == SZ_4K && (defined(CONFIG_64BIT) || MAX_HEADER > 64)
> +
> +#define NAPI_HAS_SMALL_PAGE_FRAG 1
> +
> +/* specializzed page frag allocator using a single order 0 page
> + * and slicing it into 1K sized fragment. Constrained to system
> + * with:
> + * - a very limited amount of 1K fragments fitting a single
> + *   page - to avoid excessive truesize underestimation
> + * - reasonably high truesize value for napi_get_frags()
> + *   allocation - to avoid memory usage increased compared
> + *   to kalloc, see __napi_alloc_skb()
> + *
> + */
> +struct page_frag_1k {
> +	void *va;
> +	u16 offset;
> +	bool pfmemalloc;
> +};
> +
> +static void *page_frag_alloc_1k(struct page_frag_1k *nc, gfp_t gfp)
> +{
> +	struct page *page;
> +	int offset;
> +
> +	if (likely(nc->va)) {
> +		offset = nc->offset - SZ_1K;
> +		if (likely(offset >= 0))
> +			goto out;
> +
> +		put_page(virt_to_page(nc->va));
> +	}
> +
> +	page = alloc_pages_node(NUMA_NO_NODE, gfp, 0);
> +	if (!page) {
> +		nc->va = NULL;
> +		return NULL;
> +	}
> +
> +	nc->va = page_address(page);
> +	nc->pfmemalloc = page_is_pfmemalloc(page);
> +	page_ref_add(page, PAGE_SIZE / SZ_1K);
> +	offset = PAGE_SIZE - SZ_1K;
> +
> +out:
> +	nc->offset = offset;
> +	return nc->va + offset;

So you might be better off organizing this around the offset rather
than the virtual address. As long as offset is 0 you know the page
isn't there and has to be replaced.

	offset = nc->offset - SZ_1K;
	if (offset >= 0)
		goto out;

	page = alloc_pages_node(NUMA_NO_NODE, gfp, 0);
	if (!page)
		return NULL;

	nc->va = page_address(page);
	nc->pfmemalloc = page_is_pfmemalloc(page);
	offset = PAGE_SIZE - SZ_1K;
	page_ref_add(page, offset / SZ_1K);
out:
	nc->offset = offset;
	return nc->va + offset;

That will save you from having to call put_page and cleans it up so you
only have to perform 1 conditional check instead of 2 in the fast path.

> +}
> +#else
> +#define NAPI_HAS_SMALL_PAGE_FRAG 0
> +
> +struct page_frag_1k {
> +};
> +
> +static void *page_frag_alloc_1k(struct page_frag_1k *nc, gfp_t gfp_mask)
> +{
> +	return NULL;
> +}
> +
> +#endif
> +

Rather than have this return NULL why not just point it at the
page_frag_alloc?

>  struct napi_alloc_cache {
>  	struct page_frag_cache page;
> +	struct page_frag_1k page_small;
>  	unsigned int skb_count;
>  	void *skb_cache[NAPI_SKB_CACHE_SIZE];
>  };
> @@ -143,6 +208,23 @@ struct napi_alloc_cache {
>  static DEFINE_PER_CPU(struct page_frag_cache, netdev_alloc_cache);
>  static DEFINE_PER_CPU(struct napi_alloc_cache, napi_alloc_cache);
>  
> +/* Double check that napi_get_frags() allocates skbs with
> + * skb->head being backed by slab, not a page fragment.
> + * This is to make sure bug fixed in 3226b158e67c
> + * ("net: avoid 32 x truesize under-estimation for tiny skbs")
> + * does not accidentally come back.
> + */
> +void napi_get_frags_check(struct napi_struct *napi)
> +{
> +	struct sk_buff *skb;
> +
> +	local_bh_disable();
> +	skb = napi_get_frags(napi);
> +	WARN_ON_ONCE(!NAPI_HAS_SMALL_PAGE_FRAG && skb && skb->head_frag);
> +	napi_free_frags(napi);
> +	local_bh_enable();
> +}
> +
>  void *__napi_alloc_frag_align(unsigned int fragsz, unsigned int align_mask)
>  {
>  	struct napi_alloc_cache *nc = this_cpu_ptr(&napi_alloc_cache);
> @@ -561,15 +643,39 @@ struct sk_buff *__napi_alloc_skb(struct napi_struct *napi, unsigned int len,
>  {
>  	struct napi_alloc_cache *nc;
>  	struct sk_buff *skb;
> +	bool pfmemalloc;
>  	void *data;
>  
>  	DEBUG_NET_WARN_ON_ONCE(!in_softirq());
>  	len += NET_SKB_PAD + NET_IP_ALIGN;
>  
> +	/* When the small frag allocator is available, prefer it over kmalloc
> +	 * for small fragments
> +	 */
> +	if (NAPI_HAS_SMALL_PAGE_FRAG && len <= SKB_WITH_OVERHEAD(1024)) {
> +		nc = this_cpu_ptr(&napi_alloc_cache);
> +
> +		if (sk_memalloc_socks())
> +			gfp_mask |= __GFP_MEMALLOC;
> +
> +		/* we are artificially inflating the allocation size, but
> +		 * that is not as bad as it may look like, as:
> +		 * - 'len' less then GRO_MAX_HEAD makes little sense
> +		 * - larger 'len' values lead to fragment size above 512 bytes
> +		 *   as per NAPI_HAS_SMALL_PAGE_FRAG definition
> +		 * - kmalloc would use the kmalloc-1k slab for such values
> +		 */
> +		len = SZ_1K;
> +
> +		data = page_frag_alloc_1k(&nc->page_small, gfp_mask);
> +		pfmemalloc = nc->page_small.pfmemalloc;
> +		goto check_data;
> +	}
> +

It might be better to place this code further down as a branch rather
than having to duplicate things up here such as the __GFP_MEMALLOC
setting.

You could essentially just put the lines getting the napi_alloc_cache
and adding the shared info after the sk_memalloc_socks() check. Then it
could just be an if/else block either calling page_frag_alloc or your
page_frag_alloc_1k.

>  	/* If requested length is either too small or too big,
>  	 * we use kmalloc() for skb->head allocation.
>  	 */
> -	if (len <= SKB_WITH_OVERHEAD(1024) ||
> +	if ((!NAPI_HAS_SMALL_PAGE_FRAG && len <= SKB_WITH_OVERHEAD(1024)) ||
>  	    len > SKB_WITH_OVERHEAD(PAGE_SIZE) ||
>  	    (gfp_mask & (__GFP_DIRECT_RECLAIM | GFP_DMA))) {
>  		skb = __alloc_skb(len, gfp_mask, SKB_ALLOC_RX | SKB_ALLOC_NAPI,
> @@ -587,6 +693,9 @@ struct sk_buff *__napi_alloc_skb(struct napi_struct *napi, unsigned int len,
>  		gfp_mask |= __GFP_MEMALLOC;
>  
>  	data = page_frag_alloc(&nc->page, len, gfp_mask);
> +	pfmemalloc = nc->page.pfmemalloc;
> +
> +check_data:
>  	if (unlikely(!data))
>  		return NULL;
>  
> @@ -596,8 +705,8 @@ struct sk_buff *__napi_alloc_skb(struct napi_struct *napi, unsigned int len,
>  		return NULL;
>  	}
>  
> -	if (nc->page.pfmemalloc)
> -		skb->pfmemalloc = 1;
> +	if (pfmemalloc)
> +		skb->pfmemalloc = pfmemalloc;
>  	skb->head_frag = 1;
>  
>  skb_success:

In regards to the pfmemalloc bits I wonder if it wouldn't be better to
just have them both using the page_frag_cache and just use a pointer to
that to populate the skb->pfmemalloc based on frag_cache->pfmemalloc at
the end?

Powered by blists - more mailing lists

Powered by Openwall GNU/*/Linux Powered by OpenVZ