Re: [PATCH 2/2] Revert "mm, thp: restore node-local hugepage allocations"

From: David Rientjes
Date: Wed May 15 2019 - 16:28:40 EST


On Fri, 3 May 2019, Andrea Arcangeli wrote:

> This reverts commit 2f0799a0ffc033bf3cc82d5032acc3ec633464c2.
>
> commit 2f0799a0ffc033bf3cc82d5032acc3ec633464c2 was rightfully applied
> to avoid the risk of a severe regression that was reported by the
> kernel test robot at the end of the merge window. Now we understood
> the regression was a false positive and was caused by a significant
> increase in fairness during a swap trashing benchmark. So it's safe to
> re-apply the fix and continue improving the code from there. The
> benchmark that reported the regression is very useful, but it provides
> a meaningful result only when there is no significant alteration in
> fairness during the workload. The removal of __GFP_THISNODE increased
> fairness.
>

Hi Andrea,

There was exhausting discussion subsequent to this that caused Linus to
have to revert the offending commit late in an rc series that is not
described here. This was after the offending commit, which this commit
now reintroduces, was described as causing user facing access latency
regressions and nacked. The same objection is obviously going to be made
here and I'd really prefer if this could be worked out without yet another
merge into -mm, push to Linus, and revert by Linus. There are solutions
to this issue that does not cause anybody to have performance regressions
rather than reintroducing them for a class of users that use the
overloaded MADV_HUGEPAGE for the purposes it has provided them over the
past three years.

> __GFP_THISNODE cannot be used in the generic page faults path for new
> memory allocations under the MPOL_DEFAULT mempolicy, or the allocation
> behavior significantly deviates from what the MPOL_DEFAULT semantics
> are supposed to be for THP and 4k allocations alike.
>

This isn't an argument in support of this patch, there is a difference
between (1) pages of the native page size being faulted first locally
falling back remotely and (2) hugepages being faulted first locally and
falling back to native pages locally because it has better access latency
on most platforms for workloads that do not span multiple nodes. Note
that the page allocator is unaware whether the workload spans multiple
nodes so it cannot make this distinction today, and that's what I'd prefer
to focus on rather than changing an overall policy for everybody.

> Setting THP defrag to "always" or using MADV_HUGEPAGE (with THP defrag
> set to "madvise") has never meant to provide an implicit MPOL_BIND on
> the "current" node the task is running on, causing swap storms and
> providing a much more aggressive behavior than even zone_reclaim_node
> = 3.
>

It may not have been meant to provide this, but when IBM changed this
three years ago because of performance regressions and others have started
to use MADV_HUGEPAGE with that policy in mind, it is the reality of what
the madvise advice has provided. What was meant to be semantics of
MADV_HUGEPAGE three years ago is irrelevant today if it introduces
performance regressions for users who have used the advice mode during
that past three years.

> Any workload who could have benefited from __GFP_THISNODE has now to
> enable zone_reclaim_mode=1||2||3. __GFP_THISNODE implicitly provided
> the zone_reclaim_mode behavior, but it only did so if THP was enabled:
> if THP was disabled, there would have been no chance to get any 4k
> page from the current node if the current node was full of pagecache,
> which further shows how this __GFP_THISNODE was misplaced in
> MADV_HUGEPAGE. MADV_HUGEPAGE has never been intended to provide any
> zone_reclaim_mode semantics, in fact the two are orthogonal,
> zone_reclaim_mode = 1|2|3 must work exactly the same with
> MADV_HUGEPAGE set or not.
>
> The performance characteristic of memory depends on the hardware
> details. The numbers below are obtained on Naples/EPYC architecture
> and the N/A projection extends them to show what we should aim for in
> the future as a good THP NUMA locality default. The benchmark used
> exercises random memory seeks (note: the cost of the page faults is
> not part of the measurement).
>
> D0 THP | D0 4k | D1 THP | D1 4k | D2 THP | D2 4k | D3 THP | D3 4k | ...
> 0% | +43% | +45% | +106% | +131% | +224% | N/A | N/A
>

The performance measurements that we have on Naples shows a more
significant change between D0 4k and D1 THP: it certainly is not 2% worse
access latency to a remote hugepage compared to local native pages.

> D0 means distance zero (i.e. local memory), D1 means distance
> one (i.e. intra socket memory), D2 means distance two (i.e. inter
> socket memory), etc...
>
> For the guest physical memory allocated by qemu and for guest mode kernel
> the performance characteristic of RAM is more complex and an ideal
> default could be:
>
> D0 THP | D1 THP | D0 4k | D2 THP | D1 4k | D3 THP | D2 4k | D3 4k | ...
> 0% | +58% | +101% | N/A | +222% | N/A | N/A | N/A
>
> NOTE: the N/A are projections and haven't been measured yet, the
> measurement in this case is done on a 1950x with only two NUMA nodes.
> The THP case here means THP was used both in the host and in the
> guest.
>

Yes, this is clearly understood and was never objected to when this first
came up in the thread where __GFP_THISNODE was removed or when Linus
reverted the patch.

The issue being discussed here is a result of MADV_HUGEPAGE being
overloaded: it cannot mean to control (1) how much compaction/reclaim is
done for page allocation, (2) the NUMA locality of those hugepages, and
(3) the eligibility of the memory to be collapsed into hugepages by
khugepaged all at the same time.

I suggested then that we actually define (2) concretely specifically for
the usecase that you mention. Changing the behavior of MADV_HUGEPAGE for
the past three years, however, and introducing performance regressions for
those users is not an option regardless of the intent that it had when
developed.

I suggested two options: (1) __MPOL_F_HUGE flag to set a mempolicy for
specific memory ranges so that you can define thp specific mempolicies
(Vlastimil considered this to be a lot of work, which I agreed) or (2) a
prctl() mode to specify that a workload will span multiple sockets and
benefits from remote hugepage allocation over local native pages (or
because it is faulting memory remotely that it will access locally at some
point in the future depending on cpu binding). Any prctl() mode can be
inherited across fork so it can be used for the qemu case that you suggest
and is a very simple change to make compared with (1).

Please consider methods to accomplish this goal that will not cause
existing users of MADV_HUGEPAGE to incur 13.9% access latency regressions
and have no way to workaround without MPOL_BIND that will introduce
undeserved and unnecessary oom kills because we can't specify native page
vs hugepage mempolicies independently.

I'm confident that everybody on this cc list is well aware of both sides
of this discussion and I hope that we can work together to address it to
achieve the goals of both.

Thanks.