SurrogateDataContract.cs 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127
  1. //-----------------------------------------------------------------------------
  2. // Copyright (c) Microsoft Corporation. All rights reserved.
  3. //-----------------------------------------------------------------------------
  4. namespace System.Runtime.Serialization
  5. {
  6. using System;
  7. using System.Security;
  8. using System.Security.Permissions;
  9. using System.Runtime.CompilerServices;
  10. #if USE_REFEMIT
  11. public sealed class SurrogateDataContract : DataContract
  12. #else
  13. internal sealed class SurrogateDataContract : DataContract
  14. #endif
  15. {
  16. [Fx.Tag.SecurityNote(Critical = "Holds instance of CriticalHelper which keeps state that is cached statically for serialization."
  17. + " Static fields are marked SecurityCritical or readonly to prevent data from being modified or leaked to other components in appdomain.")]
  18. [SecurityCritical]
  19. SurrogateDataContractCriticalHelper helper;
  20. [Fx.Tag.SecurityNote(Critical = "Initializes SecurityCritical field 'helper'.",
  21. Safe = "Doesn't leak anything.")]
  22. [SecuritySafeCritical]
  23. internal SurrogateDataContract(Type type, ISerializationSurrogate serializationSurrogate)
  24. : base(new SurrogateDataContractCriticalHelper(type, serializationSurrogate))
  25. {
  26. helper = base.Helper as SurrogateDataContractCriticalHelper;
  27. }
  28. internal ISerializationSurrogate SerializationSurrogate
  29. {
  30. [Fx.Tag.SecurityNote(Critical = "Fetches the critical serializationSurrogate property.",
  31. Safe = "serializationSurrogate only needs to be protected for write.")]
  32. [SecuritySafeCritical]
  33. get { return helper.SerializationSurrogate; }
  34. }
  35. public override void WriteXmlValue(XmlWriterDelegator xmlWriter, object obj, XmlObjectSerializerWriteContext context)
  36. {
  37. SerializationInfo serInfo = new SerializationInfo(UnderlyingType, XmlObjectSerializer.FormatterConverter, !context.UnsafeTypeForwardingEnabled);
  38. SerializationSurrogateGetObjectData(obj, serInfo, context.GetStreamingContext());
  39. context.WriteSerializationInfo(xmlWriter, UnderlyingType, serInfo);
  40. }
  41. [Fx.Tag.SecurityNote(Critical = "Calls the critical methods of ISurrogateSelector", Safe = "Demands for FullTrust")]
  42. [SecuritySafeCritical]
  43. [PermissionSet(SecurityAction.Demand, Unrestricted = true)]
  44. [MethodImpl(MethodImplOptions.NoInlining)]
  45. object SerializationSurrogateSetObjectData(object obj, SerializationInfo serInfo, StreamingContext context)
  46. {
  47. return SerializationSurrogate.SetObjectData(obj, serInfo, context, null);
  48. }
  49. [Fx.Tag.SecurityNote(Critical = "Calls the critical methods of IObjectReference", Safe = "Demands for FullTrust")]
  50. [SecuritySafeCritical]
  51. [PermissionSet(SecurityAction.Demand, Unrestricted = true)]
  52. [MethodImpl(MethodImplOptions.NoInlining)]
  53. internal static object GetRealObject(IObjectReference obj, StreamingContext context)
  54. {
  55. return obj.GetRealObject(context);
  56. }
  57. [Fx.Tag.SecurityNote(Critical = "Calls the critical methods of FormatterServices", Safe = "Demands for FullTrust")]
  58. [SecuritySafeCritical]
  59. [PermissionSet(SecurityAction.Demand, Unrestricted = true)]
  60. [MethodImpl(MethodImplOptions.NoInlining)]
  61. object GetUninitializedObject(Type objType)
  62. {
  63. return FormatterServices.GetUninitializedObject(objType);
  64. }
  65. [Fx.Tag.SecurityNote(Critical = "Calls the critical methods of ISerializationSurrogate", Safe = "Demands for FullTrust")]
  66. [SecuritySafeCritical]
  67. [PermissionSet(SecurityAction.Demand, Unrestricted = true)]
  68. [MethodImpl(MethodImplOptions.NoInlining)]
  69. void SerializationSurrogateGetObjectData(object obj, SerializationInfo serInfo, StreamingContext context)
  70. {
  71. SerializationSurrogate.GetObjectData(obj, serInfo, context);
  72. }
  73. public override object ReadXmlValue(XmlReaderDelegator xmlReader, XmlObjectSerializerReadContext context)
  74. {
  75. xmlReader.Read();
  76. Type objType = UnderlyingType;
  77. object obj = objType.IsArray ? Array.CreateInstance(objType.GetElementType(), 0) : GetUninitializedObject(objType);
  78. context.AddNewObject(obj);
  79. string objectId = context.GetObjectId();
  80. SerializationInfo serInfo = context.ReadSerializationInfo(xmlReader, objType);
  81. object newObj = SerializationSurrogateSetObjectData(obj, serInfo, context.GetStreamingContext());
  82. if (newObj == null)
  83. newObj = obj;
  84. if (newObj is IDeserializationCallback)
  85. ((IDeserializationCallback)newObj).OnDeserialization(null);
  86. if (newObj is IObjectReference)
  87. newObj = GetRealObject((IObjectReference)newObj, context.GetStreamingContext());
  88. context.ReplaceDeserializedObject(objectId, obj, newObj);
  89. xmlReader.ReadEndElement();
  90. return newObj;
  91. }
  92. [Fx.Tag.SecurityNote(Critical = "Holds all state used for for (de)serializing with ISerializationSurrogate."
  93. + " Since it accesses data on the base type that is cached statically, we lock down access to it.")]
  94. #if !NO_SECURITY_ATTRIBUTES
  95. [SecurityCritical(SecurityCriticalScope.Everything)]
  96. #endif
  97. class SurrogateDataContractCriticalHelper : DataContract.DataContractCriticalHelper
  98. {
  99. ISerializationSurrogate serializationSurrogate;
  100. internal SurrogateDataContractCriticalHelper(Type type, ISerializationSurrogate serializationSurrogate)
  101. : base(type)
  102. {
  103. this.serializationSurrogate = serializationSurrogate;
  104. string name, ns;
  105. DataContract.GetDefaultStableName(DataContract.GetClrTypeFullName(type), out name, out ns);
  106. SetDataContractName(CreateQualifiedName(name, ns));
  107. }
  108. internal ISerializationSurrogate SerializationSurrogate
  109. {
  110. get { return serializationSurrogate; }
  111. }
  112. }
  113. }
  114. }