mirror of
https://github.com/golang/go.git
synced 2025-12-08 06:10:04 +00:00
This change adds tests for reflect.Value.Call for calling functions using the new register-based ABI. For #40724. Change-Id: Ia9afd43e26dd80c7e36dd135a5b71acce8074801 Reviewed-on: https://go-review.googlesource.com/c/go/+/299269 Trust: Michael Knyszek <mknyszek@google.com> Run-TryBot: Michael Knyszek <mknyszek@google.com> TryBot-Result: Go Bot <gobot@golang.org> Reviewed-by: Cherry Zhang <cherryyz@google.com>
137 lines
2.7 KiB
Go
137 lines
2.7 KiB
Go
// Copyright 2012 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 reflect
|
|
|
|
import (
|
|
"sync"
|
|
"unsafe"
|
|
)
|
|
|
|
// MakeRO returns a copy of v with the read-only flag set.
|
|
func MakeRO(v Value) Value {
|
|
v.flag |= flagStickyRO
|
|
return v
|
|
}
|
|
|
|
// IsRO reports whether v's read-only flag is set.
|
|
func IsRO(v Value) bool {
|
|
return v.flag&flagStickyRO != 0
|
|
}
|
|
|
|
var (
|
|
IntArgRegs = &intArgRegs
|
|
FloatArgRegs = &floatArgRegs
|
|
FloatRegSize = &floatRegSize
|
|
)
|
|
|
|
var CallGC = &callGC
|
|
|
|
const PtrSize = ptrSize
|
|
|
|
func FuncLayout(t Type, rcvr Type) (frametype Type, argSize, retOffset uintptr, stack []byte, gc []byte, ptrs bool) {
|
|
var ft *rtype
|
|
var abi abiDesc
|
|
if rcvr != nil {
|
|
ft, _, abi = funcLayout((*funcType)(unsafe.Pointer(t.(*rtype))), rcvr.(*rtype))
|
|
} else {
|
|
ft, _, abi = funcLayout((*funcType)(unsafe.Pointer(t.(*rtype))), nil)
|
|
}
|
|
argSize = abi.stackCallArgsSize
|
|
retOffset = abi.retOffset
|
|
frametype = ft
|
|
for i := uint32(0); i < abi.stackPtrs.n; i++ {
|
|
stack = append(stack, abi.stackPtrs.data[i/8]>>(i%8)&1)
|
|
}
|
|
if ft.kind&kindGCProg != 0 {
|
|
panic("can't handle gc programs")
|
|
}
|
|
ptrs = ft.ptrdata != 0
|
|
if ptrs {
|
|
nptrs := ft.ptrdata / ptrSize
|
|
gcdata := ft.gcSlice(0, (nptrs+7)/8)
|
|
for i := uintptr(0); i < nptrs; i++ {
|
|
gc = append(gc, gcdata[i/8]>>(i%8)&1)
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
func TypeLinks() []string {
|
|
var r []string
|
|
sections, offset := typelinks()
|
|
for i, offs := range offset {
|
|
rodata := sections[i]
|
|
for _, off := range offs {
|
|
typ := (*rtype)(resolveTypeOff(unsafe.Pointer(rodata), off))
|
|
r = append(r, typ.String())
|
|
}
|
|
}
|
|
return r
|
|
}
|
|
|
|
var GCBits = gcbits
|
|
|
|
func gcbits(interface{}) []byte // provided by runtime
|
|
|
|
func MapBucketOf(x, y Type) Type {
|
|
return bucketOf(x.(*rtype), y.(*rtype))
|
|
}
|
|
|
|
func CachedBucketOf(m Type) Type {
|
|
t := m.(*rtype)
|
|
if Kind(t.kind&kindMask) != Map {
|
|
panic("not map")
|
|
}
|
|
tt := (*mapType)(unsafe.Pointer(t))
|
|
return tt.bucket
|
|
}
|
|
|
|
type EmbedWithUnexpMeth struct{}
|
|
|
|
func (EmbedWithUnexpMeth) f() {}
|
|
|
|
type pinUnexpMeth interface {
|
|
f()
|
|
}
|
|
|
|
var pinUnexpMethI = pinUnexpMeth(EmbedWithUnexpMeth{})
|
|
|
|
func FirstMethodNameBytes(t Type) *byte {
|
|
_ = pinUnexpMethI
|
|
|
|
ut := t.uncommon()
|
|
if ut == nil {
|
|
panic("type has no methods")
|
|
}
|
|
m := ut.methods()[0]
|
|
mname := t.(*rtype).nameOff(m.name)
|
|
if *mname.data(0, "name flag field")&(1<<2) == 0 {
|
|
panic("method name does not have pkgPath *string")
|
|
}
|
|
return mname.bytes
|
|
}
|
|
|
|
type OtherPkgFields struct {
|
|
OtherExported int
|
|
otherUnexported int
|
|
}
|
|
|
|
func IsExported(t Type) bool {
|
|
typ := t.(*rtype)
|
|
n := typ.nameOff(typ.str)
|
|
return n.isExported()
|
|
}
|
|
|
|
func ResolveReflectName(s string) {
|
|
resolveReflectName(newName(s, "", false))
|
|
}
|
|
|
|
type Buffer struct {
|
|
buf []byte
|
|
}
|
|
|
|
func ClearLayoutCache() {
|
|
layoutCache = sync.Map{}
|
|
}
|