-
Notifications
You must be signed in to change notification settings - Fork 17.9k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
runtime: the scavenger doesn't call MADV_NOHUGEPAGE
like it used to
#55328
Comments
Thinking about this more, what my proposed solution loses is the benefit of backing dense parts of the heap with huge pages. However, back when we worked on the scavenger, there was a key insight about having a GC'd heap: as we reach the end of a GC cycle, the whole heap is always densely packed. I wonder if we should say something like "back up the heap with At first I thought the goroutine doing mark termination could spend a little extra time post-STW to go over the (mostly-contiguous) heap and call This is in effect a very simple "huge page aware" allocation policy without needing to do any special bin-packing. We're taking advantage of the fact that we can count on heap density, thanks to the GC. |
The idea in the last comment doesn't really work. Well, it kind of does. It definitely reduces the number of VMAs, however it causes a small performance regression. The issue is that the scavenger still breaks up huge pages while the heap is growing, which I partially mitigated by adding a heuristic to not turn on the scavenger while the heap is growing. This wouldn't be so bad if the huge pages were collapsed again lazily, but because we set So, new idea: track page alloc chunk occupancy. If it exceeds 85%, mark it as This is much more direct: we're just trying to back the high-occupancy chunks with huge pages, exactly where we get the most benefit. |
Change https://go.dev/cl/436395 mentions this issue: |
I think the latest iteration of https://go.dev/cl/436395 implements a decent huge page policy. It leverages both the occupancy heuristic and general assumptions about heap density. 4 MiB chunks that are consistently densely packed (96%+ occupancy, because Scavenging for the memory limit or |
(The purely occupancy-based implementation had churn issues as well.) |
Change https://go.dev/cl/460316 mentions this issue: |
For #55328. Change-Id: I8792161f09906c08d506cc0ace9d07e76ec6baa6 Reviewed-on: https://go-review.googlesource.com/c/go/+/460316 Reviewed-by: Michael Pratt <mpratt@google.com> Run-TryBot: Michael Knyszek <mknyszek@google.com> TryBot-Result: Gopher Robot <gobot@golang.org>
Change https://go.dev/cl/526615 mentions this issue: |
For golang/go#8832. For golang/go#55328. For golang/go#61718. Change-Id: I1ee51424dc2591a84f09ca8687c113f0af3550d1 Reviewed-on: https://go-review.googlesource.com/c/website/+/526615 Auto-Submit: Michael Knyszek <mknyszek@google.com> Reviewed-by: Michael Pratt <mpratt@google.com> LUCI-TryBot-Result: Go LUCI <golang-scoped@luci-project-accounts.iam.gserviceaccount.com>
For golang/go#8832. For golang/go#55328. For golang/go#61718. Change-Id: I1ee51424dc2591a84f09ca8687c113f0af3550d1 Reviewed-on: https://go-review.googlesource.com/c/website/+/526615 Auto-Submit: Michael Knyszek <mknyszek@google.com> Reviewed-by: Michael Pratt <mpratt@google.com> LUCI-TryBot-Result: Go LUCI <golang-scoped@luci-project-accounts.iam.gserviceaccount.com>
Because the Go runtime's background scavenger releases memory in 64 KiB chunks, the heuristic in
mem_linux.go
's implementation ofsysUnusedOS
never actually callsMADV_NOHUGEPAGE
to break up huge pages that would otherwise keep a lot of memory around. The reason for this is that there's a heuristic preventing the call on everysysUnusedOS
to avoid creating too many VMAs (Linux has a low limit).The result of all this is that Linux may be keeping around huge pages transparently using up a lot more memory than intended, and reducing the effectiveness of the scavenger.
I think the fix for this is to mark all new heap memory as
MADV_NOHUGEPAGE
initially, and then only call it again in circumstances where we free memory that is likely to have hadMADV_HUGEPAGE
called on it (e.g. a large object that contains at least one aligned 2 MiB region will have this happen viasysUsed
). However, this might not be enough, and needs some more investigation.CC @golang/runtime
The text was updated successfully, but these errors were encountered: