| 
									
										
										
										
											2021-08-12 21:03:24 +02:00
										 |  |  | // 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. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | package impl | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							|  |  |  | 	"reflect" | 
					
						
							|  |  |  | 	"sync" | 
					
						
							|  |  |  | 	"sync/atomic" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-28 18:30:40 +01:00
										 |  |  | 	"google.golang.org/protobuf/reflect/protoreflect" | 
					
						
							|  |  |  | 	"google.golang.org/protobuf/runtime/protoiface" | 
					
						
							| 
									
										
										
										
											2021-08-12 21:03:24 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // ExtensionInfo implements ExtensionType. | 
					
						
							|  |  |  | // | 
					
						
							|  |  |  | // This type contains a number of exported fields for legacy compatibility. | 
					
						
							|  |  |  | // The only non-deprecated use of this type is through the methods of the | 
					
						
							|  |  |  | // ExtensionType interface. | 
					
						
							|  |  |  | type ExtensionInfo struct { | 
					
						
							|  |  |  | 	// An ExtensionInfo may exist in several stages of initialization. | 
					
						
							|  |  |  | 	// | 
					
						
							|  |  |  | 	// extensionInfoUninitialized: Some or all of the legacy exported | 
					
						
							|  |  |  | 	// fields may be set, but none of the unexported fields have been | 
					
						
							|  |  |  | 	// initialized. This is the starting state for an ExtensionInfo | 
					
						
							|  |  |  | 	// in legacy generated code. | 
					
						
							|  |  |  | 	// | 
					
						
							|  |  |  | 	// extensionInfoDescInit: The desc field is set, but other unexported fields | 
					
						
							|  |  |  | 	// may not be initialized. Legacy exported fields may or may not be set. | 
					
						
							|  |  |  | 	// This is the starting state for an ExtensionInfo in newly generated code. | 
					
						
							|  |  |  | 	// | 
					
						
							|  |  |  | 	// extensionInfoFullInit: The ExtensionInfo is fully initialized. | 
					
						
							|  |  |  | 	// This state is only entered after lazy initialization is complete. | 
					
						
							|  |  |  | 	init uint32 | 
					
						
							|  |  |  | 	mu   sync.Mutex | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	goType reflect.Type | 
					
						
							|  |  |  | 	desc   extensionTypeDescriptor | 
					
						
							|  |  |  | 	conv   Converter | 
					
						
							|  |  |  | 	info   *extensionFieldInfo // for fast-path method implementations | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// ExtendedType is a typed nil-pointer to the parent message type that | 
					
						
							|  |  |  | 	// is being extended. It is possible for this to be unpopulated in v2 | 
					
						
							|  |  |  | 	// since the message may no longer implement the MessageV1 interface. | 
					
						
							|  |  |  | 	// | 
					
						
							|  |  |  | 	// Deprecated: Use the ExtendedType method instead. | 
					
						
							| 
									
										
										
										
											2022-09-28 18:30:40 +01:00
										 |  |  | 	ExtendedType protoiface.MessageV1 | 
					
						
							| 
									
										
										
										
											2021-08-12 21:03:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	// ExtensionType is the zero value of the extension type. | 
					
						
							|  |  |  | 	// | 
					
						
							|  |  |  | 	// For historical reasons, reflect.TypeOf(ExtensionType) and the | 
					
						
							|  |  |  | 	// type returned by InterfaceOf may not be identical. | 
					
						
							|  |  |  | 	// | 
					
						
							|  |  |  | 	// Deprecated: Use InterfaceOf(xt.Zero()) instead. | 
					
						
							|  |  |  | 	ExtensionType interface{} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Field is the field number of the extension. | 
					
						
							|  |  |  | 	// | 
					
						
							|  |  |  | 	// Deprecated: Use the Descriptor().Number method instead. | 
					
						
							|  |  |  | 	Field int32 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Name is the fully qualified name of extension. | 
					
						
							|  |  |  | 	// | 
					
						
							|  |  |  | 	// Deprecated: Use the Descriptor().FullName method instead. | 
					
						
							|  |  |  | 	Name string | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Tag is the protobuf struct tag used in the v1 API. | 
					
						
							|  |  |  | 	// | 
					
						
							|  |  |  | 	// Deprecated: Do not use. | 
					
						
							|  |  |  | 	Tag string | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// Filename is the proto filename in which the extension is defined. | 
					
						
							|  |  |  | 	// | 
					
						
							|  |  |  | 	// Deprecated: Use Descriptor().ParentFile().Path() instead. | 
					
						
							|  |  |  | 	Filename string | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Stages of initialization: See the ExtensionInfo.init field. | 
					
						
							|  |  |  | const ( | 
					
						
							|  |  |  | 	extensionInfoUninitialized = 0 | 
					
						
							|  |  |  | 	extensionInfoDescInit      = 1 | 
					
						
							|  |  |  | 	extensionInfoFullInit      = 2 | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-28 18:30:40 +01:00
										 |  |  | func InitExtensionInfo(xi *ExtensionInfo, xd protoreflect.ExtensionDescriptor, goType reflect.Type) { | 
					
						
							| 
									
										
										
										
											2021-08-12 21:03:24 +02:00
										 |  |  | 	xi.goType = goType | 
					
						
							|  |  |  | 	xi.desc = extensionTypeDescriptor{xd, xi} | 
					
						
							|  |  |  | 	xi.init = extensionInfoDescInit | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-28 18:30:40 +01:00
										 |  |  | func (xi *ExtensionInfo) New() protoreflect.Value { | 
					
						
							| 
									
										
										
										
											2021-08-12 21:03:24 +02:00
										 |  |  | 	return xi.lazyInit().New() | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-09-28 18:30:40 +01:00
										 |  |  | func (xi *ExtensionInfo) Zero() protoreflect.Value { | 
					
						
							| 
									
										
										
										
											2021-08-12 21:03:24 +02:00
										 |  |  | 	return xi.lazyInit().Zero() | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-09-28 18:30:40 +01:00
										 |  |  | func (xi *ExtensionInfo) ValueOf(v interface{}) protoreflect.Value { | 
					
						
							| 
									
										
										
										
											2021-08-12 21:03:24 +02:00
										 |  |  | 	return xi.lazyInit().PBValueOf(reflect.ValueOf(v)) | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-09-28 18:30:40 +01:00
										 |  |  | func (xi *ExtensionInfo) InterfaceOf(v protoreflect.Value) interface{} { | 
					
						
							| 
									
										
										
										
											2021-08-12 21:03:24 +02:00
										 |  |  | 	return xi.lazyInit().GoValueOf(v).Interface() | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-09-28 18:30:40 +01:00
										 |  |  | func (xi *ExtensionInfo) IsValidValue(v protoreflect.Value) bool { | 
					
						
							| 
									
										
										
										
											2021-08-12 21:03:24 +02:00
										 |  |  | 	return xi.lazyInit().IsValidPB(v) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | func (xi *ExtensionInfo) IsValidInterface(v interface{}) bool { | 
					
						
							|  |  |  | 	return xi.lazyInit().IsValidGo(reflect.ValueOf(v)) | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-09-28 18:30:40 +01:00
										 |  |  | func (xi *ExtensionInfo) TypeDescriptor() protoreflect.ExtensionTypeDescriptor { | 
					
						
							| 
									
										
										
										
											2021-08-12 21:03:24 +02:00
										 |  |  | 	if atomic.LoadUint32(&xi.init) < extensionInfoDescInit { | 
					
						
							|  |  |  | 		xi.lazyInitSlow() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return &xi.desc | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (xi *ExtensionInfo) lazyInit() Converter { | 
					
						
							|  |  |  | 	if atomic.LoadUint32(&xi.init) < extensionInfoFullInit { | 
					
						
							|  |  |  | 		xi.lazyInitSlow() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return xi.conv | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | func (xi *ExtensionInfo) lazyInitSlow() { | 
					
						
							|  |  |  | 	xi.mu.Lock() | 
					
						
							|  |  |  | 	defer xi.mu.Unlock() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if xi.init == extensionInfoFullInit { | 
					
						
							|  |  |  | 		return | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	defer atomic.StoreUint32(&xi.init, extensionInfoFullInit) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if xi.desc.ExtensionDescriptor == nil { | 
					
						
							|  |  |  | 		xi.initFromLegacy() | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if !xi.desc.ExtensionDescriptor.IsPlaceholder() { | 
					
						
							|  |  |  | 		if xi.ExtensionType == nil { | 
					
						
							|  |  |  | 			xi.initToLegacy() | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		xi.conv = NewConverter(xi.goType, xi.desc.ExtensionDescriptor) | 
					
						
							|  |  |  | 		xi.info = makeExtensionFieldInfo(xi.desc.ExtensionDescriptor) | 
					
						
							|  |  |  | 		xi.info.validation = newValidationInfo(xi.desc.ExtensionDescriptor, xi.goType) | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | type extensionTypeDescriptor struct { | 
					
						
							| 
									
										
										
										
											2022-09-28 18:30:40 +01:00
										 |  |  | 	protoreflect.ExtensionDescriptor | 
					
						
							| 
									
										
										
										
											2021-08-12 21:03:24 +02:00
										 |  |  | 	xi *ExtensionInfo | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-28 18:30:40 +01:00
										 |  |  | func (xtd *extensionTypeDescriptor) Type() protoreflect.ExtensionType { | 
					
						
							| 
									
										
										
										
											2021-08-12 21:03:24 +02:00
										 |  |  | 	return xtd.xi | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-09-28 18:30:40 +01:00
										 |  |  | func (xtd *extensionTypeDescriptor) Descriptor() protoreflect.ExtensionDescriptor { | 
					
						
							| 
									
										
										
										
											2021-08-12 21:03:24 +02:00
										 |  |  | 	return xtd.ExtensionDescriptor | 
					
						
							|  |  |  | } |