2013-07-01 21:36:08 +00:00
|
|
|
// run
|
|
|
|
|
|
|
|
// Copyright 2013 The Go Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
// Test that defers do not prevent garbage collection.
|
|
|
|
|
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"runtime"
|
|
|
|
"sync"
|
|
|
|
"sync/atomic"
|
|
|
|
"time"
|
|
|
|
)
|
|
|
|
|
|
|
|
var sink func()
|
|
|
|
|
|
|
|
func main() {
|
2021-01-05 16:37:41 +00:00
|
|
|
// Does not work with gccgo, due to partially conservative GC.
|
2013-07-01 21:36:08 +00:00
|
|
|
// Try to enable when we have fully precise GC.
|
2013-12-13 01:13:27 +00:00
|
|
|
if runtime.Compiler == "gccgo" {
|
|
|
|
return
|
|
|
|
}
|
2013-07-01 21:36:08 +00:00
|
|
|
N := 10
|
|
|
|
count := int32(N)
|
|
|
|
var wg sync.WaitGroup
|
|
|
|
wg.Add(N)
|
|
|
|
for i := 0; i < N; i++ {
|
|
|
|
go func() {
|
|
|
|
defer wg.Done()
|
2014-01-24 18:35:11 +00:00
|
|
|
v := new(string)
|
2013-07-01 21:36:08 +00:00
|
|
|
f := func() {
|
2014-01-24 18:35:11 +00:00
|
|
|
if *v != "" {
|
2013-07-01 21:36:08 +00:00
|
|
|
panic("oops")
|
|
|
|
}
|
|
|
|
}
|
2014-01-24 18:35:11 +00:00
|
|
|
if *v != "" {
|
2013-07-01 21:36:08 +00:00
|
|
|
// let the compiler think f escapes
|
|
|
|
sink = f
|
|
|
|
}
|
2014-01-24 18:35:11 +00:00
|
|
|
runtime.SetFinalizer(v, func(p *string) {
|
2013-07-01 21:36:08 +00:00
|
|
|
atomic.AddInt32(&count, -1)
|
|
|
|
})
|
|
|
|
defer f()
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
wg.Wait()
|
|
|
|
for i := 0; i < 3; i++ {
|
|
|
|
time.Sleep(10 * time.Millisecond)
|
|
|
|
runtime.GC()
|
|
|
|
}
|
|
|
|
if count != 0 {
|
|
|
|
println(count, "out of", N, "finalizer are not called")
|
|
|
|
panic("not all finalizers are called")
|
|
|
|
}
|
|
|
|
}
|