Source file
src/reflect/type.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16 package reflect
17
18 import (
19 "internal/abi"
20 "internal/goarch"
21 "runtime"
22 "strconv"
23 "sync"
24 "unicode"
25 "unicode/utf8"
26 "unsafe"
27 )
28
29
30
31
32
33
34
35
36
37
38
39
40 type Type interface {
41
42
43
44
45 Align() int
46
47
48
49 FieldAlign() int
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65 Method(int) Method
66
67
68
69
70
71
72
73
74
75
76
77
78
79 MethodByName(string) (Method, bool)
80
81
82
83
84
85
86 NumMethod() int
87
88
89
90 Name() string
91
92
93
94
95
96
97 PkgPath() string
98
99
100
101 Size() uintptr
102
103
104
105
106
107
108 String() string
109
110
111 Kind() Kind
112
113
114 Implements(u Type) bool
115
116
117 AssignableTo(u Type) bool
118
119
120
121
122
123 ConvertibleTo(u Type) bool
124
125
126
127
128
129 Comparable() bool
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146 Bits() int
147
148
149
150 ChanDir() ChanDir
151
152
153
154
155
156
157
158
159
160
161
162
163
164 IsVariadic() bool
165
166
167
168 Elem() Type
169
170
171
172
173 Field(i int) StructField
174
175
176
177
178
179 FieldByIndex(index []int) StructField
180
181
182
183
184
185
186 FieldByName(name string) (StructField, bool)
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204 FieldByNameFunc(match func(string) bool) (StructField, bool)
205
206
207
208
209 In(i int) Type
210
211
212
213 Key() Type
214
215
216
217 Len() int
218
219
220
221 NumField() int
222
223
224
225 NumIn() int
226
227
228
229 NumOut() int
230
231
232
233
234 Out(i int) Type
235
236
237
238 OverflowComplex(x complex128) bool
239
240
241
242 OverflowFloat(x float64) bool
243
244
245
246 OverflowInt(x int64) bool
247
248
249
250 OverflowUint(x uint64) bool
251
252
253 CanSeq() bool
254
255
256 CanSeq2() bool
257
258 common() *abi.Type
259 uncommon() *uncommonType
260 }
261
262
263
264
265
266
267
268
269
270
275
276
277
278 type Kind uint
279
280 const (
281 Invalid Kind = iota
282 Bool
283 Int
284 Int8
285 Int16
286 Int32
287 Int64
288 Uint
289 Uint8
290 Uint16
291 Uint32
292 Uint64
293 Uintptr
294 Float32
295 Float64
296 Complex64
297 Complex128
298 Array
299 Chan
300 Func
301 Interface
302 Map
303 Pointer
304 Slice
305 String
306 Struct
307 UnsafePointer
308 )
309
310
311
312
313 const Ptr = Pointer
314
315
316
317
318
319 type uncommonType = abi.UncommonType
320
321
322 type common struct {
323 abi.Type
324 }
325
326
327
328 type rtype struct {
329 t abi.Type
330 }
331
332 func (t *rtype) common() *abi.Type {
333 return &t.t
334 }
335
336 func (t *rtype) uncommon() *abi.UncommonType {
337 return t.t.Uncommon()
338 }
339
340 type aNameOff = abi.NameOff
341 type aTypeOff = abi.TypeOff
342 type aTextOff = abi.TextOff
343
344
345 type ChanDir int
346
347 const (
348 RecvDir ChanDir = 1 << iota
349 SendDir
350 BothDir = RecvDir | SendDir
351 )
352
353
354 type arrayType = abi.ArrayType
355
356
357 type chanType = abi.ChanType
358
359
360
361
362
363
364
365
366
367
368
369
370 type funcType = abi.FuncType
371
372
373 type interfaceType struct {
374 abi.InterfaceType
375 }
376
377 func (t *interfaceType) nameOff(off aNameOff) abi.Name {
378 return toRType(&t.Type).nameOff(off)
379 }
380
381 func nameOffFor(t *abi.Type, off aNameOff) abi.Name {
382 return toRType(t).nameOff(off)
383 }
384
385 func typeOffFor(t *abi.Type, off aTypeOff) *abi.Type {
386 return toRType(t).typeOff(off)
387 }
388
389 func (t *interfaceType) typeOff(off aTypeOff) *abi.Type {
390 return toRType(&t.Type).typeOff(off)
391 }
392
393 func (t *interfaceType) common() *abi.Type {
394 return &t.Type
395 }
396
397 func (t *interfaceType) uncommon() *abi.UncommonType {
398 return t.Uncommon()
399 }
400
401
402 type ptrType struct {
403 abi.PtrType
404 }
405
406
407 type sliceType struct {
408 abi.SliceType
409 }
410
411
412 type structField = abi.StructField
413
414
415 type structType struct {
416 abi.StructType
417 }
418
419 func pkgPath(n abi.Name) string {
420 if n.Bytes == nil || *n.DataChecked(0, "name flag field")&(1<<2) == 0 {
421 return ""
422 }
423 i, l := n.ReadVarint(1)
424 off := 1 + i + l
425 if n.HasTag() {
426 i2, l2 := n.ReadVarint(off)
427 off += i2 + l2
428 }
429 var nameOff int32
430
431
432 copy((*[4]byte)(unsafe.Pointer(&nameOff))[:], (*[4]byte)(unsafe.Pointer(n.DataChecked(off, "name offset field")))[:])
433 pkgPathName := abi.Name{Bytes: (*byte)(resolveTypeOff(unsafe.Pointer(n.Bytes), nameOff))}
434 return pkgPathName.Name()
435 }
436
437 func newName(n, tag string, exported, embedded bool) abi.Name {
438 return abi.NewName(n, tag, exported, embedded)
439 }
440
441
445
446
447 type Method struct {
448
449 Name string
450
451
452
453
454
455
456 PkgPath string
457
458 Type Type
459 Func Value
460 Index int
461 }
462
463
464 func (m Method) IsExported() bool {
465 return m.PkgPath == ""
466 }
467
468
469 func (k Kind) String() string {
470 if uint(k) < uint(len(kindNames)) {
471 return kindNames[uint(k)]
472 }
473 return "kind" + strconv.Itoa(int(k))
474 }
475
476 var kindNames = []string{
477 Invalid: "invalid",
478 Bool: "bool",
479 Int: "int",
480 Int8: "int8",
481 Int16: "int16",
482 Int32: "int32",
483 Int64: "int64",
484 Uint: "uint",
485 Uint8: "uint8",
486 Uint16: "uint16",
487 Uint32: "uint32",
488 Uint64: "uint64",
489 Uintptr: "uintptr",
490 Float32: "float32",
491 Float64: "float64",
492 Complex64: "complex64",
493 Complex128: "complex128",
494 Array: "array",
495 Chan: "chan",
496 Func: "func",
497 Interface: "interface",
498 Map: "map",
499 Pointer: "ptr",
500 Slice: "slice",
501 String: "string",
502 Struct: "struct",
503 UnsafePointer: "unsafe.Pointer",
504 }
505
506
507
508
509
510
511 func resolveNameOff(ptrInModule unsafe.Pointer, off int32) unsafe.Pointer
512
513
514
515
516
517
518 func resolveTypeOff(rtype unsafe.Pointer, off int32) unsafe.Pointer
519
520
521
522
523
524
525 func resolveTextOff(rtype unsafe.Pointer, off int32) unsafe.Pointer
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541 func addReflectOff(ptr unsafe.Pointer) int32
542
543
544
545 func resolveReflectName(n abi.Name) aNameOff {
546 return aNameOff(addReflectOff(unsafe.Pointer(n.Bytes)))
547 }
548
549
550
551 func resolveReflectType(t *abi.Type) aTypeOff {
552 return aTypeOff(addReflectOff(unsafe.Pointer(t)))
553 }
554
555
556
557
558 func resolveReflectText(ptr unsafe.Pointer) aTextOff {
559 return aTextOff(addReflectOff(ptr))
560 }
561
562 func (t *rtype) nameOff(off aNameOff) abi.Name {
563 return abi.Name{Bytes: (*byte)(resolveNameOff(unsafe.Pointer(t), int32(off)))}
564 }
565
566 func (t *rtype) typeOff(off aTypeOff) *abi.Type {
567 return (*abi.Type)(resolveTypeOff(unsafe.Pointer(t), int32(off)))
568 }
569
570 func (t *rtype) textOff(off aTextOff) unsafe.Pointer {
571 return resolveTextOff(unsafe.Pointer(t), int32(off))
572 }
573
574 func textOffFor(t *abi.Type, off aTextOff) unsafe.Pointer {
575 return toRType(t).textOff(off)
576 }
577
578 func (t *rtype) String() string {
579 s := t.nameOff(t.t.Str).Name()
580 if t.t.TFlag&abi.TFlagExtraStar != 0 {
581 return s[1:]
582 }
583 return s
584 }
585
586 func (t *rtype) Size() uintptr { return t.t.Size() }
587
588 func (t *rtype) Bits() int {
589 if t == nil {
590 panic("reflect: Bits of nil Type")
591 }
592 k := t.Kind()
593 if k < Int || k > Complex128 {
594 panic("reflect: Bits of non-arithmetic Type " + t.String())
595 }
596 return int(t.t.Size_) * 8
597 }
598
599 func (t *rtype) Align() int { return t.t.Align() }
600
601 func (t *rtype) FieldAlign() int { return t.t.FieldAlign() }
602
603 func (t *rtype) Kind() Kind { return Kind(t.t.Kind()) }
604
605 func (t *rtype) exportedMethods() []abi.Method {
606 ut := t.uncommon()
607 if ut == nil {
608 return nil
609 }
610 return ut.ExportedMethods()
611 }
612
613 func (t *rtype) NumMethod() int {
614 if t.Kind() == Interface {
615 tt := (*interfaceType)(unsafe.Pointer(t))
616 return tt.NumMethod()
617 }
618 return len(t.exportedMethods())
619 }
620
621 func (t *rtype) Method(i int) (m Method) {
622 if t.Kind() == Interface {
623 tt := (*interfaceType)(unsafe.Pointer(t))
624 return tt.Method(i)
625 }
626 methods := t.exportedMethods()
627 if i < 0 || i >= len(methods) {
628 panic("reflect: Method index out of range")
629 }
630 p := methods[i]
631 pname := t.nameOff(p.Name)
632 m.Name = pname.Name()
633 fl := flag(Func)
634 mtyp := t.typeOff(p.Mtyp)
635 ft := (*funcType)(unsafe.Pointer(mtyp))
636 in := make([]Type, 0, 1+ft.NumIn())
637 in = append(in, t)
638 for _, arg := range ft.InSlice() {
639 in = append(in, toRType(arg))
640 }
641 out := make([]Type, 0, ft.NumOut())
642 for _, ret := range ft.OutSlice() {
643 out = append(out, toRType(ret))
644 }
645 mt := FuncOf(in, out, ft.IsVariadic())
646 m.Type = mt
647 tfn := t.textOff(p.Tfn)
648 fn := unsafe.Pointer(&tfn)
649 m.Func = Value{&mt.(*rtype).t, fn, fl}
650
651 m.Index = i
652 return m
653 }
654
655 func (t *rtype) MethodByName(name string) (m Method, ok bool) {
656 if t.Kind() == Interface {
657 tt := (*interfaceType)(unsafe.Pointer(t))
658 return tt.MethodByName(name)
659 }
660 ut := t.uncommon()
661 if ut == nil {
662 return Method{}, false
663 }
664
665 methods := ut.ExportedMethods()
666
667
668
669 i, j := 0, len(methods)
670 for i < j {
671 h := int(uint(i+j) >> 1)
672
673 if !(t.nameOff(methods[h].Name).Name() >= name) {
674 i = h + 1
675 } else {
676 j = h
677 }
678 }
679
680 if i < len(methods) && name == t.nameOff(methods[i].Name).Name() {
681 return t.Method(i), true
682 }
683
684 return Method{}, false
685 }
686
687 func (t *rtype) PkgPath() string {
688 if t.t.TFlag&abi.TFlagNamed == 0 {
689 return ""
690 }
691 ut := t.uncommon()
692 if ut == nil {
693 return ""
694 }
695 return t.nameOff(ut.PkgPath).Name()
696 }
697
698 func pkgPathFor(t *abi.Type) string {
699 return toRType(t).PkgPath()
700 }
701
702 func (t *rtype) Name() string {
703 if !t.t.HasName() {
704 return ""
705 }
706 s := t.String()
707 i := len(s) - 1
708 sqBrackets := 0
709 for i >= 0 && (s[i] != '.' || sqBrackets != 0) {
710 switch s[i] {
711 case ']':
712 sqBrackets++
713 case '[':
714 sqBrackets--
715 }
716 i--
717 }
718 return s[i+1:]
719 }
720
721 func nameFor(t *abi.Type) string {
722 return toRType(t).Name()
723 }
724
725 func (t *rtype) ChanDir() ChanDir {
726 if t.Kind() != Chan {
727 panic("reflect: ChanDir of non-chan type " + t.String())
728 }
729 tt := (*abi.ChanType)(unsafe.Pointer(t))
730 return ChanDir(tt.Dir)
731 }
732
733 func toRType(t *abi.Type) *rtype {
734 return (*rtype)(unsafe.Pointer(t))
735 }
736
737 func elem(t *abi.Type) *abi.Type {
738 et := t.Elem()
739 if et != nil {
740 return et
741 }
742 panic("reflect: Elem of invalid type " + stringFor(t))
743 }
744
745 func (t *rtype) Elem() Type {
746 return toType(elem(t.common()))
747 }
748
749 func (t *rtype) Field(i int) StructField {
750 if t.Kind() != Struct {
751 panic("reflect: Field of non-struct type " + t.String())
752 }
753 tt := (*structType)(unsafe.Pointer(t))
754 return tt.Field(i)
755 }
756
757 func (t *rtype) FieldByIndex(index []int) StructField {
758 if t.Kind() != Struct {
759 panic("reflect: FieldByIndex of non-struct type " + t.String())
760 }
761 tt := (*structType)(unsafe.Pointer(t))
762 return tt.FieldByIndex(index)
763 }
764
765 func (t *rtype) FieldByName(name string) (StructField, bool) {
766 if t.Kind() != Struct {
767 panic("reflect: FieldByName of non-struct type " + t.String())
768 }
769 tt := (*structType)(unsafe.Pointer(t))
770 return tt.FieldByName(name)
771 }
772
773 func (t *rtype) FieldByNameFunc(match func(string) bool) (StructField, bool) {
774 if t.Kind() != Struct {
775 panic("reflect: FieldByNameFunc of non-struct type " + t.String())
776 }
777 tt := (*structType)(unsafe.Pointer(t))
778 return tt.FieldByNameFunc(match)
779 }
780
781 func (t *rtype) Len() int {
782 if t.Kind() != Array {
783 panic("reflect: Len of non-array type " + t.String())
784 }
785 tt := (*arrayType)(unsafe.Pointer(t))
786 return int(tt.Len)
787 }
788
789 func (t *rtype) NumField() int {
790 if t.Kind() != Struct {
791 panic("reflect: NumField of non-struct type " + t.String())
792 }
793 tt := (*structType)(unsafe.Pointer(t))
794 return len(tt.Fields)
795 }
796
797 func (t *rtype) In(i int) Type {
798 if t.Kind() != Func {
799 panic("reflect: In of non-func type " + t.String())
800 }
801 tt := (*abi.FuncType)(unsafe.Pointer(t))
802 return toType(tt.InSlice()[i])
803 }
804
805 func (t *rtype) NumIn() int {
806 if t.Kind() != Func {
807 panic("reflect: NumIn of non-func type " + t.String())
808 }
809 tt := (*abi.FuncType)(unsafe.Pointer(t))
810 return tt.NumIn()
811 }
812
813 func (t *rtype) NumOut() int {
814 if t.Kind() != Func {
815 panic("reflect: NumOut of non-func type " + t.String())
816 }
817 tt := (*abi.FuncType)(unsafe.Pointer(t))
818 return tt.NumOut()
819 }
820
821 func (t *rtype) Out(i int) Type {
822 if t.Kind() != Func {
823 panic("reflect: Out of non-func type " + t.String())
824 }
825 tt := (*abi.FuncType)(unsafe.Pointer(t))
826 return toType(tt.OutSlice()[i])
827 }
828
829 func (t *rtype) IsVariadic() bool {
830 if t.Kind() != Func {
831 panic("reflect: IsVariadic of non-func type " + t.String())
832 }
833 tt := (*abi.FuncType)(unsafe.Pointer(t))
834 return tt.IsVariadic()
835 }
836
837 func (t *rtype) OverflowComplex(x complex128) bool {
838 k := t.Kind()
839 switch k {
840 case Complex64:
841 return overflowFloat32(real(x)) || overflowFloat32(imag(x))
842 case Complex128:
843 return false
844 }
845 panic("reflect: OverflowComplex of non-complex type " + t.String())
846 }
847
848 func (t *rtype) OverflowFloat(x float64) bool {
849 k := t.Kind()
850 switch k {
851 case Float32:
852 return overflowFloat32(x)
853 case Float64:
854 return false
855 }
856 panic("reflect: OverflowFloat of non-float type " + t.String())
857 }
858
859 func (t *rtype) OverflowInt(x int64) bool {
860 k := t.Kind()
861 switch k {
862 case Int, Int8, Int16, Int32, Int64:
863 bitSize := t.Size() * 8
864 trunc := (x << (64 - bitSize)) >> (64 - bitSize)
865 return x != trunc
866 }
867 panic("reflect: OverflowInt of non-int type " + t.String())
868 }
869
870 func (t *rtype) OverflowUint(x uint64) bool {
871 k := t.Kind()
872 switch k {
873 case Uint, Uintptr, Uint8, Uint16, Uint32, Uint64:
874 bitSize := t.Size() * 8
875 trunc := (x << (64 - bitSize)) >> (64 - bitSize)
876 return x != trunc
877 }
878 panic("reflect: OverflowUint of non-uint type " + t.String())
879 }
880
881 func (t *rtype) CanSeq() bool {
882 switch t.Kind() {
883 case Int8, Int16, Int32, Int64, Int, Uint8, Uint16, Uint32, Uint64, Uint, Uintptr, Array, Slice, Chan, String, Map:
884 return true
885 case Func:
886 return canRangeFunc(&t.t)
887 case Pointer:
888 return t.Elem().Kind() == Array
889 }
890 return false
891 }
892
893 func canRangeFunc(t *abi.Type) bool {
894 if t.Kind() != abi.Func {
895 return false
896 }
897 f := t.FuncType()
898 if f.InCount != 1 || f.OutCount != 0 {
899 return false
900 }
901 y := f.In(0)
902 if y.Kind() != abi.Func {
903 return false
904 }
905 yield := y.FuncType()
906 return yield.InCount == 1 && yield.OutCount == 1 && yield.Out(0).Kind() == abi.Bool
907 }
908
909 func (t *rtype) CanSeq2() bool {
910 switch t.Kind() {
911 case Array, Slice, String, Map:
912 return true
913 case Func:
914 return canRangeFunc2(&t.t)
915 case Pointer:
916 return t.Elem().Kind() == Array
917 }
918 return false
919 }
920
921 func canRangeFunc2(t *abi.Type) bool {
922 if t.Kind() != abi.Func {
923 return false
924 }
925 f := t.FuncType()
926 if f.InCount != 1 || f.OutCount != 0 {
927 return false
928 }
929 y := f.In(0)
930 if y.Kind() != abi.Func {
931 return false
932 }
933 yield := y.FuncType()
934 return yield.InCount == 2 && yield.OutCount == 1 && yield.Out(0).Kind() == abi.Bool
935 }
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955 func add(p unsafe.Pointer, x uintptr, whySafe string) unsafe.Pointer {
956 return unsafe.Pointer(uintptr(p) + x)
957 }
958
959 func (d ChanDir) String() string {
960 switch d {
961 case SendDir:
962 return "chan<-"
963 case RecvDir:
964 return "<-chan"
965 case BothDir:
966 return "chan"
967 }
968 return "ChanDir" + strconv.Itoa(int(d))
969 }
970
971
972 func (t *interfaceType) Method(i int) (m Method) {
973 if i < 0 || i >= len(t.Methods) {
974 return
975 }
976 p := &t.Methods[i]
977 pname := t.nameOff(p.Name)
978 m.Name = pname.Name()
979 if !pname.IsExported() {
980 m.PkgPath = pkgPath(pname)
981 if m.PkgPath == "" {
982 m.PkgPath = t.PkgPath.Name()
983 }
984 }
985 m.Type = toType(t.typeOff(p.Typ))
986 m.Index = i
987 return
988 }
989
990
991 func (t *interfaceType) NumMethod() int { return len(t.Methods) }
992
993
994 func (t *interfaceType) MethodByName(name string) (m Method, ok bool) {
995 if t == nil {
996 return
997 }
998 var p *abi.Imethod
999 for i := range t.Methods {
1000 p = &t.Methods[i]
1001 if t.nameOff(p.Name).Name() == name {
1002 return t.Method(i), true
1003 }
1004 }
1005 return
1006 }
1007
1008
1009 type StructField struct {
1010
1011 Name string
1012
1013
1014
1015
1016 PkgPath string
1017
1018 Type Type
1019 Tag StructTag
1020 Offset uintptr
1021 Index []int
1022 Anonymous bool
1023 }
1024
1025
1026 func (f StructField) IsExported() bool {
1027 return f.PkgPath == ""
1028 }
1029
1030
1031
1032
1033
1034
1035
1036
1037
1038 type StructTag string
1039
1040
1041
1042
1043
1044
1045 func (tag StructTag) Get(key string) string {
1046 v, _ := tag.Lookup(key)
1047 return v
1048 }
1049
1050
1051
1052
1053
1054
1055
1056 func (tag StructTag) Lookup(key string) (value string, ok bool) {
1057
1058
1059
1060 for tag != "" {
1061
1062 i := 0
1063 for i < len(tag) && tag[i] == ' ' {
1064 i++
1065 }
1066 tag = tag[i:]
1067 if tag == "" {
1068 break
1069 }
1070
1071
1072
1073
1074
1075 i = 0
1076 for i < len(tag) && tag[i] > ' ' && tag[i] != ':' && tag[i] != '"' && tag[i] != 0x7f {
1077 i++
1078 }
1079 if i == 0 || i+1 >= len(tag) || tag[i] != ':' || tag[i+1] != '"' {
1080 break
1081 }
1082 name := string(tag[:i])
1083 tag = tag[i+1:]
1084
1085
1086 i = 1
1087 for i < len(tag) && tag[i] != '"' {
1088 if tag[i] == '\\' {
1089 i++
1090 }
1091 i++
1092 }
1093 if i >= len(tag) {
1094 break
1095 }
1096 qvalue := string(tag[:i+1])
1097 tag = tag[i+1:]
1098
1099 if key == name {
1100 value, err := strconv.Unquote(qvalue)
1101 if err != nil {
1102 break
1103 }
1104 return value, true
1105 }
1106 }
1107 return "", false
1108 }
1109
1110
1111 func (t *structType) Field(i int) (f StructField) {
1112 if i < 0 || i >= len(t.Fields) {
1113 panic("reflect: Field index out of bounds")
1114 }
1115 p := &t.Fields[i]
1116 f.Type = toType(p.Typ)
1117 f.Name = p.Name.Name()
1118 f.Anonymous = p.Embedded()
1119 if !p.Name.IsExported() {
1120 f.PkgPath = t.PkgPath.Name()
1121 }
1122 if tag := p.Name.Tag(); tag != "" {
1123 f.Tag = StructTag(tag)
1124 }
1125 f.Offset = p.Offset
1126
1127
1128
1129 if i < 256 && runtime.GOOS != "js" && runtime.GOOS != "wasip1" {
1130 staticuint64s := getStaticuint64s()
1131 p := unsafe.Pointer(&(*staticuint64s)[i])
1132 if unsafe.Sizeof(int(0)) == 4 && goarch.BigEndian {
1133 p = unsafe.Add(p, 4)
1134 }
1135 f.Index = unsafe.Slice((*int)(p), 1)
1136 } else {
1137
1138
1139
1140
1141
1142
1143
1144 f.Index = []int{i}
1145 }
1146 return
1147 }
1148
1149
1150
1151
1152
1153
1154 func getStaticuint64s() *[256]uint64
1155
1156
1157
1158
1159
1160 func (t *structType) FieldByIndex(index []int) (f StructField) {
1161 f.Type = toType(&t.Type)
1162 for i, x := range index {
1163 if i > 0 {
1164 ft := f.Type
1165 if ft.Kind() == Pointer && ft.Elem().Kind() == Struct {
1166 ft = ft.Elem()
1167 }
1168 f.Type = ft
1169 }
1170 f = f.Type.Field(x)
1171 }
1172 return
1173 }
1174
1175
1176 type fieldScan struct {
1177 typ *structType
1178 index []int
1179 }
1180
1181
1182
1183 func (t *structType) FieldByNameFunc(match func(string) bool) (result StructField, ok bool) {
1184
1185
1186
1187
1188
1189
1190
1191
1192 current := []fieldScan{}
1193 next := []fieldScan{{typ: t}}
1194
1195
1196
1197
1198
1199
1200
1201 var nextCount map[*structType]int
1202
1203
1204
1205
1206
1207
1208 visited := map[*structType]bool{}
1209
1210 for len(next) > 0 {
1211 current, next = next, current[:0]
1212 count := nextCount
1213 nextCount = nil
1214
1215
1216
1217
1218
1219 for _, scan := range current {
1220 t := scan.typ
1221 if visited[t] {
1222
1223
1224
1225 continue
1226 }
1227 visited[t] = true
1228 for i := range t.Fields {
1229 f := &t.Fields[i]
1230
1231 fname := f.Name.Name()
1232 var ntyp *abi.Type
1233 if f.Embedded() {
1234
1235 ntyp = f.Typ
1236 if ntyp.Kind() == abi.Pointer {
1237 ntyp = ntyp.Elem()
1238 }
1239 }
1240
1241
1242 if match(fname) {
1243
1244 if count[t] > 1 || ok {
1245
1246 return StructField{}, false
1247 }
1248 result = t.Field(i)
1249 result.Index = nil
1250 result.Index = append(result.Index, scan.index...)
1251 result.Index = append(result.Index, i)
1252 ok = true
1253 continue
1254 }
1255
1256
1257
1258
1259 if ok || ntyp == nil || ntyp.Kind() != abi.Struct {
1260 continue
1261 }
1262 styp := (*structType)(unsafe.Pointer(ntyp))
1263 if nextCount[styp] > 0 {
1264 nextCount[styp] = 2
1265 continue
1266 }
1267 if nextCount == nil {
1268 nextCount = map[*structType]int{}
1269 }
1270 nextCount[styp] = 1
1271 if count[t] > 1 {
1272 nextCount[styp] = 2
1273 }
1274 var index []int
1275 index = append(index, scan.index...)
1276 index = append(index, i)
1277 next = append(next, fieldScan{styp, index})
1278 }
1279 }
1280 if ok {
1281 break
1282 }
1283 }
1284 return
1285 }
1286
1287
1288
1289 func (t *structType) FieldByName(name string) (f StructField, present bool) {
1290
1291 hasEmbeds := false
1292 if name != "" {
1293 for i := range t.Fields {
1294 tf := &t.Fields[i]
1295 if tf.Name.Name() == name {
1296 return t.Field(i), true
1297 }
1298 if tf.Embedded() {
1299 hasEmbeds = true
1300 }
1301 }
1302 }
1303 if !hasEmbeds {
1304 return
1305 }
1306 return t.FieldByNameFunc(func(s string) bool { return s == name })
1307 }
1308
1309
1310
1311 func TypeOf(i any) Type {
1312 return toType(abi.TypeOf(i))
1313 }
1314
1315
1316 func TypeFor[T any]() Type {
1317 return toType(abi.TypeFor[T]())
1318 }
1319
1320
1321 func rtypeOf(i any) *abi.Type {
1322 return abi.TypeOf(i)
1323 }
1324
1325
1326 var ptrMap sync.Map
1327
1328
1329
1330
1331
1332
1333
1334
1335
1336
1337 func PtrTo(t Type) Type { return PointerTo(t) }
1338
1339
1340
1341 func PointerTo(t Type) Type {
1342 return toRType(t.(*rtype).ptrTo())
1343 }
1344
1345 func (t *rtype) ptrTo() *abi.Type {
1346 at := &t.t
1347 if at.PtrToThis != 0 {
1348 return t.typeOff(at.PtrToThis)
1349 }
1350
1351
1352 if pi, ok := ptrMap.Load(t); ok {
1353 return &pi.(*ptrType).Type
1354 }
1355
1356
1357 s := "*" + t.String()
1358 for _, tt := range typesByString(s) {
1359 p := (*ptrType)(unsafe.Pointer(tt))
1360 if p.Elem != &t.t {
1361 continue
1362 }
1363 pi, _ := ptrMap.LoadOrStore(t, p)
1364 return &pi.(*ptrType).Type
1365 }
1366
1367
1368
1369 var iptr any = (*unsafe.Pointer)(nil)
1370 prototype := *(**ptrType)(unsafe.Pointer(&iptr))
1371 pp := *prototype
1372
1373 pp.Str = resolveReflectName(newName(s, "", false, false))
1374 pp.PtrToThis = 0
1375
1376
1377
1378
1379
1380
1381 pp.Hash = fnv1(t.t.Hash, '*')
1382
1383 pp.Elem = at
1384
1385 pi, _ := ptrMap.LoadOrStore(t, &pp)
1386 return &pi.(*ptrType).Type
1387 }
1388
1389 func ptrTo(t *abi.Type) *abi.Type {
1390 return toRType(t).ptrTo()
1391 }
1392
1393
1394 func fnv1(x uint32, list ...byte) uint32 {
1395 for _, b := range list {
1396 x = x*16777619 ^ uint32(b)
1397 }
1398 return x
1399 }
1400
1401 func (t *rtype) Implements(u Type) bool {
1402 if u == nil {
1403 panic("reflect: nil type passed to Type.Implements")
1404 }
1405 if u.Kind() != Interface {
1406 panic("reflect: non-interface type passed to Type.Implements")
1407 }
1408 return implements(u.common(), t.common())
1409 }
1410
1411 func (t *rtype) AssignableTo(u Type) bool {
1412 if u == nil {
1413 panic("reflect: nil type passed to Type.AssignableTo")
1414 }
1415 uu := u.common()
1416 return directlyAssignable(uu, t.common()) || implements(uu, t.common())
1417 }
1418
1419 func (t *rtype) ConvertibleTo(u Type) bool {
1420 if u == nil {
1421 panic("reflect: nil type passed to Type.ConvertibleTo")
1422 }
1423 return convertOp(u.common(), t.common()) != nil
1424 }
1425
1426 func (t *rtype) Comparable() bool {
1427 return t.t.Equal != nil
1428 }
1429
1430
1431 func implements(T, V *abi.Type) bool {
1432 if T.Kind() != abi.Interface {
1433 return false
1434 }
1435 t := (*interfaceType)(unsafe.Pointer(T))
1436 if len(t.Methods) == 0 {
1437 return true
1438 }
1439
1440
1441
1442
1443
1444
1445
1446
1447
1448
1449
1450
1451
1452 if V.Kind() == abi.Interface {
1453 v := (*interfaceType)(unsafe.Pointer(V))
1454 i := 0
1455 for j := 0; j < len(v.Methods); j++ {
1456 tm := &t.Methods[i]
1457 tmName := t.nameOff(tm.Name)
1458 vm := &v.Methods[j]
1459 vmName := nameOffFor(V, vm.Name)
1460 if vmName.Name() == tmName.Name() && typeOffFor(V, vm.Typ) == t.typeOff(tm.Typ) {
1461 if !tmName.IsExported() {
1462 tmPkgPath := pkgPath(tmName)
1463 if tmPkgPath == "" {
1464 tmPkgPath = t.PkgPath.Name()
1465 }
1466 vmPkgPath := pkgPath(vmName)
1467 if vmPkgPath == "" {
1468 vmPkgPath = v.PkgPath.Name()
1469 }
1470 if tmPkgPath != vmPkgPath {
1471 continue
1472 }
1473 }
1474 if i++; i >= len(t.Methods) {
1475 return true
1476 }
1477 }
1478 }
1479 return false
1480 }
1481
1482 v := V.Uncommon()
1483 if v == nil {
1484 return false
1485 }
1486 i := 0
1487 vmethods := v.Methods()
1488 for j := 0; j < int(v.Mcount); j++ {
1489 tm := &t.Methods[i]
1490 tmName := t.nameOff(tm.Name)
1491 vm := vmethods[j]
1492 vmName := nameOffFor(V, vm.Name)
1493 if vmName.Name() == tmName.Name() && typeOffFor(V, vm.Mtyp) == t.typeOff(tm.Typ) {
1494 if !tmName.IsExported() {
1495 tmPkgPath := pkgPath(tmName)
1496 if tmPkgPath == "" {
1497 tmPkgPath = t.PkgPath.Name()
1498 }
1499 vmPkgPath := pkgPath(vmName)
1500 if vmPkgPath == "" {
1501 vmPkgPath = nameOffFor(V, v.PkgPath).Name()
1502 }
1503 if tmPkgPath != vmPkgPath {
1504 continue
1505 }
1506 }
1507 if i++; i >= len(t.Methods) {
1508 return true
1509 }
1510 }
1511 }
1512 return false
1513 }
1514
1515
1516
1517
1518
1519 func specialChannelAssignability(T, V *abi.Type) bool {
1520
1521
1522
1523
1524 return V.ChanDir() == abi.BothDir && (nameFor(T) == "" || nameFor(V) == "") && haveIdenticalType(T.Elem(), V.Elem(), true)
1525 }
1526
1527
1528
1529
1530
1531
1532 func directlyAssignable(T, V *abi.Type) bool {
1533
1534 if T == V {
1535 return true
1536 }
1537
1538
1539
1540 if T.HasName() && V.HasName() || T.Kind() != V.Kind() {
1541 return false
1542 }
1543
1544 if T.Kind() == abi.Chan && specialChannelAssignability(T, V) {
1545 return true
1546 }
1547
1548
1549 return haveIdenticalUnderlyingType(T, V, true)
1550 }
1551
1552 func haveIdenticalType(T, V *abi.Type, cmpTags bool) bool {
1553 if cmpTags {
1554 return T == V
1555 }
1556
1557 if nameFor(T) != nameFor(V) || T.Kind() != V.Kind() || pkgPathFor(T) != pkgPathFor(V) {
1558 return false
1559 }
1560
1561 return haveIdenticalUnderlyingType(T, V, false)
1562 }
1563
1564 func haveIdenticalUnderlyingType(T, V *abi.Type, cmpTags bool) bool {
1565 if T == V {
1566 return true
1567 }
1568
1569 kind := Kind(T.Kind())
1570 if kind != Kind(V.Kind()) {
1571 return false
1572 }
1573
1574
1575
1576 if Bool <= kind && kind <= Complex128 || kind == String || kind == UnsafePointer {
1577 return true
1578 }
1579
1580
1581 switch kind {
1582 case Array:
1583 return T.Len() == V.Len() && haveIdenticalType(T.Elem(), V.Elem(), cmpTags)
1584
1585 case Chan:
1586 return V.ChanDir() == T.ChanDir() && haveIdenticalType(T.Elem(), V.Elem(), cmpTags)
1587
1588 case Func:
1589 t := (*funcType)(unsafe.Pointer(T))
1590 v := (*funcType)(unsafe.Pointer(V))
1591 if t.OutCount != v.OutCount || t.InCount != v.InCount {
1592 return false
1593 }
1594 for i := 0; i < t.NumIn(); i++ {
1595 if !haveIdenticalType(t.In(i), v.In(i), cmpTags) {
1596 return false
1597 }
1598 }
1599 for i := 0; i < t.NumOut(); i++ {
1600 if !haveIdenticalType(t.Out(i), v.Out(i), cmpTags) {
1601 return false
1602 }
1603 }
1604 return true
1605
1606 case Interface:
1607 t := (*interfaceType)(unsafe.Pointer(T))
1608 v := (*interfaceType)(unsafe.Pointer(V))
1609 if len(t.Methods) == 0 && len(v.Methods) == 0 {
1610 return true
1611 }
1612
1613
1614 return false
1615
1616 case Map:
1617 return haveIdenticalType(T.Key(), V.Key(), cmpTags) && haveIdenticalType(T.Elem(), V.Elem(), cmpTags)
1618
1619 case Pointer, Slice:
1620 return haveIdenticalType(T.Elem(), V.Elem(), cmpTags)
1621
1622 case Struct:
1623 t := (*structType)(unsafe.Pointer(T))
1624 v := (*structType)(unsafe.Pointer(V))
1625 if len(t.Fields) != len(v.Fields) {
1626 return false
1627 }
1628 if t.PkgPath.Name() != v.PkgPath.Name() {
1629 return false
1630 }
1631 for i := range t.Fields {
1632 tf := &t.Fields[i]
1633 vf := &v.Fields[i]
1634 if tf.Name.Name() != vf.Name.Name() {
1635 return false
1636 }
1637 if !haveIdenticalType(tf.Typ, vf.Typ, cmpTags) {
1638 return false
1639 }
1640 if cmpTags && tf.Name.Tag() != vf.Name.Tag() {
1641 return false
1642 }
1643 if tf.Offset != vf.Offset {
1644 return false
1645 }
1646 if tf.Embedded() != vf.Embedded() {
1647 return false
1648 }
1649 }
1650 return true
1651 }
1652
1653 return false
1654 }
1655
1656
1657
1658
1659
1660
1661
1662
1663
1664
1665
1666
1667
1668
1669
1670
1671
1672
1673
1674
1675 func typelinks() (sections []unsafe.Pointer, offset [][]int32)
1676
1677
1678
1679
1680
1681
1682
1683
1684
1685
1686 func rtypeOff(section unsafe.Pointer, off int32) *abi.Type {
1687 return (*abi.Type)(add(section, uintptr(off), "sizeof(rtype) > 0"))
1688 }
1689
1690
1691
1692
1693
1694
1695
1696
1697
1698
1699
1700
1701
1702
1703
1704
1705 func typesByString(s string) []*abi.Type {
1706 sections, offset := typelinks()
1707 var ret []*abi.Type
1708
1709 for offsI, offs := range offset {
1710 section := sections[offsI]
1711
1712
1713
1714 i, j := 0, len(offs)
1715 for i < j {
1716 h := int(uint(i+j) >> 1)
1717
1718 if !(stringFor(rtypeOff(section, offs[h])) >= s) {
1719 i = h + 1
1720 } else {
1721 j = h
1722 }
1723 }
1724
1725
1726
1727
1728
1729 for j := i; j < len(offs); j++ {
1730 typ := rtypeOff(section, offs[j])
1731 if stringFor(typ) != s {
1732 break
1733 }
1734 ret = append(ret, typ)
1735 }
1736 }
1737 return ret
1738 }
1739
1740
1741 var lookupCache sync.Map
1742
1743
1744
1745
1746 type cacheKey struct {
1747 kind Kind
1748 t1 *abi.Type
1749 t2 *abi.Type
1750 extra uintptr
1751 }
1752
1753
1754
1755
1756 var funcLookupCache struct {
1757 sync.Mutex
1758
1759
1760
1761 m sync.Map
1762 }
1763
1764
1765
1766
1767
1768
1769 func ChanOf(dir ChanDir, t Type) Type {
1770 typ := t.common()
1771
1772
1773 ckey := cacheKey{Chan, typ, nil, uintptr(dir)}
1774 if ch, ok := lookupCache.Load(ckey); ok {
1775 return ch.(*rtype)
1776 }
1777
1778
1779 if typ.Size_ >= 1<<16 {
1780 panic("reflect.ChanOf: element size too large")
1781 }
1782
1783
1784 var s string
1785 switch dir {
1786 default:
1787 panic("reflect.ChanOf: invalid dir")
1788 case SendDir:
1789 s = "chan<- " + stringFor(typ)
1790 case RecvDir:
1791 s = "<-chan " + stringFor(typ)
1792 case BothDir:
1793 typeStr := stringFor(typ)
1794 if typeStr[0] == '<' {
1795
1796
1797
1798
1799 s = "chan (" + typeStr + ")"
1800 } else {
1801 s = "chan " + typeStr
1802 }
1803 }
1804 for _, tt := range typesByString(s) {
1805 ch := (*chanType)(unsafe.Pointer(tt))
1806 if ch.Elem == typ && ch.Dir == abi.ChanDir(dir) {
1807 ti, _ := lookupCache.LoadOrStore(ckey, toRType(tt))
1808 return ti.(Type)
1809 }
1810 }
1811
1812
1813 var ichan any = (chan unsafe.Pointer)(nil)
1814 prototype := *(**chanType)(unsafe.Pointer(&ichan))
1815 ch := *prototype
1816 ch.TFlag = abi.TFlagRegularMemory
1817 ch.Dir = abi.ChanDir(dir)
1818 ch.Str = resolveReflectName(newName(s, "", false, false))
1819 ch.Hash = fnv1(typ.Hash, 'c', byte(dir))
1820 ch.Elem = typ
1821
1822 ti, _ := lookupCache.LoadOrStore(ckey, toRType(&ch.Type))
1823 return ti.(Type)
1824 }
1825
1826 var funcTypes []Type
1827 var funcTypesMutex sync.Mutex
1828
1829 func initFuncTypes(n int) Type {
1830 funcTypesMutex.Lock()
1831 defer funcTypesMutex.Unlock()
1832 if n >= len(funcTypes) {
1833 newFuncTypes := make([]Type, n+1)
1834 copy(newFuncTypes, funcTypes)
1835 funcTypes = newFuncTypes
1836 }
1837 if funcTypes[n] != nil {
1838 return funcTypes[n]
1839 }
1840
1841 funcTypes[n] = StructOf([]StructField{
1842 {
1843 Name: "FuncType",
1844 Type: TypeOf(funcType{}),
1845 },
1846 {
1847 Name: "Args",
1848 Type: ArrayOf(n, TypeOf(&rtype{})),
1849 },
1850 })
1851 return funcTypes[n]
1852 }
1853
1854
1855
1856
1857
1858
1859
1860
1861 func FuncOf(in, out []Type, variadic bool) Type {
1862 if variadic && (len(in) == 0 || in[len(in)-1].Kind() != Slice) {
1863 panic("reflect.FuncOf: last arg of variadic func must be slice")
1864 }
1865
1866
1867 var ifunc any = (func())(nil)
1868 prototype := *(**funcType)(unsafe.Pointer(&ifunc))
1869 n := len(in) + len(out)
1870
1871 if n > 128 {
1872 panic("reflect.FuncOf: too many arguments")
1873 }
1874
1875 o := New(initFuncTypes(n)).Elem()
1876 ft := (*funcType)(unsafe.Pointer(o.Field(0).Addr().Pointer()))
1877 args := unsafe.Slice((**rtype)(unsafe.Pointer(o.Field(1).Addr().Pointer())), n)[0:0:n]
1878 *ft = *prototype
1879
1880
1881 var hash uint32
1882 for _, in := range in {
1883 t := in.(*rtype)
1884 args = append(args, t)
1885 hash = fnv1(hash, byte(t.t.Hash>>24), byte(t.t.Hash>>16), byte(t.t.Hash>>8), byte(t.t.Hash))
1886 }
1887 if variadic {
1888 hash = fnv1(hash, 'v')
1889 }
1890 hash = fnv1(hash, '.')
1891 for _, out := range out {
1892 t := out.(*rtype)
1893 args = append(args, t)
1894 hash = fnv1(hash, byte(t.t.Hash>>24), byte(t.t.Hash>>16), byte(t.t.Hash>>8), byte(t.t.Hash))
1895 }
1896
1897 ft.TFlag = 0
1898 ft.Hash = hash
1899 ft.InCount = uint16(len(in))
1900 ft.OutCount = uint16(len(out))
1901 if variadic {
1902 ft.OutCount |= 1 << 15
1903 }
1904
1905
1906 if ts, ok := funcLookupCache.m.Load(hash); ok {
1907 for _, t := range ts.([]*abi.Type) {
1908 if haveIdenticalUnderlyingType(&ft.Type, t, true) {
1909 return toRType(t)
1910 }
1911 }
1912 }
1913
1914
1915 funcLookupCache.Lock()
1916 defer funcLookupCache.Unlock()
1917 if ts, ok := funcLookupCache.m.Load(hash); ok {
1918 for _, t := range ts.([]*abi.Type) {
1919 if haveIdenticalUnderlyingType(&ft.Type, t, true) {
1920 return toRType(t)
1921 }
1922 }
1923 }
1924
1925 addToCache := func(tt *abi.Type) Type {
1926 var rts []*abi.Type
1927 if rti, ok := funcLookupCache.m.Load(hash); ok {
1928 rts = rti.([]*abi.Type)
1929 }
1930 funcLookupCache.m.Store(hash, append(rts, tt))
1931 return toType(tt)
1932 }
1933
1934
1935 str := funcStr(ft)
1936 for _, tt := range typesByString(str) {
1937 if haveIdenticalUnderlyingType(&ft.Type, tt, true) {
1938 return addToCache(tt)
1939 }
1940 }
1941
1942
1943 ft.Str = resolveReflectName(newName(str, "", false, false))
1944 ft.PtrToThis = 0
1945 return addToCache(&ft.Type)
1946 }
1947 func stringFor(t *abi.Type) string {
1948 return toRType(t).String()
1949 }
1950
1951
1952 func funcStr(ft *funcType) string {
1953 repr := make([]byte, 0, 64)
1954 repr = append(repr, "func("...)
1955 for i, t := range ft.InSlice() {
1956 if i > 0 {
1957 repr = append(repr, ", "...)
1958 }
1959 if ft.IsVariadic() && i == int(ft.InCount)-1 {
1960 repr = append(repr, "..."...)
1961 repr = append(repr, stringFor((*sliceType)(unsafe.Pointer(t)).Elem)...)
1962 } else {
1963 repr = append(repr, stringFor(t)...)
1964 }
1965 }
1966 repr = append(repr, ')')
1967 out := ft.OutSlice()
1968 if len(out) == 1 {
1969 repr = append(repr, ' ')
1970 } else if len(out) > 1 {
1971 repr = append(repr, " ("...)
1972 }
1973 for i, t := range out {
1974 if i > 0 {
1975 repr = append(repr, ", "...)
1976 }
1977 repr = append(repr, stringFor(t)...)
1978 }
1979 if len(out) > 1 {
1980 repr = append(repr, ')')
1981 }
1982 return string(repr)
1983 }
1984
1985
1986
1987 func isReflexive(t *abi.Type) bool {
1988 switch Kind(t.Kind()) {
1989 case Bool, Int, Int8, Int16, Int32, Int64, Uint, Uint8, Uint16, Uint32, Uint64, Uintptr, Chan, Pointer, String, UnsafePointer:
1990 return true
1991 case Float32, Float64, Complex64, Complex128, Interface:
1992 return false
1993 case Array:
1994 tt := (*arrayType)(unsafe.Pointer(t))
1995 return isReflexive(tt.Elem)
1996 case Struct:
1997 tt := (*structType)(unsafe.Pointer(t))
1998 for _, f := range tt.Fields {
1999 if !isReflexive(f.Typ) {
2000 return false
2001 }
2002 }
2003 return true
2004 default:
2005
2006 panic("isReflexive called on non-key type " + stringFor(t))
2007 }
2008 }
2009
2010
2011 func needKeyUpdate(t *abi.Type) bool {
2012 switch Kind(t.Kind()) {
2013 case Bool, Int, Int8, Int16, Int32, Int64, Uint, Uint8, Uint16, Uint32, Uint64, Uintptr, Chan, Pointer, UnsafePointer:
2014 return false
2015 case Float32, Float64, Complex64, Complex128, Interface, String:
2016
2017
2018
2019 return true
2020 case Array:
2021 tt := (*arrayType)(unsafe.Pointer(t))
2022 return needKeyUpdate(tt.Elem)
2023 case Struct:
2024 tt := (*structType)(unsafe.Pointer(t))
2025 for _, f := range tt.Fields {
2026 if needKeyUpdate(f.Typ) {
2027 return true
2028 }
2029 }
2030 return false
2031 default:
2032
2033 panic("needKeyUpdate called on non-key type " + stringFor(t))
2034 }
2035 }
2036
2037
2038 func hashMightPanic(t *abi.Type) bool {
2039 switch Kind(t.Kind()) {
2040 case Interface:
2041 return true
2042 case Array:
2043 tt := (*arrayType)(unsafe.Pointer(t))
2044 return hashMightPanic(tt.Elem)
2045 case Struct:
2046 tt := (*structType)(unsafe.Pointer(t))
2047 for _, f := range tt.Fields {
2048 if hashMightPanic(f.Typ) {
2049 return true
2050 }
2051 }
2052 return false
2053 default:
2054 return false
2055 }
2056 }
2057
2058
2059
2060 func emitGCMask(out []byte, base uintptr, typ *abi.Type, n uintptr) {
2061 ptrs := typ.PtrBytes / goarch.PtrSize
2062 words := typ.Size_ / goarch.PtrSize
2063 mask := typ.GcSlice(0, (ptrs+7)/8)
2064 for j := uintptr(0); j < ptrs; j++ {
2065 if (mask[j/8]>>(j%8))&1 != 0 {
2066 for i := uintptr(0); i < n; i++ {
2067 k := base + i*words + j
2068 out[k/8] |= 1 << (k % 8)
2069 }
2070 }
2071 }
2072 }
2073
2074
2075
2076 func SliceOf(t Type) Type {
2077 typ := t.common()
2078
2079
2080 ckey := cacheKey{Slice, typ, nil, 0}
2081 if slice, ok := lookupCache.Load(ckey); ok {
2082 return slice.(Type)
2083 }
2084
2085
2086 s := "[]" + stringFor(typ)
2087 for _, tt := range typesByString(s) {
2088 slice := (*sliceType)(unsafe.Pointer(tt))
2089 if slice.Elem == typ {
2090 ti, _ := lookupCache.LoadOrStore(ckey, toRType(tt))
2091 return ti.(Type)
2092 }
2093 }
2094
2095
2096 var islice any = ([]unsafe.Pointer)(nil)
2097 prototype := *(**sliceType)(unsafe.Pointer(&islice))
2098 slice := *prototype
2099 slice.TFlag = 0
2100 slice.Str = resolveReflectName(newName(s, "", false, false))
2101 slice.Hash = fnv1(typ.Hash, '[')
2102 slice.Elem = typ
2103 slice.PtrToThis = 0
2104
2105 ti, _ := lookupCache.LoadOrStore(ckey, toRType(&slice.Type))
2106 return ti.(Type)
2107 }
2108
2109
2110
2111
2112 var structLookupCache struct {
2113 sync.Mutex
2114
2115
2116
2117 m sync.Map
2118 }
2119
2120 type structTypeUncommon struct {
2121 structType
2122 u uncommonType
2123 }
2124
2125
2126 func isLetter(ch rune) bool {
2127 return 'a' <= ch && ch <= 'z' || 'A' <= ch && ch <= 'Z' || ch == '_' || ch >= utf8.RuneSelf && unicode.IsLetter(ch)
2128 }
2129
2130
2131
2132
2133
2134
2135
2136 func isValidFieldName(fieldName string) bool {
2137 for i, c := range fieldName {
2138 if i == 0 && !isLetter(c) {
2139 return false
2140 }
2141
2142 if !(isLetter(c) || unicode.IsDigit(c)) {
2143 return false
2144 }
2145 }
2146
2147 return len(fieldName) > 0
2148 }
2149
2150
2151 func isRegularMemory(t Type) bool {
2152 switch t.Kind() {
2153 case Array:
2154 elem := t.Elem()
2155 if isRegularMemory(elem) {
2156 return true
2157 }
2158 return elem.Comparable() && t.Len() == 0
2159 case Int8, Int16, Int32, Int64, Int, Uint8, Uint16, Uint32, Uint64, Uint, Uintptr, Chan, Pointer, Bool, UnsafePointer:
2160 return true
2161 case Struct:
2162 num := t.NumField()
2163 switch num {
2164 case 0:
2165 return true
2166 case 1:
2167 field := t.Field(0)
2168 if field.Name == "_" {
2169 return false
2170 }
2171 return isRegularMemory(field.Type)
2172 default:
2173 for i := range num {
2174 field := t.Field(i)
2175 if field.Name == "_" || !isRegularMemory(field.Type) || isPaddedField(t, i) {
2176 return false
2177 }
2178 }
2179 return true
2180 }
2181 }
2182 return false
2183 }
2184
2185
2186
2187 func isPaddedField(t Type, i int) bool {
2188 field := t.Field(i)
2189 if i+1 < t.NumField() {
2190 return field.Offset+field.Type.Size() != t.Field(i+1).Offset
2191 }
2192 return field.Offset+field.Type.Size() != t.Size()
2193 }
2194
2195
2196
2197
2198
2199
2200
2201 func StructOf(fields []StructField) Type {
2202 var (
2203 hash = fnv1(0, []byte("struct {")...)
2204 size uintptr
2205 typalign uint8
2206 comparable = true
2207 methods []abi.Method
2208
2209 fs = make([]structField, len(fields))
2210 repr = make([]byte, 0, 64)
2211 fset = map[string]struct{}{}
2212 )
2213
2214 lastzero := uintptr(0)
2215 repr = append(repr, "struct {"...)
2216 pkgpath := ""
2217 for i, field := range fields {
2218 if field.Name == "" {
2219 panic("reflect.StructOf: field " + strconv.Itoa(i) + " has no name")
2220 }
2221 if !isValidFieldName(field.Name) {
2222 panic("reflect.StructOf: field " + strconv.Itoa(i) + " has invalid name")
2223 }
2224 if field.Type == nil {
2225 panic("reflect.StructOf: field " + strconv.Itoa(i) + " has no type")
2226 }
2227 f, fpkgpath := runtimeStructField(field)
2228 ft := f.Typ
2229 if fpkgpath != "" {
2230 if pkgpath == "" {
2231 pkgpath = fpkgpath
2232 } else if pkgpath != fpkgpath {
2233 panic("reflect.Struct: fields with different PkgPath " + pkgpath + " and " + fpkgpath)
2234 }
2235 }
2236
2237
2238 name := f.Name.Name()
2239 hash = fnv1(hash, []byte(name)...)
2240 if !f.Embedded() {
2241 repr = append(repr, (" " + name)...)
2242 } else {
2243
2244 if f.Typ.Kind() == abi.Pointer {
2245
2246 elem := ft.Elem()
2247 if k := elem.Kind(); k == abi.Pointer || k == abi.Interface {
2248 panic("reflect.StructOf: illegal embedded field type " + stringFor(ft))
2249 }
2250 }
2251
2252 switch Kind(f.Typ.Kind()) {
2253 case Interface:
2254 ift := (*interfaceType)(unsafe.Pointer(ft))
2255 for _, m := range ift.Methods {
2256 if pkgPath(ift.nameOff(m.Name)) != "" {
2257
2258 panic("reflect: embedded interface with unexported method(s) not implemented")
2259 }
2260
2261 fnStub := resolveReflectText(unsafe.Pointer(abi.FuncPCABIInternal(embeddedIfaceMethStub)))
2262 methods = append(methods, abi.Method{
2263 Name: resolveReflectName(ift.nameOff(m.Name)),
2264 Mtyp: resolveReflectType(ift.typeOff(m.Typ)),
2265 Ifn: fnStub,
2266 Tfn: fnStub,
2267 })
2268 }
2269 case Pointer:
2270 ptr := (*ptrType)(unsafe.Pointer(ft))
2271 if unt := ptr.Uncommon(); unt != nil {
2272 if i > 0 && unt.Mcount > 0 {
2273
2274 panic("reflect: embedded type with methods not implemented if type is not first field")
2275 }
2276 if len(fields) > 1 {
2277 panic("reflect: embedded type with methods not implemented if there is more than one field")
2278 }
2279 for _, m := range unt.Methods() {
2280 mname := nameOffFor(ft, m.Name)
2281 if pkgPath(mname) != "" {
2282
2283
2284 panic("reflect: embedded interface with unexported method(s) not implemented")
2285 }
2286 methods = append(methods, abi.Method{
2287 Name: resolveReflectName(mname),
2288 Mtyp: resolveReflectType(typeOffFor(ft, m.Mtyp)),
2289 Ifn: resolveReflectText(textOffFor(ft, m.Ifn)),
2290 Tfn: resolveReflectText(textOffFor(ft, m.Tfn)),
2291 })
2292 }
2293 }
2294 if unt := ptr.Elem.Uncommon(); unt != nil {
2295 for _, m := range unt.Methods() {
2296 mname := nameOffFor(ft, m.Name)
2297 if pkgPath(mname) != "" {
2298
2299
2300 panic("reflect: embedded interface with unexported method(s) not implemented")
2301 }
2302 methods = append(methods, abi.Method{
2303 Name: resolveReflectName(mname),
2304 Mtyp: resolveReflectType(typeOffFor(ptr.Elem, m.Mtyp)),
2305 Ifn: resolveReflectText(textOffFor(ptr.Elem, m.Ifn)),
2306 Tfn: resolveReflectText(textOffFor(ptr.Elem, m.Tfn)),
2307 })
2308 }
2309 }
2310 default:
2311 if unt := ft.Uncommon(); unt != nil {
2312 if i > 0 && unt.Mcount > 0 {
2313
2314 panic("reflect: embedded type with methods not implemented if type is not first field")
2315 }
2316 if len(fields) > 1 && ft.Kind_&abi.KindDirectIface != 0 {
2317 panic("reflect: embedded type with methods not implemented for non-pointer type")
2318 }
2319 for _, m := range unt.Methods() {
2320 mname := nameOffFor(ft, m.Name)
2321 if pkgPath(mname) != "" {
2322
2323
2324 panic("reflect: embedded interface with unexported method(s) not implemented")
2325 }
2326 methods = append(methods, abi.Method{
2327 Name: resolveReflectName(mname),
2328 Mtyp: resolveReflectType(typeOffFor(ft, m.Mtyp)),
2329 Ifn: resolveReflectText(textOffFor(ft, m.Ifn)),
2330 Tfn: resolveReflectText(textOffFor(ft, m.Tfn)),
2331 })
2332
2333 }
2334 }
2335 }
2336 }
2337 if _, dup := fset[name]; dup && name != "_" {
2338 panic("reflect.StructOf: duplicate field " + name)
2339 }
2340 fset[name] = struct{}{}
2341
2342 hash = fnv1(hash, byte(ft.Hash>>24), byte(ft.Hash>>16), byte(ft.Hash>>8), byte(ft.Hash))
2343
2344 repr = append(repr, (" " + stringFor(ft))...)
2345 if f.Name.HasTag() {
2346 hash = fnv1(hash, []byte(f.Name.Tag())...)
2347 repr = append(repr, (" " + strconv.Quote(f.Name.Tag()))...)
2348 }
2349 if i < len(fields)-1 {
2350 repr = append(repr, ';')
2351 }
2352
2353 comparable = comparable && (ft.Equal != nil)
2354
2355 offset := align(size, uintptr(ft.Align_))
2356 if offset < size {
2357 panic("reflect.StructOf: struct size would exceed virtual address space")
2358 }
2359 if ft.Align_ > typalign {
2360 typalign = ft.Align_
2361 }
2362 size = offset + ft.Size_
2363 if size < offset {
2364 panic("reflect.StructOf: struct size would exceed virtual address space")
2365 }
2366 f.Offset = offset
2367
2368 if ft.Size_ == 0 {
2369 lastzero = size
2370 }
2371
2372 fs[i] = f
2373 }
2374
2375 if size > 0 && lastzero == size {
2376
2377
2378
2379
2380
2381 size++
2382 if size == 0 {
2383 panic("reflect.StructOf: struct size would exceed virtual address space")
2384 }
2385 }
2386
2387 var typ *structType
2388 var ut *uncommonType
2389
2390 if len(methods) == 0 {
2391 t := new(structTypeUncommon)
2392 typ = &t.structType
2393 ut = &t.u
2394 } else {
2395
2396
2397
2398
2399
2400 tt := New(StructOf([]StructField{
2401 {Name: "S", Type: TypeOf(structType{})},
2402 {Name: "U", Type: TypeOf(uncommonType{})},
2403 {Name: "M", Type: ArrayOf(len(methods), TypeOf(methods[0]))},
2404 }))
2405
2406 typ = (*structType)(tt.Elem().Field(0).Addr().UnsafePointer())
2407 ut = (*uncommonType)(tt.Elem().Field(1).Addr().UnsafePointer())
2408
2409 copy(tt.Elem().Field(2).Slice(0, len(methods)).Interface().([]abi.Method), methods)
2410 }
2411
2412
2413
2414
2415 ut.Mcount = uint16(len(methods))
2416 ut.Xcount = ut.Mcount
2417 ut.Moff = uint32(unsafe.Sizeof(uncommonType{}))
2418
2419 if len(fs) > 0 {
2420 repr = append(repr, ' ')
2421 }
2422 repr = append(repr, '}')
2423 hash = fnv1(hash, '}')
2424 str := string(repr)
2425
2426
2427 s := align(size, uintptr(typalign))
2428 if s < size {
2429 panic("reflect.StructOf: struct size would exceed virtual address space")
2430 }
2431 size = s
2432
2433
2434 var istruct any = struct{}{}
2435 prototype := *(**structType)(unsafe.Pointer(&istruct))
2436 *typ = *prototype
2437 typ.Fields = fs
2438 if pkgpath != "" {
2439 typ.PkgPath = newName(pkgpath, "", false, false)
2440 }
2441
2442
2443 if ts, ok := structLookupCache.m.Load(hash); ok {
2444 for _, st := range ts.([]Type) {
2445 t := st.common()
2446 if haveIdenticalUnderlyingType(&typ.Type, t, true) {
2447 return toType(t)
2448 }
2449 }
2450 }
2451
2452
2453 structLookupCache.Lock()
2454 defer structLookupCache.Unlock()
2455 if ts, ok := structLookupCache.m.Load(hash); ok {
2456 for _, st := range ts.([]Type) {
2457 t := st.common()
2458 if haveIdenticalUnderlyingType(&typ.Type, t, true) {
2459 return toType(t)
2460 }
2461 }
2462 }
2463
2464 addToCache := func(t Type) Type {
2465 var ts []Type
2466 if ti, ok := structLookupCache.m.Load(hash); ok {
2467 ts = ti.([]Type)
2468 }
2469 structLookupCache.m.Store(hash, append(ts, t))
2470 return t
2471 }
2472
2473
2474 for _, t := range typesByString(str) {
2475 if haveIdenticalUnderlyingType(&typ.Type, t, true) {
2476
2477
2478
2479 return addToCache(toType(t))
2480 }
2481 }
2482
2483 typ.Str = resolveReflectName(newName(str, "", false, false))
2484 if isRegularMemory(toType(&typ.Type)) {
2485 typ.TFlag = abi.TFlagRegularMemory
2486 } else {
2487 typ.TFlag = 0
2488 }
2489 typ.Hash = hash
2490 typ.Size_ = size
2491 typ.PtrBytes = typeptrdata(&typ.Type)
2492 typ.Align_ = typalign
2493 typ.FieldAlign_ = typalign
2494 typ.PtrToThis = 0
2495 if len(methods) > 0 {
2496 typ.TFlag |= abi.TFlagUncommon
2497 }
2498
2499 if typ.PtrBytes == 0 {
2500 typ.GCData = nil
2501 } else if typ.PtrBytes <= abi.MaxPtrmaskBytes*8*goarch.PtrSize {
2502 bv := new(bitVector)
2503 addTypeBits(bv, 0, &typ.Type)
2504 typ.GCData = &bv.data[0]
2505 } else {
2506
2507
2508 typ.TFlag |= abi.TFlagGCMaskOnDemand
2509 typ.GCData = (*byte)(unsafe.Pointer(new(uintptr)))
2510 }
2511
2512 typ.Equal = nil
2513 if comparable {
2514 typ.Equal = func(p, q unsafe.Pointer) bool {
2515 for _, ft := range typ.Fields {
2516 pi := add(p, ft.Offset, "&x.field safe")
2517 qi := add(q, ft.Offset, "&x.field safe")
2518 if !ft.Typ.Equal(pi, qi) {
2519 return false
2520 }
2521 }
2522 return true
2523 }
2524 }
2525
2526 switch {
2527 case len(fs) == 1 && !fs[0].Typ.IfaceIndir():
2528
2529 typ.Kind_ |= abi.KindDirectIface
2530 default:
2531 typ.Kind_ &^= abi.KindDirectIface
2532 }
2533
2534 return addToCache(toType(&typ.Type))
2535 }
2536
2537 func embeddedIfaceMethStub() {
2538 panic("reflect: StructOf does not support methods of embedded interfaces")
2539 }
2540
2541
2542
2543
2544 func runtimeStructField(field StructField) (structField, string) {
2545 if field.Anonymous && field.PkgPath != "" {
2546 panic("reflect.StructOf: field \"" + field.Name + "\" is anonymous but has PkgPath set")
2547 }
2548
2549 if field.IsExported() {
2550
2551
2552 c := field.Name[0]
2553 if 'a' <= c && c <= 'z' || c == '_' {
2554 panic("reflect.StructOf: field \"" + field.Name + "\" is unexported but missing PkgPath")
2555 }
2556 }
2557
2558 resolveReflectType(field.Type.common())
2559 f := structField{
2560 Name: newName(field.Name, string(field.Tag), field.IsExported(), field.Anonymous),
2561 Typ: field.Type.common(),
2562 Offset: 0,
2563 }
2564 return f, field.PkgPath
2565 }
2566
2567
2568
2569
2570 func typeptrdata(t *abi.Type) uintptr {
2571 switch t.Kind() {
2572 case abi.Struct:
2573 st := (*structType)(unsafe.Pointer(t))
2574
2575 field := -1
2576 for i := range st.Fields {
2577 ft := st.Fields[i].Typ
2578 if ft.Pointers() {
2579 field = i
2580 }
2581 }
2582 if field == -1 {
2583 return 0
2584 }
2585 f := st.Fields[field]
2586 return f.Offset + f.Typ.PtrBytes
2587
2588 default:
2589 panic("reflect.typeptrdata: unexpected type, " + stringFor(t))
2590 }
2591 }
2592
2593
2594
2595
2596
2597
2598 func ArrayOf(length int, elem Type) Type {
2599 if length < 0 {
2600 panic("reflect: negative length passed to ArrayOf")
2601 }
2602
2603 typ := elem.common()
2604
2605
2606 ckey := cacheKey{Array, typ, nil, uintptr(length)}
2607 if array, ok := lookupCache.Load(ckey); ok {
2608 return array.(Type)
2609 }
2610
2611
2612 s := "[" + strconv.Itoa(length) + "]" + stringFor(typ)
2613 for _, tt := range typesByString(s) {
2614 array := (*arrayType)(unsafe.Pointer(tt))
2615 if array.Elem == typ {
2616 ti, _ := lookupCache.LoadOrStore(ckey, toRType(tt))
2617 return ti.(Type)
2618 }
2619 }
2620
2621
2622 var iarray any = [1]unsafe.Pointer{}
2623 prototype := *(**arrayType)(unsafe.Pointer(&iarray))
2624 array := *prototype
2625 array.TFlag = typ.TFlag & abi.TFlagRegularMemory
2626 array.Str = resolveReflectName(newName(s, "", false, false))
2627 array.Hash = fnv1(typ.Hash, '[')
2628 for n := uint32(length); n > 0; n >>= 8 {
2629 array.Hash = fnv1(array.Hash, byte(n))
2630 }
2631 array.Hash = fnv1(array.Hash, ']')
2632 array.Elem = typ
2633 array.PtrToThis = 0
2634 if typ.Size_ > 0 {
2635 max := ^uintptr(0) / typ.Size_
2636 if uintptr(length) > max {
2637 panic("reflect.ArrayOf: array size would exceed virtual address space")
2638 }
2639 }
2640 array.Size_ = typ.Size_ * uintptr(length)
2641 if length > 0 && typ.Pointers() {
2642 array.PtrBytes = typ.Size_*uintptr(length-1) + typ.PtrBytes
2643 } else {
2644 array.PtrBytes = 0
2645 }
2646 array.Align_ = typ.Align_
2647 array.FieldAlign_ = typ.FieldAlign_
2648 array.Len = uintptr(length)
2649 array.Slice = &(SliceOf(elem).(*rtype).t)
2650
2651 switch {
2652 case array.PtrBytes == 0:
2653
2654 array.GCData = nil
2655
2656 case length == 1:
2657
2658
2659 array.TFlag |= typ.TFlag & abi.TFlagGCMaskOnDemand
2660 array.GCData = typ.GCData
2661
2662 case array.PtrBytes <= abi.MaxPtrmaskBytes*8*goarch.PtrSize:
2663
2664 n := (array.PtrBytes/goarch.PtrSize + 7) / 8
2665
2666 n = (n + goarch.PtrSize - 1) &^ (goarch.PtrSize - 1)
2667 mask := make([]byte, n)
2668 emitGCMask(mask, 0, typ, array.Len)
2669 array.GCData = &mask[0]
2670
2671 default:
2672
2673
2674 array.TFlag |= abi.TFlagGCMaskOnDemand
2675 array.GCData = (*byte)(unsafe.Pointer(new(uintptr)))
2676 }
2677
2678 etyp := typ
2679 esize := etyp.Size()
2680
2681 array.Equal = nil
2682 if eequal := etyp.Equal; eequal != nil {
2683 array.Equal = func(p, q unsafe.Pointer) bool {
2684 for i := 0; i < length; i++ {
2685 pi := arrayAt(p, i, esize, "i < length")
2686 qi := arrayAt(q, i, esize, "i < length")
2687 if !eequal(pi, qi) {
2688 return false
2689 }
2690
2691 }
2692 return true
2693 }
2694 }
2695
2696 switch {
2697 case length == 1 && !typ.IfaceIndir():
2698
2699 array.Kind_ |= abi.KindDirectIface
2700 default:
2701 array.Kind_ &^= abi.KindDirectIface
2702 }
2703
2704 ti, _ := lookupCache.LoadOrStore(ckey, toRType(&array.Type))
2705 return ti.(Type)
2706 }
2707
2708 func appendVarint(x []byte, v uintptr) []byte {
2709 for ; v >= 0x80; v >>= 7 {
2710 x = append(x, byte(v|0x80))
2711 }
2712 x = append(x, byte(v))
2713 return x
2714 }
2715
2716
2717
2718
2719
2720
2721
2722
2723
2724
2725
2726
2727
2728
2729
2730
2731
2732
2733
2734 func toType(t *abi.Type) Type {
2735 if t == nil {
2736 return nil
2737 }
2738 return toRType(t)
2739 }
2740
2741 type layoutKey struct {
2742 ftyp *funcType
2743 rcvr *abi.Type
2744 }
2745
2746 type layoutType struct {
2747 t *abi.Type
2748 framePool *sync.Pool
2749 abid abiDesc
2750 }
2751
2752 var layoutCache sync.Map
2753
2754
2755
2756
2757
2758
2759
2760
2761 func funcLayout(t *funcType, rcvr *abi.Type) (frametype *abi.Type, framePool *sync.Pool, abid abiDesc) {
2762 if t.Kind() != abi.Func {
2763 panic("reflect: funcLayout of non-func type " + stringFor(&t.Type))
2764 }
2765 if rcvr != nil && rcvr.Kind() == abi.Interface {
2766 panic("reflect: funcLayout with interface receiver " + stringFor(rcvr))
2767 }
2768 k := layoutKey{t, rcvr}
2769 if lti, ok := layoutCache.Load(k); ok {
2770 lt := lti.(layoutType)
2771 return lt.t, lt.framePool, lt.abid
2772 }
2773
2774
2775 abid = newAbiDesc(t, rcvr)
2776
2777
2778 x := &abi.Type{
2779 Align_: goarch.PtrSize,
2780
2781
2782
2783
2784 Size_: align(abid.retOffset+abid.ret.stackBytes, goarch.PtrSize),
2785 PtrBytes: uintptr(abid.stackPtrs.n) * goarch.PtrSize,
2786 }
2787 if abid.stackPtrs.n > 0 {
2788 x.GCData = &abid.stackPtrs.data[0]
2789 }
2790
2791 var s string
2792 if rcvr != nil {
2793 s = "methodargs(" + stringFor(rcvr) + ")(" + stringFor(&t.Type) + ")"
2794 } else {
2795 s = "funcargs(" + stringFor(&t.Type) + ")"
2796 }
2797 x.Str = resolveReflectName(newName(s, "", false, false))
2798
2799
2800 framePool = &sync.Pool{New: func() any {
2801 return unsafe_New(x)
2802 }}
2803 lti, _ := layoutCache.LoadOrStore(k, layoutType{
2804 t: x,
2805 framePool: framePool,
2806 abid: abid,
2807 })
2808 lt := lti.(layoutType)
2809 return lt.t, lt.framePool, lt.abid
2810 }
2811
2812
2813 type bitVector struct {
2814 n uint32
2815 data []byte
2816 }
2817
2818
2819 func (bv *bitVector) append(bit uint8) {
2820 if bv.n%(8*goarch.PtrSize) == 0 {
2821
2822
2823
2824 for i := 0; i < goarch.PtrSize; i++ {
2825 bv.data = append(bv.data, 0)
2826 }
2827 }
2828 bv.data[bv.n/8] |= bit << (bv.n % 8)
2829 bv.n++
2830 }
2831
2832 func addTypeBits(bv *bitVector, offset uintptr, t *abi.Type) {
2833 if !t.Pointers() {
2834 return
2835 }
2836
2837 switch Kind(t.Kind_ & abi.KindMask) {
2838 case Chan, Func, Map, Pointer, Slice, String, UnsafePointer:
2839
2840 for bv.n < uint32(offset/goarch.PtrSize) {
2841 bv.append(0)
2842 }
2843 bv.append(1)
2844
2845 case Interface:
2846
2847 for bv.n < uint32(offset/goarch.PtrSize) {
2848 bv.append(0)
2849 }
2850 bv.append(1)
2851 bv.append(1)
2852
2853 case Array:
2854
2855 tt := (*arrayType)(unsafe.Pointer(t))
2856 for i := 0; i < int(tt.Len); i++ {
2857 addTypeBits(bv, offset+uintptr(i)*tt.Elem.Size_, tt.Elem)
2858 }
2859
2860 case Struct:
2861
2862 tt := (*structType)(unsafe.Pointer(t))
2863 for i := range tt.Fields {
2864 f := &tt.Fields[i]
2865 addTypeBits(bv, offset+f.Offset, f.Typ)
2866 }
2867 }
2868 }
2869
View as plain text