BsIReflectable.cpp 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124
  1. #include "BsIReflectable.h"
  2. #include "BsRTTIType.h"
  3. #include "BsException.h"
  4. namespace BansheeEngine
  5. {
  6. void IReflectable::_registerDerivedClass(RTTITypeBase* derivedClass)
  7. {
  8. if(_isTypeIdDuplicate(derivedClass->getRTTIId()))
  9. {
  10. BS_EXCEPT(InternalErrorException, "RTTI type \"" + derivedClass->getRTTIName() +
  11. "\" has a duplicate ID: " + toString(derivedClass->getRTTIId()));
  12. }
  13. getDerivedClasses().push_back(derivedClass);
  14. }
  15. std::shared_ptr<IReflectable> IReflectable::createInstanceFromTypeId(UINT32 rttiTypeId)
  16. {
  17. RTTITypeBase* type = _getRTTIfromTypeId(rttiTypeId);
  18. if(type != nullptr)
  19. return type->newRTTIObject();
  20. return nullptr;
  21. }
  22. RTTITypeBase* IReflectable::_getRTTIfromTypeId(UINT32 rttiTypeId)
  23. {
  24. Stack<RTTITypeBase*> todo;
  25. Vector<RTTITypeBase*>& rootClasses = getDerivedClasses();
  26. for(auto iter = rootClasses.begin(); iter != rootClasses.end(); ++iter)
  27. todo.push(*iter);
  28. while(!todo.empty())
  29. {
  30. RTTITypeBase* curType = todo.top();
  31. todo.pop();
  32. if(curType->getRTTIId() == rttiTypeId)
  33. return curType;
  34. Vector<RTTITypeBase*>& derivedClasses = curType->getDerivedClasses();
  35. for(auto iter = derivedClasses.begin(); iter != derivedClasses.end(); ++iter)
  36. todo.push(*iter);
  37. }
  38. return nullptr;
  39. }
  40. bool IReflectable::_isTypeIdDuplicate(UINT32 typeId)
  41. {
  42. if(typeId == TID_Abstract)
  43. return false;
  44. return IReflectable::_getRTTIfromTypeId(typeId) != nullptr;
  45. }
  46. bool IReflectable::isDerivedFrom(RTTITypeBase* base)
  47. {
  48. return getRTTI()->isDerivedFrom(base);
  49. }
  50. void IReflectable::_checkForCircularReferences()
  51. {
  52. Stack<RTTITypeBase*> todo;
  53. Vector<RTTITypeBase*> rootTypes = getDerivedClasses();
  54. for (auto& entry : rootTypes)
  55. todo.push(entry);
  56. while(!todo.empty())
  57. {
  58. RTTITypeBase* myType = todo.top();
  59. todo.pop();
  60. UINT32 myNumFields = myType->getNumFields();
  61. for (UINT32 i = 0; i < myNumFields; i++)
  62. {
  63. RTTIField* myField = myType->getField(i);
  64. if (!myField->isReflectablePtrType())
  65. continue;
  66. RTTIReflectablePtrFieldBase* myReflectablePtrField = static_cast<RTTIReflectablePtrFieldBase*>(myField);
  67. RTTITypeBase* otherType = myReflectablePtrField->getType();
  68. UINT32 otherNumFields = otherType->getNumFields();
  69. for (UINT32 j = 0; j < otherNumFields; j++)
  70. {
  71. RTTIField* otherField = otherType->getField(j);
  72. if (!otherField->isReflectablePtrType())
  73. continue;
  74. RTTIReflectablePtrFieldBase* otherReflectablePtrField = static_cast<RTTIReflectablePtrFieldBase*>(otherField);
  75. RTTITypeBase* a = myReflectablePtrField->getType();
  76. RTTITypeBase* b = otherReflectablePtrField->getType();
  77. if (myType->getRTTIId() == otherReflectablePtrField->getType()->getRTTIId() &&
  78. (myReflectablePtrField->getFlags() & RTTI_Flag_WeakRef) == 0 &&
  79. (otherReflectablePtrField->getFlags() & RTTI_Flag_WeakRef) == 0)
  80. {
  81. BS_EXCEPT(InternalErrorException, "Found circular reference on RTTI type: " + myType->getRTTIName()
  82. + " to type: " + otherType->getRTTIName() + ". Either remove one of the references or mark it"
  83. + " as a weak reference when defining the RTTI field.");
  84. }
  85. }
  86. }
  87. }
  88. }
  89. UINT32 IReflectable::getTypeId() const
  90. {
  91. return getRTTI()->getRTTIId();
  92. }
  93. const String& IReflectable::getTypeName() const
  94. {
  95. return getRTTI()->getRTTIName();
  96. }
  97. }