BsMonoMethod.cpp 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175
  1. //********************************** Banshee Engine (www.banshee3d.com) **************************************************//
  2. //**************** Copyright (c) 2016 Marko Pintera ([email protected]). All rights reserved. **********************//
  3. #include "BsMonoMethod.h"
  4. #include "BsMonoManager.h"
  5. #include "BsMonoUtil.h"
  6. #include "BsMonoClass.h"
  7. #include "Error/BsException.h"
  8. #include <mono/jit/jit.h>
  9. #include <mono/metadata/attrdefs.h>
  10. namespace bs
  11. {
  12. MonoMethod::MonoMethod(::MonoMethod* method)
  13. :mMethod(method), mCachedReturnType(nullptr), mCachedParameters(nullptr),
  14. mCachedNumParameters(0), mIsStatic(false), mHasCachedSignature(false)
  15. {
  16. }
  17. MonoMethod::~MonoMethod()
  18. {
  19. if (mCachedParameters != nullptr)
  20. bs_free(mCachedParameters);
  21. }
  22. MonoObject* MonoMethod::invoke(MonoObject* instance, void** params)
  23. {
  24. MonoObject* exception = nullptr;
  25. MonoObject* retVal = mono_runtime_invoke(mMethod, instance, params, &exception);
  26. MonoUtil::throwIfException(exception);
  27. return retVal;
  28. }
  29. MonoObject* MonoMethod::invokeVirtual(MonoObject* instance, void** params)
  30. {
  31. ::MonoMethod* virtualMethod = mono_object_get_virtual_method(instance, mMethod);
  32. MonoObject* exception = nullptr;
  33. MonoObject* retVal = mono_runtime_invoke(virtualMethod, instance, params, &exception);
  34. MonoUtil::throwIfException(exception);
  35. return retVal;
  36. }
  37. void* MonoMethod::getThunk() const
  38. {
  39. return mono_method_get_unmanaged_thunk(mMethod);
  40. }
  41. String MonoMethod::getName() const
  42. {
  43. return String(mono_method_get_name(mMethod));
  44. }
  45. MonoClass* MonoMethod::getReturnType() const
  46. {
  47. if (!mHasCachedSignature)
  48. cacheSignature();
  49. return mCachedReturnType;
  50. }
  51. UINT32 MonoMethod::getNumParameters() const
  52. {
  53. if (!mHasCachedSignature)
  54. cacheSignature();
  55. return mCachedNumParameters;
  56. }
  57. MonoClass* MonoMethod::getParameterType(UINT32 paramIdx) const
  58. {
  59. if (!mHasCachedSignature)
  60. cacheSignature();
  61. if (paramIdx >= mCachedNumParameters)
  62. BS_EXCEPT(InvalidParametersException, "Parameter index out of range. Valid range is [0, " + toString(mCachedNumParameters - 1) + "]");
  63. return mCachedParameters[paramIdx];
  64. }
  65. bool MonoMethod::isStatic() const
  66. {
  67. if (!mHasCachedSignature)
  68. cacheSignature();
  69. return mIsStatic;
  70. }
  71. bool MonoMethod::hasAttribute(MonoClass* monoClass) const
  72. {
  73. // TODO - Consider caching custom attributes or just initializing them all at load
  74. MonoCustomAttrInfo* attrInfo = mono_custom_attrs_from_method(mMethod);
  75. if (attrInfo == nullptr)
  76. return false;
  77. bool hasAttr = mono_custom_attrs_has_attr(attrInfo, monoClass->_getInternalClass()) != 0;
  78. mono_custom_attrs_free(attrInfo);
  79. return hasAttr;
  80. }
  81. MonoObject* MonoMethod::getAttribute(MonoClass* monoClass) const
  82. {
  83. // TODO - Consider caching custom attributes or just initializing them all at load
  84. MonoCustomAttrInfo* attrInfo = mono_custom_attrs_from_method(mMethod);
  85. if (attrInfo == nullptr)
  86. return nullptr;
  87. MonoObject* foundAttr = mono_custom_attrs_get_attr(attrInfo, monoClass->_getInternalClass());
  88. mono_custom_attrs_free(attrInfo);
  89. return foundAttr;
  90. }
  91. MonoMemberVisibility MonoMethod::getVisibility()
  92. {
  93. uint32_t flags = mono_method_get_flags(mMethod, nullptr) & MONO_METHOD_ATTR_ACCESS_MASK;
  94. if (flags == MONO_METHOD_ATTR_PRIVATE)
  95. return MonoMemberVisibility::Private;
  96. else if (flags == MONO_METHOD_ATTR_FAM_AND_ASSEM)
  97. return MonoMemberVisibility::ProtectedInternal;
  98. else if (flags == MONO_METHOD_ATTR_ASSEM)
  99. return MonoMemberVisibility::Internal;
  100. else if (flags == MONO_METHOD_ATTR_FAMILY)
  101. return MonoMemberVisibility::Protected;
  102. else if (flags == MONO_METHOD_ATTR_PUBLIC)
  103. return MonoMemberVisibility::Public;
  104. assert(false);
  105. return MonoMemberVisibility::Private;
  106. }
  107. void MonoMethod::cacheSignature() const
  108. {
  109. MonoMethodSignature* methodSignature = mono_method_signature(mMethod);
  110. MonoType* returnType = mono_signature_get_return_type(methodSignature);
  111. if (returnType != nullptr)
  112. {
  113. ::MonoClass* returnClass = mono_class_from_mono_type(returnType);
  114. if (returnClass != nullptr)
  115. mCachedReturnType = MonoManager::instance().findClass(returnClass);
  116. }
  117. mCachedNumParameters = (UINT32)mono_signature_get_param_count(methodSignature);
  118. if (mCachedParameters != nullptr)
  119. {
  120. bs_free(mCachedParameters);
  121. mCachedParameters = nullptr;
  122. }
  123. if (mCachedNumParameters > 0)
  124. {
  125. mCachedParameters = (MonoClass**)bs_alloc(mCachedNumParameters * sizeof(MonoClass*));
  126. void* iter = nullptr;
  127. for (UINT32 i = 0; i < mCachedNumParameters; i++)
  128. {
  129. MonoType* curParamType = mono_signature_get_params(methodSignature, &iter);
  130. ::MonoClass* rawClass = mono_class_from_mono_type(curParamType);
  131. mCachedParameters[i] = MonoManager::instance().findClass(rawClass);
  132. }
  133. }
  134. mIsStatic = !mono_signature_is_instance(methodSignature);
  135. mHasCachedSignature = true;
  136. }
  137. }