BsUtility.cpp 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145
  1. #include "BsUtility.h"
  2. #include "BsRTTIType.h"
  3. namespace BansheeEngine
  4. {
  5. Vector<ResourceDependency> Utility::findResourceDependencies(IReflectable& obj, bool recursive)
  6. {
  7. Map<String, ResourceDependency> dependencies;
  8. findResourceDependenciesInternal(obj, recursive, dependencies);
  9. Vector<ResourceDependency> dependencyList(dependencies.size());
  10. UINT32 i = 0;
  11. for (auto& entry : dependencies)
  12. {
  13. dependencyList[i] = entry.second;
  14. i++;
  15. }
  16. return dependencyList;
  17. }
  18. void Utility::findResourceDependenciesInternal(IReflectable& obj, bool recursive, Map<String, ResourceDependency>& dependencies)
  19. {
  20. RTTITypeBase* rtti = obj.getRTTI();
  21. rtti->onSerializationStarted(&obj);
  22. UINT32 numFields = rtti->getNumFields();
  23. for (UINT32 i = 0; i < numFields; i++)
  24. {
  25. RTTIField* field = rtti->getField(i);
  26. if ((field->getFlags() & RTTI_Flag_SkipInReferenceSearch) != 0)
  27. continue;
  28. if (field->isReflectableType())
  29. {
  30. RTTIReflectableFieldBase* reflectableField = static_cast<RTTIReflectableFieldBase*>(field);
  31. if (reflectableField->getType()->getRTTIId() == TID_ResourceHandle)
  32. {
  33. if (reflectableField->isArray())
  34. {
  35. UINT32 numElements = reflectableField->getArraySize(&obj);
  36. for (UINT32 j = 0; j < numElements; j++)
  37. {
  38. HResource resource = (HResource&)reflectableField->getArrayValue(&obj, j);
  39. if (!resource.getUUID().empty())
  40. {
  41. ResourceDependency& dependency = dependencies[resource.getUUID()];
  42. dependency.resource = resource;
  43. dependency.numReferences++;
  44. }
  45. }
  46. }
  47. else
  48. {
  49. HResource resource = (HResource&)reflectableField->getValue(&obj);
  50. if (!resource.getUUID().empty())
  51. {
  52. ResourceDependency& dependency = dependencies[resource.getUUID()];
  53. dependency.resource = resource;
  54. dependency.numReferences++;
  55. }
  56. }
  57. }
  58. else if (recursive)
  59. {
  60. // Optimization, no need to retrieve its value and go deeper if it has no
  61. // reflectable children that may hold the reference.
  62. if (hasReflectableChildren(reflectableField->getType()))
  63. {
  64. if (reflectableField->isArray())
  65. {
  66. UINT32 numElements = reflectableField->getArraySize(&obj);
  67. for (UINT32 j = 0; j < numElements; j++)
  68. {
  69. IReflectable& childObj = reflectableField->getArrayValue(&obj, j);
  70. findResourceDependenciesInternal(childObj, true, dependencies);
  71. }
  72. }
  73. else
  74. {
  75. IReflectable& childObj = reflectableField->getValue(&obj);
  76. findResourceDependenciesInternal(childObj, true, dependencies);
  77. }
  78. }
  79. }
  80. }
  81. else if (field->isReflectablePtrType() && recursive)
  82. {
  83. RTTIReflectablePtrFieldBase* reflectablePtrField = static_cast<RTTIReflectablePtrFieldBase*>(field);
  84. // Optimization, no need to retrieve its value and go deeper if it has no
  85. // reflectable children that may hold the reference.
  86. if (hasReflectableChildren(reflectablePtrField->getType()))
  87. {
  88. if (reflectablePtrField->isArray())
  89. {
  90. UINT32 numElements = reflectablePtrField->getArraySize(&obj);
  91. for (UINT32 j = 0; j < numElements; j++)
  92. {
  93. SPtr<IReflectable> childObj = reflectablePtrField->getArrayValue(&obj, j);
  94. if (childObj != nullptr)
  95. findResourceDependenciesInternal(*childObj, true, dependencies);
  96. }
  97. }
  98. else
  99. {
  100. SPtr<IReflectable> childObj = reflectablePtrField->getValue(&obj);
  101. if (childObj != nullptr)
  102. findResourceDependenciesInternal(*childObj, true, dependencies);
  103. }
  104. }
  105. }
  106. }
  107. rtti->onSerializationEnded(&obj);
  108. }
  109. bool Utility::hasReflectableChildren(RTTITypeBase* type)
  110. {
  111. UINT32 numFields = type->getNumFields();
  112. for (UINT32 i = 0; i < numFields; i++)
  113. {
  114. RTTIField* field = type->getField(i);
  115. if (field->isReflectableType() || field->isReflectablePtrType())
  116. return true;
  117. }
  118. const Vector<RTTITypeBase*>& derivedClasses = type->getDerivedClasses();
  119. for (auto& derivedClass : derivedClasses)
  120. {
  121. numFields = derivedClass->getNumFields();
  122. for (UINT32 i = 0; i < numFields; i++)
  123. {
  124. RTTIField* field = derivedClass->getField(i);
  125. if (field->isReflectableType() || field->isReflectablePtrType())
  126. return true;
  127. }
  128. }
  129. return false;
  130. }
  131. }