BsIReflectable.cpp 3.7 KB

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