BsIReflectable.cpp 3.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133
  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. SPtr<IReflectable> output;
  22. if(type != nullptr)
  23. {
  24. output = type->newRTTIObject();
  25. output->mRTTIData = nullptr;
  26. }
  27. return output;
  28. }
  29. RTTITypeBase* IReflectable::_getRTTIfromTypeId(UINT32 rttiTypeId)
  30. {
  31. Stack<RTTITypeBase*> todo;
  32. Vector<RTTITypeBase*>& rootClasses = getDerivedClasses();
  33. for(auto iter = rootClasses.begin(); iter != rootClasses.end(); ++iter)
  34. todo.push(*iter);
  35. while(!todo.empty())
  36. {
  37. RTTITypeBase* curType = todo.top();
  38. todo.pop();
  39. if(curType->getRTTIId() == rttiTypeId)
  40. return curType;
  41. Vector<RTTITypeBase*>& derivedClasses = curType->getDerivedClasses();
  42. for(auto iter = derivedClasses.begin(); iter != derivedClasses.end(); ++iter)
  43. todo.push(*iter);
  44. }
  45. return nullptr;
  46. }
  47. bool IReflectable::_isTypeIdDuplicate(UINT32 typeId)
  48. {
  49. if(typeId == TID_Abstract)
  50. return false;
  51. return IReflectable::_getRTTIfromTypeId(typeId) != nullptr;
  52. }
  53. bool IReflectable::isDerivedFrom(RTTITypeBase* base)
  54. {
  55. return getRTTI()->isDerivedFrom(base);
  56. }
  57. void IReflectable::_checkForCircularReferences()
  58. {
  59. Stack<RTTITypeBase*> todo;
  60. Vector<RTTITypeBase*> rootTypes = getDerivedClasses();
  61. for (auto& entry : rootTypes)
  62. todo.push(entry);
  63. while(!todo.empty())
  64. {
  65. RTTITypeBase* myType = todo.top();
  66. todo.pop();
  67. UINT32 myNumFields = myType->getNumFields();
  68. for (UINT32 i = 0; i < myNumFields; i++)
  69. {
  70. RTTIField* myField = myType->getField(i);
  71. if (!myField->isReflectablePtrType())
  72. continue;
  73. RTTIReflectablePtrFieldBase* myReflectablePtrField = static_cast<RTTIReflectablePtrFieldBase*>(myField);
  74. RTTITypeBase* otherType = myReflectablePtrField->getType();
  75. UINT32 otherNumFields = otherType->getNumFields();
  76. for (UINT32 j = 0; j < otherNumFields; j++)
  77. {
  78. RTTIField* otherField = otherType->getField(j);
  79. if (!otherField->isReflectablePtrType())
  80. continue;
  81. RTTIReflectablePtrFieldBase* otherReflectablePtrField = static_cast<RTTIReflectablePtrFieldBase*>(otherField);
  82. if (myType->getRTTIId() == otherReflectablePtrField->getType()->getRTTIId() &&
  83. (myReflectablePtrField->getFlags() & RTTI_Flag_WeakRef) == 0 &&
  84. (otherReflectablePtrField->getFlags() & RTTI_Flag_WeakRef) == 0)
  85. {
  86. BS_EXCEPT(InternalErrorException, "Found circular reference on RTTI type: " + myType->getRTTIName()
  87. + " to type: " + otherType->getRTTIName() + ". Either remove one of the references or mark it"
  88. + " as a weak reference when defining the RTTI field.");
  89. }
  90. }
  91. }
  92. }
  93. }
  94. UINT32 IReflectable::getTypeId() const
  95. {
  96. return getRTTI()->getRTTIId();
  97. }
  98. const String& IReflectable::getTypeName() const
  99. {
  100. return getRTTI()->getRTTIName();
  101. }
  102. RTTITypeBase* IReflectable::getRTTIStatic()
  103. {
  104. return IReflectableRTTI::instance();
  105. }
  106. }