[<prev] [next>] [<thread-prev] [day] [month] [year] [list]
Message-ID: <20120213111058.GW5938@suse.de>
Date: Mon, 13 Feb 2012 11:10:58 +0000
From: Mel Gorman <mgorman@...e.de>
To: Christoph Lameter <cl@...ux.com>
Cc: Andrew Morton <akpm@...ux-foundation.org>,
Linux-MM <linux-mm@...ck.org>,
Linux-Netdev <netdev@...r.kernel.org>,
LKML <linux-kernel@...r.kernel.org>,
David Miller <davem@...emloft.net>, Neil Brown <neilb@...e.de>,
Peter Zijlstra <a.p.zijlstra@...llo.nl>,
Pekka Enberg <penberg@...helsinki.fi>
Subject: Re: [PATCH 02/15] mm: sl[au]b: Add knowledge of PFMEMALLOC reserve
pages
On Fri, Feb 10, 2012 at 03:01:37PM -0600, Christoph Lameter wrote:
> On Fri, 10 Feb 2012, Mel Gorman wrote:
>
> > I have an updated version of this 02/15 patch below. It passed testing
> > and is a lot less invasive than the previous release. As you suggested,
> > it uses page flags and the bulk of the complexity is only executed if
> > someone is using network-backed storage.
>
> Hmmm.. hmm... Still modifies the hotpaths of the allocators for a
> pretty exotic feature.
>
> > > On top of that you want to add
> > > special code in various subsystems to also do that over the network.
> > > Sigh. I think we agreed a while back that we want to limit the amount of
> > > I/O triggered from reclaim paths?
> >
> > Specifically we wanted to reduce or stop page reclaim calling ->writepage()
> > for file-backed pages because it generated awful IO patterns and deep
> > call stacks. We still write anonymous pages from page reclaim because we
> > do not have a dedicated thread for writing to swap. It is expected that
> > the call stack for writing to network storage would be less than a
> > filesystem.
> >
> > > AFAICT many filesystems do not support
> > > writeout from reclaim anymore because of all the issues that arise at that
> > > level.
> > >
> >
> > NBD is a block device so filesystem restrictions like you mention do not
> > apply. In NFS, the direct_IO paths are used to write pages not
> > ->writepage so again the restriction does not apply.
>
> Block devices are a little simpler ok. But it is still not a desirable
> thing to do (just think about raid and other complex filesystems that may
> also have to do allocations).
Swap IO is never desirable but it has to happen somehow and right now, we
only initiate swap IO from direct reclaim or kswapd. For complex filesystems,
it is mandatory if they are using direct_IO like I do for NFS that they
pin the necessary structures in advance to avoid any allocations in their
reclaim path. I do not expect RAID to be used over network-based swap files.
> I do not think that block device writers
> code with the VM in mind. In the case of network devices as block devices
> we have a pretty serious problem since the network subsystem is certainly
> not designed to be called from VM reclaim code that may be triggered
> arbitrarily from deeply nested other code in the kernel. Implementing
> something like this invites breakage all over the place to show up.
>
The whole point of the series is to allow the possibility of using
network-based swap devices starting with NBD and with NFS in the related
series. swap-over-NFS has been used for the last few years by enterprise
distros and while bugs do get reported, they are rare.
As the person that introduced this, I would support it in mainline for
NBD and NFS if it was merged.
> > index 8b3b8cf..6a3fa1c 100644
> > --- a/mm/page_alloc.c
> > +++ b/mm/page_alloc.c
> > @@ -695,6 +695,7 @@ static bool free_pages_prepare(struct page *page, unsigned int order)
> > trace_mm_page_free(page, order);
> > kmemcheck_free_shadow(page, order);
> >
> > + page->pfmemalloc = false;
> > if (PageAnon(page))
> > page->mapping = NULL;
> > for (i = 0; i < (1 << order); i++)
> > @@ -1221,6 +1222,7 @@ void free_hot_cold_page(struct page *page, int cold)
> >
> > migratetype = get_pageblock_migratetype(page);
> > set_page_private(page, migratetype);
> > + page->pfmemalloc = false;
> > local_irq_save(flags);
> > if (unlikely(wasMlocked))
> > free_page_mlock(page);
>
> page allocator hotpaths affected.
>
I can remove these but then page->pfmemalloc has to be set on the allocation
side. It's a single write to a dirty cache line that is already local to
the processor. It's not measurable although I accept that the page
allocator paths could do with a diet in general.
> > diff --git a/mm/slab.c b/mm/slab.c
> > index f0bd785..f322dc2 100644
> > --- a/mm/slab.c
> > +++ b/mm/slab.c
> > @@ -123,6 +123,8 @@
> >
> > #include <trace/events/kmem.h>
> >
> > +#include "internal.h"
> > +
> > /*
> > * DEBUG - 1 for kmem_cache_create() to honour; SLAB_RED_ZONE & SLAB_POISON.
> > * 0 for faster, smaller code (especially in the critical paths).
> > @@ -151,6 +153,12 @@
> > #define ARCH_KMALLOC_FLAGS SLAB_HWCACHE_ALIGN
> > #endif
> >
> > +/*
> > + * true if a page was allocated from pfmemalloc reserves for network-based
> > + * swap
> > + */
> > +static bool pfmemalloc_active;
>
> Implying an additional cacheline use in critical slab paths?
This was the alternative to altering the slub structures.
> Hopefully grouped with other variables already in cache.
>
This is the expectation. I considered tagging it read_mostly but didn't
at the time. I will now because it is genuinely expected to be
read-mostly and in the case where it is being written to, we're also
using network-based swap and the cost of a cache miss will be negligible
in comparison to swapping under memory pressure to a network.
> > @@ -3243,23 +3380,35 @@ static inline void *____cache_alloc(struct kmem_cache *cachep, gfp_t flags)
> > {
> > void *objp;
> > struct array_cache *ac;
> > + bool force_refill = false;
>
> ... hitting the hotpath here.
>
> > @@ -3693,12 +3845,12 @@ static inline void __cache_free(struct kmem_cache *cachep, void *objp,
> >
> > if (likely(ac->avail < ac->limit)) {
> > STATS_INC_FREEHIT(cachep);
> > - ac->entry[ac->avail++] = objp;
> > + ac_put_obj(cachep, ac, objp);
> > return;
> > } else {
> > STATS_INC_FREEMISS(cachep);
> > cache_flusharray(cachep, ac);
> > - ac->entry[ac->avail++] = objp;
> > + ac_put_obj(cachep, ac, objp);
> > }
> > }
>
> and here.
>
The impact of ac_put_obj() is reduced in a later patch and becomes just
an additional read of a global variable. There was not an obvious way to
me to ensure pfmemalloc pages were not used for !pfmemalloc allocations
without having some sort of impact.
>
> > diff --git a/mm/slub.c b/mm/slub.c
> > index 4907563..8eed0de 100644
> > --- a/mm/slub.c
> > +++ b/mm/slub.c
>
> > @@ -2304,8 +2327,8 @@ redo:
> > barrier();
> >
> > object = c->freelist;
> > - if (unlikely(!object || !node_match(c, node)))
> > -
> > + if (unlikely(!object || !node_match(c, node) ||
> > + !pfmemalloc_match(c, gfpflags)))
> > object = __slab_alloc(s, gfpflags, node, addr, c);
> >
> > else {
>
>
> Modification to hotpath. That could be fixed here by forcing pfmemalloc
> (like debug allocs) to always go to the slow path and checking in there
> instead. Just keep c->freelist == NULL.
>
I picked up your patch for this, thanks.
--
Mel Gorman
SUSE Labs
--
To unsubscribe from this list: send the line "unsubscribe netdev" in
the body of a message to majordomo@...r.kernel.org
More majordomo info at http://vger.kernel.org/majordomo-info.html
Powered by blists - more mailing lists