// 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" "time" ) func main() { // Does not work on gccgo due to partially conservative GC. // Try to enable when we have fully precise GC. if runtime.Compiler == "gccgo" { return } const N = 100 finalized := make(chan int32, N) for i := 0; i < N; i++ { x := new(int32) // subject to tiny alloc *x = int32(i) // the closure must be big enough to be combined runtime.SetFinalizer(x, func(p *int32) { finalized <- *p }) } runtime.GC() count := 0 done := make([]bool, N) timeout := time.After(5*time.Second) for { select { case <-timeout: println("timeout,", count, "finalized so far") panic("not all finalizers are called") case x := <-finalized: // 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 x < 0 || x >= N { println("got", x) panic("corrupted") } if done[x] { println("got", x) panic("already finalized") } done[x] = true count++ if count > N/10*9 { // 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. return } } } }