mirror of
https://github.com/mjl-/mox.git
synced 2025-01-15 09:56:27 +03:00
210 lines
5.3 KiB
Go
210 lines
5.3 KiB
Go
// Copyright 2019 The Go Authors. All rights reserved.
|
|
// Use of this source code is governed by a BSD-style
|
|
// license that can be found in the LICENSE file.
|
|
|
|
//go:build purego || appengine
|
|
// +build purego appengine
|
|
|
|
package impl
|
|
|
|
import (
|
|
"reflect"
|
|
|
|
"google.golang.org/protobuf/encoding/protowire"
|
|
)
|
|
|
|
func sizeEnum(p pointer, f *coderFieldInfo, _ marshalOptions) (size int) {
|
|
v := p.v.Elem().Int()
|
|
return f.tagsize + protowire.SizeVarint(uint64(v))
|
|
}
|
|
|
|
func appendEnum(b []byte, p pointer, f *coderFieldInfo, opts marshalOptions) ([]byte, error) {
|
|
v := p.v.Elem().Int()
|
|
b = protowire.AppendVarint(b, f.wiretag)
|
|
b = protowire.AppendVarint(b, uint64(v))
|
|
return b, nil
|
|
}
|
|
|
|
func consumeEnum(b []byte, p pointer, wtyp protowire.Type, f *coderFieldInfo, _ unmarshalOptions) (out unmarshalOutput, err error) {
|
|
if wtyp != protowire.VarintType {
|
|
return out, errUnknown
|
|
}
|
|
v, n := protowire.ConsumeVarint(b)
|
|
if n < 0 {
|
|
return out, errDecode
|
|
}
|
|
p.v.Elem().SetInt(int64(v))
|
|
out.n = n
|
|
return out, nil
|
|
}
|
|
|
|
func mergeEnum(dst, src pointer, _ *coderFieldInfo, _ mergeOptions) {
|
|
dst.v.Elem().Set(src.v.Elem())
|
|
}
|
|
|
|
var coderEnum = pointerCoderFuncs{
|
|
size: sizeEnum,
|
|
marshal: appendEnum,
|
|
unmarshal: consumeEnum,
|
|
merge: mergeEnum,
|
|
}
|
|
|
|
func sizeEnumNoZero(p pointer, f *coderFieldInfo, opts marshalOptions) (size int) {
|
|
if p.v.Elem().Int() == 0 {
|
|
return 0
|
|
}
|
|
return sizeEnum(p, f, opts)
|
|
}
|
|
|
|
func appendEnumNoZero(b []byte, p pointer, f *coderFieldInfo, opts marshalOptions) ([]byte, error) {
|
|
if p.v.Elem().Int() == 0 {
|
|
return b, nil
|
|
}
|
|
return appendEnum(b, p, f, opts)
|
|
}
|
|
|
|
func mergeEnumNoZero(dst, src pointer, _ *coderFieldInfo, _ mergeOptions) {
|
|
if src.v.Elem().Int() != 0 {
|
|
dst.v.Elem().Set(src.v.Elem())
|
|
}
|
|
}
|
|
|
|
var coderEnumNoZero = pointerCoderFuncs{
|
|
size: sizeEnumNoZero,
|
|
marshal: appendEnumNoZero,
|
|
unmarshal: consumeEnum,
|
|
merge: mergeEnumNoZero,
|
|
}
|
|
|
|
func sizeEnumPtr(p pointer, f *coderFieldInfo, opts marshalOptions) (size int) {
|
|
return sizeEnum(pointer{p.v.Elem()}, f, opts)
|
|
}
|
|
|
|
func appendEnumPtr(b []byte, p pointer, f *coderFieldInfo, opts marshalOptions) ([]byte, error) {
|
|
return appendEnum(b, pointer{p.v.Elem()}, f, opts)
|
|
}
|
|
|
|
func consumeEnumPtr(b []byte, p pointer, wtyp protowire.Type, f *coderFieldInfo, opts unmarshalOptions) (out unmarshalOutput, err error) {
|
|
if wtyp != protowire.VarintType {
|
|
return out, errUnknown
|
|
}
|
|
if p.v.Elem().IsNil() {
|
|
p.v.Elem().Set(reflect.New(p.v.Elem().Type().Elem()))
|
|
}
|
|
return consumeEnum(b, pointer{p.v.Elem()}, wtyp, f, opts)
|
|
}
|
|
|
|
func mergeEnumPtr(dst, src pointer, _ *coderFieldInfo, _ mergeOptions) {
|
|
if !src.v.Elem().IsNil() {
|
|
v := reflect.New(dst.v.Type().Elem().Elem())
|
|
v.Elem().Set(src.v.Elem().Elem())
|
|
dst.v.Elem().Set(v)
|
|
}
|
|
}
|
|
|
|
var coderEnumPtr = pointerCoderFuncs{
|
|
size: sizeEnumPtr,
|
|
marshal: appendEnumPtr,
|
|
unmarshal: consumeEnumPtr,
|
|
merge: mergeEnumPtr,
|
|
}
|
|
|
|
func sizeEnumSlice(p pointer, f *coderFieldInfo, opts marshalOptions) (size int) {
|
|
s := p.v.Elem()
|
|
for i, llen := 0, s.Len(); i < llen; i++ {
|
|
size += protowire.SizeVarint(uint64(s.Index(i).Int())) + f.tagsize
|
|
}
|
|
return size
|
|
}
|
|
|
|
func appendEnumSlice(b []byte, p pointer, f *coderFieldInfo, opts marshalOptions) ([]byte, error) {
|
|
s := p.v.Elem()
|
|
for i, llen := 0, s.Len(); i < llen; i++ {
|
|
b = protowire.AppendVarint(b, f.wiretag)
|
|
b = protowire.AppendVarint(b, uint64(s.Index(i).Int()))
|
|
}
|
|
return b, nil
|
|
}
|
|
|
|
func consumeEnumSlice(b []byte, p pointer, wtyp protowire.Type, f *coderFieldInfo, opts unmarshalOptions) (out unmarshalOutput, err error) {
|
|
s := p.v.Elem()
|
|
if wtyp == protowire.BytesType {
|
|
b, n := protowire.ConsumeBytes(b)
|
|
if n < 0 {
|
|
return out, errDecode
|
|
}
|
|
for len(b) > 0 {
|
|
v, n := protowire.ConsumeVarint(b)
|
|
if n < 0 {
|
|
return out, errDecode
|
|
}
|
|
rv := reflect.New(s.Type().Elem()).Elem()
|
|
rv.SetInt(int64(v))
|
|
s.Set(reflect.Append(s, rv))
|
|
b = b[n:]
|
|
}
|
|
out.n = n
|
|
return out, nil
|
|
}
|
|
if wtyp != protowire.VarintType {
|
|
return out, errUnknown
|
|
}
|
|
v, n := protowire.ConsumeVarint(b)
|
|
if n < 0 {
|
|
return out, errDecode
|
|
}
|
|
rv := reflect.New(s.Type().Elem()).Elem()
|
|
rv.SetInt(int64(v))
|
|
s.Set(reflect.Append(s, rv))
|
|
out.n = n
|
|
return out, nil
|
|
}
|
|
|
|
func mergeEnumSlice(dst, src pointer, _ *coderFieldInfo, _ mergeOptions) {
|
|
dst.v.Elem().Set(reflect.AppendSlice(dst.v.Elem(), src.v.Elem()))
|
|
}
|
|
|
|
var coderEnumSlice = pointerCoderFuncs{
|
|
size: sizeEnumSlice,
|
|
marshal: appendEnumSlice,
|
|
unmarshal: consumeEnumSlice,
|
|
merge: mergeEnumSlice,
|
|
}
|
|
|
|
func sizeEnumPackedSlice(p pointer, f *coderFieldInfo, opts marshalOptions) (size int) {
|
|
s := p.v.Elem()
|
|
llen := s.Len()
|
|
if llen == 0 {
|
|
return 0
|
|
}
|
|
n := 0
|
|
for i := 0; i < llen; i++ {
|
|
n += protowire.SizeVarint(uint64(s.Index(i).Int()))
|
|
}
|
|
return f.tagsize + protowire.SizeBytes(n)
|
|
}
|
|
|
|
func appendEnumPackedSlice(b []byte, p pointer, f *coderFieldInfo, opts marshalOptions) ([]byte, error) {
|
|
s := p.v.Elem()
|
|
llen := s.Len()
|
|
if llen == 0 {
|
|
return b, nil
|
|
}
|
|
b = protowire.AppendVarint(b, f.wiretag)
|
|
n := 0
|
|
for i := 0; i < llen; i++ {
|
|
n += protowire.SizeVarint(uint64(s.Index(i).Int()))
|
|
}
|
|
b = protowire.AppendVarint(b, uint64(n))
|
|
for i := 0; i < llen; i++ {
|
|
b = protowire.AppendVarint(b, uint64(s.Index(i).Int()))
|
|
}
|
|
return b, nil
|
|
}
|
|
|
|
var coderEnumPackedSlice = pointerCoderFuncs{
|
|
size: sizeEnumPackedSlice,
|
|
marshal: appendEnumPackedSlice,
|
|
unmarshal: consumeEnumSlice,
|
|
merge: mergeEnumSlice,
|
|
}
|