BsUtility.cpp 4.6 KB

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