You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 

139 lines
3.0 KiB

//go:build unsafe
// +build unsafe
package protocol
import (
"reflect"
"unsafe"
)
type iface struct {
typ unsafe.Pointer
ptr unsafe.Pointer
}
type slice struct {
ptr unsafe.Pointer
len int
cap int
}
type index uintptr
type _type struct {
ptr unsafe.Pointer
}
func typeOf(x interface{}) _type {
return _type{ptr: ((*iface)(unsafe.Pointer(&x))).typ}
}
func elemTypeOf(x interface{}) _type {
return makeType(reflect.TypeOf(x).Elem())
}
func makeType(t reflect.Type) _type {
return _type{ptr: ((*iface)(unsafe.Pointer(&t))).ptr}
}
type value struct {
ptr unsafe.Pointer
}
func nonAddressableValueOf(x interface{}) value {
return valueOf(x)
}
func valueOf(x interface{}) value {
return value{ptr: ((*iface)(unsafe.Pointer(&x))).ptr}
}
func makeValue(t reflect.Type) value {
return value{ptr: unsafe.Pointer(reflect.New(t).Pointer())}
}
func (v value) bool() bool { return *(*bool)(v.ptr) }
func (v value) int8() int8 { return *(*int8)(v.ptr) }
func (v value) int16() int16 { return *(*int16)(v.ptr) }
func (v value) int32() int32 { return *(*int32)(v.ptr) }
func (v value) int64() int64 { return *(*int64)(v.ptr) }
func (v value) string() string { return *(*string)(v.ptr) }
func (v value) bytes() []byte { return *(*[]byte)(v.ptr) }
func (v value) iface(t reflect.Type) interface{} {
return *(*interface{})(unsafe.Pointer(&iface{
typ: ((*iface)(unsafe.Pointer(&t))).ptr,
ptr: v.ptr,
}))
}
func (v value) array(t reflect.Type) array {
return array{
size: uintptr(t.Size()),
elem: ((*slice)(v.ptr)).ptr,
len: ((*slice)(v.ptr)).len,
}
}
func (v value) setBool(b bool) { *(*bool)(v.ptr) = b }
func (v value) setInt8(i int8) { *(*int8)(v.ptr) = i }
func (v value) setInt16(i int16) { *(*int16)(v.ptr) = i }
func (v value) setInt32(i int32) { *(*int32)(v.ptr) = i }
func (v value) setInt64(i int64) { *(*int64)(v.ptr) = i }
func (v value) setString(s string) { *(*string)(v.ptr) = s }
func (v value) setBytes(b []byte) { *(*[]byte)(v.ptr) = b }
func (v value) setArray(a array) { *(*slice)(v.ptr) = slice{ptr: a.elem, len: a.len, cap: a.len} }
func (v value) fieldByIndex(i index) value {
return value{ptr: unsafe.Pointer(uintptr(v.ptr) + uintptr(i))}
}
type array struct {
elem unsafe.Pointer
size uintptr
len int
}
var (
emptyArray struct{}
)
func makeArray(t reflect.Type, n int) array {
var elem unsafe.Pointer
var size = uintptr(t.Size())
if n == 0 {
elem = unsafe.Pointer(&emptyArray)
} else {
elem = unsafe_NewArray(((*iface)(unsafe.Pointer(&t))).ptr, n)
}
return array{elem: elem, size: size, len: n}
}
func (a array) index(i int) value {
return value{ptr: unsafe.Pointer(uintptr(a.elem) + (uintptr(i) * a.size))}
}
func (a array) length() int { return a.len }
func (a array) isNil() bool { return a.elem == nil }
func indexOf(s reflect.StructField) index { return index(s.Offset) }
func bytesToString(b []byte) string { return *(*string)(unsafe.Pointer(&b)) }
//go:linkname unsafe_NewArray reflect.unsafe_NewArray
func unsafe_NewArray(rtype unsafe.Pointer, length int) unsafe.Pointer