|  | // run | 
|  |  | 
|  | // Copyright 2009 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 basic operation of finalizers. | 
|  |  | 
|  | package main | 
|  |  | 
|  | import ( | 
|  | "runtime" | 
|  | "time" | 
|  | ) | 
|  |  | 
|  | const N = 250 | 
|  |  | 
|  | type A struct { | 
|  | b *B | 
|  | n int | 
|  | } | 
|  |  | 
|  | type B struct { | 
|  | n int | 
|  | } | 
|  |  | 
|  | var i int | 
|  | var nfinal int | 
|  | var final [N]int | 
|  |  | 
|  | // the unused return is to test finalizers with return values | 
|  | func finalA(a *A) (unused [N]int) { | 
|  | if final[a.n] != 0 { | 
|  | println("finalA", a.n, final[a.n]) | 
|  | panic("fail") | 
|  | } | 
|  | final[a.n] = 1 | 
|  | return | 
|  | } | 
|  |  | 
|  | func finalB(b *B) { | 
|  | if final[b.n] != 1 { | 
|  | println("finalB", b.n, final[b.n]) | 
|  | panic("fail") | 
|  | } | 
|  | final[b.n] = 2 | 
|  | nfinal++ | 
|  | } | 
|  |  | 
|  | func nofinalB(b *B) { | 
|  | panic("nofinalB run") | 
|  | } | 
|  |  | 
|  | func main() { | 
|  | runtime.GOMAXPROCS(4) | 
|  | for i = 0; i < N; i++ { | 
|  | b := &B{i} | 
|  | a := &A{b, i} | 
|  | c := new(B) | 
|  | runtime.SetFinalizer(c, nofinalB) | 
|  | runtime.SetFinalizer(b, finalB) | 
|  | runtime.SetFinalizer(a, finalA) | 
|  | runtime.SetFinalizer(c, nil) | 
|  | } | 
|  | for i := 0; i < N; i++ { | 
|  | runtime.GC() | 
|  | runtime.Gosched() | 
|  | time.Sleep(1e6) | 
|  | if nfinal >= N*8/10 { | 
|  | break | 
|  | } | 
|  | } | 
|  | if nfinal < N*8/10 { | 
|  | println("not enough finalizing:", nfinal, "/", N) | 
|  | panic("fail") | 
|  | } | 
|  | } |