mirror of
https://github.com/golang/go
synced 2024-11-23 13:40:04 -07:00
63 lines
1.5 KiB
Go
63 lines
1.5 KiB
Go
|
// 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")
|
||
|
}
|
||
|
}
|
||
|
|