DataMember.cs 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349
  1. //-----------------------------------------------------------------------------
  2. // Copyright (c) Microsoft Corporation. All rights reserved.
  3. //-----------------------------------------------------------------------------
  4. namespace System.Runtime.Serialization
  5. {
  6. using System;
  7. using System.Collections.Generic;
  8. using System.Reflection;
  9. using System.Security;
  10. class DataMember
  11. {
  12. [Fx.Tag.SecurityNote(Critical = "Holds instance of CriticalHelper which keeps state that is cached statically for serialization."
  13. + " Static fields are marked SecurityCritical or readonly to prevent data from being modified or leaked to other components in appdomain.")]
  14. [SecurityCritical]
  15. CriticalHelper helper;
  16. [Fx.Tag.SecurityNote(Critical = "Initializes SecurityCritical field 'helper'.",
  17. Safe = "Doesn't leak anything.")]
  18. [SecuritySafeCritical]
  19. internal DataMember()
  20. {
  21. helper = new CriticalHelper();
  22. }
  23. [Fx.Tag.SecurityNote(Critical = "Initializes SecurityCritical field 'helper'.",
  24. Safe = "Doesn't leak anything.")]
  25. [SecuritySafeCritical]
  26. internal DataMember(MemberInfo memberInfo)
  27. {
  28. helper = new CriticalHelper(memberInfo);
  29. }
  30. [Fx.Tag.SecurityNote(Critical = "Initializes SecurityCritical field 'helper'.",
  31. Safe = "Doesn't leak anything.")]
  32. [SecuritySafeCritical]
  33. internal DataMember(string name)
  34. {
  35. helper = new CriticalHelper(name);
  36. }
  37. [Fx.Tag.SecurityNote(Critical = "Initializes SecurityCritical field 'helper'.",
  38. Safe = "Doesn't leak anything.")]
  39. [SecuritySafeCritical]
  40. internal DataMember(DataContract memberTypeContract, string name, bool isNullable, bool isRequired, bool emitDefaultValue, int order)
  41. {
  42. helper = new CriticalHelper(memberTypeContract, name, isNullable, isRequired, emitDefaultValue, order);
  43. }
  44. internal MemberInfo MemberInfo
  45. {
  46. [SecuritySafeCritical]
  47. get { return helper.MemberInfo; }
  48. }
  49. internal string Name
  50. {
  51. [SecuritySafeCritical]
  52. get { return helper.Name; }
  53. [SecurityCritical]
  54. set { helper.Name = value; }
  55. }
  56. internal int Order
  57. {
  58. [SecuritySafeCritical]
  59. get { return helper.Order; }
  60. [SecurityCritical]
  61. set { helper.Order = value; }
  62. }
  63. internal bool IsRequired
  64. {
  65. [SecuritySafeCritical]
  66. get { return helper.IsRequired; }
  67. [SecurityCritical]
  68. set { helper.IsRequired = value; }
  69. }
  70. internal bool EmitDefaultValue
  71. {
  72. [SecuritySafeCritical]
  73. get { return helper.EmitDefaultValue; }
  74. [SecurityCritical]
  75. set { helper.EmitDefaultValue = value; }
  76. }
  77. internal bool IsNullable
  78. {
  79. [SecuritySafeCritical]
  80. get { return helper.IsNullable; }
  81. [SecurityCritical]
  82. set { helper.IsNullable = value; }
  83. }
  84. internal bool IsGetOnlyCollection
  85. {
  86. [SecuritySafeCritical]
  87. get { return helper.IsGetOnlyCollection; }
  88. [SecurityCritical]
  89. set { helper.IsGetOnlyCollection = value; }
  90. }
  91. internal Type MemberType
  92. {
  93. [SecuritySafeCritical]
  94. get { return helper.MemberType; }
  95. }
  96. internal DataContract MemberTypeContract
  97. {
  98. [SecuritySafeCritical]
  99. get { return helper.MemberTypeContract; }
  100. [SecurityCritical]
  101. set { helper.MemberTypeContract = value; }
  102. }
  103. internal bool HasConflictingNameAndType
  104. {
  105. [SecuritySafeCritical]
  106. get { return helper.HasConflictingNameAndType; }
  107. [SecurityCritical]
  108. set { helper.HasConflictingNameAndType = value; }
  109. }
  110. internal DataMember ConflictingMember
  111. {
  112. [SecuritySafeCritical]
  113. get { return helper.ConflictingMember; }
  114. [SecurityCritical]
  115. set { helper.ConflictingMember = value; }
  116. }
  117. [Fx.Tag.SecurityNote(Critical = "Critical.")]
  118. #if !NO_SECURITY_ATTRIBUTES
  119. [SecurityCritical(SecurityCriticalScope.Everything)]
  120. #endif
  121. class CriticalHelper
  122. {
  123. DataContract memberTypeContract;
  124. string name;
  125. int order;
  126. bool isRequired;
  127. bool emitDefaultValue;
  128. bool isNullable;
  129. bool isGetOnlyCollection = false;
  130. MemberInfo memberInfo;
  131. bool hasConflictingNameAndType;
  132. DataMember conflictingMember;
  133. internal CriticalHelper()
  134. {
  135. this.emitDefaultValue = Globals.DefaultEmitDefaultValue;
  136. }
  137. internal CriticalHelper(MemberInfo memberInfo)
  138. {
  139. this.emitDefaultValue = Globals.DefaultEmitDefaultValue;
  140. this.memberInfo = memberInfo;
  141. }
  142. internal CriticalHelper(string name)
  143. {
  144. this.Name = name;
  145. }
  146. internal CriticalHelper(DataContract memberTypeContract, string name, bool isNullable, bool isRequired, bool emitDefaultValue, int order)
  147. {
  148. this.MemberTypeContract = memberTypeContract;
  149. this.Name = name;
  150. this.IsNullable = isNullable;
  151. this.IsRequired = isRequired;
  152. this.EmitDefaultValue = emitDefaultValue;
  153. this.Order = order;
  154. }
  155. internal MemberInfo MemberInfo
  156. {
  157. get { return memberInfo; }
  158. }
  159. internal string Name
  160. {
  161. get { return name; }
  162. set { name = value; }
  163. }
  164. internal int Order
  165. {
  166. get { return order; }
  167. set { order = value; }
  168. }
  169. internal bool IsRequired
  170. {
  171. get { return isRequired; }
  172. set { isRequired = value; }
  173. }
  174. internal bool EmitDefaultValue
  175. {
  176. get { return emitDefaultValue; }
  177. set { emitDefaultValue = value; }
  178. }
  179. internal bool IsNullable
  180. {
  181. get { return isNullable; }
  182. set { isNullable = value; }
  183. }
  184. internal bool IsGetOnlyCollection
  185. {
  186. get { return isGetOnlyCollection; }
  187. set { isGetOnlyCollection = value; }
  188. }
  189. internal Type MemberType
  190. {
  191. get
  192. {
  193. FieldInfo field = MemberInfo as FieldInfo;
  194. if (field != null)
  195. return field.FieldType;
  196. return ((PropertyInfo)MemberInfo).PropertyType;
  197. }
  198. }
  199. internal DataContract MemberTypeContract
  200. {
  201. get
  202. {
  203. if (memberTypeContract == null)
  204. {
  205. if (MemberInfo != null)
  206. {
  207. if (this.IsGetOnlyCollection)
  208. {
  209. memberTypeContract = DataContract.GetGetOnlyCollectionDataContract(DataContract.GetId(MemberType.TypeHandle), MemberType.TypeHandle, MemberType, SerializationMode.SharedContract);
  210. }
  211. else
  212. {
  213. memberTypeContract = DataContract.GetDataContract(MemberType);
  214. }
  215. }
  216. }
  217. return memberTypeContract;
  218. }
  219. set
  220. {
  221. memberTypeContract = value;
  222. }
  223. }
  224. internal bool HasConflictingNameAndType
  225. {
  226. get { return this.hasConflictingNameAndType; }
  227. set { this.hasConflictingNameAndType = value; }
  228. }
  229. internal DataMember ConflictingMember
  230. {
  231. get { return this.conflictingMember; }
  232. set { this.conflictingMember = value; }
  233. }
  234. }
  235. #if !NO_DYNAMIC_CODEGEN
  236. [Fx.Tag.SecurityNote(Miscellaneous = "RequiresReview - checks member visibility to calculate if access to it requires MemberAccessPermission for serialization."
  237. + " Since this information is used to determine whether to give the generated code access"
  238. + " permissions to private members, any changes to the logic should be reviewed.")]
  239. internal bool RequiresMemberAccessForGet()
  240. {
  241. MemberInfo memberInfo = MemberInfo;
  242. FieldInfo field = memberInfo as FieldInfo;
  243. if (field != null)
  244. {
  245. return DataContract.FieldRequiresMemberAccess(field);
  246. }
  247. else
  248. {
  249. PropertyInfo property = (PropertyInfo)memberInfo;
  250. MethodInfo getMethod = property.GetGetMethod(true /*nonPublic*/);
  251. if (getMethod != null)
  252. return DataContract.MethodRequiresMemberAccess(getMethod) || !DataContract.IsTypeVisible(property.PropertyType);
  253. }
  254. return false;
  255. }
  256. [Fx.Tag.SecurityNote(Miscellaneous = "RequiresReview - checks member visibility to calculate if access to it requires MemberAccessPermission for deserialization."
  257. + " Since this information is used to determine whether to give the generated code access"
  258. + " permissions to private members, any changes to the logic should be reviewed.")]
  259. internal bool RequiresMemberAccessForSet()
  260. {
  261. MemberInfo memberInfo = MemberInfo;
  262. FieldInfo field = memberInfo as FieldInfo;
  263. if (field != null)
  264. {
  265. return DataContract.FieldRequiresMemberAccess(field);
  266. }
  267. else
  268. {
  269. PropertyInfo property = (PropertyInfo)memberInfo;
  270. MethodInfo setMethod = property.GetSetMethod(true /*nonPublic*/);
  271. if (setMethod != null)
  272. return DataContract.MethodRequiresMemberAccess(setMethod) || !DataContract.IsTypeVisible(property.PropertyType);
  273. }
  274. return false;
  275. }
  276. #endif
  277. internal DataMember BindGenericParameters(DataContract[] paramContracts, Dictionary<DataContract, DataContract> boundContracts)
  278. {
  279. DataContract memberTypeContract = this.MemberTypeContract.BindGenericParameters(paramContracts, boundContracts);
  280. DataMember boundDataMember = new DataMember(memberTypeContract,
  281. this.Name,
  282. !memberTypeContract.IsValueType,
  283. this.IsRequired,
  284. this.EmitDefaultValue,
  285. this.Order);
  286. return boundDataMember;
  287. }
  288. internal bool Equals(object other, Dictionary<DataContractPairKey, object> checkedContracts)
  289. {
  290. if ((object)this == other)
  291. return true;
  292. DataMember dataMember = other as DataMember;
  293. if (dataMember != null)
  294. {
  295. // Note: comparison does not use Order hint since it influences element order but does not specify exact order
  296. bool thisIsNullable = (MemberTypeContract == null) ? false : !MemberTypeContract.IsValueType;
  297. bool dataMemberIsNullable = (dataMember.MemberTypeContract == null) ? false : !dataMember.MemberTypeContract.IsValueType;
  298. return (Name == dataMember.Name
  299. && (IsNullable || thisIsNullable) == (dataMember.IsNullable || dataMemberIsNullable)
  300. && IsRequired == dataMember.IsRequired
  301. && EmitDefaultValue == dataMember.EmitDefaultValue
  302. && MemberTypeContract.Equals(dataMember.MemberTypeContract, checkedContracts));
  303. }
  304. return false;
  305. }
  306. public override int GetHashCode()
  307. {
  308. return base.GetHashCode();
  309. }
  310. }
  311. }