blob: 8aacd25d35f9ecc14c435cc196550f0f8af98a10 [file] [log] [blame]
Joe Tsaifa02f4e2018-09-12 16:20:37 -07001// Copyright 2018 The Go Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style
3// license that can be found in the LICENSE file.
4
5package impl
6
7import (
8 "fmt"
Joe Tsai060cdac2019-04-22 11:44:49 -07009 "math"
Joe Tsaifa02f4e2018-09-12 16:20:37 -070010 "reflect"
11
Damien Neilc37adef2019-04-01 13:49:56 -070012 "google.golang.org/protobuf/internal/encoding/wire"
Damien Neile89e6242019-05-13 23:55:40 -070013 pvalue "google.golang.org/protobuf/internal/value"
14 pref "google.golang.org/protobuf/reflect/protoreflect"
Joe Tsai21ade492019-05-22 13:42:54 -040015 piface "google.golang.org/protobuf/runtime/protoiface"
Joe Tsaifa02f4e2018-09-12 16:20:37 -070016)
17
18type fieldInfo struct {
Joe Tsai378c1322019-04-25 23:48:08 -070019 fieldDesc pref.FieldDescriptor
20
Damien Neilc37adef2019-04-01 13:49:56 -070021 // These fields are used for protobuf reflection support.
Damien Neil97e7f572018-12-07 14:28:33 -080022 has func(pointer) bool
Joe Tsai378c1322019-04-25 23:48:08 -070023 clear func(pointer)
Damien Neil97e7f572018-12-07 14:28:33 -080024 get func(pointer) pref.Value
25 set func(pointer, pref.Value)
Joe Tsai378c1322019-04-25 23:48:08 -070026 mutable func(pointer) pref.Value
Damien Neil97e7f572018-12-07 14:28:33 -080027 newMessage func() pref.Message
Damien Neilc37adef2019-04-01 13:49:56 -070028
29 // These fields are used for fast-path functions.
Damien Neil5322bdb2019-04-09 15:57:05 -070030 funcs pointerCoderFuncs // fast-path per-field functions
31 num pref.FieldNumber // field number
32 offset offset // struct field offset
33 wiretag uint64 // field tag (number + wire type)
34 tagsize int // size of the varint-encoded tag
35 isPointer bool // true if IsNil may be called on the struct field
36 isRequired bool // true if field is required
Joe Tsaifa02f4e2018-09-12 16:20:37 -070037}
38
Joe Tsaifa02f4e2018-09-12 16:20:37 -070039func fieldInfoForOneof(fd pref.FieldDescriptor, fs reflect.StructField, ot reflect.Type) fieldInfo {
Joe Tsai2c870bb2018-10-17 11:46:52 -070040 ft := fs.Type
41 if ft.Kind() != reflect.Interface {
42 panic(fmt.Sprintf("invalid type: got %v, want interface kind", ft))
43 }
44 if ot.Kind() != reflect.Struct {
45 panic(fmt.Sprintf("invalid type: got %v, want struct kind", ot))
46 }
47 if !reflect.PtrTo(ot).Implements(ft) {
48 panic(fmt.Sprintf("invalid type: %v does not implement %v", ot, ft))
49 }
Joe Tsai21ade492019-05-22 13:42:54 -040050 conv, _ := newConverter(ot.Field(0).Type, fd.Kind())
Joe Tsai378c1322019-04-25 23:48:08 -070051 var frozenEmpty pref.Value
52 if conv.NewMessage != nil {
53 frozenEmpty = pref.ValueOf(frozenMessage{conv.NewMessage()})
54 }
55
Joe Tsai2c870bb2018-10-17 11:46:52 -070056 // TODO: Implement unsafe fast path?
Joe Tsai378c1322019-04-25 23:48:08 -070057 fieldOffset := offsetOf(fs)
Joe Tsai2c870bb2018-10-17 11:46:52 -070058 return fieldInfo{
59 // NOTE: The logic below intentionally assumes that oneof fields are
60 // well-formatted. That is, the oneof interface never contains a
61 // typed nil pointer to one of the wrapper structs.
62
Joe Tsai378c1322019-04-25 23:48:08 -070063 fieldDesc: fd,
Joe Tsai2c870bb2018-10-17 11:46:52 -070064 has: func(p pointer) bool {
Joe Tsai6cf80c42018-12-01 04:57:09 -080065 if p.IsNil() {
66 return false
67 }
68 rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem()
Joe Tsai2c870bb2018-10-17 11:46:52 -070069 if rv.IsNil() || rv.Elem().Type().Elem() != ot {
70 return false
71 }
72 return true
73 },
Joe Tsai378c1322019-04-25 23:48:08 -070074 clear: func(p pointer) {
75 rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem()
76 if rv.IsNil() || rv.Elem().Type().Elem() != ot {
77 return
78 }
79 rv.Set(reflect.Zero(rv.Type()))
80 },
Joe Tsai2c870bb2018-10-17 11:46:52 -070081 get: func(p pointer) pref.Value {
Joe Tsai6cf80c42018-12-01 04:57:09 -080082 if p.IsNil() {
Joe Tsai378c1322019-04-25 23:48:08 -070083 if frozenEmpty.IsValid() {
84 return frozenEmpty
85 }
Joe Tsai6cf80c42018-12-01 04:57:09 -080086 return defaultValueOf(fd)
87 }
88 rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem()
Joe Tsai2c870bb2018-10-17 11:46:52 -070089 if rv.IsNil() || rv.Elem().Type().Elem() != ot {
Joe Tsai378c1322019-04-25 23:48:08 -070090 if frozenEmpty.IsValid() {
91 return frozenEmpty
92 }
Joe Tsai6cf80c42018-12-01 04:57:09 -080093 return defaultValueOf(fd)
Joe Tsai2c870bb2018-10-17 11:46:52 -070094 }
95 rv = rv.Elem().Elem().Field(0)
Joe Tsai88bc5a72018-11-05 11:42:22 -080096 return conv.PBValueOf(rv)
Joe Tsai2c870bb2018-10-17 11:46:52 -070097 },
98 set: func(p pointer, v pref.Value) {
Joe Tsai6cf80c42018-12-01 04:57:09 -080099 rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem()
Joe Tsai2c870bb2018-10-17 11:46:52 -0700100 if rv.IsNil() || rv.Elem().Type().Elem() != ot {
101 rv.Set(reflect.New(ot))
102 }
103 rv = rv.Elem().Elem().Field(0)
Joe Tsai88bc5a72018-11-05 11:42:22 -0800104 rv.Set(conv.GoValueOf(v))
Joe Tsai2c870bb2018-10-17 11:46:52 -0700105 },
Joe Tsai378c1322019-04-25 23:48:08 -0700106 mutable: func(p pointer) pref.Value {
107 if conv.NewMessage == nil {
108 panic("invalid Mutable on field with non-composite type")
109 }
Joe Tsai6cf80c42018-12-01 04:57:09 -0800110 rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem()
Joe Tsai2c870bb2018-10-17 11:46:52 -0700111 if rv.IsNil() || rv.Elem().Type().Elem() != ot {
Joe Tsai378c1322019-04-25 23:48:08 -0700112 rv.Set(reflect.New(ot))
Joe Tsai2c870bb2018-10-17 11:46:52 -0700113 }
Joe Tsai378c1322019-04-25 23:48:08 -0700114 rv = rv.Elem().Elem().Field(0)
115 if rv.IsNil() {
116 rv.Set(conv.GoValueOf(pref.ValueOf(conv.NewMessage())))
117 }
118 return conv.PBValueOf(rv)
Joe Tsai2c870bb2018-10-17 11:46:52 -0700119 },
Joe Tsaib2f66be2019-05-22 00:42:45 -0400120 newMessage: conv.NewMessage,
121 offset: fieldOffset,
122 isPointer: true,
Joe Tsai2c870bb2018-10-17 11:46:52 -0700123 }
Joe Tsaifa02f4e2018-09-12 16:20:37 -0700124}
125
126func fieldInfoForMap(fd pref.FieldDescriptor, fs reflect.StructField) fieldInfo {
Joe Tsaibbfaeb72018-10-17 00:27:21 +0000127 ft := fs.Type
128 if ft.Kind() != reflect.Map {
129 panic(fmt.Sprintf("invalid type: got %v, want map kind", ft))
130 }
Joe Tsai21ade492019-05-22 13:42:54 -0400131 keyConv, _ := newConverter(ft.Key(), fd.MapKey().Kind())
132 valConv, _ := newConverter(ft.Elem(), fd.MapValue().Kind())
Damien Neilc37adef2019-04-01 13:49:56 -0700133 wiretag := wire.EncodeTag(fd.Number(), wireTypes[fd.Kind()])
Joe Tsai378c1322019-04-25 23:48:08 -0700134 frozenEmpty := pref.ValueOf(frozenMap{
135 pvalue.MapOf(reflect.Zero(reflect.PtrTo(fs.Type)).Interface(), keyConv, valConv),
136 })
137
Joe Tsaibbfaeb72018-10-17 00:27:21 +0000138 // TODO: Implement unsafe fast path?
Joe Tsai378c1322019-04-25 23:48:08 -0700139 fieldOffset := offsetOf(fs)
Joe Tsaibbfaeb72018-10-17 00:27:21 +0000140 return fieldInfo{
Joe Tsai378c1322019-04-25 23:48:08 -0700141 fieldDesc: fd,
Joe Tsaibbfaeb72018-10-17 00:27:21 +0000142 has: func(p pointer) bool {
Joe Tsai6cf80c42018-12-01 04:57:09 -0800143 if p.IsNil() {
144 return false
145 }
146 rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem()
Joe Tsaibbfaeb72018-10-17 00:27:21 +0000147 return rv.Len() > 0
148 },
Joe Tsai378c1322019-04-25 23:48:08 -0700149 clear: func(p pointer) {
150 rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem()
151 rv.Set(reflect.Zero(rv.Type()))
152 },
Joe Tsaibbfaeb72018-10-17 00:27:21 +0000153 get: func(p pointer) pref.Value {
Joe Tsai6cf80c42018-12-01 04:57:09 -0800154 if p.IsNil() {
Joe Tsai378c1322019-04-25 23:48:08 -0700155 return frozenEmpty
Joe Tsai6cf80c42018-12-01 04:57:09 -0800156 }
Joe Tsai378c1322019-04-25 23:48:08 -0700157 rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem()
158 if rv.IsNil() {
159 return frozenEmpty
160 }
161 return pref.ValueOf(pvalue.MapOf(rv.Addr().Interface(), keyConv, valConv))
Joe Tsaibbfaeb72018-10-17 00:27:21 +0000162 },
163 set: func(p pointer, v pref.Value) {
Joe Tsai6cf80c42018-12-01 04:57:09 -0800164 rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem()
Joe Tsaiba0ef9a2018-11-29 14:54:05 -0800165 rv.Set(reflect.ValueOf(v.Map().(pvalue.Unwrapper).ProtoUnwrap()).Elem())
Joe Tsaibbfaeb72018-10-17 00:27:21 +0000166 },
Joe Tsai378c1322019-04-25 23:48:08 -0700167 mutable: func(p pointer) pref.Value {
168 v := p.Apply(fieldOffset).AsIfaceOf(fs.Type)
169 return pref.ValueOf(pvalue.MapOf(v, keyConv, valConv))
Joe Tsaibbfaeb72018-10-17 00:27:21 +0000170 },
Damien Neilc37adef2019-04-01 13:49:56 -0700171 funcs: encoderFuncsForMap(fd, ft),
172 offset: fieldOffset,
173 wiretag: wiretag,
174 tagsize: wire.SizeVarint(wiretag),
175 isPointer: true,
Joe Tsaibbfaeb72018-10-17 00:27:21 +0000176 }
Joe Tsaifa02f4e2018-09-12 16:20:37 -0700177}
178
Joe Tsai4b7aff62018-11-14 14:05:19 -0800179func fieldInfoForList(fd pref.FieldDescriptor, fs reflect.StructField) fieldInfo {
Joe Tsai91e14662018-09-13 13:24:35 -0700180 ft := fs.Type
181 if ft.Kind() != reflect.Slice {
182 panic(fmt.Sprintf("invalid type: got %v, want slice kind", ft))
183 }
Joe Tsai21ade492019-05-22 13:42:54 -0400184 conv, _ := newConverter(ft.Elem(), fd.Kind())
Damien Neilc37adef2019-04-01 13:49:56 -0700185 var wiretag uint64
186 if !fd.IsPacked() {
187 wiretag = wire.EncodeTag(fd.Number(), wireTypes[fd.Kind()])
188 } else {
189 wiretag = wire.EncodeTag(fd.Number(), wire.BytesType)
190 }
Joe Tsai378c1322019-04-25 23:48:08 -0700191 frozenEmpty := pref.ValueOf(frozenList{
192 pvalue.ListOf(reflect.Zero(reflect.PtrTo(fs.Type)).Interface(), conv),
193 })
194
Joe Tsai91e14662018-09-13 13:24:35 -0700195 // TODO: Implement unsafe fast path?
Joe Tsai378c1322019-04-25 23:48:08 -0700196 fieldOffset := offsetOf(fs)
Joe Tsai91e14662018-09-13 13:24:35 -0700197 return fieldInfo{
Joe Tsai378c1322019-04-25 23:48:08 -0700198 fieldDesc: fd,
Joe Tsai91e14662018-09-13 13:24:35 -0700199 has: func(p pointer) bool {
Joe Tsai6cf80c42018-12-01 04:57:09 -0800200 if p.IsNil() {
201 return false
202 }
203 rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem()
Joe Tsai91e14662018-09-13 13:24:35 -0700204 return rv.Len() > 0
205 },
Joe Tsai378c1322019-04-25 23:48:08 -0700206 clear: func(p pointer) {
207 rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem()
208 rv.Set(reflect.Zero(rv.Type()))
209 },
Joe Tsai91e14662018-09-13 13:24:35 -0700210 get: func(p pointer) pref.Value {
Joe Tsai6cf80c42018-12-01 04:57:09 -0800211 if p.IsNil() {
Joe Tsai378c1322019-04-25 23:48:08 -0700212 return frozenEmpty
Joe Tsai6cf80c42018-12-01 04:57:09 -0800213 }
Joe Tsai378c1322019-04-25 23:48:08 -0700214 rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem()
215 if rv.Len() == 0 {
216 return frozenEmpty
217 }
218 return pref.ValueOf(pvalue.ListOf(rv.Addr().Interface(), conv))
Joe Tsai91e14662018-09-13 13:24:35 -0700219 },
220 set: func(p pointer, v pref.Value) {
Joe Tsai6cf80c42018-12-01 04:57:09 -0800221 rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem()
Joe Tsaiba0ef9a2018-11-29 14:54:05 -0800222 rv.Set(reflect.ValueOf(v.List().(pvalue.Unwrapper).ProtoUnwrap()).Elem())
Joe Tsai91e14662018-09-13 13:24:35 -0700223 },
Joe Tsai378c1322019-04-25 23:48:08 -0700224 mutable: func(p pointer) pref.Value {
225 v := p.Apply(fieldOffset).AsIfaceOf(fs.Type)
226 return pref.ValueOf(pvalue.ListOf(v, conv))
Joe Tsai91e14662018-09-13 13:24:35 -0700227 },
Damien Neilc37adef2019-04-01 13:49:56 -0700228 funcs: fieldCoder(fd, ft),
229 offset: fieldOffset,
230 wiretag: wiretag,
231 tagsize: wire.SizeVarint(wiretag),
232 isPointer: true,
Joe Tsai91e14662018-09-13 13:24:35 -0700233 }
Joe Tsaifa02f4e2018-09-12 16:20:37 -0700234}
235
236var emptyBytes = reflect.ValueOf([]byte{})
237
238func fieldInfoForScalar(fd pref.FieldDescriptor, fs reflect.StructField) fieldInfo {
239 ft := fs.Type
Damien Neilc37adef2019-04-01 13:49:56 -0700240 funcs := fieldCoder(fd, ft)
Joe Tsaifa02f4e2018-09-12 16:20:37 -0700241 nullable := fd.Syntax() == pref.Proto2
242 if nullable {
243 if ft.Kind() != reflect.Ptr && ft.Kind() != reflect.Slice {
244 panic(fmt.Sprintf("invalid type: got %v, want pointer", ft))
245 }
246 if ft.Kind() == reflect.Ptr {
247 ft = ft.Elem()
248 }
249 }
Joe Tsai21ade492019-05-22 13:42:54 -0400250 conv, _ := newConverter(ft, fd.Kind())
Damien Neilc37adef2019-04-01 13:49:56 -0700251 wiretag := wire.EncodeTag(fd.Number(), wireTypes[fd.Kind()])
Joe Tsai378c1322019-04-25 23:48:08 -0700252
Joe Tsaifa02f4e2018-09-12 16:20:37 -0700253 // TODO: Implement unsafe fast path?
Joe Tsai378c1322019-04-25 23:48:08 -0700254 fieldOffset := offsetOf(fs)
Joe Tsaifa02f4e2018-09-12 16:20:37 -0700255 return fieldInfo{
Joe Tsai378c1322019-04-25 23:48:08 -0700256 fieldDesc: fd,
Joe Tsaifa02f4e2018-09-12 16:20:37 -0700257 has: func(p pointer) bool {
Joe Tsai6cf80c42018-12-01 04:57:09 -0800258 if p.IsNil() {
259 return false
260 }
261 rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem()
Joe Tsaifa02f4e2018-09-12 16:20:37 -0700262 if nullable {
263 return !rv.IsNil()
264 }
265 switch rv.Kind() {
266 case reflect.Bool:
267 return rv.Bool()
268 case reflect.Int32, reflect.Int64:
Joe Tsai44e389c2018-11-19 15:27:09 -0800269 return rv.Int() != 0
Joe Tsaifa02f4e2018-09-12 16:20:37 -0700270 case reflect.Uint32, reflect.Uint64:
Joe Tsai44e389c2018-11-19 15:27:09 -0800271 return rv.Uint() != 0
Joe Tsaifa02f4e2018-09-12 16:20:37 -0700272 case reflect.Float32, reflect.Float64:
Joe Tsai060cdac2019-04-22 11:44:49 -0700273 return rv.Float() != 0 || math.Signbit(rv.Float())
Joe Tsaifa02f4e2018-09-12 16:20:37 -0700274 case reflect.String, reflect.Slice:
275 return rv.Len() > 0
276 default:
277 panic(fmt.Sprintf("invalid type: %v", rv.Type())) // should never happen
278 }
279 },
Joe Tsai378c1322019-04-25 23:48:08 -0700280 clear: func(p pointer) {
281 rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem()
282 rv.Set(reflect.Zero(rv.Type()))
283 },
Joe Tsaifa02f4e2018-09-12 16:20:37 -0700284 get: func(p pointer) pref.Value {
Joe Tsai6cf80c42018-12-01 04:57:09 -0800285 if p.IsNil() {
286 return defaultValueOf(fd)
287 }
288 rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem()
Joe Tsaifa02f4e2018-09-12 16:20:37 -0700289 if nullable {
290 if rv.IsNil() {
Joe Tsai6cf80c42018-12-01 04:57:09 -0800291 return defaultValueOf(fd)
Joe Tsaifa02f4e2018-09-12 16:20:37 -0700292 }
293 if rv.Kind() == reflect.Ptr {
294 rv = rv.Elem()
295 }
296 }
Joe Tsai88bc5a72018-11-05 11:42:22 -0800297 return conv.PBValueOf(rv)
Joe Tsaifa02f4e2018-09-12 16:20:37 -0700298 },
299 set: func(p pointer, v pref.Value) {
Joe Tsai6cf80c42018-12-01 04:57:09 -0800300 rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem()
Joe Tsaifa02f4e2018-09-12 16:20:37 -0700301 if nullable && rv.Kind() == reflect.Ptr {
302 if rv.IsNil() {
303 rv.Set(reflect.New(ft))
304 }
305 rv = rv.Elem()
306 }
Joe Tsai88bc5a72018-11-05 11:42:22 -0800307 rv.Set(conv.GoValueOf(v))
Joe Tsaifa02f4e2018-09-12 16:20:37 -0700308 if nullable && rv.Kind() == reflect.Slice && rv.IsNil() {
309 rv.Set(emptyBytes)
310 }
311 },
Damien Neil5322bdb2019-04-09 15:57:05 -0700312 funcs: funcs,
313 offset: fieldOffset,
314 isPointer: nullable,
315 isRequired: fd.Cardinality() == pref.Required,
316 wiretag: wiretag,
317 tagsize: wire.SizeVarint(wiretag),
Joe Tsaifa02f4e2018-09-12 16:20:37 -0700318 }
319}
320
321func fieldInfoForMessage(fd pref.FieldDescriptor, fs reflect.StructField) fieldInfo {
Joe Tsaice6edd32018-10-19 16:27:46 -0700322 ft := fs.Type
Joe Tsai21ade492019-05-22 13:42:54 -0400323 conv, _ := newConverter(ft, fd.Kind())
Damien Neilc37adef2019-04-01 13:49:56 -0700324 wiretag := wire.EncodeTag(fd.Number(), wireTypes[fd.Kind()])
Joe Tsai378c1322019-04-25 23:48:08 -0700325 frozenEmpty := pref.ValueOf(frozenMessage{conv.NewMessage()})
326
327 // TODO: Implement unsafe fast path?
328 fieldOffset := offsetOf(fs)
Joe Tsaice6edd32018-10-19 16:27:46 -0700329 return fieldInfo{
Joe Tsai378c1322019-04-25 23:48:08 -0700330 fieldDesc: fd,
Joe Tsaice6edd32018-10-19 16:27:46 -0700331 has: func(p pointer) bool {
Joe Tsai6cf80c42018-12-01 04:57:09 -0800332 if p.IsNil() {
333 return false
334 }
335 rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem()
Joe Tsaice6edd32018-10-19 16:27:46 -0700336 return !rv.IsNil()
337 },
Joe Tsai378c1322019-04-25 23:48:08 -0700338 clear: func(p pointer) {
339 rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem()
340 rv.Set(reflect.Zero(rv.Type()))
341 },
Joe Tsaice6edd32018-10-19 16:27:46 -0700342 get: func(p pointer) pref.Value {
Joe Tsai6cf80c42018-12-01 04:57:09 -0800343 if p.IsNil() {
Joe Tsai378c1322019-04-25 23:48:08 -0700344 return frozenEmpty
Joe Tsai6cf80c42018-12-01 04:57:09 -0800345 }
346 rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem()
Joe Tsaif6d4a422018-11-19 14:26:06 -0800347 if rv.IsNil() {
Joe Tsai378c1322019-04-25 23:48:08 -0700348 return frozenEmpty
Joe Tsaif6d4a422018-11-19 14:26:06 -0800349 }
Joe Tsai88bc5a72018-11-05 11:42:22 -0800350 return conv.PBValueOf(rv)
Joe Tsaice6edd32018-10-19 16:27:46 -0700351 },
352 set: func(p pointer, v pref.Value) {
Joe Tsai6cf80c42018-12-01 04:57:09 -0800353 rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem()
Joe Tsai88bc5a72018-11-05 11:42:22 -0800354 rv.Set(conv.GoValueOf(v))
Joe Tsaif6d4a422018-11-19 14:26:06 -0800355 if rv.IsNil() {
356 panic("invalid nil pointer")
357 }
Joe Tsaice6edd32018-10-19 16:27:46 -0700358 },
Joe Tsai378c1322019-04-25 23:48:08 -0700359 mutable: func(p pointer) pref.Value {
Joe Tsai6cf80c42018-12-01 04:57:09 -0800360 rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem()
Joe Tsai378c1322019-04-25 23:48:08 -0700361 if rv.IsNil() {
362 rv.Set(conv.GoValueOf(pref.ValueOf(conv.NewMessage())))
363 }
364 return conv.PBValueOf(rv)
Joe Tsaice6edd32018-10-19 16:27:46 -0700365 },
Joe Tsaib2f66be2019-05-22 00:42:45 -0400366 newMessage: conv.NewMessage,
367 funcs: fieldCoder(fd, ft),
368 offset: fieldOffset,
369 isPointer: true,
Damien Neil5322bdb2019-04-09 15:57:05 -0700370 isRequired: fd.Cardinality() == pref.Required,
Joe Tsaib2f66be2019-05-22 00:42:45 -0400371 wiretag: wiretag,
372 tagsize: wire.SizeVarint(wiretag),
Joe Tsaifa02f4e2018-09-12 16:20:37 -0700373 }
374}
Joe Tsai6cf80c42018-12-01 04:57:09 -0800375
Joe Tsai4ec39c72019-04-03 13:40:53 -0700376type oneofInfo struct {
Joe Tsai378c1322019-04-25 23:48:08 -0700377 oneofDesc pref.OneofDescriptor
378 which func(pointer) pref.FieldNumber
Joe Tsai4ec39c72019-04-03 13:40:53 -0700379}
380
381func makeOneofInfo(od pref.OneofDescriptor, fs reflect.StructField, wrappersByType map[reflect.Type]pref.FieldNumber) *oneofInfo {
382 fieldOffset := offsetOf(fs)
383 return &oneofInfo{
Joe Tsai378c1322019-04-25 23:48:08 -0700384 oneofDesc: od,
Joe Tsai4ec39c72019-04-03 13:40:53 -0700385 which: func(p pointer) pref.FieldNumber {
386 if p.IsNil() {
387 return 0
388 }
389 rv := p.Apply(fieldOffset).AsValueOf(fs.Type).Elem()
390 if rv.IsNil() {
391 return 0
392 }
393 return wrappersByType[rv.Elem().Type().Elem()]
394 },
395 }
396}
Joe Tsaib2f66be2019-05-22 00:42:45 -0400397
Joe Tsai21ade492019-05-22 13:42:54 -0400398var (
399 enumIfaceV2 = reflect.TypeOf((*pref.Enum)(nil)).Elem()
400 messageIfaceV1 = reflect.TypeOf((*piface.MessageV1)(nil)).Elem()
401 messageIfaceV2 = reflect.TypeOf((*pref.ProtoMessage)(nil)).Elem()
402)
403
404func newConverter(t reflect.Type, k pref.Kind) (conv pvalue.Converter, isLegacy bool) {
405 switch k {
406 case pref.EnumKind:
407 if t.Kind() == reflect.Int32 && !t.Implements(enumIfaceV2) {
408 return pvalue.Converter{
409 PBValueOf: func(v reflect.Value) pref.Value {
410 if v.Type() != t {
411 panic(fmt.Sprintf("invalid type: got %v, want %v", v.Type(), t))
412 }
413 return pref.ValueOf(pref.EnumNumber(v.Int()))
414 },
415 GoValueOf: func(v pref.Value) reflect.Value {
416 return reflect.ValueOf(v.Enum()).Convert(t)
417 },
418 NewEnum: func(n pref.EnumNumber) pref.Enum {
419 return legacyWrapEnum(reflect.ValueOf(n).Convert(t))
420 },
421 }, true
422 }
423 case pref.MessageKind, pref.GroupKind:
424 if t.Kind() == reflect.Ptr && t.Implements(messageIfaceV1) && !t.Implements(messageIfaceV2) {
425 return pvalue.Converter{
426 PBValueOf: func(v reflect.Value) pref.Value {
427 if v.Type() != t {
428 panic(fmt.Sprintf("invalid type: got %v, want %v", v.Type(), t))
429 }
430 return pref.ValueOf(Export{}.MessageOf(v.Interface()))
431 },
432 GoValueOf: func(v pref.Value) reflect.Value {
433 rv := reflect.ValueOf(v.Message().(pvalue.Unwrapper).ProtoUnwrap())
434 if rv.Type() != t {
435 panic(fmt.Sprintf("invalid type: got %v, want %v", rv.Type(), t))
436 }
437 return rv
438 },
439 NewMessage: func() pref.Message {
440 return legacyWrapMessage(reflect.New(t.Elem())).ProtoReflect()
441 },
442 }, true
443 }
Joe Tsaib2f66be2019-05-22 00:42:45 -0400444 }
Joe Tsai21ade492019-05-22 13:42:54 -0400445 return pvalue.NewConverter(t, k), false
Joe Tsaib2f66be2019-05-22 00:42:45 -0400446}
Joe Tsai378c1322019-04-25 23:48:08 -0700447
448// defaultValueOf returns the default value for the field.
449func defaultValueOf(fd pref.FieldDescriptor) pref.Value {
450 if fd == nil {
451 return pref.Value{}
452 }
453 pv := fd.Default() // invalid Value for messages and repeated fields
454 if fd.Kind() == pref.BytesKind && pv.IsValid() && len(pv.Bytes()) > 0 {
455 return pref.ValueOf(append([]byte(nil), pv.Bytes()...)) // copy default bytes for safety
456 }
457 return pv
458}
459
460// frozenValueOf returns a frozen version of any composite value.
461func frozenValueOf(v pref.Value) pref.Value {
462 switch v := v.Interface().(type) {
463 case pref.Message:
464 if _, ok := v.(frozenMessage); !ok {
465 return pref.ValueOf(frozenMessage{v})
466 }
467 case pref.List:
468 if _, ok := v.(frozenList); !ok {
469 return pref.ValueOf(frozenList{v})
470 }
471 case pref.Map:
472 if _, ok := v.(frozenMap); !ok {
473 return pref.ValueOf(frozenMap{v})
474 }
475 }
476 return v
477}
478
479type frozenMessage struct{ pref.Message }
480
481func (m frozenMessage) ProtoReflect() pref.Message { return m }
482func (m frozenMessage) Interface() pref.ProtoMessage { return m }
483func (m frozenMessage) Range(f func(pref.FieldDescriptor, pref.Value) bool) {
484 m.Message.Range(func(fd pref.FieldDescriptor, v pref.Value) bool {
485 return f(fd, frozenValueOf(v))
486 })
487}
488func (m frozenMessage) Get(fd pref.FieldDescriptor) pref.Value {
489 v := m.Message.Get(fd)
490 return frozenValueOf(v)
491}
492func (frozenMessage) Clear(pref.FieldDescriptor) { panic("invalid on read-only Message") }
493func (frozenMessage) Set(pref.FieldDescriptor, pref.Value) { panic("invalid on read-only Message") }
494func (frozenMessage) Mutable(pref.FieldDescriptor) pref.Value { panic("invalid on read-only Message") }
495func (frozenMessage) SetUnknown(pref.RawFields) { panic("invalid on read-only Message") }
496
497type frozenList struct{ pref.List }
498
499func (ls frozenList) Get(i int) pref.Value {
500 v := ls.List.Get(i)
501 return frozenValueOf(v)
502}
503func (frozenList) Set(i int, v pref.Value) { panic("invalid on read-only List") }
504func (frozenList) Append(v pref.Value) { panic("invalid on read-only List") }
505func (frozenList) Truncate(i int) { panic("invalid on read-only List") }
506
507type frozenMap struct{ pref.Map }
508
509func (ms frozenMap) Get(k pref.MapKey) pref.Value {
510 v := ms.Map.Get(k)
511 return frozenValueOf(v)
512}
513func (ms frozenMap) Range(f func(pref.MapKey, pref.Value) bool) {
514 ms.Map.Range(func(k pref.MapKey, v pref.Value) bool {
515 return f(k, frozenValueOf(v))
516 })
517}
518func (frozenMap) Set(k pref.MapKey, v pref.Value) { panic("invalid n read-only Map") }
519func (frozenMap) Clear(k pref.MapKey) { panic("invalid on read-only Map") }