1
2
3
4
5 package abi
6
7 import (
8 "unsafe"
9 )
10
11
12
13
14
15
16
17
18
19
20 type Type struct {
21 Size_ uintptr
22 PtrBytes uintptr
23 Hash uint32
24 TFlag TFlag
25 Align_ uint8
26 FieldAlign_ uint8
27 Kind_ Kind
28
29
30 Equal func(unsafe.Pointer, unsafe.Pointer) bool
31
32
33
34
35
36
37
38
39
40
41
42 GCData *byte
43 Str NameOff
44 PtrToThis TypeOff
45 }
46
47
48
49 type Kind uint8
50
51 const (
52 Invalid Kind = iota
53 Bool
54 Int
55 Int8
56 Int16
57 Int32
58 Int64
59 Uint
60 Uint8
61 Uint16
62 Uint32
63 Uint64
64 Uintptr
65 Float32
66 Float64
67 Complex64
68 Complex128
69 Array
70 Chan
71 Func
72 Interface
73 Map
74 Pointer
75 Slice
76 String
77 Struct
78 UnsafePointer
79 )
80
81 const (
82
83 KindDirectIface Kind = 1 << 5
84 KindMask Kind = (1 << 5) - 1
85 )
86
87
88
89 type TFlag uint8
90
91 const (
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107 TFlagUncommon TFlag = 1 << 0
108
109
110
111
112
113 TFlagExtraStar TFlag = 1 << 1
114
115
116 TFlagNamed TFlag = 1 << 2
117
118
119
120 TFlagRegularMemory TFlag = 1 << 3
121
122
123
124
125
126
127 TFlagGCMaskOnDemand TFlag = 1 << 4
128 )
129
130
131 type NameOff int32
132
133
134 type TypeOff int32
135
136
137 type TextOff int32
138
139
140 func (k Kind) String() string {
141 if int(k) < len(kindNames) {
142 return kindNames[k]
143 }
144 return kindNames[0]
145 }
146
147 var kindNames = []string{
148 Invalid: "invalid",
149 Bool: "bool",
150 Int: "int",
151 Int8: "int8",
152 Int16: "int16",
153 Int32: "int32",
154 Int64: "int64",
155 Uint: "uint",
156 Uint8: "uint8",
157 Uint16: "uint16",
158 Uint32: "uint32",
159 Uint64: "uint64",
160 Uintptr: "uintptr",
161 Float32: "float32",
162 Float64: "float64",
163 Complex64: "complex64",
164 Complex128: "complex128",
165 Array: "array",
166 Chan: "chan",
167 Func: "func",
168 Interface: "interface",
169 Map: "map",
170 Pointer: "ptr",
171 Slice: "slice",
172 String: "string",
173 Struct: "struct",
174 UnsafePointer: "unsafe.Pointer",
175 }
176
177
178 func TypeOf(a any) *Type {
179 eface := *(*EmptyInterface)(unsafe.Pointer(&a))
180
181
182
183
184
185 return (*Type)(NoEscape(unsafe.Pointer(eface.Type)))
186 }
187
188
189 func TypeFor[T any]() *Type {
190 var v T
191 if t := TypeOf(v); t != nil {
192 return t
193 }
194 return TypeOf((*T)(nil)).Elem()
195 }
196
197 func (t *Type) Kind() Kind { return t.Kind_ & KindMask }
198
199 func (t *Type) HasName() bool {
200 return t.TFlag&TFlagNamed != 0
201 }
202
203
204 func (t *Type) Pointers() bool { return t.PtrBytes != 0 }
205
206
207 func (t *Type) IfaceIndir() bool {
208 return t.Kind_&KindDirectIface == 0
209 }
210
211
212 func (t *Type) IsDirectIface() bool {
213 return t.Kind_&KindDirectIface != 0
214 }
215
216 func (t *Type) GcSlice(begin, end uintptr) []byte {
217 if t.TFlag&TFlagGCMaskOnDemand != 0 {
218 panic("GcSlice can't handle on-demand gcdata types")
219 }
220 return unsafe.Slice(t.GCData, int(end))[begin:]
221 }
222
223
224 type Method struct {
225 Name NameOff
226 Mtyp TypeOff
227 Ifn TextOff
228 Tfn TextOff
229 }
230
231
232
233
234
235 type UncommonType struct {
236 PkgPath NameOff
237 Mcount uint16
238 Xcount uint16
239 Moff uint32
240 _ uint32
241 }
242
243 func (t *UncommonType) Methods() []Method {
244 if t.Mcount == 0 {
245 return nil
246 }
247 return (*[1 << 16]Method)(addChecked(unsafe.Pointer(t), uintptr(t.Moff), "t.mcount > 0"))[:t.Mcount:t.Mcount]
248 }
249
250 func (t *UncommonType) ExportedMethods() []Method {
251 if t.Xcount == 0 {
252 return nil
253 }
254 return (*[1 << 16]Method)(addChecked(unsafe.Pointer(t), uintptr(t.Moff), "t.xcount > 0"))[:t.Xcount:t.Xcount]
255 }
256
257
258
259
260
261
262
263
264 func addChecked(p unsafe.Pointer, x uintptr, whySafe string) unsafe.Pointer {
265 return unsafe.Pointer(uintptr(p) + x)
266 }
267
268
269 type Imethod struct {
270 Name NameOff
271 Typ TypeOff
272 }
273
274
275 type ArrayType struct {
276 Type
277 Elem *Type
278 Slice *Type
279 Len uintptr
280 }
281
282
283 func (t *Type) Len() int {
284 if t.Kind() == Array {
285 return int((*ArrayType)(unsafe.Pointer(t)).Len)
286 }
287 return 0
288 }
289
290 func (t *Type) Common() *Type {
291 return t
292 }
293
294 type ChanDir int
295
296 const (
297 RecvDir ChanDir = 1 << iota
298 SendDir
299 BothDir = RecvDir | SendDir
300 InvalidDir ChanDir = 0
301 )
302
303
304 type ChanType struct {
305 Type
306 Elem *Type
307 Dir ChanDir
308 }
309
310 type structTypeUncommon struct {
311 StructType
312 u UncommonType
313 }
314
315
316 func (t *Type) ChanDir() ChanDir {
317 if t.Kind() == Chan {
318 ch := (*ChanType)(unsafe.Pointer(t))
319 return ch.Dir
320 }
321 return InvalidDir
322 }
323
324
325 func (t *Type) Uncommon() *UncommonType {
326 if t.TFlag&TFlagUncommon == 0 {
327 return nil
328 }
329 switch t.Kind() {
330 case Struct:
331 return &(*structTypeUncommon)(unsafe.Pointer(t)).u
332 case Pointer:
333 type u struct {
334 PtrType
335 u UncommonType
336 }
337 return &(*u)(unsafe.Pointer(t)).u
338 case Func:
339 type u struct {
340 FuncType
341 u UncommonType
342 }
343 return &(*u)(unsafe.Pointer(t)).u
344 case Slice:
345 type u struct {
346 SliceType
347 u UncommonType
348 }
349 return &(*u)(unsafe.Pointer(t)).u
350 case Array:
351 type u struct {
352 ArrayType
353 u UncommonType
354 }
355 return &(*u)(unsafe.Pointer(t)).u
356 case Chan:
357 type u struct {
358 ChanType
359 u UncommonType
360 }
361 return &(*u)(unsafe.Pointer(t)).u
362 case Map:
363 type u struct {
364 mapType
365 u UncommonType
366 }
367 return &(*u)(unsafe.Pointer(t)).u
368 case Interface:
369 type u struct {
370 InterfaceType
371 u UncommonType
372 }
373 return &(*u)(unsafe.Pointer(t)).u
374 default:
375 type u struct {
376 Type
377 u UncommonType
378 }
379 return &(*u)(unsafe.Pointer(t)).u
380 }
381 }
382
383
384 func (t *Type) Elem() *Type {
385 switch t.Kind() {
386 case Array:
387 tt := (*ArrayType)(unsafe.Pointer(t))
388 return tt.Elem
389 case Chan:
390 tt := (*ChanType)(unsafe.Pointer(t))
391 return tt.Elem
392 case Map:
393 tt := (*mapType)(unsafe.Pointer(t))
394 return tt.Elem
395 case Pointer:
396 tt := (*PtrType)(unsafe.Pointer(t))
397 return tt.Elem
398 case Slice:
399 tt := (*SliceType)(unsafe.Pointer(t))
400 return tt.Elem
401 }
402 return nil
403 }
404
405
406 func (t *Type) StructType() *StructType {
407 if t.Kind() != Struct {
408 return nil
409 }
410 return (*StructType)(unsafe.Pointer(t))
411 }
412
413
414 func (t *Type) MapType() *mapType {
415 if t.Kind() != Map {
416 return nil
417 }
418 return (*mapType)(unsafe.Pointer(t))
419 }
420
421
422 func (t *Type) ArrayType() *ArrayType {
423 if t.Kind() != Array {
424 return nil
425 }
426 return (*ArrayType)(unsafe.Pointer(t))
427 }
428
429
430 func (t *Type) FuncType() *FuncType {
431 if t.Kind() != Func {
432 return nil
433 }
434 return (*FuncType)(unsafe.Pointer(t))
435 }
436
437
438 func (t *Type) InterfaceType() *InterfaceType {
439 if t.Kind() != Interface {
440 return nil
441 }
442 return (*InterfaceType)(unsafe.Pointer(t))
443 }
444
445
446 func (t *Type) Size() uintptr { return t.Size_ }
447
448
449 func (t *Type) Align() int { return int(t.Align_) }
450
451 func (t *Type) FieldAlign() int { return int(t.FieldAlign_) }
452
453 type InterfaceType struct {
454 Type
455 PkgPath Name
456 Methods []Imethod
457 }
458
459 func (t *Type) ExportedMethods() []Method {
460 ut := t.Uncommon()
461 if ut == nil {
462 return nil
463 }
464 return ut.ExportedMethods()
465 }
466
467 func (t *Type) NumMethod() int {
468 if t.Kind() == Interface {
469 tt := (*InterfaceType)(unsafe.Pointer(t))
470 return tt.NumMethod()
471 }
472 return len(t.ExportedMethods())
473 }
474
475
476 func (t *InterfaceType) NumMethod() int { return len(t.Methods) }
477
478 func (t *Type) Key() *Type {
479 if t.Kind() == Map {
480 return (*mapType)(unsafe.Pointer(t)).Key
481 }
482 return nil
483 }
484
485 type SliceType struct {
486 Type
487 Elem *Type
488 }
489
490
491
492
493
494
495
496
497
498
499
500
501 type FuncType struct {
502 Type
503 InCount uint16
504 OutCount uint16
505 }
506
507 func (t *FuncType) In(i int) *Type {
508 return t.InSlice()[i]
509 }
510
511 func (t *FuncType) NumIn() int {
512 return int(t.InCount)
513 }
514
515 func (t *FuncType) NumOut() int {
516 return int(t.OutCount & (1<<15 - 1))
517 }
518
519 func (t *FuncType) Out(i int) *Type {
520 return (t.OutSlice()[i])
521 }
522
523 func (t *FuncType) InSlice() []*Type {
524 uadd := unsafe.Sizeof(*t)
525 if t.TFlag&TFlagUncommon != 0 {
526 uadd += unsafe.Sizeof(UncommonType{})
527 }
528 if t.InCount == 0 {
529 return nil
530 }
531 return (*[1 << 16]*Type)(addChecked(unsafe.Pointer(t), uadd, "t.inCount > 0"))[:t.InCount:t.InCount]
532 }
533 func (t *FuncType) OutSlice() []*Type {
534 outCount := uint16(t.NumOut())
535 if outCount == 0 {
536 return nil
537 }
538 uadd := unsafe.Sizeof(*t)
539 if t.TFlag&TFlagUncommon != 0 {
540 uadd += unsafe.Sizeof(UncommonType{})
541 }
542 return (*[1 << 17]*Type)(addChecked(unsafe.Pointer(t), uadd, "outCount > 0"))[t.InCount : t.InCount+outCount : t.InCount+outCount]
543 }
544
545 func (t *FuncType) IsVariadic() bool {
546 return t.OutCount&(1<<15) != 0
547 }
548
549 type PtrType struct {
550 Type
551 Elem *Type
552 }
553
554 type StructField struct {
555 Name Name
556 Typ *Type
557 Offset uintptr
558 }
559
560 func (f *StructField) Embedded() bool {
561 return f.Name.IsEmbedded()
562 }
563
564 type StructType struct {
565 Type
566 PkgPath Name
567 Fields []StructField
568 }
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596 type Name struct {
597 Bytes *byte
598 }
599
600
601
602 func (n Name) DataChecked(off int, whySafe string) *byte {
603 return (*byte)(addChecked(unsafe.Pointer(n.Bytes), uintptr(off), whySafe))
604 }
605
606
607
608 func (n Name) Data(off int) *byte {
609 return (*byte)(addChecked(unsafe.Pointer(n.Bytes), uintptr(off), "the runtime doesn't need to give you a reason"))
610 }
611
612
613 func (n Name) IsExported() bool {
614 return (*n.Bytes)&(1<<0) != 0
615 }
616
617
618 func (n Name) HasTag() bool {
619 return (*n.Bytes)&(1<<1) != 0
620 }
621
622
623 func (n Name) IsEmbedded() bool {
624 return (*n.Bytes)&(1<<3) != 0
625 }
626
627
628
629 func (n Name) ReadVarint(off int) (int, int) {
630 v := 0
631 for i := 0; ; i++ {
632 x := *n.DataChecked(off+i, "read varint")
633 v += int(x&0x7f) << (7 * i)
634 if x&0x80 == 0 {
635 return i + 1, v
636 }
637 }
638 }
639
640
641 func (n Name) IsBlank() bool {
642 if n.Bytes == nil {
643 return false
644 }
645 _, l := n.ReadVarint(1)
646 return l == 1 && *n.Data(2) == '_'
647 }
648
649
650
651
652 func writeVarint(buf []byte, n int) int {
653 for i := 0; ; i++ {
654 b := byte(n & 0x7f)
655 n >>= 7
656 if n == 0 {
657 buf[i] = b
658 return i + 1
659 }
660 buf[i] = b | 0x80
661 }
662 }
663
664
665 func (n Name) Name() string {
666 if n.Bytes == nil {
667 return ""
668 }
669 i, l := n.ReadVarint(1)
670 return unsafe.String(n.DataChecked(1+i, "non-empty string"), l)
671 }
672
673
674 func (n Name) Tag() string {
675 if !n.HasTag() {
676 return ""
677 }
678 i, l := n.ReadVarint(1)
679 i2, l2 := n.ReadVarint(1 + i + l)
680 return unsafe.String(n.DataChecked(1+i+l+i2, "non-empty string"), l2)
681 }
682
683 func NewName(n, tag string, exported, embedded bool) Name {
684 if len(n) >= 1<<29 {
685 panic("abi.NewName: name too long: " + n[:1024] + "...")
686 }
687 if len(tag) >= 1<<29 {
688 panic("abi.NewName: tag too long: " + tag[:1024] + "...")
689 }
690 var nameLen [10]byte
691 var tagLen [10]byte
692 nameLenLen := writeVarint(nameLen[:], len(n))
693 tagLenLen := writeVarint(tagLen[:], len(tag))
694
695 var bits byte
696 l := 1 + nameLenLen + len(n)
697 if exported {
698 bits |= 1 << 0
699 }
700 if len(tag) > 0 {
701 l += tagLenLen + len(tag)
702 bits |= 1 << 1
703 }
704 if embedded {
705 bits |= 1 << 3
706 }
707
708 b := make([]byte, l)
709 b[0] = bits
710 copy(b[1:], nameLen[:nameLenLen])
711 copy(b[1+nameLenLen:], n)
712 if len(tag) > 0 {
713 tb := b[1+nameLenLen+len(n):]
714 copy(tb, tagLen[:tagLenLen])
715 copy(tb[tagLenLen:], tag)
716 }
717
718 return Name{Bytes: &b[0]}
719 }
720
721 const (
722 TraceArgsLimit = 10
723 TraceArgsMaxDepth = 5
724
725
726
727
728
729
730 TraceArgsMaxLen = (TraceArgsMaxDepth*3+2)*TraceArgsLimit + 1
731 )
732
733
734
735
736
737
738
739
740
741
742
743
744
745 const (
746 TraceArgsEndSeq = 0xff
747 TraceArgsStartAgg = 0xfe
748 TraceArgsEndAgg = 0xfd
749 TraceArgsDotdotdot = 0xfc
750 TraceArgsOffsetTooLarge = 0xfb
751 TraceArgsSpecial = 0xf0
752 )
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783 const MaxPtrmaskBytes = 2048
784
View as plain text