246 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			246 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| //
 | |
| // Author:
 | |
| //   Jb Evain (jbevain@gmail.com)
 | |
| //
 | |
| // Copyright (c) 2008 - 2015 Jb Evain
 | |
| // Copyright (c) 2008 - 2011 Novell, Inc.
 | |
| //
 | |
| // Licensed under the MIT/X11 license.
 | |
| //
 | |
| 
 | |
| using MonoFN.Collections.Generic;
 | |
| using System.Text;
 | |
| using System.Threading;
 | |
| 
 | |
| namespace MonoFN.Cecil {
 | |
| 
 | |
| 	public sealed class PropertyDefinition : PropertyReference, IMemberDefinition, IConstantProvider {
 | |
| 
 | |
| 		bool? has_this;
 | |
| 		ushort attributes;
 | |
| 
 | |
| 		Collection<CustomAttribute> custom_attributes;
 | |
| 
 | |
| 		internal MethodDefinition get_method;
 | |
| 		internal MethodDefinition set_method;
 | |
| 		internal Collection<MethodDefinition> other_methods;
 | |
| 
 | |
| 		object constant = Mixin.NotResolved;
 | |
| 
 | |
| 		public PropertyAttributes Attributes {
 | |
| 			get { return (PropertyAttributes)attributes; }
 | |
| 			set { attributes = (ushort)value; }
 | |
| 		}
 | |
| 
 | |
| 		public bool HasThis {
 | |
| 			get {
 | |
| 				if (has_this.HasValue)
 | |
| 					return has_this.Value;
 | |
| 
 | |
| 				if (GetMethod != null)
 | |
| 					return get_method.HasThis;
 | |
| 
 | |
| 				if (SetMethod != null)
 | |
| 					return set_method.HasThis;
 | |
| 
 | |
| 				return false;
 | |
| 			}
 | |
| 			set { has_this = value; }
 | |
| 		}
 | |
| 
 | |
| 		public bool HasCustomAttributes {
 | |
| 			get {
 | |
| 				if (custom_attributes != null)
 | |
| 					return custom_attributes.Count > 0;
 | |
| 
 | |
| 				return this.GetHasCustomAttributes (Module);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		public Collection<CustomAttribute> CustomAttributes {
 | |
| 			get { return custom_attributes ?? (this.GetCustomAttributes (ref custom_attributes, Module)); }
 | |
| 		}
 | |
| 
 | |
| 		public MethodDefinition GetMethod {
 | |
| 			get {
 | |
| 				if (get_method != null)
 | |
| 					return get_method;
 | |
| 
 | |
| 				InitializeMethods ();
 | |
| 				return get_method;
 | |
| 			}
 | |
| 			set { get_method = value; }
 | |
| 		}
 | |
| 
 | |
| 		public MethodDefinition SetMethod {
 | |
| 			get {
 | |
| 				if (set_method != null)
 | |
| 					return set_method;
 | |
| 
 | |
| 				InitializeMethods ();
 | |
| 				return set_method;
 | |
| 			}
 | |
| 			set { set_method = value; }
 | |
| 		}
 | |
| 
 | |
| 		public bool HasOtherMethods {
 | |
| 			get {
 | |
| 				if (other_methods != null)
 | |
| 					return other_methods.Count > 0;
 | |
| 
 | |
| 				InitializeMethods ();
 | |
| 				return !other_methods.IsNullOrEmpty ();
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		public Collection<MethodDefinition> OtherMethods {
 | |
| 			get {
 | |
| 				if (other_methods != null)
 | |
| 					return other_methods;
 | |
| 
 | |
| 				InitializeMethods ();
 | |
| 
 | |
| 				if (other_methods != null)
 | |
| 					return other_methods;
 | |
| 
 | |
| 				Interlocked.CompareExchange (ref other_methods, new Collection<MethodDefinition> (), null);
 | |
| 				return other_methods;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		public bool HasParameters {
 | |
| 			get {
 | |
| 				InitializeMethods ();
 | |
| 
 | |
| 				if (get_method != null)
 | |
| 					return get_method.HasParameters;
 | |
| 
 | |
| 				if (set_method != null)
 | |
| 					return set_method.HasParameters && set_method.Parameters.Count > 1;
 | |
| 
 | |
| 				return false;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		public override Collection<ParameterDefinition> Parameters {
 | |
| 			get {
 | |
| 				InitializeMethods ();
 | |
| 
 | |
| 				if (get_method != null)
 | |
| 					return MirrorParameters (get_method, 0);
 | |
| 
 | |
| 				if (set_method != null)
 | |
| 					return MirrorParameters (set_method, 1);
 | |
| 
 | |
| 				return new Collection<ParameterDefinition> ();
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		static Collection<ParameterDefinition> MirrorParameters (MethodDefinition method, int bound)
 | |
| 		{
 | |
| 			var parameters = new Collection<ParameterDefinition> ();
 | |
| 			if (!method.HasParameters)
 | |
| 				return parameters;
 | |
| 
 | |
| 			var original_parameters = method.Parameters;
 | |
| 			var end = original_parameters.Count - bound;
 | |
| 
 | |
| 			for (int i = 0; i < end; i++)
 | |
| 				parameters.Add (original_parameters [i]);
 | |
| 
 | |
| 			return parameters;
 | |
| 		}
 | |
| 
 | |
| 		public bool HasConstant {
 | |
| 			get {
 | |
| 				this.ResolveConstant (ref constant, Module);
 | |
| 
 | |
| 				return constant != Mixin.NoValue;
 | |
| 			}
 | |
| 			set { if (!value) constant = Mixin.NoValue; }
 | |
| 		}
 | |
| 
 | |
| 		public object Constant {
 | |
| 			get { return HasConstant ? constant : null; }
 | |
| 			set { constant = value; }
 | |
| 		}
 | |
| 
 | |
| 		#region PropertyAttributes
 | |
| 
 | |
| 		public bool IsSpecialName {
 | |
| 			get { return attributes.GetAttributes ((ushort)PropertyAttributes.SpecialName); }
 | |
| 			set { attributes = attributes.SetAttributes ((ushort)PropertyAttributes.SpecialName, value); }
 | |
| 		}
 | |
| 
 | |
| 		public bool IsRuntimeSpecialName {
 | |
| 			get { return attributes.GetAttributes ((ushort)PropertyAttributes.RTSpecialName); }
 | |
| 			set { attributes = attributes.SetAttributes ((ushort)PropertyAttributes.RTSpecialName, value); }
 | |
| 		}
 | |
| 
 | |
| 		public bool HasDefault {
 | |
| 			get { return attributes.GetAttributes ((ushort)PropertyAttributes.HasDefault); }
 | |
| 			set { attributes = attributes.SetAttributes ((ushort)PropertyAttributes.HasDefault, value); }
 | |
| 		}
 | |
| 
 | |
| 		#endregion
 | |
| 
 | |
| 		public new TypeDefinition DeclaringType {
 | |
| 			get { return (TypeDefinition)base.DeclaringType; }
 | |
| 			set { base.DeclaringType = value; }
 | |
| 		}
 | |
| 
 | |
| 		public override bool IsDefinition {
 | |
| 			get { return true; }
 | |
| 		}
 | |
| 
 | |
| 		public override string FullName {
 | |
| 			get {
 | |
| 				var builder = new StringBuilder ();
 | |
| 				builder.Append (PropertyType.ToString ());
 | |
| 				builder.Append (' ');
 | |
| 				builder.Append (MemberFullName ());
 | |
| 				builder.Append ('(');
 | |
| 				if (HasParameters) {
 | |
| 					var parameters = Parameters;
 | |
| 					for (int i = 0; i < parameters.Count; i++) {
 | |
| 						if (i > 0)
 | |
| 							builder.Append (',');
 | |
| 						builder.Append (parameters [i].ParameterType.FullName);
 | |
| 					}
 | |
| 				}
 | |
| 				builder.Append (')');
 | |
| 				return builder.ToString ();
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		public PropertyDefinition (string name, PropertyAttributes attributes, TypeReference propertyType)
 | |
| 			: base (name, propertyType)
 | |
| 		{
 | |
| 			this.attributes = (ushort)attributes;
 | |
| 			this.token = new MetadataToken (TokenType.Property);
 | |
| 		}
 | |
| 
 | |
| 		void InitializeMethods ()
 | |
| 		{
 | |
| 			var module = this.Module;
 | |
| 			if (module == null)
 | |
| 				return;
 | |
| 
 | |
| 			lock (module.SyncRoot) {
 | |
| 				if (get_method != null || set_method != null)
 | |
| 					return;
 | |
| 
 | |
| 				if (!module.HasImage ())
 | |
| 					return;
 | |
| 
 | |
| 				module.Read (this, (property, reader) => reader.ReadMethods (property));
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		public override PropertyDefinition Resolve ()
 | |
| 		{
 | |
| 			return this;
 | |
| 		}
 | |
| 	}
 | |
| }
 |