2011-10-27 01:57:58 +02:00
|
|
|
// 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"
|
|
|
|
"testing"
|
2013-11-06 20:49:01 +01:00
|
|
|
"time"
|
2014-07-19 10:53:52 +02:00
|
|
|
"unsafe"
|
2011-10-27 01:57:58 +02:00
|
|
|
)
|
|
|
|
|
2013-11-06 20:49:01 +01:00
|
|
|
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")
|
|
|
|
}
|
2015-01-24 00:22:29 +01:00
|
|
|
if runtime.Compiler == "gccgo" {
|
|
|
|
t.Skip("skipping for gccgo")
|
|
|
|
}
|
2013-11-06 20:49:01 +01:00
|
|
|
|
|
|
|
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))) }},
|
|
|
|
}
|
|
|
|
|
2015-01-15 01:27:56 +01:00
|
|
|
for i, tt := range finalizerTests {
|
2014-06-05 01:15:33 +02:00
|
|
|
done := make(chan bool, 1)
|
2013-11-06 20:49:01 +01:00
|
|
|
go func() {
|
2015-01-15 01:27:56 +01:00
|
|
|
// allocate struct with pointer to avoid hitting tinyalloc.
|
|
|
|
// Otherwise we can't be sure when the allocation will
|
|
|
|
// be freed.
|
|
|
|
type T struct {
|
|
|
|
v int
|
|
|
|
p unsafe.Pointer
|
|
|
|
}
|
|
|
|
v := &new(T).v
|
2013-11-06 20:49:01 +01:00
|
|
|
*v = 97531
|
|
|
|
runtime.SetFinalizer(tt.convert(v), tt.finalizer)
|
|
|
|
v = nil
|
2014-06-05 01:15:33 +02:00
|
|
|
done <- true
|
2013-11-06 20:49:01 +01:00
|
|
|
}()
|
2014-06-05 01:15:33 +02:00
|
|
|
<-done
|
2013-11-06 20:49:01 +01:00
|
|
|
runtime.GC()
|
|
|
|
select {
|
|
|
|
case <-ch:
|
|
|
|
case <-time.After(time.Second * 4):
|
2015-01-15 01:27:56 +01:00
|
|
|
t.Errorf("#%d: finalizer for type %T didn't run", i, tt.finalizer)
|
2013-11-06 20:49:01 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
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")
|
|
|
|
}
|
2015-01-24 00:22:29 +01:00
|
|
|
if runtime.Compiler == "gccgo" {
|
|
|
|
t.Skip("skipping for gccgo")
|
|
|
|
}
|
2013-11-06 20:49:01 +01:00
|
|
|
ch := make(chan bool)
|
2014-06-05 01:15:33 +02:00
|
|
|
done := make(chan bool, 1)
|
2013-11-06 20:49:01 +01:00
|
|
|
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
|
2014-06-05 01:15:33 +02:00
|
|
|
done <- true
|
2013-11-06 20:49:01 +01:00
|
|
|
}()
|
2014-06-05 01:15:33 +02:00
|
|
|
<-done
|
2013-11-06 20:49:01 +01:00
|
|
|
runtime.GC()
|
|
|
|
select {
|
|
|
|
case <-ch:
|
|
|
|
case <-time.After(4 * time.Second):
|
|
|
|
t.Errorf("finalizer for type *bigValue didn't run")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-10-27 01:57:58 +02:00
|
|
|
func fin(v *int) {
|
|
|
|
}
|
|
|
|
|
2014-06-05 01:15:33 +02:00
|
|
|
// Verify we don't crash at least. golang.org/issue/6857
|
|
|
|
func TestFinalizerZeroSizedStruct(t *testing.T) {
|
|
|
|
type Z struct{}
|
|
|
|
z := new(Z)
|
|
|
|
runtime.SetFinalizer(z, func(*Z) {})
|
|
|
|
}
|
|
|
|
|
2011-10-27 01:57:58 +02:00
|
|
|
func BenchmarkFinalizer(b *testing.B) {
|
2014-07-19 10:53:52 +02:00
|
|
|
const Batch = 1000
|
|
|
|
b.RunParallel(func(pb *testing.PB) {
|
|
|
|
var data [Batch]*int
|
|
|
|
for i := 0; i < Batch; i++ {
|
|
|
|
data[i] = new(int)
|
|
|
|
}
|
|
|
|
for pb.Next() {
|
|
|
|
for i := 0; i < Batch; i++ {
|
|
|
|
runtime.SetFinalizer(data[i], fin)
|
2011-10-27 01:57:58 +02:00
|
|
|
}
|
2014-07-19 10:53:52 +02:00
|
|
|
for i := 0; i < Batch; i++ {
|
|
|
|
runtime.SetFinalizer(data[i], nil)
|
2011-10-27 01:57:58 +02:00
|
|
|
}
|
2014-07-19 10:53:52 +02:00
|
|
|
}
|
|
|
|
})
|
2011-10-27 01:57:58 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
func BenchmarkFinalizerRun(b *testing.B) {
|
2014-07-19 10:53:52 +02:00
|
|
|
b.RunParallel(func(pb *testing.PB) {
|
|
|
|
for pb.Next() {
|
|
|
|
v := new(int)
|
|
|
|
runtime.SetFinalizer(v, fin)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
// One chunk must be exactly one sizeclass in size.
|
|
|
|
// It should be a sizeclass not used much by others, so we
|
|
|
|
// have a greater chance of finding adjacent ones.
|
|
|
|
// size class 19: 320 byte objects, 25 per page, 1 page alloc at a time
|
|
|
|
const objsize = 320
|
|
|
|
|
|
|
|
type objtype [objsize]byte
|
|
|
|
|
|
|
|
func adjChunks() (*objtype, *objtype) {
|
|
|
|
var s []*objtype
|
|
|
|
|
|
|
|
for {
|
|
|
|
c := new(objtype)
|
|
|
|
for _, d := range s {
|
|
|
|
if uintptr(unsafe.Pointer(c))+unsafe.Sizeof(*c) == uintptr(unsafe.Pointer(d)) {
|
|
|
|
return c, d
|
2011-10-27 01:57:58 +02:00
|
|
|
}
|
2014-07-19 10:53:52 +02:00
|
|
|
if uintptr(unsafe.Pointer(d))+unsafe.Sizeof(*c) == uintptr(unsafe.Pointer(c)) {
|
|
|
|
return d, c
|
|
|
|
}
|
|
|
|
}
|
|
|
|
s = append(s, c)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure an empty slice on the stack doesn't pin the next object in memory.
|
|
|
|
func TestEmptySlice(t *testing.T) {
|
2015-01-24 00:22:29 +01:00
|
|
|
if runtime.Compiler == "gccgo" {
|
|
|
|
t.Skip("skipping for gccgo")
|
|
|
|
}
|
2014-07-19 10:53:52 +02:00
|
|
|
x, y := adjChunks()
|
|
|
|
|
|
|
|
// the pointer inside xs points to y.
|
|
|
|
xs := x[objsize:] // change objsize to objsize-1 and the test passes
|
|
|
|
|
|
|
|
fin := make(chan bool, 1)
|
|
|
|
runtime.SetFinalizer(y, func(z *objtype) { fin <- true })
|
|
|
|
runtime.GC()
|
|
|
|
select {
|
|
|
|
case <-fin:
|
|
|
|
case <-time.After(4 * time.Second):
|
|
|
|
t.Errorf("finalizer of next object in memory didn't run")
|
|
|
|
}
|
|
|
|
xsglobal = xs // keep empty slice alive until here
|
|
|
|
}
|
|
|
|
|
|
|
|
var xsglobal []byte
|
|
|
|
|
|
|
|
func adjStringChunk() (string, *objtype) {
|
|
|
|
b := make([]byte, objsize)
|
|
|
|
for {
|
|
|
|
s := string(b)
|
|
|
|
t := new(objtype)
|
|
|
|
p := *(*uintptr)(unsafe.Pointer(&s))
|
|
|
|
q := uintptr(unsafe.Pointer(t))
|
|
|
|
if p+objsize == q {
|
|
|
|
return s, t
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure an empty string on the stack doesn't pin the next object in memory.
|
|
|
|
func TestEmptyString(t *testing.T) {
|
|
|
|
if runtime.Compiler == "gccgo" {
|
|
|
|
t.Skip("skipping for gccgo")
|
|
|
|
}
|
|
|
|
|
|
|
|
x, y := adjStringChunk()
|
|
|
|
|
|
|
|
ss := x[objsize:] // change objsize to objsize-1 and the test passes
|
|
|
|
fin := make(chan bool, 1)
|
|
|
|
// set finalizer on string contents of y
|
|
|
|
runtime.SetFinalizer(y, func(z *objtype) { fin <- true })
|
|
|
|
runtime.GC()
|
|
|
|
select {
|
|
|
|
case <-fin:
|
|
|
|
case <-time.After(4 * time.Second):
|
|
|
|
t.Errorf("finalizer of next string in memory didn't run")
|
2011-10-27 01:57:58 +02:00
|
|
|
}
|
2014-07-19 10:53:52 +02:00
|
|
|
ssglobal = ss // keep 0-length string live until here
|
|
|
|
}
|
|
|
|
|
|
|
|
var ssglobal string
|
|
|
|
|
|
|
|
// Test for issue 7656.
|
|
|
|
func TestFinalizerOnGlobal(t *testing.T) {
|
|
|
|
runtime.SetFinalizer(Foo1, func(p *Object1) {})
|
|
|
|
runtime.SetFinalizer(Foo2, func(p *Object2) {})
|
|
|
|
runtime.SetFinalizer(Foo1, nil)
|
|
|
|
runtime.SetFinalizer(Foo2, nil)
|
2011-10-27 01:57:58 +02:00
|
|
|
}
|
2014-07-19 10:53:52 +02:00
|
|
|
|
|
|
|
type Object1 struct {
|
|
|
|
Something []byte
|
|
|
|
}
|
|
|
|
|
|
|
|
type Object2 struct {
|
|
|
|
Something byte
|
|
|
|
}
|
|
|
|
|
|
|
|
var (
|
|
|
|
Foo2 = &Object2{}
|
|
|
|
Foo1 = &Object1{}
|
|
|
|
)
|