Commit b8903339 authored by Austin Clements's avatar Austin Clements

runtime: clean up gcMarkDone

This improves the documentation comment on gcMarkDone, replaces a
recursive call with a simple goto, and disables preemption before
stopping the world in accordance with the documentation comment on
stopTheWorldWithSema.

Updates #13363, but, sadly, doesn't fix it.

Change-Id: I6cb2a5836b35685bf82f7b1ce7e48a7625906656
Reviewed-on: https://go-review.googlesource.com/17149Reviewed-by: default avatarRick Hudson <rlh@golang.org>
Run-TryBot: Austin Clements <austin@google.com>
TryBot-Result: Gobot Gobot <gobot@golang.org>
parent a7c09ad4
...@@ -1027,7 +1027,15 @@ func gcStart(mode gcMode, forceTrigger bool) { ...@@ -1027,7 +1027,15 @@ func gcStart(mode gcMode, forceTrigger bool) {
// active work buffers in assists and background workers; however, // active work buffers in assists and background workers; however,
// work may still be cached in per-P work buffers. In mark 2, per-P // work may still be cached in per-P work buffers. In mark 2, per-P
// caches are disabled. // caches are disabled.
//
// The calling context must be preemptible.
//
// Note that it is explicitly okay to have write barriers in this
// function because completion of concurrent mark is best-effort
// anyway. Any work created by write barriers here will be cleaned up
// by mark termination.
func gcMarkDone() { func gcMarkDone() {
top:
semacquire(&work.markDoneSema, false) semacquire(&work.markDoneSema, false)
// Re-check transition condition under transition lock. // Re-check transition condition under transition lock.
...@@ -1090,15 +1098,17 @@ func gcMarkDone() { ...@@ -1090,15 +1098,17 @@ func gcMarkDone() {
incnwait := atomic.Xadd(&work.nwait, +1) incnwait := atomic.Xadd(&work.nwait, +1)
if incnwait == work.nproc && !gcMarkWorkAvailable(nil) { if incnwait == work.nproc && !gcMarkWorkAvailable(nil) {
// This recursion is safe because the call // This loop will make progress because
// can't take this same "if" branch. // gcBlackenPromptly is now true, so it won't
gcMarkDone() // take this same "if" branch.
goto top
} }
} else { } else {
// Transition to mark termination. // Transition to mark termination.
now := nanotime() now := nanotime()
work.tMarkTerm = now work.tMarkTerm = now
work.pauseStart = now work.pauseStart = now
getg().m.preemptoff = "gcing"
systemstack(stopTheWorldWithSema) systemstack(stopTheWorldWithSema)
// The gcphase is _GCmark, it will transition to _GCmarktermination // The gcphase is _GCmark, it will transition to _GCmarktermination
// below. The important thing is that the wb remains active until // below. The important thing is that the wb remains active until
......
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