// 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") } }