// Copyright 2011 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 runtime_test import ( "runtime" "sync" "sync/atomic" "testing" "time" ) type Tintptr *int // assignable to *int type Tint int // *Tint implements Tinter, interface{} func (t *Tint) m() {} type Tinter interface { m() } func TestFinalizerType(t *testing.T) { if runtime.GOARCH != "amd64" { t.Skipf("Skipping on non-amd64 machine") } ch := make(chan bool, 10) finalize := func(x *int) { if *x != 97531 { t.Errorf("finalizer %d, want %d", *x, 97531) } ch <- true } var finalizerTests = []struct { convert func(*int) interface{} finalizer interface{} }{ {func(x *int) interface{} { return x }, func(v *int) { finalize(v) }}, {func(x *int) interface{} { return Tintptr(x) }, func(v Tintptr) { finalize(v) }}, {func(x *int) interface{} { return Tintptr(x) }, func(v *int) { finalize(v) }}, {func(x *int) interface{} { return (*Tint)(x) }, func(v *Tint) { finalize((*int)(v)) }}, {func(x *int) interface{} { return (*Tint)(x) }, func(v Tinter) { finalize((*int)(v.(*Tint))) }}, } for _, tt := range finalizerTests { go func() { v := new(int) *v = 97531 runtime.SetFinalizer(tt.convert(v), tt.finalizer) v = nil }() time.Sleep(1 * time.Second) runtime.GC() select { case <-ch: case <-time.After(time.Second * 4): t.Errorf("finalizer for type %T didn't run", tt.finalizer) } } } type bigValue struct { fill uint64 it bool up string } func TestFinalizerInterfaceBig(t *testing.T) { if runtime.GOARCH != "amd64" { t.Skipf("Skipping on non-amd64 machine") } ch := make(chan bool) go func() { v := &bigValue{0xDEADBEEFDEADBEEF, true, "It matters not how strait the gate"} old := *v runtime.SetFinalizer(v, func(v interface{}) { i, ok := v.(*bigValue) if !ok { t.Errorf("finalizer called with type %T, want *bigValue", v) } if *i != old { t.Errorf("finalizer called with %+v, want %+v", *i, old) } close(ch) }) v = nil }() time.Sleep(1 * time.Second) runtime.GC() select { case <-ch: case <-time.After(4 * time.Second): t.Errorf("finalizer for type *bigValue didn't run") } } func fin(v *int) { } func BenchmarkFinalizer(b *testing.B) { const CallsPerSched = 1000 procs := runtime.GOMAXPROCS(-1) N := int32(b.N / CallsPerSched) var wg sync.WaitGroup wg.Add(procs) for p := 0; p < procs; p++ { go func() { var data [CallsPerSched]*int for i := 0; i < CallsPerSched; i++ { data[i] = new(int) } for atomic.AddInt32(&N, -1) >= 0 { runtime.Gosched() for i := 0; i < CallsPerSched; i++ { runtime.SetFinalizer(data[i], fin) } for i := 0; i < CallsPerSched; i++ { runtime.SetFinalizer(data[i], nil) } } wg.Done() }() } wg.Wait() } func BenchmarkFinalizerRun(b *testing.B) { const CallsPerSched = 1000 procs := runtime.GOMAXPROCS(-1) N := int32(b.N / CallsPerSched) var wg sync.WaitGroup wg.Add(procs) for p := 0; p < procs; p++ { go func() { for atomic.AddInt32(&N, -1) >= 0 { runtime.Gosched() for i := 0; i < CallsPerSched; i++ { v := new(int) runtime.SetFinalizer(v, fin) } runtime.GC() } wg.Done() }() } wg.Wait() }