Commit 608c1b0d authored by Austin Clements's avatar Austin Clements

runtime: scan objects with finalizers concurrently

This reduces pause time by ~25% relative to tip and by ~50% relative
to Go 1.5.1.

Currently one of the steps of STW mark termination is to loop (in
parallel) over all spans to find objects with finalizers in order to
mark all objects reachable from these objects and to treat the
finalizer special as a root. Unfortunately, even if there are no
finalizers at all, this loop takes roughly 1 ms/heap GB/core, so
multi-gigabyte heaps can quickly push our STW time past 10ms.

Fix this by moving this scan from mark termination to concurrent scan,
where it can run in parallel with mutators. The loop itself could also
be optimized, but this cost is small compared to concurrent marking.

Making this scan concurrent introduces two complications:

1) The scan currently walks the specials list of each span without
locking it, which is safe only with the world stopped. We fix this by
speculatively checking if a span has any specials (the vast majority
won't) and then locking the specials list only if there are specials
to check.

2) An object can have a finalizer set after concurrent scan, in which
case it won't have been marked appropriately by concurrent scan. If
the finalizer is a closure and is only reachable from the special, it
could be swept before it is run. Likewise, if the object is not marked
yet when the finalizer is set and then becomes unreachable before it
is marked, other objects reachable only from it may be swept before
the finalizer function is run. We fix this issue by making
addfinalizer ensure the same marking invariants as markroot does.

For multi-gigabyte heaps, this reduces max pause time by 20%–30%
relative to tip (depending on GOMAXPROCS) and by ~50% relative to Go
1.5.1 (where this loop was neither concurrent nor parallel). Here are
the results for the garbage benchmark:

               ---------------- max pause ----------------
Heap   Procs   Concurrent scan   STW parallel scan   1.5.1
24GB     12         18ms              23ms            37ms
24GB      4         18ms              25ms            37ms
 4GB      4         3.8ms            4.9ms           6.9ms

In all cases, 95%ile pause time is similar to the max pause time. This
also improves mean STW time by 10%–30%.

Fixes #11485.

