mirror of
				https://github.com/superseriousbusiness/gotosocial.git
				synced 2025-10-31 03:02:25 -05: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 | ||
|  | } |