BsManagedSerializableObject.cpp 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211
  1. //********************************** Banshee Engine (www.banshee3d.com) **************************************************//
  2. //**************** Copyright (c) 2016 Marko Pintera ([email protected]). All rights reserved. **********************//
  3. #include "Serialization/BsManagedSerializableObject.h"
  4. #include "RTTI/BsManagedSerializableObjectRTTI.h"
  5. #include "Serialization/BsManagedSerializableObjectInfo.h"
  6. #include "Serialization/BsManagedSerializableField.h"
  7. #include "Serialization/BsScriptAssemblyManager.h"
  8. #include "BsMonoField.h"
  9. #include "BsMonoClass.h"
  10. #include "BsMonoUtil.h"
  11. namespace bs
  12. {
  13. inline size_t ManagedSerializableObject::Hash::operator()(const ManagedSerializableFieldKey& x) const
  14. {
  15. size_t seed = 0;
  16. hash_combine(seed, (UINT32)x.mFieldId);
  17. hash_combine(seed, (UINT32)x.mTypeId);
  18. return seed;
  19. }
  20. inline bool ManagedSerializableObject::Equals::operator()(const ManagedSerializableFieldKey& a, const ManagedSerializableFieldKey& b) const
  21. {
  22. return a.mFieldId == b.mFieldId && a.mTypeId == b.mTypeId;
  23. }
  24. ManagedSerializableObject::ManagedSerializableObject(const ConstructPrivately& dummy)
  25. :mManagedInstance(nullptr)
  26. {
  27. }
  28. ManagedSerializableObject::ManagedSerializableObject(const ConstructPrivately& dummy, SPtr<ManagedSerializableObjectInfo> objInfo, MonoObject* managedInstance)
  29. :mManagedInstance(managedInstance), mObjInfo(objInfo)
  30. {
  31. }
  32. SPtr<ManagedSerializableObject> ManagedSerializableObject::createFromExisting(MonoObject* managedInstance)
  33. {
  34. if(managedInstance == nullptr)
  35. return nullptr;
  36. String elementNs;
  37. String elementTypeName;
  38. MonoUtil::getClassName(managedInstance, elementNs, elementTypeName);
  39. SPtr<ManagedSerializableObjectInfo> objInfo;
  40. if(!ScriptAssemblyManager::instance().getSerializableObjectInfo(elementNs, elementTypeName, objInfo))
  41. return nullptr;
  42. return bs_shared_ptr_new<ManagedSerializableObject>(ConstructPrivately(), objInfo, managedInstance);
  43. }
  44. SPtr<ManagedSerializableObject> ManagedSerializableObject::createNew(const SPtr<ManagedSerializableTypeInfoObject>& type)
  45. {
  46. SPtr<ManagedSerializableObjectInfo> currentObjInfo = nullptr;
  47. // See if this type even still exists
  48. if (!ScriptAssemblyManager::instance().getSerializableObjectInfo(type->mTypeNamespace, type->mTypeName, currentObjInfo))
  49. return nullptr;
  50. return bs_shared_ptr_new<ManagedSerializableObject>(ConstructPrivately(), currentObjInfo, createManagedInstance(type));
  51. }
  52. MonoObject* ManagedSerializableObject::createManagedInstance(const SPtr<ManagedSerializableTypeInfoObject>& type)
  53. {
  54. SPtr<ManagedSerializableObjectInfo> currentObjInfo = nullptr;
  55. // See if this type even still exists
  56. if (!ScriptAssemblyManager::instance().getSerializableObjectInfo(type->mTypeNamespace, type->mTypeName, currentObjInfo))
  57. return nullptr;
  58. // Never call constructor for structs. For classes we could call it if we could guarantee it always has one,
  59. // but at the moment we don't call constructor for them at all (which is probably fine).
  60. return currentObjInfo->mMonoClass->createInstance(false);
  61. }
  62. SPtr<ManagedSerializableObject> ManagedSerializableObject::createEmpty()
  63. {
  64. return bs_shared_ptr_new<ManagedSerializableObject>(ConstructPrivately());
  65. }
  66. void ManagedSerializableObject::serialize()
  67. {
  68. if (mManagedInstance == nullptr)
  69. return;
  70. mCachedData.clear();
  71. SPtr<ManagedSerializableObjectInfo> curType = mObjInfo;
  72. while (curType != nullptr)
  73. {
  74. for (auto& field : curType->mFields)
  75. {
  76. if (field.second->isSerializable())
  77. {
  78. ManagedSerializableFieldKey key(field.second->mParentTypeId, field.second->mFieldId);
  79. mCachedData[key] = getFieldData(field.second);
  80. }
  81. }
  82. curType = curType->mBaseClass;
  83. }
  84. // Serialize children
  85. for (auto& fieldEntry : mCachedData)
  86. fieldEntry.second->serialize();
  87. mManagedInstance = nullptr;
  88. }
  89. void ManagedSerializableObject::deserialize()
  90. {
  91. // See if this type even still exists
  92. SPtr<ManagedSerializableObjectInfo> currentObjInfo = nullptr;
  93. if (!ScriptAssemblyManager::instance().getSerializableObjectInfo(mObjInfo->mTypeInfo->mTypeNamespace, mObjInfo->mTypeInfo->mTypeName, currentObjInfo))
  94. {
  95. mManagedInstance = nullptr;
  96. mCachedData.clear();
  97. return;
  98. }
  99. deserialize(createManagedInstance(currentObjInfo->mTypeInfo), currentObjInfo);
  100. }
  101. void ManagedSerializableObject::deserialize(MonoObject* instance, const SPtr<ManagedSerializableObjectInfo>& objInfo)
  102. {
  103. mManagedInstance = instance;
  104. if (mManagedInstance == nullptr)
  105. {
  106. mCachedData.clear();
  107. return;
  108. }
  109. // Deserialize children
  110. for (auto& fieldEntry : mCachedData)
  111. fieldEntry.second->deserialize();
  112. // Scan all fields and ensure the fields still exist
  113. UINT32 i = 0;
  114. SPtr<ManagedSerializableObjectInfo> curType = mObjInfo;
  115. while (curType != nullptr)
  116. {
  117. for (auto& field : curType->mFields)
  118. {
  119. if (field.second->isSerializable())
  120. {
  121. UINT32 fieldId = field.second->mFieldId;
  122. UINT32 typeID = field.second->mParentTypeId;
  123. ManagedSerializableFieldKey key(typeID, fieldId);
  124. SPtr<ManagedSerializableMemberInfo> matchingFieldInfo = objInfo->findMatchingField(field.second, curType->mTypeInfo);
  125. if (matchingFieldInfo != nullptr)
  126. setFieldData(matchingFieldInfo, mCachedData[key]);
  127. i++;
  128. }
  129. }
  130. curType = curType->mBaseClass;
  131. }
  132. mObjInfo = objInfo;
  133. mCachedData.clear();
  134. }
  135. void ManagedSerializableObject::setFieldData(const SPtr<ManagedSerializableMemberInfo>& fieldInfo, const SPtr<ManagedSerializableFieldData>& val)
  136. {
  137. if (mManagedInstance != nullptr)
  138. fieldInfo->setValue(mManagedInstance, val->getValue(fieldInfo->mTypeInfo));
  139. else
  140. {
  141. ManagedSerializableFieldKey key(fieldInfo->mParentTypeId, fieldInfo->mFieldId);
  142. mCachedData[key] = val;
  143. }
  144. }
  145. SPtr<ManagedSerializableFieldData> ManagedSerializableObject::getFieldData(const SPtr<ManagedSerializableMemberInfo>& fieldInfo) const
  146. {
  147. if (mManagedInstance != nullptr)
  148. {
  149. MonoObject* fieldValue = fieldInfo->getValue(mManagedInstance);
  150. return ManagedSerializableFieldData::create(fieldInfo->mTypeInfo, fieldValue);
  151. }
  152. else
  153. {
  154. ManagedSerializableFieldKey key(fieldInfo->mParentTypeId, fieldInfo->mFieldId);
  155. auto iterFind = mCachedData.find(key);
  156. if (iterFind != mCachedData.end())
  157. return iterFind->second;
  158. return nullptr;
  159. }
  160. }
  161. RTTITypeBase* ManagedSerializableObject::getRTTIStatic()
  162. {
  163. return ManagedSerializableObjectRTTI::instance();
  164. }
  165. RTTITypeBase* ManagedSerializableObject::getRTTI() const
  166. {
  167. return ManagedSerializableObject::getRTTIStatic();
  168. }
  169. }