blob: d492f5681308e4e94ad4204753241da616dde2da (
plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
|
//go:build go1.24 && !go1.26
package xunsafe
import (
"reflect"
"unsafe"
)
func init() {
// TypeOf(reflect.Type{}) == *struct{ abi.Type{} }
t := reflect.TypeOf(reflect.TypeOf(0)).Elem()
if t.Size() != unsafe.Sizeof(Abi_Type{}) {
panic("Abi_Type{} not in sync with abi.Type{}")
}
}
const (
// see: go/src/internal/abi/type.go
Abi_KindDirectIface uint8 = 1 << 5
Abi_KindMask uint8 = (1 << 5) - 1
)
// Abi_Type is a copy of the memory layout of abi.Type{}.
//
// see: go/src/internal/abi/type.go
type Abi_Type struct {
_ uintptr
PtrBytes uintptr
_ uint32
_ uint8
_ uint8
_ uint8
Kind_ uint8
_ func(unsafe.Pointer, unsafe.Pointer) bool
_ *byte
_ int32
_ int32
}
// Abi_EmptyInterface is a copy of the memory layout of abi.EmptyInterface{},
// which is to say also the memory layout of any method-less interface.
//
// see: go/src/internal/abi/iface.go
type Abi_EmptyInterface struct {
Type *Abi_Type
Data unsafe.Pointer
}
// Abi_NonEmptyInterface is a copy of the memory layout of abi.NonEmptyInterface{},
// which is to say also the memory layout of any interface containing method(s).
//
// see: go/src/internal/abi/iface.go on 1.25+
// see: go/src/reflect/value.go on 1.24
type Abi_NonEmptyInterface struct {
ITab uintptr
Data unsafe.Pointer
}
// see: go/src/internal/abi/type.go Type.Kind()
func Abi_Type_Kind(t reflect.Type) uint8 {
iface := (*Abi_NonEmptyInterface)(unsafe.Pointer(&t))
atype := (*Abi_Type)(unsafe.Pointer(iface.Data))
return atype.Kind_ & Abi_KindMask
}
// see: go/src/internal/abi/type.go Type.IfaceIndir()
func Abi_Type_IfaceIndir(t reflect.Type) bool {
iface := (*Abi_NonEmptyInterface)(unsafe.Pointer(&t))
atype := (*Abi_Type)(unsafe.Pointer(iface.Data))
return atype.Kind_&Abi_KindDirectIface == 0
}
// PackIface packs a new reflect.nonEmptyInterface{} using shielded
// itab and data pointer, returning a pointer for caller casting.
func PackIface(itab uintptr, word unsafe.Pointer) unsafe.Pointer {
return unsafe.Pointer(&Abi_NonEmptyInterface{
ITab: itab,
Data: word,
})
}
// GetIfaceITab generates a new value of given type,
// casts it to the generic param interface type, and
// returns the .itab portion of the abi.NonEmptyInterface{}.
// this is useful for later calls to PackIface for known type.
func GetIfaceITab[I any](t reflect.Type) uintptr {
s := reflect.New(t).Elem().Interface().(I)
i := (*Abi_NonEmptyInterface)(unsafe.Pointer(&s))
return i.ITab
}
// UnpackEface returns the .Data portion of an abi.EmptyInterface{}.
func UnpackEface(a any) unsafe.Pointer {
return (*Abi_EmptyInterface)(unsafe.Pointer((&a))).Data
}
// see: go/src/internal/unsafeheader/unsafeheader.go
type Unsafeheader_Slice struct {
Data unsafe.Pointer
Len int
Cap int
}
// see: go/src/internal/unsafeheader/unsafeheader.go
type Unsafeheader_String struct {
Data unsafe.Pointer
Len int
}
|