gotosocial/vendor/github.com/bytedance/sonic/internal/rt/fastvalue.go

262 lines
5 KiB
Go
Raw Normal View History

2023-02-25 12:12:40 +00:00
/*
* Copyright 2021 ByteDance Inc.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package rt
import (
"reflect"
"unsafe"
2023-02-25 12:12:40 +00:00
)
var (
reflectRtypeItab = findReflectRtypeItab()
2023-02-25 12:12:40 +00:00
)
// GoType.KindFlags const
2023-02-25 12:12:40 +00:00
const (
F_direct = 1 << 5
F_kind_mask = (1 << 5) - 1
2023-02-25 12:12:40 +00:00
)
// GoType.Flags const
const (
tflagUncommon uint8 = 1 << 0
tflagExtraStar uint8 = 1 << 1
tflagNamed uint8 = 1 << 2
tflagRegularMemory uint8 = 1 << 3
)
2023-02-25 12:12:40 +00:00
type GoType struct {
Size uintptr
PtrData uintptr
Hash uint32
Flags uint8
Align uint8
FieldAlign uint8
KindFlags uint8
Traits unsafe.Pointer
GCData *byte
Str int32
PtrToSelf int32
2023-02-25 12:12:40 +00:00
}
func (self *GoType) IsNamed() bool {
return (self.Flags & tflagNamed) != 0
}
2023-02-25 12:12:40 +00:00
func (self *GoType) Kind() reflect.Kind {
return reflect.Kind(self.KindFlags & F_kind_mask)
2023-02-25 12:12:40 +00:00
}
func (self *GoType) Pack() (t reflect.Type) {
(*GoIface)(unsafe.Pointer(&t)).Itab = reflectRtypeItab
(*GoIface)(unsafe.Pointer(&t)).Value = unsafe.Pointer(self)
return
2023-02-25 12:12:40 +00:00
}
func (self *GoType) String() string {
return self.Pack().String()
2023-02-25 12:12:40 +00:00
}
func (self *GoType) Indirect() bool {
return self.KindFlags&F_direct == 0
2023-02-25 12:12:40 +00:00
}
type GoMap struct {
Count int
Flags uint8
B uint8
Overflow uint16
Hash0 uint32
Buckets unsafe.Pointer
OldBuckets unsafe.Pointer
Evacuate uintptr
Extra unsafe.Pointer
2023-02-25 12:12:40 +00:00
}
type GoMapIterator struct {
K unsafe.Pointer
V unsafe.Pointer
T *GoMapType
H *GoMap
Buckets unsafe.Pointer
Bptr *unsafe.Pointer
Overflow *[]unsafe.Pointer
OldOverflow *[]unsafe.Pointer
StartBucket uintptr
Offset uint8
Wrapped bool
B uint8
I uint8
Bucket uintptr
CheckBucket uintptr
2023-02-25 12:12:40 +00:00
}
type GoItab struct {
it unsafe.Pointer
Vt *GoType
hv uint32
_ [4]byte
fn [1]uintptr
2023-02-25 12:12:40 +00:00
}
type GoIface struct {
Itab *GoItab
Value unsafe.Pointer
2023-02-25 12:12:40 +00:00
}
type GoEface struct {
Type *GoType
Value unsafe.Pointer
2023-02-25 12:12:40 +00:00
}
func (self GoEface) Pack() (v interface{}) {
*(*GoEface)(unsafe.Pointer(&v)) = self
return
2023-02-25 12:12:40 +00:00
}
type GoPtrType struct {
GoType
Elem *GoType
2023-02-25 12:12:40 +00:00
}
type GoMapType struct {
GoType
Key *GoType
Elem *GoType
Bucket *GoType
Hasher func(unsafe.Pointer, uintptr) uintptr
KeySize uint8
ElemSize uint8
BucketSize uint16
Flags uint32
2023-02-25 12:12:40 +00:00
}
func (self *GoMapType) IndirectElem() bool {
return self.Flags&2 != 0
2023-02-25 12:12:40 +00:00
}
type GoStructType struct {
GoType
Pkg *byte
Fields []GoStructField
2023-02-25 12:12:40 +00:00
}
type GoStructField struct {
Name *byte
Type *GoType
OffEmbed uintptr
2023-02-25 12:12:40 +00:00
}
type GoInterfaceType struct {
GoType
PkgPath *byte
Methods []GoInterfaceMethod
2023-02-25 12:12:40 +00:00
}
type GoInterfaceMethod struct {
Name int32
Type int32
2023-02-25 12:12:40 +00:00
}
type GoSlice struct {
Ptr unsafe.Pointer
Len int
Cap int
2023-02-25 12:12:40 +00:00
}
type GoString struct {
Ptr unsafe.Pointer
Len int
2023-02-25 12:12:40 +00:00
}
func PtrElem(t *GoType) *GoType {
return (*GoPtrType)(unsafe.Pointer(t)).Elem
2023-02-25 12:12:40 +00:00
}
func MapType(t *GoType) *GoMapType {
return (*GoMapType)(unsafe.Pointer(t))
2023-02-25 12:12:40 +00:00
}
func IfaceType(t *GoType) *GoInterfaceType {
return (*GoInterfaceType)(unsafe.Pointer(t))
2023-02-25 12:12:40 +00:00
}
func UnpackType(t reflect.Type) *GoType {
return (*GoType)((*GoIface)(unsafe.Pointer(&t)).Value)
2023-02-25 12:12:40 +00:00
}
func UnpackEface(v interface{}) GoEface {
return *(*GoEface)(unsafe.Pointer(&v))
2023-02-25 12:12:40 +00:00
}
func UnpackIface(v interface{}) GoIface {
return *(*GoIface)(unsafe.Pointer(&v))
2023-02-25 12:12:40 +00:00
}
func findReflectRtypeItab() *GoItab {
v := reflect.TypeOf(struct{}{})
return (*GoIface)(unsafe.Pointer(&v)).Itab
2023-02-25 12:12:40 +00:00
}
func AssertI2I2(t *GoType, i GoIface) (r GoIface) {
inter := IfaceType(t)
tab := i.Itab
if tab == nil {
return
}
if (*GoInterfaceType)(tab.it) != inter {
tab = GetItab(inter, tab.Vt, true)
if tab == nil {
return
}
}
r.Itab = tab
r.Value = i.Value
return
}
func (t *GoType) IsInt64() bool {
return t.Kind() == reflect.Int64 || (t.Kind() == reflect.Int && t.Size == 8)
}
func (t *GoType) IsInt32() bool {
return t.Kind() == reflect.Int32 || (t.Kind() == reflect.Int && t.Size == 4)
}
//go:nosplit
func (t *GoType) IsUint64() bool {
isUint := t.Kind() == reflect.Uint || t.Kind() == reflect.Uintptr
return t.Kind() == reflect.Uint64 || (isUint && t.Size == 8)
}
//go:nosplit
func (t *GoType) IsUint32() bool {
isUint := t.Kind() == reflect.Uint || t.Kind() == reflect.Uintptr
return t.Kind() == reflect.Uint32 || (isUint && t.Size == 4)
}
//go:nosplit
func PtrAdd(ptr unsafe.Pointer, offset uintptr) unsafe.Pointer {
return unsafe.Pointer(uintptr(ptr) + offset)
}
//go:noescape
//go:linkname GetItab runtime.getitab
func GetItab(inter *GoInterfaceType, typ *GoType, canfail bool) *GoItab