|  | // Copyright 2024 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. | 
|  |  | 
|  | package impl | 
|  |  | 
|  | import ( | 
|  | "fmt" | 
|  | "math" | 
|  | "reflect" | 
|  | "strings" | 
|  | "sync/atomic" | 
|  |  | 
|  | "google.golang.org/protobuf/internal/filedesc" | 
|  | "google.golang.org/protobuf/reflect/protoreflect" | 
|  | ) | 
|  |  | 
|  | type opaqueStructInfo struct { | 
|  | structInfo | 
|  | } | 
|  |  | 
|  | // isOpaque determines whether a protobuf message type is on the Opaque API.  It | 
|  | // checks whether the type is a Go struct that protoc-gen-go would generate. | 
|  | // | 
|  | // This function only detects newly generated messages from the v2 | 
|  | // implementation of protoc-gen-go. It is unable to classify generated messages | 
|  | // that are too old or those that are generated by a different generator | 
|  | // such as protoc-gen-gogo. | 
|  | func isOpaque(t reflect.Type) bool { | 
|  | // The current detection mechanism is to simply check the first field | 
|  | // for a struct tag with the "protogen" key. | 
|  | if t.Kind() == reflect.Struct && t.NumField() > 0 { | 
|  | pgt := t.Field(0).Tag.Get("protogen") | 
|  | return strings.HasPrefix(pgt, "opaque.") | 
|  | } | 
|  | return false | 
|  | } | 
|  |  | 
|  | func opaqueInitHook(mi *MessageInfo) bool { | 
|  | mt := mi.GoReflectType.Elem() | 
|  | si := opaqueStructInfo{ | 
|  | structInfo: mi.makeStructInfo(mt), | 
|  | } | 
|  |  | 
|  | if !isOpaque(mt) { | 
|  | return false | 
|  | } | 
|  |  | 
|  | defer atomic.StoreUint32(&mi.initDone, 1) | 
|  |  | 
|  | mi.fields = map[protoreflect.FieldNumber]*fieldInfo{} | 
|  | fds := mi.Desc.Fields() | 
|  | for i := 0; i < fds.Len(); i++ { | 
|  | fd := fds.Get(i) | 
|  | fs := si.fieldsByNumber[fd.Number()] | 
|  | var fi fieldInfo | 
|  | usePresence, _ := filedesc.UsePresenceForField(fd) | 
|  |  | 
|  | switch { | 
|  | case fd.ContainingOneof() != nil && !fd.ContainingOneof().IsSynthetic(): | 
|  | // Oneofs are no different for opaque. | 
|  | fi = fieldInfoForOneof(fd, si.oneofsByName[fd.ContainingOneof().Name()], mi.Exporter, si.oneofWrappersByNumber[fd.Number()]) | 
|  | case fd.IsMap(): | 
|  | fi = mi.fieldInfoForMapOpaque(si, fd, fs) | 
|  | case fd.IsList() && fd.Message() == nil && usePresence: | 
|  | fi = mi.fieldInfoForScalarListOpaque(si, fd, fs) | 
|  | case fd.IsList() && fd.Message() == nil: | 
|  | // Proto3 lists without presence can use same access methods as open | 
|  | fi = fieldInfoForList(fd, fs, mi.Exporter) | 
|  | case fd.IsList() && usePresence: | 
|  | fi = mi.fieldInfoForMessageListOpaque(si, fd, fs) | 
|  | case fd.IsList(): | 
|  | // Proto3 opaque messages that does not need presence bitmap. | 
|  | // Different representation than open struct, but same logic | 
|  | fi = mi.fieldInfoForMessageListOpaqueNoPresence(si, fd, fs) | 
|  | case fd.Message() != nil && usePresence: | 
|  | fi = mi.fieldInfoForMessageOpaque(si, fd, fs) | 
|  | case fd.Message() != nil: | 
|  | // Proto3 messages without presence can use same access methods as open | 
|  | fi = fieldInfoForMessage(fd, fs, mi.Exporter) | 
|  | default: | 
|  | fi = mi.fieldInfoForScalarOpaque(si, fd, fs) | 
|  | } | 
|  | mi.fields[fd.Number()] = &fi | 
|  | } | 
|  | mi.oneofs = map[protoreflect.Name]*oneofInfo{} | 
|  | for i := 0; i < mi.Desc.Oneofs().Len(); i++ { | 
|  | od := mi.Desc.Oneofs().Get(i) | 
|  | mi.oneofs[od.Name()] = makeOneofInfoOpaque(mi, od, si.structInfo, mi.Exporter) | 
|  | } | 
|  |  | 
|  | mi.denseFields = make([]*fieldInfo, fds.Len()*2) | 
|  | for i := 0; i < fds.Len(); i++ { | 
|  | if fd := fds.Get(i); int(fd.Number()) < len(mi.denseFields) { | 
|  | mi.denseFields[fd.Number()] = mi.fields[fd.Number()] | 
|  | } | 
|  | } | 
|  |  | 
|  | for i := 0; i < fds.Len(); { | 
|  | fd := fds.Get(i) | 
|  | if od := fd.ContainingOneof(); od != nil && !fd.ContainingOneof().IsSynthetic() { | 
|  | mi.rangeInfos = append(mi.rangeInfos, mi.oneofs[od.Name()]) | 
|  | i += od.Fields().Len() | 
|  | } else { | 
|  | mi.rangeInfos = append(mi.rangeInfos, mi.fields[fd.Number()]) | 
|  | i++ | 
|  | } | 
|  | } | 
|  |  | 
|  | mi.makeExtensionFieldsFunc(mt, si.structInfo) | 
|  | mi.makeUnknownFieldsFunc(mt, si.structInfo) | 
|  | mi.makeOpaqueCoderMethods(mt, si) | 
|  | mi.makeFieldTypes(si.structInfo) | 
|  |  | 
|  | return true | 
|  | } | 
|  |  | 
|  | func makeOneofInfoOpaque(mi *MessageInfo, od protoreflect.OneofDescriptor, si structInfo, x exporter) *oneofInfo { | 
|  | oi := &oneofInfo{oneofDesc: od} | 
|  | if od.IsSynthetic() { | 
|  | fd := od.Fields().Get(0) | 
|  | index, _ := presenceIndex(mi.Desc, fd) | 
|  | oi.which = func(p pointer) protoreflect.FieldNumber { | 
|  | if p.IsNil() { | 
|  | return 0 | 
|  | } | 
|  | if !mi.present(p, index) { | 
|  | return 0 | 
|  | } | 
|  | return od.Fields().Get(0).Number() | 
|  | } | 
|  | return oi | 
|  | } | 
|  | // Dispatch to non-opaque oneof implementation for non-synthetic oneofs. | 
|  | return makeOneofInfo(od, si, x) | 
|  | } | 
|  |  | 
|  | func (mi *MessageInfo) fieldInfoForMapOpaque(si opaqueStructInfo, fd protoreflect.FieldDescriptor, fs reflect.StructField) fieldInfo { | 
|  | ft := fs.Type | 
|  | if ft.Kind() != reflect.Map { | 
|  | panic(fmt.Sprintf("invalid type: got %v, want map kind", ft)) | 
|  | } | 
|  | fieldOffset := offsetOf(fs) | 
|  | conv := NewConverter(ft, fd) | 
|  | return fieldInfo{ | 
|  | fieldDesc: fd, | 
|  | has: func(p pointer) bool { | 
|  | if p.IsNil() { | 
|  | return false | 
|  | } | 
|  | // Don't bother checking presence bits, since we need to | 
|  | // look at the map length even if the presence bit is set. | 
|  | rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem() | 
|  | return rv.Len() > 0 | 
|  | }, | 
|  | clear: func(p pointer) { | 
|  | rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem() | 
|  | rv.Set(reflect.Zero(rv.Type())) | 
|  | }, | 
|  | get: func(p pointer) protoreflect.Value { | 
|  | if p.IsNil() { | 
|  | return conv.Zero() | 
|  | } | 
|  | rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem() | 
|  | if rv.Len() == 0 { | 
|  | return conv.Zero() | 
|  | } | 
|  | return conv.PBValueOf(rv) | 
|  | }, | 
|  | set: func(p pointer, v protoreflect.Value) { | 
|  | pv := conv.GoValueOf(v) | 
|  | if pv.IsNil() { | 
|  | panic(fmt.Sprintf("invalid value: setting map field to read-only value")) | 
|  | } | 
|  | rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem() | 
|  | rv.Set(pv) | 
|  | }, | 
|  | mutable: func(p pointer) protoreflect.Value { | 
|  | v := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem() | 
|  | if v.IsNil() { | 
|  | v.Set(reflect.MakeMap(fs.Type)) | 
|  | } | 
|  | return conv.PBValueOf(v) | 
|  | }, | 
|  | newField: func() protoreflect.Value { | 
|  | return conv.New() | 
|  | }, | 
|  | } | 
|  | } | 
|  |  | 
|  | func (mi *MessageInfo) fieldInfoForScalarListOpaque(si opaqueStructInfo, fd protoreflect.FieldDescriptor, fs reflect.StructField) fieldInfo { | 
|  | ft := fs.Type | 
|  | if ft.Kind() != reflect.Slice { | 
|  | panic(fmt.Sprintf("invalid type: got %v, want slice kind", ft)) | 
|  | } | 
|  | conv := NewConverter(reflect.PtrTo(ft), fd) | 
|  | fieldOffset := offsetOf(fs) | 
|  | index, _ := presenceIndex(mi.Desc, fd) | 
|  | return fieldInfo{ | 
|  | fieldDesc: fd, | 
|  | has: func(p pointer) bool { | 
|  | if p.IsNil() { | 
|  | return false | 
|  | } | 
|  | rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem() | 
|  | return rv.Len() > 0 | 
|  | }, | 
|  | clear: func(p pointer) { | 
|  | rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem() | 
|  | rv.Set(reflect.Zero(rv.Type())) | 
|  | }, | 
|  | get: func(p pointer) protoreflect.Value { | 
|  | if p.IsNil() { | 
|  | return conv.Zero() | 
|  | } | 
|  | rv := p.Apply(fieldOffset).AsValueOf(fs.Type) | 
|  | if rv.Elem().Len() == 0 { | 
|  | return conv.Zero() | 
|  | } | 
|  | return conv.PBValueOf(rv) | 
|  | }, | 
|  | set: func(p pointer, v protoreflect.Value) { | 
|  | pv := conv.GoValueOf(v) | 
|  | if pv.IsNil() { | 
|  | panic(fmt.Sprintf("invalid value: setting repeated field to read-only value")) | 
|  | } | 
|  | mi.setPresent(p, index) | 
|  | rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem() | 
|  | rv.Set(pv.Elem()) | 
|  | }, | 
|  | mutable: func(p pointer) protoreflect.Value { | 
|  | mi.setPresent(p, index) | 
|  | return conv.PBValueOf(p.Apply(fieldOffset).AsValueOf(fs.Type)) | 
|  | }, | 
|  | newField: func() protoreflect.Value { | 
|  | return conv.New() | 
|  | }, | 
|  | } | 
|  | } | 
|  |  | 
|  | func (mi *MessageInfo) fieldInfoForMessageListOpaque(si opaqueStructInfo, fd protoreflect.FieldDescriptor, fs reflect.StructField) fieldInfo { | 
|  | ft := fs.Type | 
|  | if ft.Kind() != reflect.Ptr || ft.Elem().Kind() != reflect.Slice { | 
|  | panic(fmt.Sprintf("invalid type: got %v, want slice kind", ft)) | 
|  | } | 
|  | conv := NewConverter(ft, fd) | 
|  | fieldOffset := offsetOf(fs) | 
|  | index, _ := presenceIndex(mi.Desc, fd) | 
|  | fieldNumber := fd.Number() | 
|  | return fieldInfo{ | 
|  | fieldDesc: fd, | 
|  | has: func(p pointer) bool { | 
|  | if p.IsNil() { | 
|  | return false | 
|  | } | 
|  | if !mi.present(p, index) { | 
|  | return false | 
|  | } | 
|  | sp := p.Apply(fieldOffset).AtomicGetPointer() | 
|  | if sp.IsNil() { | 
|  | // Lazily unmarshal this field. | 
|  | mi.lazyUnmarshal(p, fieldNumber) | 
|  | sp = p.Apply(fieldOffset).AtomicGetPointer() | 
|  | } | 
|  | rv := sp.AsValueOf(fs.Type.Elem()) | 
|  | return rv.Elem().Len() > 0 | 
|  | }, | 
|  | clear: func(p pointer) { | 
|  | fp := p.Apply(fieldOffset) | 
|  | sp := fp.AtomicGetPointer() | 
|  | if sp.IsNil() { | 
|  | sp = fp.AtomicSetPointerIfNil(pointerOfValue(reflect.New(fs.Type.Elem()))) | 
|  | mi.setPresent(p, index) | 
|  | } | 
|  | rv := sp.AsValueOf(fs.Type.Elem()) | 
|  | rv.Elem().Set(reflect.Zero(rv.Type().Elem())) | 
|  | }, | 
|  | get: func(p pointer) protoreflect.Value { | 
|  | if p.IsNil() { | 
|  | return conv.Zero() | 
|  | } | 
|  | if !mi.present(p, index) { | 
|  | return conv.Zero() | 
|  | } | 
|  | sp := p.Apply(fieldOffset).AtomicGetPointer() | 
|  | if sp.IsNil() { | 
|  | // Lazily unmarshal this field. | 
|  | mi.lazyUnmarshal(p, fieldNumber) | 
|  | sp = p.Apply(fieldOffset).AtomicGetPointer() | 
|  | } | 
|  | rv := sp.AsValueOf(fs.Type.Elem()) | 
|  | if rv.Elem().Len() == 0 { | 
|  | return conv.Zero() | 
|  | } | 
|  | return conv.PBValueOf(rv) | 
|  | }, | 
|  | set: func(p pointer, v protoreflect.Value) { | 
|  | fp := p.Apply(fieldOffset) | 
|  | sp := fp.AtomicGetPointer() | 
|  | if sp.IsNil() { | 
|  | sp = fp.AtomicSetPointerIfNil(pointerOfValue(reflect.New(fs.Type.Elem()))) | 
|  | mi.setPresent(p, index) | 
|  | } | 
|  | rv := sp.AsValueOf(fs.Type.Elem()) | 
|  | val := conv.GoValueOf(v) | 
|  | if val.IsNil() { | 
|  | panic(fmt.Sprintf("invalid value: setting repeated field to read-only value")) | 
|  | } else { | 
|  | rv.Elem().Set(val.Elem()) | 
|  | } | 
|  | }, | 
|  | mutable: func(p pointer) protoreflect.Value { | 
|  | fp := p.Apply(fieldOffset) | 
|  | sp := fp.AtomicGetPointer() | 
|  | if sp.IsNil() { | 
|  | if mi.present(p, index) { | 
|  | // Lazily unmarshal this field. | 
|  | mi.lazyUnmarshal(p, fieldNumber) | 
|  | sp = p.Apply(fieldOffset).AtomicGetPointer() | 
|  | } else { | 
|  | sp = fp.AtomicSetPointerIfNil(pointerOfValue(reflect.New(fs.Type.Elem()))) | 
|  | mi.setPresent(p, index) | 
|  | } | 
|  | } | 
|  | rv := sp.AsValueOf(fs.Type.Elem()) | 
|  | return conv.PBValueOf(rv) | 
|  | }, | 
|  | newField: func() protoreflect.Value { | 
|  | return conv.New() | 
|  | }, | 
|  | } | 
|  | } | 
|  |  | 
|  | func (mi *MessageInfo) fieldInfoForMessageListOpaqueNoPresence(si opaqueStructInfo, fd protoreflect.FieldDescriptor, fs reflect.StructField) fieldInfo { | 
|  | ft := fs.Type | 
|  | if ft.Kind() != reflect.Ptr || ft.Elem().Kind() != reflect.Slice { | 
|  | panic(fmt.Sprintf("invalid type: got %v, want slice kind", ft)) | 
|  | } | 
|  | conv := NewConverter(ft, fd) | 
|  | fieldOffset := offsetOf(fs) | 
|  | return fieldInfo{ | 
|  | fieldDesc: fd, | 
|  | has: func(p pointer) bool { | 
|  | if p.IsNil() { | 
|  | return false | 
|  | } | 
|  | rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem() | 
|  | if rv.IsNil() { | 
|  | return false | 
|  | } | 
|  | return rv.Elem().Len() > 0 | 
|  | }, | 
|  | clear: func(p pointer) { | 
|  | rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem() | 
|  | if !rv.IsNil() { | 
|  | rv.Elem().Set(reflect.Zero(rv.Type().Elem())) | 
|  | } | 
|  | }, | 
|  | get: func(p pointer) protoreflect.Value { | 
|  | if p.IsNil() { | 
|  | return conv.Zero() | 
|  | } | 
|  | rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem() | 
|  | if rv.IsNil() { | 
|  | return conv.Zero() | 
|  | } | 
|  | if rv.Elem().Len() == 0 { | 
|  | return conv.Zero() | 
|  | } | 
|  | return conv.PBValueOf(rv) | 
|  | }, | 
|  | set: func(p pointer, v protoreflect.Value) { | 
|  | rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem() | 
|  | if rv.IsNil() { | 
|  | rv.Set(reflect.New(fs.Type.Elem())) | 
|  | } | 
|  | val := conv.GoValueOf(v) | 
|  | if val.IsNil() { | 
|  | panic(fmt.Sprintf("invalid value: setting repeated field to read-only value")) | 
|  | } else { | 
|  | rv.Elem().Set(val.Elem()) | 
|  | } | 
|  | }, | 
|  | mutable: func(p pointer) protoreflect.Value { | 
|  | rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem() | 
|  | if rv.IsNil() { | 
|  | rv.Set(reflect.New(fs.Type.Elem())) | 
|  | } | 
|  | return conv.PBValueOf(rv) | 
|  | }, | 
|  | newField: func() protoreflect.Value { | 
|  | return conv.New() | 
|  | }, | 
|  | } | 
|  | } | 
|  |  | 
|  | func (mi *MessageInfo) fieldInfoForScalarOpaque(si opaqueStructInfo, fd protoreflect.FieldDescriptor, fs reflect.StructField) fieldInfo { | 
|  | ft := fs.Type | 
|  | nullable := fd.HasPresence() | 
|  | if oneof := fd.ContainingOneof(); oneof != nil && oneof.IsSynthetic() { | 
|  | nullable = true | 
|  | } | 
|  | deref := false | 
|  | if nullable && ft.Kind() == reflect.Ptr { | 
|  | ft = ft.Elem() | 
|  | deref = true | 
|  | } | 
|  | conv := NewConverter(ft, fd) | 
|  | fieldOffset := offsetOf(fs) | 
|  | index, _ := presenceIndex(mi.Desc, fd) | 
|  | var getter func(p pointer) protoreflect.Value | 
|  | if !nullable { | 
|  | getter = getterForDirectScalar(fd, fs, conv, fieldOffset) | 
|  | } else { | 
|  | getter = getterForOpaqueNullableScalar(mi, index, fd, fs, conv, fieldOffset) | 
|  | } | 
|  | return fieldInfo{ | 
|  | fieldDesc: fd, | 
|  | has: func(p pointer) bool { | 
|  | if p.IsNil() { | 
|  | return false | 
|  | } | 
|  | if nullable { | 
|  | return mi.present(p, index) | 
|  | } | 
|  | rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem() | 
|  | switch rv.Kind() { | 
|  | case reflect.Bool: | 
|  | return rv.Bool() | 
|  | case reflect.Int32, reflect.Int64: | 
|  | return rv.Int() != 0 | 
|  | case reflect.Uint32, reflect.Uint64: | 
|  | return rv.Uint() != 0 | 
|  | case reflect.Float32, reflect.Float64: | 
|  | return rv.Float() != 0 || math.Signbit(rv.Float()) | 
|  | case reflect.String, reflect.Slice: | 
|  | return rv.Len() > 0 | 
|  | default: | 
|  | panic(fmt.Sprintf("invalid type: %v", rv.Type())) // should never happen | 
|  | } | 
|  | }, | 
|  | clear: func(p pointer) { | 
|  | if nullable { | 
|  | mi.clearPresent(p, index) | 
|  | } | 
|  | // This is only valuable for bytes and strings, but we do it unconditionally. | 
|  | rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem() | 
|  | rv.Set(reflect.Zero(rv.Type())) | 
|  | }, | 
|  | get: getter, | 
|  | // TODO: Implement unsafe fast path for set? | 
|  | set: func(p pointer, v protoreflect.Value) { | 
|  | rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem() | 
|  | if deref { | 
|  | if rv.IsNil() { | 
|  | rv.Set(reflect.New(ft)) | 
|  | } | 
|  | rv = rv.Elem() | 
|  | } | 
|  |  | 
|  | rv.Set(conv.GoValueOf(v)) | 
|  | if nullable && rv.Kind() == reflect.Slice && rv.IsNil() { | 
|  | rv.Set(emptyBytes) | 
|  | } | 
|  | if nullable { | 
|  | mi.setPresent(p, index) | 
|  | } | 
|  | }, | 
|  | newField: func() protoreflect.Value { | 
|  | return conv.New() | 
|  | }, | 
|  | } | 
|  | } | 
|  |  | 
|  | func (mi *MessageInfo) fieldInfoForMessageOpaque(si opaqueStructInfo, fd protoreflect.FieldDescriptor, fs reflect.StructField) fieldInfo { | 
|  | ft := fs.Type | 
|  | conv := NewConverter(ft, fd) | 
|  | fieldOffset := offsetOf(fs) | 
|  | index, _ := presenceIndex(mi.Desc, fd) | 
|  | fieldNumber := fd.Number() | 
|  | elemType := fs.Type.Elem() | 
|  | return fieldInfo{ | 
|  | fieldDesc: fd, | 
|  | has: func(p pointer) bool { | 
|  | if p.IsNil() { | 
|  | return false | 
|  | } | 
|  | return mi.present(p, index) | 
|  | }, | 
|  | clear: func(p pointer) { | 
|  | mi.clearPresent(p, index) | 
|  | p.Apply(fieldOffset).AtomicSetNilPointer() | 
|  | }, | 
|  | get: func(p pointer) protoreflect.Value { | 
|  | if p.IsNil() || !mi.present(p, index) { | 
|  | return conv.Zero() | 
|  | } | 
|  | fp := p.Apply(fieldOffset) | 
|  | mp := fp.AtomicGetPointer() | 
|  | if mp.IsNil() { | 
|  | // Lazily unmarshal this field. | 
|  | mi.lazyUnmarshal(p, fieldNumber) | 
|  | mp = fp.AtomicGetPointer() | 
|  | } | 
|  | rv := mp.AsValueOf(elemType) | 
|  | return conv.PBValueOf(rv) | 
|  | }, | 
|  | set: func(p pointer, v protoreflect.Value) { | 
|  | val := pointerOfValue(conv.GoValueOf(v)) | 
|  | if val.IsNil() { | 
|  | panic("invalid nil pointer") | 
|  | } | 
|  | p.Apply(fieldOffset).AtomicSetPointer(val) | 
|  | mi.setPresent(p, index) | 
|  | }, | 
|  | mutable: func(p pointer) protoreflect.Value { | 
|  | fp := p.Apply(fieldOffset) | 
|  | mp := fp.AtomicGetPointer() | 
|  | if mp.IsNil() { | 
|  | if mi.present(p, index) { | 
|  | // Lazily unmarshal this field. | 
|  | mi.lazyUnmarshal(p, fieldNumber) | 
|  | mp = fp.AtomicGetPointer() | 
|  | } else { | 
|  | mp = pointerOfValue(conv.GoValueOf(conv.New())) | 
|  | fp.AtomicSetPointer(mp) | 
|  | mi.setPresent(p, index) | 
|  | } | 
|  | } | 
|  | return conv.PBValueOf(mp.AsValueOf(fs.Type.Elem())) | 
|  | }, | 
|  | newMessage: func() protoreflect.Message { | 
|  | return conv.New().Message() | 
|  | }, | 
|  | newField: func() protoreflect.Value { | 
|  | return conv.New() | 
|  | }, | 
|  | } | 
|  | } | 
|  |  | 
|  | // A presenceList wraps a List, updating presence bits as necessary when the | 
|  | // list contents change. | 
|  | type presenceList struct { | 
|  | pvalueList | 
|  | setPresence func(bool) | 
|  | } | 
|  | type pvalueList interface { | 
|  | protoreflect.List | 
|  | //Unwrapper | 
|  | } | 
|  |  | 
|  | func (list presenceList) Append(v protoreflect.Value) { | 
|  | list.pvalueList.Append(v) | 
|  | list.setPresence(true) | 
|  | } | 
|  | func (list presenceList) Truncate(i int) { | 
|  | list.pvalueList.Truncate(i) | 
|  | list.setPresence(i > 0) | 
|  | } | 
|  |  | 
|  | // presenceIndex returns the index to pass to presence functions. | 
|  | // | 
|  | // TODO: field.Desc.Index() would be simpler, and would give space to record the presence of oneof fields. | 
|  | func presenceIndex(md protoreflect.MessageDescriptor, fd protoreflect.FieldDescriptor) (uint32, presenceSize) { | 
|  | found := false | 
|  | var index, numIndices uint32 | 
|  | for i := 0; i < md.Fields().Len(); i++ { | 
|  | f := md.Fields().Get(i) | 
|  | if f == fd { | 
|  | found = true | 
|  | index = numIndices | 
|  | } | 
|  | if f.ContainingOneof() == nil || isLastOneofField(f) { | 
|  | numIndices++ | 
|  | } | 
|  | } | 
|  | if !found { | 
|  | panic(fmt.Sprintf("BUG: %v not in %v", fd.Name(), md.FullName())) | 
|  | } | 
|  | return index, presenceSize(numIndices) | 
|  | } | 
|  |  | 
|  | func isLastOneofField(fd protoreflect.FieldDescriptor) bool { | 
|  | fields := fd.ContainingOneof().Fields() | 
|  | return fields.Get(fields.Len()-1) == fd | 
|  | } | 
|  |  | 
|  | func (mi *MessageInfo) setPresent(p pointer, index uint32) { | 
|  | p.Apply(mi.presenceOffset).PresenceInfo().SetPresent(index, mi.presenceSize) | 
|  | } | 
|  |  | 
|  | func (mi *MessageInfo) clearPresent(p pointer, index uint32) { | 
|  | p.Apply(mi.presenceOffset).PresenceInfo().ClearPresent(index) | 
|  | } | 
|  |  | 
|  | func (mi *MessageInfo) present(p pointer, index uint32) bool { | 
|  | return p.Apply(mi.presenceOffset).PresenceInfo().Present(index) | 
|  | } |