aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorAndrew Gerrand <adg@golang.org>2013-06-05 10:52:00 +1000
committerAndrew Gerrand <adg@golang.org>2013-06-05 10:52:00 +1000
commitff7cb872a455a370beed530294838ab151e965c6 (patch)
tree413dcad2e7d8e1e3b9ed7b817d76020573255a97
parentb9b37d1292d9a3e21a3fe0f86ad54c3c1a9cb770 (diff)
downloadgo-ff7cb872a455a370beed530294838ab151e965c6.tar.gz
go-ff7cb872a455a370beed530294838ab151e965c6.zip
[release-branch.go1.1] runtime: zeroize g->fnstart to not prevent GC of the closure
««« CL 9557043 / 2c128d417029 runtime: zeroize g->fnstart to not prevent GC of the closure Fixes #5493. R=golang-dev, minux.ma, iant CC=golang-dev https://golang.org/cl/9557043 »»» R=dvyukov, iant, minux.ma, bradfitz, dave CC=golang-dev https://golang.org/cl/10031043
-rw-r--r--src/pkg/runtime/proc.c1
-rw-r--r--test/fixedbugs/issue5493.go52
2 files changed, 53 insertions, 0 deletions
diff --git a/src/pkg/runtime/proc.c b/src/pkg/runtime/proc.c
index 018a453d62..31876b62a9 100644
--- a/src/pkg/runtime/proc.c
+++ b/src/pkg/runtime/proc.c
@@ -1232,6 +1232,7 @@ static void
goexit0(G *gp)
{
gp->status = Gdead;
+ gp->fnstart = nil;
gp->m = nil;
gp->lockedm = nil;
m->curg = nil;
diff --git a/test/fixedbugs/issue5493.go b/test/fixedbugs/issue5493.go
new file mode 100644
index 0000000000..fe571bc085
--- /dev/null
+++ b/test/fixedbugs/issue5493.go
@@ -0,0 +1,52 @@
+// 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.
+
+package main
+
+import (
+ "runtime"
+ "sync"
+ "sync/atomic"
+ "time"
+)
+
+const N = 10
+var count int64
+
+func run() error {
+ f1 := func() {}
+ f2 := func() {
+ func() {
+ f1()
+ }()
+ }
+ runtime.SetFinalizer(&f1, func(f *func()) {
+ atomic.AddInt64(&count, -1)
+ })
+ go f2()
+ return nil
+}
+
+func main() {
+ count = N
+ var wg sync.WaitGroup
+ wg.Add(N)
+ for i := 0; i < N; i++ {
+ go func() {
+ run()
+ wg.Done()
+ }()
+ }
+ wg.Wait()
+ for i := 0; i < 2*N; i++ {
+ time.Sleep(10 * time.Millisecond)
+ runtime.GC()
+ }
+ if count != 0 {
+ panic("not all finalizers are called")
+ }
+}
+