mirror of
https://github.com/superseriousbusiness/gotosocial.git
synced 2025-01-07 06:38:57 +01:00
251 lines
5.2 KiB
Go
251 lines
5.2 KiB
Go
|
package mangler
|
||
|
|
||
|
import (
|
||
|
"reflect"
|
||
|
"unsafe"
|
||
|
|
||
|
"github.com/modern-go/reflect2"
|
||
|
)
|
||
|
|
||
|
type (
|
||
|
byteser interface{ Bytes() []byte }
|
||
|
stringer interface{ String() string }
|
||
|
binarymarshaler interface{ MarshalBinary() ([]byte, error) }
|
||
|
textmarshaler interface{ MarshalText() ([]byte, error) }
|
||
|
jsonmarshaler interface{ MarshalJSON() ([]byte, error) }
|
||
|
)
|
||
|
|
||
|
func append_uint16(b []byte, u uint16) []byte {
|
||
|
return append(b, // LE
|
||
|
byte(u),
|
||
|
byte(u>>8),
|
||
|
)
|
||
|
}
|
||
|
|
||
|
func append_uint32(b []byte, u uint32) []byte {
|
||
|
return append(b, // LE
|
||
|
byte(u),
|
||
|
byte(u>>8),
|
||
|
byte(u>>16),
|
||
|
byte(u>>24),
|
||
|
)
|
||
|
}
|
||
|
|
||
|
func append_uint64(b []byte, u uint64) []byte {
|
||
|
return append(b, // LE
|
||
|
byte(u),
|
||
|
byte(u>>8),
|
||
|
byte(u>>16),
|
||
|
byte(u>>24),
|
||
|
byte(u>>32),
|
||
|
byte(u>>40),
|
||
|
byte(u>>48),
|
||
|
byte(u>>56),
|
||
|
)
|
||
|
}
|
||
|
|
||
|
func deref_ptr_mangler(rtype reflect.Type, mangle Mangler, count int) Mangler {
|
||
|
if rtype == nil || mangle == nil || count == 0 {
|
||
|
panic("bad input")
|
||
|
}
|
||
|
|
||
|
// Get reflect2's type for later
|
||
|
// unsafe interface data repacking,
|
||
|
type2 := reflect2.Type2(rtype)
|
||
|
|
||
|
return func(buf []byte, value any) []byte {
|
||
|
// Get raw value data.
|
||
|
ptr := eface_data(value)
|
||
|
|
||
|
// Deref n - 1 number times.
|
||
|
for i := 0; i < count-1; i++ {
|
||
|
|
||
|
if ptr == nil {
|
||
|
// Check for nil values
|
||
|
buf = append(buf, '0')
|
||
|
return buf
|
||
|
}
|
||
|
|
||
|
// Further deref ptr
|
||
|
buf = append(buf, '1')
|
||
|
ptr = *(*unsafe.Pointer)(ptr)
|
||
|
}
|
||
|
|
||
|
if ptr == nil {
|
||
|
// Final nil value check.
|
||
|
buf = append(buf, '0')
|
||
|
return buf
|
||
|
}
|
||
|
|
||
|
// Repack and mangle fully deref'd
|
||
|
value = type2.UnsafeIndirect(ptr)
|
||
|
buf = append(buf, '1')
|
||
|
return mangle(buf, value)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func iter_slice_mangler(rtype reflect.Type, mangle Mangler) Mangler {
|
||
|
if rtype == nil || mangle == nil {
|
||
|
panic("bad input")
|
||
|
}
|
||
|
|
||
|
// Get reflect2's type for later
|
||
|
// unsafe slice data manipulation.
|
||
|
slice2 := reflect2.Type2(rtype).(*reflect2.UnsafeSliceType)
|
||
|
|
||
|
return func(buf []byte, value any) []byte {
|
||
|
// Get raw value data.
|
||
|
ptr := eface_data(value)
|
||
|
|
||
|
// Get length of slice value.
|
||
|
n := slice2.UnsafeLengthOf(ptr)
|
||
|
|
||
|
for i := 0; i < n; i++ {
|
||
|
// Mangle data at each slice index.
|
||
|
e := slice2.UnsafeGetIndex(ptr, i)
|
||
|
buf = mangle(buf, e)
|
||
|
buf = append(buf, ',')
|
||
|
}
|
||
|
|
||
|
if n > 0 {
|
||
|
// Drop final comma.
|
||
|
buf = buf[:len(buf)-1]
|
||
|
}
|
||
|
|
||
|
return buf
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func iter_array_mangler(rtype reflect.Type, mangle Mangler) Mangler {
|
||
|
if rtype == nil || mangle == nil {
|
||
|
panic("bad input")
|
||
|
}
|
||
|
|
||
|
// Get reflect2's type for later
|
||
|
// unsafe slice data manipulation.
|
||
|
array2 := reflect2.Type2(rtype).(*reflect2.UnsafeArrayType)
|
||
|
n := array2.Len()
|
||
|
|
||
|
return func(buf []byte, value any) []byte {
|
||
|
// Get raw value data.
|
||
|
ptr := eface_data(value)
|
||
|
|
||
|
for i := 0; i < n; i++ {
|
||
|
// Mangle data at each slice index.
|
||
|
e := array2.UnsafeGetIndex(ptr, i)
|
||
|
buf = mangle(buf, e)
|
||
|
buf = append(buf, ',')
|
||
|
}
|
||
|
|
||
|
if n > 0 {
|
||
|
// Drop final comma.
|
||
|
buf = buf[:len(buf)-1]
|
||
|
}
|
||
|
|
||
|
return buf
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func iter_map_mangler(rtype reflect.Type, kmangle, emangle Mangler) Mangler {
|
||
|
if rtype == nil || kmangle == nil || emangle == nil {
|
||
|
panic("bad input")
|
||
|
}
|
||
|
|
||
|
// Get reflect2's type for later
|
||
|
// unsafe map data manipulation.
|
||
|
map2 := reflect2.Type2(rtype).(*reflect2.UnsafeMapType)
|
||
|
key2, elem2 := map2.Key(), map2.Elem()
|
||
|
|
||
|
return func(buf []byte, value any) []byte {
|
||
|
// Get raw value data.
|
||
|
ptr := eface_data(value)
|
||
|
ptr = indirect_ptr(ptr)
|
||
|
|
||
|
// Create iterator for map value.
|
||
|
iter := map2.UnsafeIterate(ptr)
|
||
|
|
||
|
// Check if empty map.
|
||
|
empty := !iter.HasNext()
|
||
|
|
||
|
for iter.HasNext() {
|
||
|
// Get key + elem data as ifaces.
|
||
|
kptr, eptr := iter.UnsafeNext()
|
||
|
key := key2.UnsafeIndirect(kptr)
|
||
|
elem := elem2.UnsafeIndirect(eptr)
|
||
|
|
||
|
// Mangle data for key + elem.
|
||
|
buf = kmangle(buf, key)
|
||
|
buf = append(buf, ':')
|
||
|
buf = emangle(buf, elem)
|
||
|
buf = append(buf, ',')
|
||
|
}
|
||
|
|
||
|
if !empty {
|
||
|
// Drop final comma.
|
||
|
buf = buf[:len(buf)-1]
|
||
|
}
|
||
|
|
||
|
return buf
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func iter_struct_mangler(rtype reflect.Type, manglers []Mangler) Mangler {
|
||
|
if rtype == nil || len(manglers) != rtype.NumField() {
|
||
|
panic("bad input")
|
||
|
}
|
||
|
|
||
|
type field struct {
|
||
|
type2 reflect2.Type
|
||
|
field *reflect2.UnsafeStructField
|
||
|
mangle Mangler
|
||
|
}
|
||
|
|
||
|
// Get reflect2's type for later
|
||
|
// unsafe struct field data access.
|
||
|
struct2 := reflect2.Type2(rtype).(*reflect2.UnsafeStructType)
|
||
|
|
||
|
// Bundle together the fields and manglers.
|
||
|
fields := make([]field, rtype.NumField())
|
||
|
for i := range fields {
|
||
|
fields[i].field = struct2.Field(i).(*reflect2.UnsafeStructField)
|
||
|
fields[i].type2 = fields[i].field.Type()
|
||
|
fields[i].mangle = manglers[i]
|
||
|
if fields[i].type2 == nil ||
|
||
|
fields[i].field == nil ||
|
||
|
fields[i].mangle == nil {
|
||
|
panic("bad input")
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return func(buf []byte, value any) []byte {
|
||
|
// Get raw value data.
|
||
|
ptr := eface_data(value)
|
||
|
|
||
|
for i := range fields {
|
||
|
// Get struct field as iface via offset.
|
||
|
fptr := fields[i].field.UnsafeGet(ptr)
|
||
|
field := fields[i].type2.UnsafeIndirect(fptr)
|
||
|
|
||
|
// Mangle the struct field data.
|
||
|
buf = fields[i].mangle(buf, field)
|
||
|
buf = append(buf, ',')
|
||
|
}
|
||
|
|
||
|
if len(fields) > 0 {
|
||
|
// Drop final comma.
|
||
|
buf = buf[:len(buf)-1]
|
||
|
}
|
||
|
|
||
|
return buf
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func indirect_ptr(p unsafe.Pointer) unsafe.Pointer {
|
||
|
return unsafe.Pointer(&p)
|
||
|
}
|
||
|
|
||
|
func eface_data(a any) unsafe.Pointer {
|
||
|
type eface struct{ _, data unsafe.Pointer }
|
||
|
return (*eface)(unsafe.Pointer(&a)).data
|
||
|
}
|