mirror of
https://github.com/golang/go
synced 2024-11-17 16:04:47 -07:00
reflect: keep pointer in aggregate-typed args live in Call
When register ABI is used, reflect.Value.Call prepares the call arguments in a memory representation of the argument registers. It has special handling to keep the pointers in arguments live. Currently, this handles pointer-typed arguments. But when an argument is an aggregate-type that contains pointers and passed in registers, it currently doesn't keep the pointers live. Do so in this CL. May fix #49363. Change-Id: Ic6a0c5fdf9375ef02f7c03fbe9345e2e98c9353d Reviewed-on: https://go-review.googlesource.com/c/go/+/363358 Trust: Cherry Mui <cherryyz@google.com> Run-TryBot: Cherry Mui <cherryyz@google.com> TryBot-Result: Go Bot <gobot@golang.org> Reviewed-by: Michael Knyszek <mknyszek@google.com>
This commit is contained in:
parent
e9f0381a80
commit
23adc139bf
@ -44,6 +44,24 @@ type RegArgs struct {
|
||||
ReturnIsPtr IntArgRegBitmap
|
||||
}
|
||||
|
||||
func (r *RegArgs) Dump() {
|
||||
print("Ints:")
|
||||
for _, x := range r.Ints {
|
||||
print(" ", x)
|
||||
}
|
||||
println()
|
||||
print("Floats:")
|
||||
for _, x := range r.Floats {
|
||||
print(" ", x)
|
||||
}
|
||||
println()
|
||||
print("Ptrs:")
|
||||
for _, x := range r.Ptrs {
|
||||
print(" ", x)
|
||||
}
|
||||
println()
|
||||
}
|
||||
|
||||
// IntRegArgAddr returns a pointer inside of r.Ints[reg] that is appropriately
|
||||
// offset for an argument of size argSize.
|
||||
//
|
||||
|
@ -6478,6 +6478,29 @@ func TestCallMethodJump(t *testing.T) {
|
||||
*CallGC = false
|
||||
}
|
||||
|
||||
func TestCallArgLive(t *testing.T) {
|
||||
type T struct{ X, Y *string } // pointerful aggregate
|
||||
|
||||
F := func(t T) { *t.X = "ok" }
|
||||
|
||||
// In reflect.Value.Call, trigger a garbage collection in reflect.call
|
||||
// between marshaling argument and the actual call.
|
||||
*CallGC = true
|
||||
|
||||
x := new(string)
|
||||
runtime.SetFinalizer(x, func(p *string) {
|
||||
if *p != "ok" {
|
||||
t.Errorf("x dead prematurely")
|
||||
}
|
||||
})
|
||||
v := T{x, nil}
|
||||
|
||||
ValueOf(F).Call([]Value{ValueOf(v)})
|
||||
|
||||
// Stop garbage collecting during reflect.call.
|
||||
*CallGC = false
|
||||
}
|
||||
|
||||
func TestMakeFuncStackCopy(t *testing.T) {
|
||||
target := func(in []Value) []Value {
|
||||
runtime.GC()
|
||||
|
@ -352,7 +352,7 @@ func (v Value) CallSlice(in []Value) []Value {
|
||||
return v.call("CallSlice", in)
|
||||
}
|
||||
|
||||
var callGC bool // for testing; see TestCallMethodJump
|
||||
var callGC bool // for testing; see TestCallMethodJump and TestCallArgLive
|
||||
|
||||
const debugReflectCall = false
|
||||
|
||||
@ -509,12 +509,16 @@ func (v Value) call(op string, in []Value) []Value {
|
||||
// Copy values to "integer registers."
|
||||
if v.flag&flagIndir != 0 {
|
||||
offset := add(v.ptr, st.offset, "precomputed value offset")
|
||||
intToReg(®Args, st.ireg, st.size, offset)
|
||||
} else {
|
||||
if st.kind == abiStepPointer {
|
||||
// Duplicate this pointer in the pointer area of the
|
||||
// register space. Otherwise, there's the potential for
|
||||
// this to be the last reference to v.ptr.
|
||||
regArgs.Ptrs[st.ireg] = *(*unsafe.Pointer)(offset)
|
||||
}
|
||||
intToReg(®Args, st.ireg, st.size, offset)
|
||||
} else {
|
||||
if st.kind == abiStepPointer {
|
||||
// See the comment in abiStepPointer case above.
|
||||
regArgs.Ptrs[st.ireg] = v.ptr
|
||||
}
|
||||
regArgs.Ints[st.ireg] = uintptr(v.ptr)
|
||||
@ -539,6 +543,15 @@ func (v Value) call(op string, in []Value) []Value {
|
||||
// Mark pointers in registers for the return path.
|
||||
regArgs.ReturnIsPtr = abi.outRegPtrs
|
||||
|
||||
if debugReflectCall {
|
||||
regArgs.Dump()
|
||||
}
|
||||
|
||||
// For testing; see TestCallArgLive.
|
||||
if callGC {
|
||||
runtime.GC()
|
||||
}
|
||||
|
||||
// Call.
|
||||
call(frametype, fn, stackArgs, uint32(frametype.size), uint32(abi.retOffset), uint32(frameSize), ®Args)
|
||||
|
||||
|
Loading…
Reference in New Issue
Block a user