You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			442 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			442 lines
		
	
	
		
			16 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | //------------------------------------------------------------------------------ | ||
|  | // <copyright file="AttributeCollection.cs" company="Microsoft"> | ||
|  | //     Copyright (c) Microsoft Corporation.  All rights reserved. | ||
|  | // </copyright>                                                                 | ||
|  | //------------------------------------------------------------------------------ | ||
|  | 
 | ||
|  | using System.Diagnostics.CodeAnalysis; | ||
|  | 
 | ||
|  | /* | ||
|  |  This class has the HostProtectionAttribute. The purpose of this attribute is to enforce host-specific programming model guidelines, not security behavior.  | ||
|  |  Suppress FxCop message - BUT REVISIT IF ADDING NEW SECURITY ATTRIBUTES. | ||
|  | */ | ||
|  | [assembly: SuppressMessage("Microsoft.Security", "CA2112:SecuredTypesShouldNotExposeFields", Scope="type", Target="System.ComponentModel.AttributeCollection")] | ||
|  | [assembly: SuppressMessage("Microsoft.Security", "CA2112:SecuredTypesShouldNotExposeFields", Scope="type", Target="System.ComponentModel.AttributeCollection")] | ||
|  | [assembly: SuppressMessage("Microsoft.Security", "CA2123:OverrideLinkDemandsShouldBeIdenticalToBase", Scope="member", Target="System.ComponentModel.AttributeCollection.CopyTo(System.Array,System.Int32):System.Void")] | ||
|  | [assembly: SuppressMessage("Microsoft.Security", "CA2123:OverrideLinkDemandsShouldBeIdenticalToBase", Scope="member", Target="System.ComponentModel.AttributeCollection.System.Collections.IEnumerable.GetEnumerator():System.Collections.IEnumerator")] | ||
|  | [assembly: SuppressMessage("Microsoft.Security", "CA2123:OverrideLinkDemandsShouldBeIdenticalToBase", Scope="member", Target="System.ComponentModel.AttributeCollection.System.Collections.ICollection.get_IsSynchronized():System.Boolean")] | ||
|  | [assembly: SuppressMessage("Microsoft.Security", "CA2123:OverrideLinkDemandsShouldBeIdenticalToBase", Scope="member", Target="System.ComponentModel.AttributeCollection.System.Collections.ICollection.get_Count():System.Int32")] | ||
|  | [assembly: SuppressMessage("Microsoft.Security", "CA2123:OverrideLinkDemandsShouldBeIdenticalToBase", Scope="member", Target="System.ComponentModel.AttributeCollection.System.Collections.ICollection.get_SyncRoot():System.Object")] | ||
|  | [assembly: SuppressMessage("Microsoft.Security", "CA2122:DoNotIndirectlyExposeMethodsWithLinkDemands", Scope="member", Target="System.ComponentModel.AttributeCollection.Contains(System.Attribute):System.Boolean")] | ||
|  | [assembly: SuppressMessage("Microsoft.Security", "CA2122:DoNotIndirectlyExposeMethodsWithLinkDemands", Scope="member", Target="System.ComponentModel.AttributeCollection.CopyTo(System.Array,System.Int32):System.Void")] | ||
|  | [assembly: SuppressMessage("Microsoft.Security", "CA2122:DoNotIndirectlyExposeMethodsWithLinkDemands", Scope="member", Target="System.ComponentModel.AttributeCollection..ctor(System.Attribute[])")]
 | ||
|  | [assembly: SuppressMessage("Microsoft.Security", "CA2122:DoNotIndirectlyExposeMethodsWithLinkDemands", Scope="member", Target="System.ComponentModel.AttributeCollection.GetEnumerator():System.Collections.IEnumerator")] | ||
|  | [assembly: SuppressMessage("Microsoft.Security", "CA2122:DoNotIndirectlyExposeMethodsWithLinkDemands", Scope="member", Target="System.ComponentModel.AttributeCollection.get_Item(System.Type):System.Attribute")] | ||
|  | [assembly: SuppressMessage("Microsoft.Security", "CA2122:DoNotIndirectlyExposeMethodsWithLinkDemands", Scope="member", Target="System.ComponentModel.AttributeCollection.get_Item(System.Int32):System.Attribute")] | ||
|  | [assembly: SuppressMessage("Microsoft.Security", "CA2122:DoNotIndirectlyExposeMethodsWithLinkDemands", Scope="member", Target="System.ComponentModel.AttributeCollection.get_Count():System.Int32")] | ||
|  | 
 | ||
|  | namespace System.ComponentModel  | ||
|  | { | ||
|  | 
 | ||
|  |     using System.Reflection; | ||
|  |     using System.Diagnostics; | ||
|  |     using System.Collections; | ||
|  | 
 | ||
|  |      | ||
|  |     /// <devdoc> | ||
|  |     ///     Represents a collection of attributes. | ||
|  |     /// </devdoc> | ||
|  |     [System.Runtime.InteropServices.ComVisibleAttribute(true)] | ||
|  |     [System.Security.Permissions.HostProtection(Synchronization=true)] | ||
|  |     public class AttributeCollection : ICollection | ||
|  |     { | ||
|  | 
 | ||
|  |         /// <devdoc> | ||
|  |         ///     An empty AttributeCollection that can used instead of creating a new one. | ||
|  |         /// </devdoc> | ||
|  |         public static readonly AttributeCollection Empty = new AttributeCollection((Attribute[])null); | ||
|  |         private static Hashtable _defaultAttributes; | ||
|  | 
 | ||
|  |         private Attribute[] _attributes; | ||
|  |          | ||
|  |         private static object internalSyncObject = new object(); | ||
|  |          | ||
|  |         private struct AttributeEntry | ||
|  |         { | ||
|  |             public Type type; | ||
|  |             public int index; | ||
|  |         } | ||
|  | 
 | ||
|  |         private const int FOUND_TYPES_LIMIT = 5; | ||
|  | 
 | ||
|  |         private AttributeEntry[] _foundAttributeTypes; | ||
|  | 		 | ||
|  | 		private int _index = 0; | ||
|  | 
 | ||
|  |         /// <devdoc> | ||
|  |         ///     Creates a new AttributeCollection. | ||
|  |         /// </devdoc> | ||
|  |         public AttributeCollection(params Attribute[] attributes) | ||
|  |         { | ||
|  |             if (attributes == null) | ||
|  |             { | ||
|  |                 attributes = new Attribute[0]; | ||
|  |             } | ||
|  |             _attributes = attributes; | ||
|  | 
 | ||
|  |             for (int idx = 0; idx < attributes.Length; idx++) | ||
|  |             { | ||
|  |                 if (attributes[idx] == null) | ||
|  |                 { | ||
|  |                     throw new ArgumentNullException("attributes"); | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         protected AttributeCollection() | ||
|  |         { | ||
|  |         } | ||
|  | 
 | ||
|  |          | ||
|  |         /// <devdoc> | ||
|  |         ///     Creates a new AttributeCollection from an existing AttributeCollection | ||
|  |         /// </devdoc> | ||
|  |         public static AttributeCollection FromExisting(AttributeCollection existing, params Attribute[] newAttributes) | ||
|  |         { | ||
|  |             // VSWhidbey #75418 | ||
|  |             // This method should be a constructor, but making it one introduces a breaking change. | ||
|  |             //  | ||
|  |             if (existing == null) | ||
|  |             { | ||
|  |                 throw new ArgumentNullException("existing"); | ||
|  |             } | ||
|  | 
 | ||
|  |             if (newAttributes == null) | ||
|  |             { | ||
|  |                 newAttributes = new Attribute[0]; | ||
|  |             } | ||
|  | 
 | ||
|  |             Attribute[] newArray = new Attribute[existing.Count + newAttributes.Length]; | ||
|  |             int actualCount = existing.Count; | ||
|  |             existing.CopyTo(newArray, 0); | ||
|  | 
 | ||
|  |             for (int idx = 0; idx < newAttributes.Length; idx++) | ||
|  |             { | ||
|  |                 if (newAttributes[idx] == null) | ||
|  |                 { | ||
|  |                     throw new ArgumentNullException("newAttributes"); | ||
|  |                 } | ||
|  | 
 | ||
|  |                 // We must see if this attribute is already in the existing | ||
|  |                 // array.  If it is, we replace it. | ||
|  |                 bool match = false; | ||
|  |                 for (int existingIdx = 0; existingIdx < existing.Count; existingIdx++) | ||
|  |                 { | ||
|  |                     if (newArray[existingIdx].TypeId.Equals(newAttributes[idx].TypeId)) | ||
|  |                     { | ||
|  |                         match = true; | ||
|  |                         newArray[existingIdx] = newAttributes[idx]; | ||
|  |                         break; | ||
|  |                     } | ||
|  |                 } | ||
|  | 
 | ||
|  |                 if (!match) | ||
|  |                 { | ||
|  |                     newArray[actualCount++] = newAttributes[idx]; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             // Now, if we collapsed some attributes, create a new array. | ||
|  |             // | ||
|  | 
 | ||
|  |             Attribute[] attributes = null; | ||
|  |             if (actualCount < newArray.Length) | ||
|  |             { | ||
|  |                 attributes = new Attribute[actualCount]; | ||
|  |                 Array.Copy(newArray, 0, attributes, 0, actualCount); | ||
|  |             } | ||
|  |             else | ||
|  |             { | ||
|  |                 attributes = newArray; | ||
|  |             } | ||
|  | 
 | ||
|  |             return new AttributeCollection(attributes); | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <devdoc> | ||
|  |         ///     Gets the attributes collection. | ||
|  |         /// </devdoc> | ||
|  |         [SuppressMessage("Microsoft.Performance", "CA1819:PropertiesShouldNotReturnArrays", | ||
|  |             Justification = "Matches constructor input type")] | ||
|  |         protected virtual Attribute[] Attributes | ||
|  |         { | ||
|  |             get | ||
|  |             { | ||
|  |                 return _attributes; | ||
|  |             } | ||
|  |         } | ||
|  |          | ||
|  |         /// <devdoc> | ||
|  |         ///     Gets the number of attributes. | ||
|  |         /// </devdoc> | ||
|  |         public int Count  | ||
|  |         { | ||
|  |             get  | ||
|  |             { | ||
|  |                 return Attributes.Length; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <devdoc> | ||
|  |         ///     Gets the attribute with the specified index number. | ||
|  |         /// </devdoc> | ||
|  |         public virtual Attribute this[int index]  | ||
|  |         { | ||
|  |             get  | ||
|  |             { | ||
|  |                 return Attributes[index]; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <devdoc> | ||
|  |         ///    Gets the attribute with the specified type. | ||
|  |         /// </devdoc> | ||
|  |         public virtual Attribute this[Type attributeType]  | ||
|  |         { | ||
|  |             get  | ||
|  |             { | ||
|  |                 lock (internalSyncObject) { | ||
|  |                     // 2 passes here for perf.  Really!  first pass, we just  | ||
|  |                     // check equality, and if we don't find it, then we | ||
|  |                     // do the IsAssignableFrom dance.   Turns out that's | ||
|  |                     // a relatively expensive call and we try to avoid it | ||
|  |                     // since we rarely encounter derived attribute types | ||
|  |                     // and this list is usually short.  | ||
|  |                     // | ||
|  |                     if (_foundAttributeTypes == null) | ||
|  |                     { | ||
|  |                         _foundAttributeTypes = new AttributeEntry[FOUND_TYPES_LIMIT]; | ||
|  |                     } | ||
|  | 
 | ||
|  |                     int ind = 0; | ||
|  |       | ||
|  |                     for (; ind < FOUND_TYPES_LIMIT; ind++) | ||
|  |                     { | ||
|  |                         if (_foundAttributeTypes[ind].type == attributeType) | ||
|  |                         { | ||
|  |                            int index = _foundAttributeTypes[ind].index; | ||
|  |                            if (index != -1) | ||
|  |                            { | ||
|  |                                return Attributes[index]; | ||
|  |                            } | ||
|  |                            else{ | ||
|  |                                return GetDefaultAttribute(attributeType); | ||
|  |                            }    | ||
|  |                         } | ||
|  |                         if (_foundAttributeTypes[ind].type == null) | ||
|  |                             break; | ||
|  |                     } | ||
|  | 
 | ||
|  |                     ind = _index++; | ||
|  | 
 | ||
|  |                     if (_index >= FOUND_TYPES_LIMIT) | ||
|  |                     { | ||
|  |     	                _index = 0; | ||
|  |                     } | ||
|  | 
 | ||
|  |                     _foundAttributeTypes[ind].type = attributeType; | ||
|  | 
 | ||
|  |                     int count = Attributes.Length; | ||
|  |                      | ||
|  | 
 | ||
|  |                     for (int i = 0; i < count; i++) | ||
|  |                     { | ||
|  |                         Attribute attribute = Attributes[i]; | ||
|  |                         Type aType = attribute.GetType(); | ||
|  |                         if (aType == attributeType) | ||
|  |                         {         | ||
|  |                             _foundAttributeTypes[ind].index = i; | ||
|  |                             return attribute; | ||
|  |                         } | ||
|  |                     } | ||
|  | 
 | ||
|  |                     // now check the hierarchies. | ||
|  |                     for (int i = 0; i < count; i++) | ||
|  |                     { | ||
|  |                         Attribute attribute = Attributes[i]; | ||
|  |                         Type aType = attribute.GetType(); | ||
|  |                         if (attributeType.IsAssignableFrom(aType)) | ||
|  |                         { | ||
|  |                             _foundAttributeTypes[ind].index = i; | ||
|  |                             return attribute; | ||
|  |                         } | ||
|  |                     } | ||
|  |                      | ||
|  |                     _foundAttributeTypes[ind].index = -1; | ||
|  |                     return GetDefaultAttribute(attributeType); | ||
|  |                 } | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <devdoc> | ||
|  |         ///     Determines if this collection of attributes has the specified attribute. | ||
|  |         /// </devdoc> | ||
|  |         public bool Contains(Attribute attribute) | ||
|  |         { | ||
|  |             Attribute attr = this[attribute.GetType()]; | ||
|  |             if (attr != null && attr.Equals(attribute)) | ||
|  |             { | ||
|  |                 return true; | ||
|  |             } | ||
|  |             return false; | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <devdoc> | ||
|  |         ///     Determines if this attribute collection contains the all | ||
|  |         ///     the specified attributes in the attribute array. | ||
|  |         /// </devdoc> | ||
|  |         public bool Contains(Attribute[] attributes) | ||
|  |         { | ||
|  | 
 | ||
|  |             if (attributes == null) | ||
|  |             { | ||
|  |                 return true; | ||
|  |             } | ||
|  | 
 | ||
|  |             for (int i = 0; i < attributes.Length; i++) | ||
|  |             { | ||
|  |                 if (!Contains(attributes[i])) | ||
|  |                 { | ||
|  |                     return false; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             return true; | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <devdoc> | ||
|  |         ///     Returns the default value for an attribute.  This uses the following hurestic: | ||
|  |         ///     1.  It looks for a public static field named "Default". | ||
|  |         /// </devdoc> | ||
|  |         protected Attribute GetDefaultAttribute(Type attributeType) | ||
|  |         { | ||
|  |             lock (internalSyncObject) | ||
|  |             { | ||
|  |                 if (_defaultAttributes == null) | ||
|  |                 { | ||
|  |                     _defaultAttributes = new Hashtable(); | ||
|  |                 } | ||
|  | 
 | ||
|  |                 // If we have already encountered this, use what's in the | ||
|  |                 // table. | ||
|  |                 if (_defaultAttributes.ContainsKey(attributeType)) | ||
|  |                 { | ||
|  |                     return(Attribute)_defaultAttributes[attributeType]; | ||
|  |                 } | ||
|  | 
 | ||
|  |                 Attribute attr = null; | ||
|  | 
 | ||
|  |                 // Nope, not in the table, so do the legwork to discover the default value. | ||
|  |                 Type reflect = TypeDescriptor.GetReflectionType(attributeType); | ||
|  |                 System.Reflection.FieldInfo field = reflect.GetField("Default", BindingFlags.Public | BindingFlags.Static | BindingFlags.GetField); | ||
|  |                 if (field != null && field.IsStatic) | ||
|  |                 { | ||
|  |                     attr = (Attribute)field.GetValue(null); | ||
|  |                 } | ||
|  |                 else | ||
|  |                 { | ||
|  |                     ConstructorInfo ci = reflect.UnderlyingSystemType.GetConstructor(new Type[0]); | ||
|  |                     if (ci != null) | ||
|  |                     { | ||
|  |                         attr = (Attribute)ci.Invoke(new object[0]); | ||
|  | 
 | ||
|  |                         // If we successfully created, verify that it is the | ||
|  |                         // default.  Attributes don't have to abide by this rule. | ||
|  |                         if (!attr.IsDefaultAttribute()) | ||
|  |                         { | ||
|  |                             attr = null; | ||
|  |                         } | ||
|  |                     } | ||
|  |                 } | ||
|  | 
 | ||
|  |                 _defaultAttributes[attributeType] = attr; | ||
|  |                 return attr; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <devdoc> | ||
|  |         ///     Gets an enumerator for this collection. | ||
|  |         /// </devdoc> | ||
|  |         public IEnumerator GetEnumerator() | ||
|  |         { | ||
|  |             return Attributes.GetEnumerator(); | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <devdoc> | ||
|  |         ///     Determines if a specified attribute is the same as an attribute | ||
|  |         ///     in the collection. | ||
|  |         /// </devdoc> | ||
|  |         public bool Matches(Attribute attribute) | ||
|  |         { | ||
|  |             for (int i = 0; i < Attributes.Length; i++) | ||
|  |             { | ||
|  |                 if (Attributes[i].Match(attribute)) | ||
|  |                 { | ||
|  |                     return true; | ||
|  |                 } | ||
|  |             } | ||
|  |             return false; | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <devdoc> | ||
|  |         ///     Determines if the attributes in the specified array are | ||
|  |         ///     the same as the attributes in the collection. | ||
|  |         /// </devdoc> | ||
|  |         public bool Matches(Attribute[] attributes) | ||
|  |         { | ||
|  |             for (int i = 0; i < attributes.Length; i++) | ||
|  |             { | ||
|  |                 if (!Matches(attributes[i])) | ||
|  |                 { | ||
|  |                     return false; | ||
|  |                 } | ||
|  |             } | ||
|  | 
 | ||
|  |             return true; | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <internalonly/> | ||
|  |         int ICollection.Count  | ||
|  |         { | ||
|  |             get  | ||
|  |             { | ||
|  |                 return Count; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  | 
 | ||
|  |         /// <internalonly/> | ||
|  |         bool ICollection.IsSynchronized  | ||
|  |         { | ||
|  |             get  | ||
|  |             { | ||
|  |                 return false; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <internalonly/> | ||
|  |         object ICollection.SyncRoot  | ||
|  |         { | ||
|  |             get  | ||
|  |             { | ||
|  |                 return null; | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <devdoc> | ||
|  |         ///     Copies this collection to an array. | ||
|  |         /// </devdoc> | ||
|  |         public void CopyTo(Array array, int index) | ||
|  |         { | ||
|  |             Array.Copy(Attributes, 0, array, index, Attributes.Length); | ||
|  |         } | ||
|  | 
 | ||
|  |         /// <internalonly/> | ||
|  |         IEnumerator IEnumerable.GetEnumerator() | ||
|  |         { | ||
|  |             return GetEnumerator(); | ||
|  |         } | ||
|  |     } | ||
|  | } | ||
|  | 
 |