You cannot select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
	
	
		
			157 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			Go
		
	
		
		
			
		
	
	
			157 lines
		
	
	
		
			4.6 KiB
		
	
	
	
		
			Go
		
	
| 
											2 years ago
										 | // 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" | ||
|  | 
 | ||
|  | 	"google.golang.org/protobuf/reflect/protoreflect" | ||
|  | 	"google.golang.org/protobuf/runtime/protoiface" | ||
|  | ) | ||
|  | 
 | ||
|  | // 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.
 | ||
|  | 	ExtendedType protoiface.MessageV1 | ||
|  | 
 | ||
|  | 	// 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 | ||
|  | ) | ||
|  | 
 | ||
|  | func InitExtensionInfo(xi *ExtensionInfo, xd protoreflect.ExtensionDescriptor, goType reflect.Type) { | ||
|  | 	xi.goType = goType | ||
|  | 	xi.desc = extensionTypeDescriptor{xd, xi} | ||
|  | 	xi.init = extensionInfoDescInit | ||
|  | } | ||
|  | 
 | ||
|  | func (xi *ExtensionInfo) New() protoreflect.Value { | ||
|  | 	return xi.lazyInit().New() | ||
|  | } | ||
|  | func (xi *ExtensionInfo) Zero() protoreflect.Value { | ||
|  | 	return xi.lazyInit().Zero() | ||
|  | } | ||
|  | func (xi *ExtensionInfo) ValueOf(v interface{}) protoreflect.Value { | ||
|  | 	return xi.lazyInit().PBValueOf(reflect.ValueOf(v)) | ||
|  | } | ||
|  | func (xi *ExtensionInfo) InterfaceOf(v protoreflect.Value) interface{} { | ||
|  | 	return xi.lazyInit().GoValueOf(v).Interface() | ||
|  | } | ||
|  | func (xi *ExtensionInfo) IsValidValue(v protoreflect.Value) bool { | ||
|  | 	return xi.lazyInit().IsValidPB(v) | ||
|  | } | ||
|  | func (xi *ExtensionInfo) IsValidInterface(v interface{}) bool { | ||
|  | 	return xi.lazyInit().IsValidGo(reflect.ValueOf(v)) | ||
|  | } | ||
|  | func (xi *ExtensionInfo) TypeDescriptor() protoreflect.ExtensionTypeDescriptor { | ||
|  | 	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 { | ||
|  | 	protoreflect.ExtensionDescriptor | ||
|  | 	xi *ExtensionInfo | ||
|  | } | ||
|  | 
 | ||
|  | func (xtd *extensionTypeDescriptor) Type() protoreflect.ExtensionType { | ||
|  | 	return xtd.xi | ||
|  | } | ||
|  | func (xtd *extensionTypeDescriptor) Descriptor() protoreflect.ExtensionDescriptor { | ||
|  | 	return xtd.ExtensionDescriptor | ||
|  | } |