// run // Copyright 2016 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. // Previously, cmd/compile would rewrite // // check(unsafe.Pointer(testMeth(1).Pointer()), unsafe.Pointer(testMeth(2).Pointer())) // // to // // var autotmp_1 uintptr = testMeth(1).Pointer() // var autotmp_2 uintptr = testMeth(2).Pointer() // check(unsafe.Pointer(autotmp_1), unsafe.Pointer(autotmp_2)) // // However, that means autotmp_1 is the only reference to the int // variable containing the value "1", but it's not a pointer type, // so it was at risk of being garbage collected by the evaluation of // testMeth(2).Pointer(), even though package unsafe's documentation // says the original code was allowed. // // Now cmd/compile rewrites it to // // var autotmp_1 unsafe.Pointer = unsafe.Pointer(testMeth(1).Pointer()) // var autotmp_2 unsafe.Pointer = unsafe.Pointer(testMeth(2).Pointer()) // check(autotmp_1, autotmp_2) // // to ensure the pointed-to variables are visible to the GC. package main import ( "fmt" "reflect" "runtime" "unsafe" ) func main() { // Test all the different ways we can invoke reflect.Value.Pointer. // Direct method invocation. check(unsafe.Pointer(testMeth(1).Pointer()), unsafe.Pointer(testMeth(2).Pointer())) // Invocation via method expression. check(unsafe.Pointer(reflect.Value.Pointer(testMeth(1))), unsafe.Pointer(reflect.Value.Pointer(testMeth(2)))) // Invocation via interface. check(unsafe.Pointer(testInter(1).Pointer()), unsafe.Pointer(testInter(2).Pointer())) // Invocation via method value. check(unsafe.Pointer(testFunc(1)()), unsafe.Pointer(testFunc(2)())) } func check(p, q unsafe.Pointer) { a, b := *(*int)(p), *(*int)(q) if a != 1 || b != 2 { fmt.Printf("got %v, %v; expected 1, 2\n", a, b) } } func testMeth(x int) reflect.Value { // Force GC to run. runtime.GC() return reflect.ValueOf(&x) } type Pointerer interface { Pointer() uintptr } func testInter(x int) Pointerer { return testMeth(x) } func testFunc(x int) func() uintptr { return testMeth(x).Pointer }