ObjectManager.cs 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548
  1. //
  2. // System.Runtime.Serialization.ObjectIDGenerator.cs
  3. //
  4. // Author: Lluis Sanchez Gual ([email protected])
  5. //
  6. // (C) 2003 Lluis Sanchez Gual
  7. //
  8. using System;
  9. using System.Collections;
  10. using System.Reflection;
  11. namespace System.Runtime.Serialization
  12. {
  13. public class ObjectManager
  14. {
  15. // All objects are chained in the same order as they have been registered
  16. ObjectRecord _objectRecordChain = null;
  17. ObjectRecord _lastObjectRecord = null;
  18. ArrayList _deserializedRecords = new ArrayList();
  19. Hashtable _objectRecords = new Hashtable();
  20. bool _finalFixup = false;
  21. ISurrogateSelector _selector;
  22. StreamingContext _context;
  23. int _registeredObjectsCount = 0;
  24. public ObjectManager(ISurrogateSelector selector, StreamingContext context)
  25. {
  26. _selector = selector;
  27. _context = context;
  28. }
  29. public virtual void DoFixups()
  30. {
  31. _finalFixup = true;
  32. try
  33. {
  34. if (_registeredObjectsCount < _objectRecords.Count)
  35. throw new SerializationException ("There are some fixups that refer to objects that have not been registered");
  36. ObjectRecord last = _lastObjectRecord;
  37. bool firstCicle = true;
  38. // Solve al pending fixups of all objects
  39. ObjectRecord record = _objectRecordChain;
  40. while (record != null)
  41. {
  42. bool ready = !(record.IsUnsolvedObjectReference && firstCicle);
  43. if (ready) ready = record.DoFixups (true, this, true);
  44. if (ready) ready = record.LoadData(this, _selector, _context);
  45. ObjectRecord next;
  46. if (ready)
  47. {
  48. if (record.OriginalObject is IDeserializationCallback)
  49. _deserializedRecords.Add (record);
  50. next = record.Next;
  51. }
  52. else
  53. {
  54. // There must be an unresolved IObjectReference instance.
  55. // Chain the record at the end so it is solved later
  56. if ((record.ObjectInstance is IObjectReference) && !firstCicle)
  57. {
  58. if (record.Status == ObjectRecordStatus.ReferenceSolvingDelayed)
  59. throw new SerializationException ("The object with ID " + record.ObjectID + " could not be resolved");
  60. else
  61. record.Status = ObjectRecordStatus.ReferenceSolvingDelayed;
  62. }
  63. if (record != _lastObjectRecord) {
  64. next = record.Next;
  65. record.Next = null;
  66. _lastObjectRecord.Next = record;
  67. _lastObjectRecord = record;
  68. }
  69. else
  70. next = record;
  71. }
  72. if (record == last) firstCicle = false;
  73. record = next;
  74. }
  75. }
  76. finally
  77. {
  78. _finalFixup = false;
  79. }
  80. }
  81. internal ObjectRecord GetObjectRecord (long objectID)
  82. {
  83. ObjectRecord rec = (ObjectRecord)_objectRecords[objectID];
  84. if (rec == null)
  85. {
  86. if (_finalFixup) throw new SerializationException ("The object with Id " + objectID + " has not been registered");
  87. rec = new ObjectRecord();
  88. rec.ObjectID = objectID;
  89. _objectRecords[objectID] = rec;
  90. }
  91. if (!rec.IsRegistered && _finalFixup) throw new SerializationException ("The object with Id " + objectID + " has not been registered");
  92. return rec;
  93. }
  94. public virtual object GetObject (long objectID)
  95. {
  96. if (objectID <= 0) throw new ArgumentOutOfRangeException("objectID","The objectID parameter is less than or equal to zero");
  97. ObjectRecord rec = (ObjectRecord)_objectRecords[objectID];
  98. if (rec == null || !rec.IsRegistered) return null;
  99. else return rec.ObjectInstance;
  100. }
  101. public virtual void RaiseDeserializationEvent ()
  102. {
  103. for (int i = _deserializedRecords.Count-1; i >= 0; i--)
  104. {
  105. ObjectRecord record = (ObjectRecord) _deserializedRecords [i];
  106. IDeserializationCallback obj = record.OriginalObject as IDeserializationCallback;
  107. if (obj != null) obj.OnDeserialization (this);
  108. }
  109. }
  110. private void AddFixup (BaseFixupRecord record)
  111. {
  112. record.ObjectToBeFixed.ChainFixup (record, true);
  113. record.ObjectRequired.ChainFixup (record, false);
  114. }
  115. public virtual void RecordArrayElementFixup (long arrayToBeFixed, int index, long objectRequired)
  116. {
  117. if (arrayToBeFixed <= 0) throw new ArgumentOutOfRangeException("arrayToBeFixed","The arrayToBeFixed parameter is less than or equal to zero");
  118. if (objectRequired <= 0) throw new ArgumentOutOfRangeException("objectRequired","The objectRequired parameter is less than or equal to zero");
  119. ArrayFixupRecord record = new ArrayFixupRecord(GetObjectRecord(arrayToBeFixed), index, GetObjectRecord(objectRequired));
  120. AddFixup (record);
  121. }
  122. public virtual void RecordArrayElementFixup (long arrayToBeFixed, int[] indices, long objectRequired)
  123. {
  124. if (arrayToBeFixed <= 0) throw new ArgumentOutOfRangeException("arrayToBeFixed","The arrayToBeFixed parameter is less than or equal to zero");
  125. if (objectRequired <= 0) throw new ArgumentOutOfRangeException("objectRequired","The objectRequired parameter is less than or equal to zero");
  126. if (indices == null) throw new ArgumentNullException("indices");
  127. MultiArrayFixupRecord record = new MultiArrayFixupRecord (GetObjectRecord(arrayToBeFixed), indices, GetObjectRecord(objectRequired));
  128. AddFixup (record);
  129. }
  130. public virtual void RecordDelayedFixup (long objectToBeFixed, string memberName, long objectRequired)
  131. {
  132. if (objectToBeFixed <= 0) throw new ArgumentOutOfRangeException("objectToBeFixed","The objectToBeFixed parameter is less than or equal to zero");
  133. if (objectRequired <= 0) throw new ArgumentOutOfRangeException("objectRequired","The objectRequired parameter is less than or equal to zero");
  134. if (memberName == null) throw new ArgumentNullException("memberName");
  135. DelayedFixupRecord record = new DelayedFixupRecord (GetObjectRecord(objectToBeFixed), memberName, GetObjectRecord(objectRequired));
  136. AddFixup (record);
  137. }
  138. public virtual void RecordFixup (long objectToBeFixed, MemberInfo member, long objectRequired)
  139. {
  140. if (objectToBeFixed <= 0) throw new ArgumentOutOfRangeException("objectToBeFixed","The objectToBeFixed parameter is less than or equal to zero");
  141. if (objectRequired <= 0) throw new ArgumentOutOfRangeException("objectRequired","The objectRequired parameter is less than or equal to zero");
  142. if (member == null) throw new ArgumentNullException("member");
  143. FixupRecord record = new FixupRecord (GetObjectRecord(objectToBeFixed), member, GetObjectRecord(objectRequired));
  144. AddFixup (record);
  145. }
  146. private void RegisterObjectInternal (object obj, ObjectRecord record)
  147. {
  148. if (obj == null) throw new ArgumentNullException("obj");
  149. if (record.IsRegistered)
  150. {
  151. if (record.OriginalObject != obj) throw new SerializationException ("An object with Id " + record.ObjectID + " has already been registered");
  152. else return;
  153. }
  154. record.ObjectInstance = obj;
  155. record.OriginalObject = obj;
  156. if (obj is IObjectReference) record.Status = ObjectRecordStatus.ReferenceUnsolved;
  157. else record.Status = ObjectRecordStatus.ReferenceSolved;
  158. record.DoFixups (true, this, false);
  159. record.DoFixups (false, this, false);
  160. _registeredObjectsCount++;
  161. // Adds the object to the chain of registered objects. This chain
  162. // is needed to be able to to perform the final fixups in the right order
  163. if (_objectRecordChain == null)
  164. {
  165. _objectRecordChain = record;
  166. _lastObjectRecord = record;
  167. }
  168. else
  169. {
  170. _lastObjectRecord.Next = record;
  171. _lastObjectRecord = record;
  172. }
  173. }
  174. public virtual void RegisterObject (object obj, long objectID)
  175. {
  176. RegisterObjectInternal (obj, GetObjectRecord (objectID));
  177. }
  178. public void RegisterObject (object obj, long objectID, SerializationInfo info)
  179. {
  180. if (objectID <= 0) throw new ArgumentOutOfRangeException("objectID","The objectID parameter is less than or equal to zero");
  181. ObjectRecord record = GetObjectRecord (objectID);
  182. record.Info = info;
  183. RegisterObjectInternal (obj, record);
  184. }
  185. public void RegisterObject (object obj, long objectID, SerializationInfo info, long idOfContainingObj, MemberInfo member)
  186. {
  187. RegisterObject (obj, objectID, info, idOfContainingObj, member, null);
  188. }
  189. public void RegisterObject( object obj, long objectID, SerializationInfo info, long idOfContainingObj, MemberInfo member, int[] arrayIndex)
  190. {
  191. if (objectID <= 0) throw new ArgumentOutOfRangeException("objectID","The objectID parameter is less than or equal to zero");
  192. ObjectRecord record = GetObjectRecord (objectID);
  193. record.Info = info;
  194. record.IdOfContainingObj = idOfContainingObj;
  195. record.Member = member;
  196. record.ArrayIndex = arrayIndex;
  197. RegisterObjectInternal (obj, record);
  198. }
  199. }
  200. // Fixup types. There is a fixup class for each fixup type.
  201. // BaseFixupRecord
  202. // Base class for all fixups
  203. internal abstract class BaseFixupRecord
  204. {
  205. public BaseFixupRecord(ObjectRecord objectToBeFixed, ObjectRecord objectRequired)
  206. {
  207. ObjectToBeFixed = objectToBeFixed;
  208. ObjectRequired = objectRequired;
  209. }
  210. public bool DoFixup (ObjectManager manager, bool strict)
  211. {
  212. if (ObjectToBeFixed.IsRegistered && ObjectRequired.IsInstanceReady)
  213. {
  214. FixupImpl (manager);
  215. return true;
  216. }
  217. else if (strict)
  218. {
  219. if (!ObjectToBeFixed.IsRegistered) throw new SerializationException ("An object with ID " + ObjectToBeFixed.ObjectID + " was included in a fixup, but it has not been registered");
  220. else if (!ObjectRequired.IsRegistered) throw new SerializationException ("An object with ID " + ObjectRequired.ObjectID + " was included in a fixup, but it has not been registered");
  221. else return false;
  222. }
  223. else
  224. return false;
  225. }
  226. protected abstract void FixupImpl (ObjectManager manager);
  227. internal protected ObjectRecord ObjectToBeFixed;
  228. internal protected ObjectRecord ObjectRequired;
  229. public BaseFixupRecord NextSameContainer;
  230. public BaseFixupRecord NextSameRequired;
  231. }
  232. // ArrayFixupRecord
  233. // Fixup for assigning a value to one position of an array
  234. internal class ArrayFixupRecord : BaseFixupRecord
  235. {
  236. int _index;
  237. public ArrayFixupRecord (ObjectRecord objectToBeFixed, int index, ObjectRecord objectRequired): base (objectToBeFixed, objectRequired) {
  238. _index = index;
  239. }
  240. protected override void FixupImpl (ObjectManager manager) {
  241. Array array = (Array)ObjectToBeFixed.ObjectInstance;
  242. array.SetValue (ObjectRequired.ObjectInstance, _index);
  243. }
  244. }
  245. // MultiArrayFixupRecord
  246. // Fixup for assigning a value to several positions of an array
  247. internal class MultiArrayFixupRecord : BaseFixupRecord
  248. {
  249. int[] _indices;
  250. public MultiArrayFixupRecord (ObjectRecord objectToBeFixed, int[] indices, ObjectRecord objectRequired): base (objectToBeFixed, objectRequired) {
  251. _indices = indices;
  252. }
  253. protected override void FixupImpl (ObjectManager manager) {
  254. ObjectToBeFixed.SetArrayValue (manager, ObjectRequired.ObjectInstance, _indices);
  255. }
  256. }
  257. // FixupRecord
  258. // Fixup for assigning a value to a member of an object
  259. internal class FixupRecord: BaseFixupRecord
  260. {
  261. public MemberInfo _member;
  262. public FixupRecord (ObjectRecord objectToBeFixed, MemberInfo member, ObjectRecord objectRequired): base (objectToBeFixed, objectRequired) {
  263. _member = member;
  264. }
  265. protected override void FixupImpl (ObjectManager manager) {
  266. ObjectToBeFixed.SetMemberValue (manager, _member, ObjectRequired.ObjectInstance);
  267. }
  268. }
  269. // DelayedFixupRecord
  270. // Fixup for assigning a value to a SerializationInfo of an object
  271. internal class DelayedFixupRecord: BaseFixupRecord
  272. {
  273. public string _memberName;
  274. public DelayedFixupRecord (ObjectRecord objectToBeFixed, string memberName, ObjectRecord objectRequired): base (objectToBeFixed, objectRequired) {
  275. _memberName = memberName;
  276. }
  277. protected override void FixupImpl (ObjectManager manager) {
  278. ObjectToBeFixed.SetMemberValue (manager, _memberName, ObjectRequired.ObjectInstance);
  279. }
  280. }
  281. // Object Record
  282. public enum ObjectRecordStatus: byte { Unregistered, ReferenceUnsolved, ReferenceSolvingDelayed, ReferenceSolved }
  283. internal class ObjectRecord
  284. {
  285. public ObjectRecordStatus Status = ObjectRecordStatus.Unregistered;
  286. public object OriginalObject;
  287. public object ObjectInstance;
  288. public long ObjectID;
  289. public SerializationInfo Info;
  290. public long IdOfContainingObj;
  291. public MemberInfo Member;
  292. public int[] ArrayIndex;
  293. public BaseFixupRecord FixupChainAsContainer;
  294. public BaseFixupRecord FixupChainAsRequired;
  295. public ObjectRecord Next;
  296. public void SetMemberValue (ObjectManager manager, MemberInfo member, object value)
  297. {
  298. if (member is FieldInfo)
  299. ((FieldInfo)member).SetValue (ObjectInstance, value);
  300. else if (member is PropertyInfo)
  301. ((PropertyInfo)member).SetValue (ObjectInstance, value, null);
  302. else throw new SerializationException ("Cannot perform fixup");
  303. if (Member != null)
  304. {
  305. ObjectRecord containerRecord = manager.GetObjectRecord (IdOfContainingObj);
  306. if (containerRecord.IsRegistered)
  307. containerRecord.SetMemberValue (manager, Member, ObjectInstance);
  308. }
  309. else if (ArrayIndex != null)
  310. {
  311. ObjectRecord containerRecord = manager.GetObjectRecord (IdOfContainingObj);
  312. if (containerRecord.IsRegistered)
  313. containerRecord.SetArrayValue (manager, ObjectInstance, ArrayIndex);
  314. }
  315. }
  316. public void SetArrayValue (ObjectManager manager, object value, int[] indices)
  317. {
  318. ((Array)ObjectInstance).SetValue (value, indices);
  319. }
  320. public void SetMemberValue (ObjectManager manager, string memberName, object value)
  321. {
  322. if (Info == null) throw new SerializationException ("Cannot perform fixup");
  323. Info.AddValue (memberName, value, value.GetType());
  324. }
  325. public bool IsInstanceReady
  326. {
  327. // Returns true if this object is ready to be assigned to a parent object.
  328. get
  329. {
  330. if (!IsRegistered) return false;
  331. if (IsUnsolvedObjectReference) return false;
  332. // Embedded value objects cannot be assigned to their containers until fully completed
  333. if (Member != null && (HasPendingFixups || Info != null))
  334. return false;
  335. return true;
  336. }
  337. }
  338. public bool IsUnsolvedObjectReference
  339. {
  340. get {
  341. return Status != ObjectRecordStatus.ReferenceSolved;
  342. }
  343. }
  344. public bool IsRegistered
  345. {
  346. get {
  347. return Status != ObjectRecordStatus.Unregistered;
  348. }
  349. }
  350. public bool DoFixups (bool asContainer, ObjectManager manager, bool strict)
  351. {
  352. BaseFixupRecord prevFixup = null;
  353. BaseFixupRecord fixup = asContainer ? FixupChainAsContainer : FixupChainAsRequired;
  354. bool allFixed = true;
  355. while (fixup != null)
  356. {
  357. if (fixup.DoFixup (manager, strict))
  358. {
  359. UnchainFixup (fixup, prevFixup, asContainer);
  360. if (asContainer) fixup.ObjectRequired.RemoveFixup (fixup, false);
  361. else fixup.ObjectToBeFixed.RemoveFixup (fixup, true);
  362. }
  363. else
  364. {
  365. prevFixup = fixup;
  366. allFixed = false;
  367. }
  368. fixup = asContainer ? fixup.NextSameContainer : fixup.NextSameRequired;
  369. }
  370. return allFixed;
  371. }
  372. public void RemoveFixup (BaseFixupRecord fixupToRemove, bool asContainer)
  373. {
  374. BaseFixupRecord prevFixup = null;
  375. BaseFixupRecord fixup = asContainer ? FixupChainAsContainer : FixupChainAsRequired;
  376. while (fixup != null)
  377. {
  378. if (fixup == fixupToRemove)
  379. {
  380. UnchainFixup (fixup, prevFixup, asContainer);
  381. return;
  382. }
  383. prevFixup = fixup;
  384. fixup = asContainer ? fixup.NextSameContainer : fixup.NextSameRequired;
  385. }
  386. }
  387. private void UnchainFixup (BaseFixupRecord fixup, BaseFixupRecord prevFixup, bool asContainer)
  388. {
  389. if (prevFixup == null) {
  390. if (asContainer) FixupChainAsContainer = fixup.NextSameContainer;
  391. else FixupChainAsRequired = fixup.NextSameRequired;
  392. }
  393. else {
  394. if (asContainer) prevFixup.NextSameContainer = fixup.NextSameContainer;
  395. else prevFixup.NextSameRequired = fixup.NextSameRequired;
  396. }
  397. }
  398. public void ChainFixup (BaseFixupRecord fixup, bool asContainer)
  399. {
  400. if (asContainer)
  401. {
  402. fixup.NextSameContainer = FixupChainAsContainer;
  403. FixupChainAsContainer = fixup;
  404. }
  405. else
  406. {
  407. fixup.NextSameRequired = FixupChainAsRequired;
  408. FixupChainAsRequired = fixup;
  409. }
  410. }
  411. public bool LoadData (ObjectManager manager, ISurrogateSelector selector, StreamingContext context)
  412. {
  413. if (Info != null)
  414. {
  415. if (ObjectInstance is ISerializable)
  416. {
  417. object[] pars = new object[] {Info, context};
  418. ConstructorInfo con = ObjectInstance.GetType().GetConstructor (BindingFlags.Instance | BindingFlags.Public | BindingFlags.NonPublic, null, new Type[] { typeof (SerializationInfo), typeof (StreamingContext) }, null );
  419. if (con == null) throw new SerializationException ("The constructor to deserialize an object of type " + ObjectInstance.GetType().FullName + " was not found.");
  420. con.Invoke (ObjectInstance, pars);
  421. }
  422. else
  423. {
  424. ISurrogateSelector foundSelector;
  425. ISerializationSurrogate surrogate = selector.GetSurrogate (ObjectInstance.GetType(), context, out foundSelector);
  426. if (surrogate == null) throw new SerializationException ("No surrogate selector was found for type " + ObjectInstance.GetType().FullName);
  427. surrogate.SetObjectData (ObjectInstance, Info, context, foundSelector);
  428. }
  429. Info = null;
  430. }
  431. if (ObjectInstance is IObjectReference && Status != ObjectRecordStatus.ReferenceSolved)
  432. {
  433. try {
  434. ObjectInstance = ((IObjectReference)ObjectInstance).GetRealObject(context);
  435. Status = ObjectRecordStatus.ReferenceSolved;
  436. }
  437. catch (NullReferenceException) {
  438. // Give a second chance
  439. return false;
  440. }
  441. }
  442. if (Member != null)
  443. {
  444. // If this object is a value object embedded in another object, the parent
  445. // object must be updated
  446. ObjectRecord containerRecord = manager.GetObjectRecord (IdOfContainingObj);
  447. containerRecord.SetMemberValue (manager, Member, ObjectInstance);
  448. }
  449. else if (ArrayIndex != null)
  450. {
  451. ObjectRecord containerRecord = manager.GetObjectRecord (IdOfContainingObj);
  452. containerRecord.SetArrayValue (manager, ObjectInstance, ArrayIndex);
  453. }
  454. return true;
  455. }
  456. public bool HasPendingFixups
  457. {
  458. get { return FixupChainAsContainer != null; }
  459. }
  460. }
  461. }