summaryrefslogtreecommitdiff
path: root/test/tinyfin.go
diff options
context:
space:
mode:
Diffstat (limited to 'test/tinyfin.go')
-rw-r--r--test/tinyfin.go62
1 files changed, 62 insertions, 0 deletions
diff --git a/test/tinyfin.go b/test/tinyfin.go
new file mode 100644
index 000000000..8fb109fc0
--- /dev/null
+++ b/test/tinyfin.go
@@ -0,0 +1,62 @@
+// run
+
+// Copyright 2014 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 finalizers work for tiny (combined) allocations.
+
+package main
+
+import (
+ "runtime"
+ "sync/atomic"
+ "time"
+)
+
+func main() {
+ // Does not work on 32-bits due to partially conservative GC.
+ // Try to enable when we have fully precise GC.
+ if runtime.GOARCH != "amd64" {
+ return
+ }
+ // Likewise for gccgo.
+ if runtime.Compiler == "gccgo" {
+ return
+ }
+ N := int32(100)
+ count := N
+ done := make([]bool, N)
+ for i := int32(0); i < N; i++ {
+ x := i // subject to tiny alloc
+ // the closure must be big enough to be combined
+ runtime.SetFinalizer(&x, func(p *int32) {
+ // Check that p points to the correct subobject of the tiny allocation.
+ // It's a bit tricky, because we can't capture another variable
+ // with the expected value (it would be combined as well).
+ if *p < 0 || *p >= N {
+ println("got", *p)
+ panic("corrupted")
+ }
+ if done[*p] {
+ println("got", *p)
+ panic("already finalized")
+ }
+ done[*p] = true
+ atomic.AddInt32(&count, -1)
+ })
+ }
+ for i := 0; i < 4; i++ {
+ runtime.GC()
+ time.Sleep(10 * time.Millisecond)
+ }
+ // Some of the finalizers may not be executed,
+ // if the outermost allocations are combined with something persistent.
+ // Currently 4 int32's are combined into a 16-byte block,
+ // ensure that most of them are finalized.
+ if count >= N/4 {
+ println(count, "out of", N, "finalizer are not called")
+ panic("not all finalizers are called")
+ }
+}
+