Commit 2ab6d017 authored by Michael Anthony Knyszek's avatar Michael Anthony Knyszek Committed by Michael Knyszek

runtime: throw if scavenge necessary during coalescing

Currently when coalescing if two adjacent spans are scavenged, we
subtract their sizes from memstats and re-scavenge the new combined
span. This is wasteful however, since the realignment semantics make
this case of having to re-scavenge impossible.

In realign() inside of coalesce(), there was also a bug: on systems
where physPageSize > pageSize, we wouldn't realign because a condition
had the wrong sign. This wasteful re-scavenging has been masking this
bug this whole time. So, this change fixes that first.

Then this change gets rid of the needsScavenge logic and instead checks
explicitly for the possibility of unscavenged pages near the physical
page boundary. If the possibility exists, it throws. The intent of
throwing here is to catch changes to the runtime which cause this
invariant to no longer hold, at which point it would likely be
appropriate to scavenge the additional pages (and only the additional
pages) at that point.

Change-Id: I185e3d7b53e36e90cf9ace5fa297a9e8008d75f3
Reviewed-on: https://go-review.googlesource.com/c/go/+/158377
Run-TryBot: Michael Knyszek <mknyszek@google.com>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: default avatarAustin Clements <austin@google.com>
parent 512b3c63
...@@ -428,28 +428,40 @@ func (s *mspan) physPageBounds() (uintptr, uintptr) { ...@@ -428,28 +428,40 @@ func (s *mspan) physPageBounds() (uintptr, uintptr) {
} }
func (h *mheap) coalesce(s *mspan) { func (h *mheap) coalesce(s *mspan) {
// We scavenge s at the end after coalescing if s or anything
// it merged with is marked scavenged.
needsScavenge := false
prescavenged := s.released() // number of bytes already scavenged.
// merge is a helper which merges other into s, deletes references to other // merge is a helper which merges other into s, deletes references to other
// in heap metadata, and then discards it. other must be adjacent to s. // in heap metadata, and then discards it. other must be adjacent to s.
merge := func(other *mspan) { merge := func(a, b, other *mspan) {
// Caller must ensure a.startAddr < b.startAddr and that either a or
// b is s. a and b must be adjacent. other is whichever of the two is
// not s.
if pageSize < physPageSize && a.scavenged && b.scavenged {
// If we're merging two scavenged spans on systems where
// pageSize < physPageSize, then their boundary should always be on
// a physical page boundary, due to the realignment that happens
// during coalescing. Throw if this case is no longer true, which
// means the implementation should probably be changed to scavenge
// along the boundary.
_, start := a.physPageBounds()
end, _ := b.physPageBounds()
if start != end {
println("runtime: a.base=", hex(a.base()), "a.npages=", a.npages)
println("runtime: b.base=", hex(b.base()), "b.npages=", b.npages)
println("runtime: physPageSize=", physPageSize, "pageSize=", pageSize)
throw("neighboring scavenged spans boundary is not a physical page boundary")
}
}
// Adjust s via base and npages and also in heap metadata. // Adjust s via base and npages and also in heap metadata.
s.npages += other.npages s.npages += other.npages
s.needzero |= other.needzero s.needzero |= other.needzero
if other.startAddr < s.startAddr { if a == s {
h.setSpan(s.base()+s.npages*pageSize-1, s)
} else {
s.startAddr = other.startAddr s.startAddr = other.startAddr
h.setSpan(s.base(), s) h.setSpan(s.base(), s)
} else {
h.setSpan(s.base()+s.npages*pageSize-1, s)
} }
// If before or s are scavenged, then we need to scavenge the final coalesced span.
needsScavenge = needsScavenge || other.scavenged || s.scavenged
prescavenged += other.released()
// The size is potentially changing so the treap needs to delete adjacent nodes and // The size is potentially changing so the treap needs to delete adjacent nodes and
// insert back as a combined node. // insert back as a combined node.
if other.scavenged { if other.scavenged {
...@@ -468,9 +480,9 @@ func (h *mheap) coalesce(s *mspan) { ...@@ -468,9 +480,9 @@ func (h *mheap) coalesce(s *mspan) {
// b is s. a and b must be adjacent. other is whichever of the two is // b is s. a and b must be adjacent. other is whichever of the two is
// not s. // not s.
// If pageSize <= physPageSize then spans are always aligned // If pageSize >= physPageSize then spans are always aligned
// to physical page boundaries, so just exit. // to physical page boundaries, so just exit.
if pageSize <= physPageSize { if pageSize >= physPageSize {
return return
} }
// Since we're resizing other, we must remove it from the treap. // Since we're resizing other, we must remove it from the treap.
...@@ -505,7 +517,7 @@ func (h *mheap) coalesce(s *mspan) { ...@@ -505,7 +517,7 @@ func (h *mheap) coalesce(s *mspan) {
// Coalesce with earlier, later spans. // Coalesce with earlier, later spans.
if before := spanOf(s.base() - 1); before != nil && before.state == mSpanFree { if before := spanOf(s.base() - 1); before != nil && before.state == mSpanFree {
if s.scavenged == before.scavenged { if s.scavenged == before.scavenged {
merge(before) merge(before, s, before)
} else { } else {
realign(before, s, before) realign(before, s, before)
} }
...@@ -514,26 +526,11 @@ func (h *mheap) coalesce(s *mspan) { ...@@ -514,26 +526,11 @@ func (h *mheap) coalesce(s *mspan) {
// Now check to see if next (greater addresses) span is free and can be coalesced. // Now check to see if next (greater addresses) span is free and can be coalesced.
if after := spanOf(s.base() + s.npages*pageSize); after != nil && after.state == mSpanFree { if after := spanOf(s.base() + s.npages*pageSize); after != nil && after.state == mSpanFree {
if s.scavenged == after.scavenged { if s.scavenged == after.scavenged {
merge(after) merge(s, after, after)
} else { } else {
realign(s, after, after) realign(s, after, after)
} }
} }
if needsScavenge {
// When coalescing spans, some physical pages which
// were not returned to the OS previously because
// they were only partially covered by the span suddenly
// become available for scavenging. We want to make sure
// those holes are filled in, and the span is properly
// scavenged. Rather than trying to detect those holes
// directly, we collect how many bytes were already
// scavenged above and subtract that from heap_released
// before re-scavenging the entire newly-coalesced span,
// which will implicitly bump up heap_released.
memstats.heap_released -= uint64(prescavenged)
s.scavenge()
}
} }
func (s *mspan) scavenge() uintptr { func (s *mspan) scavenge() uintptr {
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment