You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			510 lines
		
	
	
		
			21 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			510 lines
		
	
	
		
			21 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| //-----------------------------------------------------------------------------
 | |
| // Copyright (c) Microsoft Corporation.  All rights reserved.
 | |
| //-----------------------------------------------------------------------------
 | |
| namespace System.Runtime.Serialization
 | |
| {
 | |
|     using System;
 | |
|     using System.Collections;
 | |
|     using System.Collections.Generic;
 | |
|     using System.Diagnostics.CodeAnalysis;
 | |
|     using System.Globalization;
 | |
|     using System.Reflection;
 | |
|     using System.Threading;
 | |
|     using System.Text;
 | |
|     using System.Xml;
 | |
|     using System.Security;
 | |
| 
 | |
| #if USE_REFEMIT
 | |
|     public sealed class EnumDataContract : DataContract
 | |
| #else
 | |
|     internal sealed class EnumDataContract : DataContract
 | |
| #endif
 | |
|     {
 | |
|         [Fx.Tag.SecurityNote(Critical = "Holds instance of CriticalHelper which keeps state that is cached statically for serialization."
 | |
|             + " Static fields are marked SecurityCritical or readonly to prevent data from being modified or leaked to other components in appdomain.")]
 | |
|         [SecurityCritical]
 | |
|         EnumDataContractCriticalHelper helper;
 | |
| 
 | |
|         [Fx.Tag.SecurityNote(Critical = "Initializes SecurityCritical field 'helper'.",
 | |
|             Safe = "Doesn't leak anything.")]
 | |
|         [SecuritySafeCritical]
 | |
|         internal EnumDataContract()
 | |
|             : base(new EnumDataContractCriticalHelper())
 | |
|         {
 | |
|             helper = base.Helper as EnumDataContractCriticalHelper;
 | |
|         }
 | |
| 
 | |
|         [Fx.Tag.SecurityNote(Critical = "Initializes SecurityCritical field 'helper'.",
 | |
|             Safe = "Doesn't leak anything.")]
 | |
|         [SecuritySafeCritical]
 | |
|         internal EnumDataContract(Type type)
 | |
|             : base(new EnumDataContractCriticalHelper(type))
 | |
|         {
 | |
|             helper = base.Helper as EnumDataContractCriticalHelper;
 | |
|         }
 | |
| 
 | |
|         [Fx.Tag.SecurityNote(Critical = "Accesses SecurityCritical static cache to look up base contract name for a type.",
 | |
|             Safe = "Read only access.")]
 | |
|         [SecuritySafeCritical]
 | |
|         static internal XmlQualifiedName GetBaseContractName(Type type)
 | |
|         {
 | |
|             return EnumDataContractCriticalHelper.GetBaseContractName(type);
 | |
|         }
 | |
| 
 | |
|         [Fx.Tag.SecurityNote(Critical = "Accesses SecurityCritical static cache to look up a base contract name.",
 | |
|             Safe = "Read only access.")]
 | |
|         [SecuritySafeCritical]
 | |
|         static internal Type GetBaseType(XmlQualifiedName baseContractName)
 | |
|         {
 | |
|             return EnumDataContractCriticalHelper.GetBaseType(baseContractName);
 | |
|         }
 | |
| 
 | |
|         internal XmlQualifiedName BaseContractName
 | |
|         {
 | |
|             [Fx.Tag.SecurityNote(Critical = "Fetches the critical BaseContractName property.",
 | |
|                 Safe = "BaseContractName only needs to be protected for write.")]
 | |
|             [SecuritySafeCritical]
 | |
|             get { return helper.BaseContractName; }
 | |
| 
 | |
|             [Fx.Tag.SecurityNote(Critical = "Sets the critical BaseContractName property.")]
 | |
|             [SecurityCritical]
 | |
|             set { helper.BaseContractName = value; }
 | |
|         }
 | |
| 
 | |
|         internal List<DataMember> Members
 | |
|         {
 | |
|             [Fx.Tag.SecurityNote(Critical = "Fetches the critical Members property.",
 | |
|                 Safe = "Members only needs to be protected for write.")]
 | |
|             [SecuritySafeCritical]
 | |
|             get { return helper.Members; }
 | |
| 
 | |
|             [Fx.Tag.SecurityNote(Critical = "Sets the critical Members property.")]
 | |
|             [SecurityCritical]
 | |
|             set { helper.Members = value; }
 | |
|         }
 | |
| 
 | |
|         internal List<long> Values
 | |
|         {
 | |
|             [Fx.Tag.SecurityNote(Critical = "Fetches the critical Values property.",
 | |
|                 Safe = "Values only needs to be protected for write.")]
 | |
|             [SecuritySafeCritical]
 | |
|             get { return helper.Values; }
 | |
| 
 | |
|             [Fx.Tag.SecurityNote(Critical = "Sets the critical Values property.")]
 | |
|             [SecurityCritical]
 | |
|             set { helper.Values = value; }
 | |
|         }
 | |
| 
 | |
|         internal bool IsFlags
 | |
|         {
 | |
|             [Fx.Tag.SecurityNote(Critical = "Fetches the critical IsFlags property.",
 | |
|                 Safe = "IsFlags only needs to be protected for write.")]
 | |
|             [SecuritySafeCritical]
 | |
|             get { return helper.IsFlags; }
 | |
| 
 | |
|             [Fx.Tag.SecurityNote(Critical = "Sets the critical IsFlags property.")]
 | |
|             [SecurityCritical]
 | |
|             set { helper.IsFlags = value; }
 | |
|         }
 | |
| 
 | |
|         internal bool IsULong
 | |
|         {
 | |
|             [Fx.Tag.SecurityNote(Critical = "Fetches the critical IsULong property.",
 | |
|                 Safe = "IsULong only needs to be protected for write.")]
 | |
|             [SecuritySafeCritical]
 | |
|             get { return helper.IsULong; }
 | |
|         }
 | |
| 
 | |
|         XmlDictionaryString[] ChildElementNames
 | |
|         {
 | |
|             [Fx.Tag.SecurityNote(Critical = "Fetches the critical ChildElementNames property.",
 | |
|                 Safe = "ChildElementNames only needs to be protected for write.")]
 | |
|             [SecuritySafeCritical]
 | |
|             get { return helper.ChildElementNames; }
 | |
|         }
 | |
| 
 | |
|         internal override bool CanContainReferences
 | |
|         {
 | |
|             get { return false; }
 | |
|         }
 | |
| 
 | |
|         [Fx.Tag.SecurityNote(Critical = "Holds all state used for (de)serializing enums."
 | |
|             + " Since the data is cached statically, we lock down access to it.")]
 | |
| #if !NO_SECURITY_ATTRIBUTES
 | |
|         [SecurityCritical(SecurityCriticalScope.Everything)]
 | |
| #endif
 | |
|         class EnumDataContractCriticalHelper : DataContract.DataContractCriticalHelper
 | |
|         {
 | |
|             static Dictionary<Type, XmlQualifiedName> typeToName;
 | |
|             static Dictionary<XmlQualifiedName, Type> nameToType;
 | |
| 
 | |
|             XmlQualifiedName baseContractName;
 | |
|             List<DataMember> members;
 | |
|             List<long> values;
 | |
|             bool isULong;
 | |
|             bool isFlags;
 | |
|             bool hasDataContract;
 | |
|             XmlDictionaryString[] childElementNames;
 | |
| 
 | |
|             static EnumDataContractCriticalHelper()
 | |
|             {
 | |
|                 typeToName = new Dictionary<Type, XmlQualifiedName>();
 | |
|                 nameToType = new Dictionary<XmlQualifiedName, Type>();
 | |
|                 Add(typeof(sbyte), "byte");
 | |
|                 Add(typeof(byte), "unsignedByte");
 | |
|                 Add(typeof(short), "short");
 | |
|                 Add(typeof(ushort), "unsignedShort");
 | |
|                 Add(typeof(int), "int");
 | |
|                 Add(typeof(uint), "unsignedInt");
 | |
|                 Add(typeof(long), "long");
 | |
|                 Add(typeof(ulong), "unsignedLong");
 | |
|             }
 | |
| 
 | |
|             [SuppressMessage(FxCop.Category.Usage, "CA2301:EmbeddableTypesInContainersRule", MessageId = "typeToName", Justification = "No need to support type equivalence here.")]
 | |
|             static internal void Add(Type type, string localName)
 | |
|             {
 | |
|                 XmlQualifiedName stableName = CreateQualifiedName(localName, Globals.SchemaNamespace);
 | |
|                 typeToName.Add(type, stableName);
 | |
|                 nameToType.Add(stableName, type);
 | |
|             }
 | |
| 
 | |
|             static internal XmlQualifiedName GetBaseContractName(Type type)
 | |
|             {
 | |
|                 XmlQualifiedName retVal = null;
 | |
|                 typeToName.TryGetValue(type, out retVal);
 | |
|                 return retVal;
 | |
|             }
 | |
| 
 | |
|             static internal Type GetBaseType(XmlQualifiedName baseContractName)
 | |
|             {
 | |
|                 Type retVal = null;
 | |
|                 nameToType.TryGetValue(baseContractName, out retVal);
 | |
|                 return retVal;
 | |
|             }
 | |
| 
 | |
|             internal EnumDataContractCriticalHelper()
 | |
|             {
 | |
|                 IsValueType = true;
 | |
|             }
 | |
| 
 | |
|             internal EnumDataContractCriticalHelper(Type type)
 | |
|                 : base(type)
 | |
|             {
 | |
|                 this.StableName = DataContract.GetStableName(type, out hasDataContract);
 | |
|                 Type baseType = Enum.GetUnderlyingType(type);
 | |
|                 baseContractName = GetBaseContractName(baseType);
 | |
|                 ImportBaseType(baseType);
 | |
|                 IsFlags = type.IsDefined(Globals.TypeOfFlagsAttribute, false);
 | |
|                 ImportDataMembers();
 | |
| 
 | |
|                 XmlDictionary dictionary = new XmlDictionary(2 + Members.Count);
 | |
|                 Name = dictionary.Add(StableName.Name);
 | |
|                 Namespace = dictionary.Add(StableName.Namespace);
 | |
|                 childElementNames = new XmlDictionaryString[Members.Count];
 | |
|                 for (int i = 0; i < Members.Count; i++)
 | |
|                     childElementNames[i] = dictionary.Add(Members[i].Name);
 | |
| 
 | |
|                 DataContractAttribute dataContractAttribute;
 | |
|                 if (TryGetDCAttribute(type, out dataContractAttribute))
 | |
|                 {
 | |
|                     if (dataContractAttribute.IsReference)
 | |
|                     {
 | |
|                         DataContract.ThrowInvalidDataContractException(
 | |
|                                 SR.GetString(SR.EnumTypeCannotHaveIsReference,
 | |
|                                     DataContract.GetClrTypeFullName(type),
 | |
|                                     dataContractAttribute.IsReference,
 | |
|                                     false),
 | |
|                                 type);
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             internal XmlQualifiedName BaseContractName
 | |
|             {
 | |
|                 get
 | |
|                 {
 | |
|                     return baseContractName;
 | |
|                 }
 | |
|                 set
 | |
|                 {
 | |
|                     baseContractName = value;
 | |
|                     Type baseType = GetBaseType(baseContractName);
 | |
|                     if (baseType == null)
 | |
|                         ThrowInvalidDataContractException(SR.GetString(SR.InvalidEnumBaseType, value.Name, value.Namespace, StableName.Name, StableName.Namespace));
 | |
|                     ImportBaseType(baseType);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             internal List<DataMember> Members
 | |
|             {
 | |
|                 get { return members; }
 | |
|                 set { members = value; }
 | |
|             }
 | |
| 
 | |
|             internal List<long> Values
 | |
|             {
 | |
|                 get { return values; }
 | |
|                 set { values = value; }
 | |
|             }
 | |
| 
 | |
|             internal bool IsFlags
 | |
|             {
 | |
|                 get { return isFlags; }
 | |
|                 set { isFlags = value; }
 | |
|             }
 | |
| 
 | |
|             internal bool IsULong
 | |
|             {
 | |
|                 get { return isULong; }
 | |
|             }
 | |
| 
 | |
|             internal XmlDictionaryString[] ChildElementNames
 | |
|             {
 | |
|                 get { return childElementNames; }
 | |
|             }
 | |
| 
 | |
|             void ImportBaseType(Type baseType)
 | |
|             {
 | |
|                 isULong = (baseType == Globals.TypeOfULong);
 | |
|             }
 | |
| 
 | |
|             void ImportDataMembers()
 | |
|             {
 | |
|                 Type type = this.UnderlyingType;
 | |
|                 FieldInfo[] fields = type.GetFields(BindingFlags.Static | BindingFlags.Public);
 | |
|                 Dictionary<string, DataMember> memberValuesTable = new Dictionary<string, DataMember>();
 | |
|                 List<DataMember> tempMembers = new List<DataMember>(fields.Length);
 | |
|                 List<long> tempValues = new List<long>(fields.Length);
 | |
| 
 | |
|                 for (int i = 0; i < fields.Length; i++)
 | |
|                 {
 | |
|                     FieldInfo field = fields[i];
 | |
|                     bool enumMemberValid = false;
 | |
|                     if (hasDataContract)
 | |
|                     {
 | |
|                         object[] memberAttributes = field.GetCustomAttributes(Globals.TypeOfEnumMemberAttribute, false);
 | |
|                         if (memberAttributes != null && memberAttributes.Length > 0)
 | |
|                         {
 | |
|                             if (memberAttributes.Length > 1)
 | |
|                                 ThrowInvalidDataContractException(SR.GetString(SR.TooManyEnumMembers, DataContract.GetClrTypeFullName(field.DeclaringType), field.Name));
 | |
|                             EnumMemberAttribute memberAttribute = (EnumMemberAttribute)memberAttributes[0];
 | |
| 
 | |
|                             DataMember memberContract = new DataMember(field);
 | |
|                             if (memberAttribute.IsValueSetExplicitly)
 | |
|                             {
 | |
|                                 if (memberAttribute.Value == null || memberAttribute.Value.Length == 0)
 | |
|                                     ThrowInvalidDataContractException(SR.GetString(SR.InvalidEnumMemberValue, field.Name, DataContract.GetClrTypeFullName(type)));
 | |
|                                 memberContract.Name = memberAttribute.Value;
 | |
|                             }
 | |
|                             else
 | |
|                                 memberContract.Name = field.Name;
 | |
|                             ClassDataContract.CheckAndAddMember(tempMembers, memberContract, memberValuesTable);
 | |
|                             enumMemberValid = true;
 | |
|                         }
 | |
| 
 | |
|                         object[] dataMemberAttributes = field.GetCustomAttributes(Globals.TypeOfDataMemberAttribute, false);
 | |
|                         if (dataMemberAttributes != null && dataMemberAttributes.Length > 0)
 | |
|                             ThrowInvalidDataContractException(SR.GetString(SR.DataMemberOnEnumField, DataContract.GetClrTypeFullName(field.DeclaringType), field.Name));
 | |
|                     }
 | |
|                     else
 | |
|                     {
 | |
|                         if (!field.IsNotSerialized)
 | |
|                         {
 | |
|                             DataMember memberContract = new DataMember(field);
 | |
|                             memberContract.Name = field.Name;
 | |
|                             ClassDataContract.CheckAndAddMember(tempMembers, memberContract, memberValuesTable);
 | |
|                             enumMemberValid = true;
 | |
|                         }
 | |
|                     }
 | |
| 
 | |
|                     if (enumMemberValid)
 | |
|                     {
 | |
|                         object enumValue = field.GetValue(null);
 | |
|                         if (isULong)
 | |
|                             tempValues.Add((long)((IConvertible)enumValue).ToUInt64(null));
 | |
|                         else
 | |
|                             tempValues.Add(((IConvertible)enumValue).ToInt64(null));
 | |
|                     }
 | |
|                 }
 | |
| 
 | |
|                 Thread.MemoryBarrier();
 | |
|                 members = tempMembers;
 | |
|                 values = tempValues;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         internal void WriteEnumValue(XmlWriterDelegator writer, object value)
 | |
|         {
 | |
|             long longValue = IsULong ? (long)((IConvertible)value).ToUInt64(null) : ((IConvertible)value).ToInt64(null);
 | |
|             for (int i = 0; i < Values.Count; i++)
 | |
|             {
 | |
|                 if (longValue == Values[i])
 | |
|                 {
 | |
|                     writer.WriteString(ChildElementNames[i].Value);
 | |
|                     return;
 | |
|                 }
 | |
|             }
 | |
|             if (IsFlags)
 | |
|             {
 | |
|                 int zeroIndex = -1;
 | |
|                 bool noneWritten = true;
 | |
|                 for (int i = 0; i < Values.Count; i++)
 | |
|                 {
 | |
|                     long current = Values[i];
 | |
|                     if (current == 0)
 | |
|                     {
 | |
|                         zeroIndex = i;
 | |
|                         continue;
 | |
|                     }
 | |
|                     if (longValue == 0)
 | |
|                         break;
 | |
|                     if ((current & longValue) == current)
 | |
|                     {
 | |
|                         if (noneWritten)
 | |
|                             noneWritten = false;
 | |
|                         else
 | |
|                             writer.WriteString(DictionaryGlobals.Space.Value);
 | |
| 
 | |
|                         writer.WriteString(ChildElementNames[i].Value);
 | |
|                         longValue &= ~current;
 | |
|                     }
 | |
|                 }
 | |
|                 // enforce that enum value was completely parsed
 | |
|                 if (longValue != 0)
 | |
|                     throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(XmlObjectSerializer.CreateSerializationException(SR.GetString(SR.InvalidEnumValueOnWrite, value, DataContract.GetClrTypeFullName(UnderlyingType))));
 | |
| 
 | |
|                 if (noneWritten && zeroIndex >= 0)
 | |
|                     writer.WriteString(ChildElementNames[zeroIndex].Value);
 | |
|             }
 | |
|             else
 | |
|                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(XmlObjectSerializer.CreateSerializationException(SR.GetString(SR.InvalidEnumValueOnWrite, value, DataContract.GetClrTypeFullName(UnderlyingType))));
 | |
|         }
 | |
| 
 | |
|         internal object ReadEnumValue(XmlReaderDelegator reader)
 | |
|         {
 | |
|             string stringValue = reader.ReadElementContentAsString();
 | |
|             long longValue = 0;
 | |
|             int i = 0;
 | |
|             if (IsFlags)
 | |
|             {
 | |
|                 // Skip initial spaces
 | |
|                 for (; i < stringValue.Length; i++)
 | |
|                     if (stringValue[i] != ' ')
 | |
|                         break;
 | |
| 
 | |
|                 // Read space-delimited values
 | |
|                 int startIndex = i;
 | |
|                 int count = 0;
 | |
|                 for (; i < stringValue.Length; i++)
 | |
|                 {
 | |
|                     if (stringValue[i] == ' ')
 | |
|                     {
 | |
|                         count = i - startIndex;
 | |
|                         if (count > 0)
 | |
|                             longValue |= ReadEnumValue(stringValue, startIndex, count);
 | |
|                         for (++i; i < stringValue.Length; i++)
 | |
|                             if (stringValue[i] != ' ')
 | |
|                                 break;
 | |
|                         startIndex = i;
 | |
|                         if (i == stringValue.Length)
 | |
|                             break;
 | |
|                     }
 | |
|                 }
 | |
|                 count = i - startIndex;
 | |
|                 if (count > 0)
 | |
|                     longValue |= ReadEnumValue(stringValue, startIndex, count);
 | |
|             }
 | |
|             else
 | |
|             {
 | |
|                 if (stringValue.Length == 0)
 | |
|                     throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(XmlObjectSerializer.CreateSerializationException(SR.GetString(SR.InvalidEnumValueOnRead, stringValue, DataContract.GetClrTypeFullName(UnderlyingType))));
 | |
|                 longValue = ReadEnumValue(stringValue, 0, stringValue.Length);
 | |
|             }
 | |
| 
 | |
|             if (IsULong)
 | |
|                 return Enum.ToObject(UnderlyingType, (ulong)longValue);
 | |
|             return Enum.ToObject(UnderlyingType, longValue);
 | |
|         }
 | |
| 
 | |
|         long ReadEnumValue(string value, int index, int count)
 | |
|         {
 | |
|             for (int i = 0; i < Members.Count; i++)
 | |
|             {
 | |
|                 string memberName = Members[i].Name;
 | |
|                 if (memberName.Length == count && String.CompareOrdinal(value, index, memberName, 0, count) == 0)
 | |
|                 {
 | |
|                     return Values[i];
 | |
|                 }
 | |
|             }
 | |
|             throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(XmlObjectSerializer.CreateSerializationException(SR.GetString(SR.InvalidEnumValueOnRead, value.Substring(index, count), DataContract.GetClrTypeFullName(UnderlyingType))));
 | |
|         }
 | |
| 
 | |
|         internal string GetStringFromEnumValue(long value)
 | |
|         {
 | |
|             if (IsULong)
 | |
|                 return XmlConvert.ToString((ulong)value);
 | |
|             else
 | |
|                 return XmlConvert.ToString(value);
 | |
|         }
 | |
| 
 | |
|         internal long GetEnumValueFromString(string value)
 | |
|         {
 | |
|             if (IsULong)
 | |
|                 return (long)XmlConverter.ToUInt64(value);
 | |
|             else
 | |
|                 return XmlConverter.ToInt64(value);
 | |
|         }
 | |
| 
 | |
|         internal override bool Equals(object other, Dictionary<DataContractPairKey, object> checkedContracts)
 | |
|         {
 | |
|             if (IsEqualOrChecked(other, checkedContracts))
 | |
|                 return true;
 | |
| 
 | |
|             if (base.Equals(other, null))
 | |
|             {
 | |
|                 EnumDataContract dataContract = other as EnumDataContract;
 | |
|                 if (dataContract != null)
 | |
|                 {
 | |
|                     if (Members.Count != dataContract.Members.Count || Values.Count != dataContract.Values.Count)
 | |
|                         return false;
 | |
|                     string[] memberNames1 = new string[Members.Count], memberNames2 = new string[Members.Count];
 | |
|                     for (int i = 0; i < Members.Count; i++)
 | |
|                     {
 | |
|                         memberNames1[i] = Members[i].Name;
 | |
|                         memberNames2[i] = dataContract.Members[i].Name;
 | |
|                     }
 | |
|                     Array.Sort(memberNames1);
 | |
|                     Array.Sort(memberNames2);
 | |
|                     for (int i = 0; i < Members.Count; i++)
 | |
|                     {
 | |
|                         if (memberNames1[i] != memberNames2[i])
 | |
|                             return false;
 | |
|                     }
 | |
| 
 | |
|                     return (IsFlags == dataContract.IsFlags);
 | |
|                 }
 | |
|             }
 | |
|             return false;
 | |
|         }
 | |
| 
 | |
|         public override int GetHashCode()
 | |
|         {
 | |
|             return base.GetHashCode();
 | |
|         }
 | |
| 
 | |
|         public override void WriteXmlValue(XmlWriterDelegator xmlWriter, object obj, XmlObjectSerializerWriteContext context)
 | |
|         {
 | |
|             WriteEnumValue(xmlWriter, obj);
 | |
|         }
 | |
| 
 | |
|         public override object ReadXmlValue(XmlReaderDelegator xmlReader, XmlObjectSerializerReadContext context)
 | |
|         {
 | |
|             object obj = ReadEnumValue(xmlReader);
 | |
|             if (context != null)
 | |
|                 context.AddNewObject(obj);
 | |
|             return obj;
 | |
|         }
 | |
| 
 | |
|     }
 | |
| }
 |