]> Cypherpunks repositories - gostls13.git/commitdiff
reflect: use a bigger object when we need a finalizer to run
authorKeith Randall <khr@golang.org>
Wed, 8 Aug 2018 03:59:04 +0000 (20:59 -0700)
committerKeith Randall <khr@golang.org>
Fri, 24 Aug 2018 22:17:54 +0000 (22:17 +0000)
If an object is allocated as part of a tinyalloc, then other live
objects in the same tinyalloc chunk keep the finalizer from being run,
even if the object that has the finalizer is dead.

Make sure the object we're setting the finalizer on is big enough
to not trigger tinyalloc allocation.

Fixes #26857
Update #21717

Change-Id: I56ad8679426283237ebff20a0da6c9cf64eb1c27
Reviewed-on: https://go-review.googlesource.com/128475
Run-TryBot: Keith Randall <khr@golang.org>
TryBot-Result: Gobot Gobot <gobot@golang.org>
Reviewed-by: Austin Clements <austin@google.com>
Reviewed-by: Cherry Zhang <cherryyz@google.com>
src/reflect/all_test.go

index 33bd75fda508673d4a46b220c5a0e17dfd0a28a3..c616b37008badd5e356bf8a7ef6adbe1825be2ef 100644 (file)
@@ -1693,9 +1693,9 @@ func TestCallReturnsEmpty(t *testing.T) {
        // nonzero-sized frame and zero-sized return value.
        runtime.GC()
        var finalized uint32
-       f := func() (emptyStruct, *int) {
-               i := new(int)
-               runtime.SetFinalizer(i, func(*int) { atomic.StoreUint32(&finalized, 1) })
+       f := func() (emptyStruct, *[2]int64) {
+               i := new([2]int64) // big enough to not be tinyalloc'd, so finalizer always runs when i dies
+               runtime.SetFinalizer(i, func(*[2]int64) { atomic.StoreUint32(&finalized, 1) })
                return emptyStruct{}, i
        }
        v := ValueOf(f).Call(nil)[0] // out[0] should not alias out[1]'s memory, so the finalizer should run.