BsMonoMethod.cpp 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151
  1. #include "BsMonoMethod.h"
  2. #include "BsMonoManager.h"
  3. #include "BsMonoUtil.h"
  4. #include "BsMonoClass.h"
  5. namespace BansheeEngine
  6. {
  7. MonoMethod::MonoMethod(::MonoMethod* method)
  8. :mMethod(method), mCachedReturnType(nullptr), mCachedParameters(nullptr),
  9. mCachedNumParameters(0), mIsStatic(false), mHasCachedSignature(false)
  10. {
  11. mThunk = mono_method_get_unmanaged_thunk(mMethod);
  12. }
  13. MonoMethod::~MonoMethod()
  14. {
  15. if (mCachedParameters != nullptr)
  16. bs_free(mCachedParameters);
  17. }
  18. MonoObject* MonoMethod::invoke(MonoObject* instance, void** params)
  19. {
  20. MonoObject* exception = nullptr;
  21. MonoObject* retVal = mono_runtime_invoke(mMethod, instance, params, &exception);
  22. MonoUtil::throwIfException(exception);
  23. return retVal;
  24. }
  25. MonoObject* MonoMethod::invokeVirtual(MonoObject* instance, void** params)
  26. {
  27. ::MonoMethod* virtualMethod = mono_object_get_virtual_method(instance, mMethod);
  28. MonoObject* exception = nullptr;
  29. MonoObject* retVal = mono_runtime_invoke(virtualMethod, instance, params, &exception);
  30. MonoUtil::throwIfException(exception);
  31. return retVal;
  32. }
  33. void* MonoMethod::getThunk() const
  34. {
  35. return mThunk;
  36. }
  37. String MonoMethod::getName() const
  38. {
  39. return String(mono_method_get_name(mMethod));
  40. }
  41. MonoClass* MonoMethod::getReturnType() const
  42. {
  43. if (!mHasCachedSignature)
  44. cacheSignature();
  45. return mCachedReturnType;
  46. }
  47. UINT32 MonoMethod::getNumParameters() const
  48. {
  49. if (!mHasCachedSignature)
  50. cacheSignature();
  51. return mCachedNumParameters;
  52. }
  53. MonoClass* MonoMethod::getParameterType(UINT32 paramIdx) const
  54. {
  55. if (!mHasCachedSignature)
  56. cacheSignature();
  57. if (paramIdx >= mCachedNumParameters)
  58. BS_EXCEPT(InvalidParametersException, "Parameter index out of range. Valid range is [0, " + toString(mCachedNumParameters - 1) + "]");
  59. return mCachedParameters[paramIdx];
  60. }
  61. bool MonoMethod::isStatic() const
  62. {
  63. if (!mHasCachedSignature)
  64. cacheSignature();
  65. return mIsStatic;
  66. }
  67. bool MonoMethod::hasAttribute(MonoClass* monoClass) const
  68. {
  69. // TODO - Consider caching custom attributes or just initializing them all at load
  70. MonoCustomAttrInfo* attrInfo = mono_custom_attrs_from_method(mMethod);
  71. if (attrInfo == nullptr)
  72. return false;
  73. bool hasAttr = mono_custom_attrs_has_attr(attrInfo, monoClass->_getInternalClass()) != 0;
  74. mono_custom_attrs_free(attrInfo);
  75. return hasAttr;
  76. }
  77. MonoObject* MonoMethod::getAttribute(MonoClass* monoClass) const
  78. {
  79. // TODO - Consider caching custom attributes or just initializing them all at load
  80. MonoCustomAttrInfo* attrInfo = mono_custom_attrs_from_method(mMethod);
  81. if (attrInfo == nullptr)
  82. return nullptr;
  83. MonoObject* foundAttr = mono_custom_attrs_get_attr(attrInfo, monoClass->_getInternalClass());
  84. mono_custom_attrs_free(attrInfo);
  85. return foundAttr;
  86. }
  87. void MonoMethod::cacheSignature() const
  88. {
  89. MonoMethodSignature* methodSignature = mono_method_signature(mMethod);
  90. MonoType* returnType = mono_signature_get_return_type(methodSignature);
  91. if (returnType != nullptr)
  92. {
  93. ::MonoClass* returnClass = mono_class_from_mono_type(returnType);
  94. if (returnClass != nullptr)
  95. mCachedReturnType = MonoManager::instance().findClass(returnClass);
  96. }
  97. mCachedNumParameters = (UINT32)mono_signature_get_param_count(methodSignature);
  98. if (mCachedParameters != nullptr)
  99. {
  100. bs_free(mCachedParameters);
  101. mCachedParameters = nullptr;
  102. }
  103. if (mCachedNumParameters > 0)
  104. {
  105. mCachedParameters = (MonoClass**)bs_alloc(mCachedNumParameters * sizeof(MonoClass*));
  106. void* iter = nullptr;
  107. for (UINT32 i = 0; i < mCachedNumParameters; i++)
  108. {
  109. MonoType* curParamType = mono_signature_get_params(methodSignature, &iter);
  110. ::MonoClass* rawClass = mono_class_from_mono_type(curParamType);
  111. mCachedParameters[i] = MonoManager::instance().findClass(rawClass);
  112. }
  113. }
  114. mIsStatic = !mono_signature_is_instance(methodSignature);
  115. mHasCachedSignature = true;
  116. }
  117. }