Change-Id: I9359d8c3d120a51d23d924b52bf853a1299b1dfd
Reviewed-on: https://go-review.googlesource.com/14982Reviewed-by: default avatarRick Hudson <rlh@golang.org>
Run-TryBot: Austin Clements <austin@google.com>
TryBot-Result: Gobot Gobot <gobot@golang.org>
parent fbd2660a
...@@ -763,6 +763,12 @@ var work struct { ...@@ -763,6 +763,12 @@ var work struct {
alldone note alldone note
markfor *parfor markfor *parfor
// finalizersDone indicates that finalizers and objects with
// finalizers have been scanned by markroot. During concurrent
// GC, this happens during the concurrent scan phase. During
// STW GC, this happens during mark termination.
finalizersDone bool
bgMarkReady note // signal background mark worker has started bgMarkReady note // signal background mark worker has started
bgMarkDone uint32 // cas to 1 when at a background mark completion point bgMarkDone uint32 // cas to 1 when at a background mark completion point
// Background mark completion signaling // Background mark completion signaling
...@@ -938,6 +944,8 @@ func gc(mode gcMode) { ...@@ -938,6 +944,8 @@ func gc(mode gcMode) {
gcResetMarkState() gcResetMarkState()
work.finalizersDone = false
if mode == gcBackgroundMode { // Do as much work concurrently as possible if mode == gcBackgroundMode { // Do as much work concurrently as possible
gcController.startCycle() gcController.startCycle()
heapGoal = gcController.heapGoal heapGoal = gcController.heapGoal
...@@ -970,6 +978,10 @@ func gc(mode gcMode) { ...@@ -970,6 +978,10 @@ func gc(mode gcMode) {
// boundaries where there are up-pointers. // boundaries where there are up-pointers.
setGCPhase(_GCscan) setGCPhase(_GCscan)
// markrootSpans uses work.spans, so make sure
// it is up to date.
gcCopySpans()
gcBgMarkPrepare() // Must happen before assist enable. gcBgMarkPrepare() // Must happen before assist enable.
// At this point all Ps have enabled the write // At this point all Ps have enabled the write
...@@ -1037,6 +1049,10 @@ func gc(mode gcMode) { ...@@ -1037,6 +1049,10 @@ func gc(mode gcMode) {
// below. The important thing is that the wb remains active until // below. The important thing is that the wb remains active until
// all marking is complete. This includes writes made by the GC. // all marking is complete. This includes writes made by the GC.
// markroot is done now, so record that objects with
// finalizers have been scanned.
work.finalizersDone = true
// Flush the gcWork caches. This must be done before // Flush the gcWork caches. This must be done before
// endCycle since endCycle depends on statistics kept // endCycle since endCycle depends on statistics kept
// in these caches. // in these caches.
...@@ -1442,6 +1458,10 @@ func gcMark(start_time int64) { ...@@ -1442,6 +1458,10 @@ func gcMark(start_time int64) {
notesleep(&work.alldone) notesleep(&work.alldone)
} }
// markroot is done now, so record that objects with
// finalizers have been scanned.
work.finalizersDone = true
for i := 0; i < int(gomaxprocs); i++ { for i := 0; i < int(gomaxprocs); i++ {
if allp[i].gcw.wbuf != 0 { if allp[i].gcw.wbuf != 0 {
throw("P has cached GC work at end of mark termination") throw("P has cached GC work at end of mark termination")
......
...@@ -118,9 +118,38 @@ func markroot(desc *parfor, i uint32) { ...@@ -118,9 +118,38 @@ func markroot(desc *parfor, i uint32) {
// //
//go:nowritebarrier //go:nowritebarrier
func markrootSpans(gcw *gcWork, shard int) { func markrootSpans(gcw *gcWork, shard int) {
// Objects with finalizers have two GC-related invariants:
//
// 1) Everything reachable from the object must be marked.
// This ensures that when we pass the object to its finalizer,
// everything the finalizer can reach will be retained.
//
// 2) Finalizer specials (which are not in the garbage
// collected heap) are roots. In practice, this means the fn
// field must be scanned.
//
// TODO(austin): There are several ideas for making this more
// efficient in issue #11485.
// We process objects with finalizers only during the first
// markroot pass. In concurrent GC, this happens during
// concurrent scan and we depend on addfinalizer to ensure the
// above invariants for objects that get finalizers after
// concurrent scan. In STW GC, this will happen during mark
// termination.
if work.finalizersDone {
return
}
sg := mheap_.sweepgen sg := mheap_.sweepgen
startSpan := shard * len(work.spans) / _RootSpansShards startSpan := shard * len(work.spans) / _RootSpansShards
endSpan := (shard + 1) * len(work.spans) / _RootSpansShards endSpan := (shard + 1) * len(work.spans) / _RootSpansShards
// Note that work.spans may not include spans that were
// allocated between entering the scan phase and now. This is
// okay because any objects with finalizers in those spans
// must have been allocated and given finalizers after we
// entered the scan phase, so addfinalizer will have ensured
// the above invariants for them.
for _, s := range work.spans[startSpan:endSpan] { for _, s := range work.spans[startSpan:endSpan] {
if s.state != mSpanInUse { if s.state != mSpanInUse {
continue continue
...@@ -130,6 +159,22 @@ func markrootSpans(gcw *gcWork, shard int) { ...@@ -130,6 +159,22 @@ func markrootSpans(gcw *gcWork, shard int) {
print("sweep ", s.sweepgen, " ", sg, "\n") print("sweep ", s.sweepgen, " ", sg, "\n")
throw("gc: unswept span") throw("gc: unswept span")
} }
// Speculatively check if there are any specials
// without acquiring the span lock. This may race with
// adding the first special to a span, but in that
// case addfinalizer will observe that the GC is
// active (which is globally synchronized) and ensure
// the above invariants. We may also ensure the
// invariants, but it's okay to scan an object twice.
if s.specials == nil {
continue
}
// Lock the specials to prevent a special from being
// removed from the list while we're traversing it.
lock(&s.speciallock)
for sp := s.specials; sp != nil; sp = sp.next { for sp := s.specials; sp != nil; sp = sp.next {
if sp.kind != _KindSpecialFinalizer { if sp.kind != _KindSpecialFinalizer {
continue continue
...@@ -139,11 +184,17 @@ func markrootSpans(gcw *gcWork, shard int) { ...@@ -139,11 +184,17 @@ func markrootSpans(gcw *gcWork, shard int) {
spf := (*specialfinalizer)(unsafe.Pointer(sp)) spf := (*specialfinalizer)(unsafe.Pointer(sp))
// A finalizer can be set for an inner byte of an object, find object beginning. // A finalizer can be set for an inner byte of an object, find object beginning.
p := uintptr(s.start<<_PageShift) + uintptr(spf.special.offset)/s.elemsize*s.elemsize p := uintptr(s.start<<_PageShift) + uintptr(spf.special.offset)/s.elemsize*s.elemsize
if gcphase != _GCscan {
scanobject(p, gcw) // scanned during mark termination // Mark everything that can be reached from
} // the object (but *not* the object itself or
// we'll never collect it).
scanobject(p, gcw)
// The special itself is a root.
scanblock(uintptr(unsafe.Pointer(&spf.fn)), ptrSize, &oneptrmask[0], gcw) scanblock(uintptr(unsafe.Pointer(&spf.fn)), ptrSize, &oneptrmask[0], gcw)
} }
unlock(&s.speciallock)
} }
} }
......
...@@ -930,7 +930,8 @@ func addspecial(p unsafe.Pointer, s *special) bool { ...@@ -930,7 +930,8 @@ func addspecial(p unsafe.Pointer, s *special) bool {
} }
// Ensure that the span is swept. // Ensure that the span is swept.
// GC accesses specials list w/o locks. And it's just much safer. // Sweeping accesses the specials list w/o locks, so we have
// to synchronize with it. And it's just much safer.
mp := acquirem() mp := acquirem()
mSpan_EnsureSwept(span) mSpan_EnsureSwept(span)
...@@ -977,7 +978,8 @@ func removespecial(p unsafe.Pointer, kind uint8) *special { ...@@ -977,7 +978,8 @@ func removespecial(p unsafe.Pointer, kind uint8) *special {
} }
// Ensure that the span is swept. // Ensure that the span is swept.
// GC accesses specials list w/o locks. And it's just much safer. // Sweeping accesses the specials list w/o locks, so we have
// to synchronize with it. And it's just much safer.
mp := acquirem() mp := acquirem()
mSpan_EnsureSwept(span) mSpan_EnsureSwept(span)
...@@ -1025,6 +1027,25 @@ func addfinalizer(p unsafe.Pointer, f *funcval, nret uintptr, fint *_type, ot *p ...@@ -1025,6 +1027,25 @@ func addfinalizer(p unsafe.Pointer, f *funcval, nret uintptr, fint *_type, ot *p
s.fint = fint s.fint = fint
s.ot = ot s.ot = ot
if addspecial(p, &s.special) { if addspecial(p, &s.special) {
// This is responsible for maintaining the same
// GC-related invariants as markrootSpans in any
// situation where it's possible that markrootSpans
// has already run but mark termination hasn't yet.
if gcphase != _GCoff {
_, base, _ := findObject(p)
mp := acquirem()
gcw := &mp.p.ptr().gcw
// Mark everything reachable from the object
// so it's retained for the finalizer.
scanobject(uintptr(base), gcw)
// Mark the finalizer itself, since the
// special isn't part of the GC'd heap.
scanblock(uintptr(unsafe.Pointer(&s.fn)), ptrSize, &oneptrmask[0], gcw)
if gcBlackenPromptly {
gcw.dispose()
}
releasem(mp)
}
return true return true
} }
......
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