BsIReflectable.cpp 3.5 KB

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