Commit 227fff2e authored by Austin Clements's avatar Austin Clements

runtime/debug: don't trigger a GC on SetGCPercent

Currently SetGCPercent forces a GC in order to recompute GC pacing.
Since we can now recompute pacing on the fly using gcSetTriggerRatio,
change SetGCPercent (really runtime.setGCPercent) to go through
gcSetTriggerRatio and not trigger a GC.

Fixes #19076.

Change-Id: Ib30d7ab1bb3b55219535b9f238108f3d45a1b522
Reviewed-on: https://go-review.googlesource.com/39835
Run-TryBot: Austin Clements <austin@google.com>
Reviewed-by: default avatarRick Hudson <rlh@golang.org>
parent d9308cbb
...@@ -89,11 +89,7 @@ func ReadGCStats(stats *GCStats) { ...@@ -89,11 +89,7 @@ func ReadGCStats(stats *GCStats) {
// at startup, or 100 if the variable is not set. // at startup, or 100 if the variable is not set.
// A negative percentage disables garbage collection. // A negative percentage disables garbage collection.
func SetGCPercent(percent int) int { func SetGCPercent(percent int) int {
old := setGCPercent(int32(percent)) return int(setGCPercent(int32(percent)))
if percent >= 0 {
runtime.GC()
}
return int(old)
} }
// FreeOSMemory forces a garbage collection followed by an // FreeOSMemory forces a garbage collection followed by an
......
...@@ -160,6 +160,8 @@ func TestSetGCPercent(t *testing.T) { ...@@ -160,6 +160,8 @@ func TestSetGCPercent(t *testing.T) {
runtime.ReadMemStats(&ms) runtime.ReadMemStats(&ms)
ngc1 := ms.NumGC ngc1 := ms.NumGC
SetGCPercent(10) SetGCPercent(10)
// It may require an allocation to actually force the GC.
setGCPercentSink = make([]byte, 1<<20)
runtime.ReadMemStats(&ms) runtime.ReadMemStats(&ms)
ngc2 := ms.NumGC ngc2 := ms.NumGC
if ngc1 == ngc2 { if ngc1 == ngc2 {
......
...@@ -178,20 +178,21 @@ func gcinit() { ...@@ -178,20 +178,21 @@ func gcinit() {
throw("size of Workbuf is suboptimal") throw("size of Workbuf is suboptimal")
} }
_ = setGCPercent(readgogc()) // No sweep on the first cycle.
mheap_.sweepdone = 1
// Set a reasonable initial GC trigger. // Set a reasonable initial GC trigger.
memstats.triggerRatio = 7 / 8.0 memstats.triggerRatio = 7 / 8.0
memstats.gc_trigger = heapminimum
// Compute the goal heap size based on the trigger: // Fake a heap_marked value so it looks like a trigger at
// trigger = marked * (1 + triggerRatio) // heapminimum is the appropriate growth from heap_marked.
// marked = trigger / (1 + triggerRatio) // This will go into computing the initial GC goal.
// goal = marked * (1 + GOGC/100) memstats.heap_marked = uint64(float64(heapminimum) / (1 + memstats.triggerRatio))
// = trigger / (1 + triggerRatio) * (1 + GOGC/100)
memstats.next_gc = uint64(float64(memstats.gc_trigger) / (1 + memstats.triggerRatio) * (1 + float64(gcpercent)/100)) // Set gcpercent from the environment. This will also compute
if gcpercent < 0 { // and set the GC trigger and goal.
memstats.next_gc = ^uint64(0) _ = setGCPercent(readgogc())
}
work.startSema = 1 work.startSema = 1
work.markDoneSema = 1 work.markDoneSema = 1
} }
...@@ -226,12 +227,8 @@ func setGCPercent(in int32) (out int32) { ...@@ -226,12 +227,8 @@ func setGCPercent(in int32) (out int32) {
} }
gcpercent = in gcpercent = in
heapminimum = defaultHeapMinimum * uint64(gcpercent) / 100 heapminimum = defaultHeapMinimum * uint64(gcpercent) / 100
if memstats.triggerRatio > float64(gcpercent)/100 { // Update pacing in response to gcpercent change.
memstats.triggerRatio = float64(gcpercent) / 100 gcSetTriggerRatio(memstats.triggerRatio)
}
// This is either in gcinit or followed by a STW GC, both of
// which will reset other stats like memstats.gc_trigger and
// memstats.next_gc to appropriate values.
unlock(&mheap_.lock) unlock(&mheap_.lock)
return out return out
} }
......
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