暂无描述
您最多选择25个主题 主题必须以字母或数字开头,可以包含连字符 (-),并且长度不得超过35个字符

ThreadPoolMs.cpp 4.5KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132
  1. #include "il2cpp-api.h"
  2. #include "il2cpp-config.h"
  3. #include "utils/dynamic_array.h"
  4. #include "vm/ThreadPoolMs.h"
  5. #include "vm/Domain.h"
  6. #include "vm/Array.h"
  7. #include "vm/Object.h"
  8. #include "vm/Runtime.h"
  9. #include "os/Atomic.h"
  10. #include "gc/WriteBarrier.h"
  11. #include "mono/ThreadPool/threadpool-ms.h"
  12. namespace il2cpp
  13. {
  14. namespace vm
  15. {
  16. Il2CppAsyncResult* ThreadPoolMs::DelegateBeginInvoke(Il2CppDelegate* delegate, void** params, Il2CppDelegate* asyncCallback, Il2CppObject* state)
  17. {
  18. int numParams = delegate->method->parameters_count;
  19. il2cpp::utils::dynamic_array<void*> newParams(numParams + 2);
  20. for (int i = 0; i < numParams; ++i)
  21. newParams[i] = params[i];
  22. newParams[numParams] = asyncCallback;
  23. newParams[numParams + 1] = state;
  24. return threadpool_ms_begin_invoke(il2cpp::vm::Domain::GetCurrent(), (Il2CppObject*)delegate, const_cast<MethodInfo*>(delegate->method), newParams.data());
  25. }
  26. Il2CppObject* ThreadPoolMs::DelegateEndInvoke(Il2CppAsyncResult* asyncResult, void **out_args)
  27. {
  28. Il2CppArray *arrayOutArgs;
  29. Il2CppObject *exc, *retVal;
  30. retVal = threadpool_ms_end_invoke(asyncResult, &arrayOutArgs, &exc);
  31. if (exc)
  32. il2cpp_raise_exception((Il2CppException*)exc);
  33. if (out_args)
  34. {
  35. const MethodInfo *method = asyncResult->async_delegate->method;
  36. void** outArgsPtr = (void**)il2cpp_array_addr(arrayOutArgs, Il2CppObject*, 0);
  37. il2cpp_array_size_t arrayOutArgsIndex = 0;
  38. for (size_t methodParameterIndex = 0; methodParameterIndex < method->parameters_count; methodParameterIndex++)
  39. {
  40. const Il2CppType* paramType = method->parameters[methodParameterIndex];
  41. // Assume that arrayOutArgs only contains parameters that are passed by reference.
  42. if (!paramType->byref)
  43. continue;
  44. IL2CPP_ASSERT(arrayOutArgsIndex < arrayOutArgs->max_length);
  45. Il2CppClass *paramClass = il2cpp_class_from_type(paramType);
  46. if (paramClass->byval_arg.valuetype)
  47. {
  48. IL2CPP_ASSERT(paramClass->native_size > 0 && "EndInvoke: Invalid native_size found when trying to copy a value type in the out_args.");
  49. // NOTE(gab): in case of value types, we need to copy the data over.
  50. memcpy(out_args[arrayOutArgsIndex], il2cpp::vm::Object::Unbox((Il2CppObject*)outArgsPtr[arrayOutArgsIndex]), paramClass->native_size);
  51. }
  52. else
  53. {
  54. *((void**)out_args[arrayOutArgsIndex]) = outArgsPtr[arrayOutArgsIndex];
  55. }
  56. arrayOutArgsIndex++;
  57. }
  58. }
  59. return retVal;
  60. }
  61. Il2CppObject* ThreadPoolMs::MessageInvoke(Il2CppObject *target, Il2CppMethodMessage *msg, Il2CppObject **exc, Il2CppArray **out_args)
  62. {
  63. static Il2CppClass *object_array_klass = NULL;
  64. MethodInfo *method;
  65. Il2CppObject *ret;
  66. Il2CppArray *arr;
  67. int i, j, outarg_count = 0;
  68. method = (MethodInfo*)msg->method->method;
  69. for (i = 0; i < method->parameters_count; i++)
  70. {
  71. if (method->parameters[i]->byref)
  72. outarg_count++;
  73. }
  74. if (!object_array_klass)
  75. {
  76. Il2CppClass *klass;
  77. klass = il2cpp_array_class_get(il2cpp_defaults.object_class, 1);
  78. IL2CPP_ASSERT(klass);
  79. os::Atomic::FullMemoryBarrier();
  80. object_array_klass = klass;
  81. }
  82. arr = il2cpp_array_new_specific(object_array_klass, outarg_count);
  83. il2cpp::gc::WriteBarrier::GenericStore(out_args, arr);
  84. il2cpp::gc::WriteBarrier::GenericStoreNull(exc);
  85. ret = vm::Runtime::InvokeArray(method, method->klass->byval_arg.valuetype ? il2cpp_object_unbox(target) : target, method->parameters_count > 0 ? msg->args : NULL, (Il2CppException**)exc);
  86. for (i = 0, j = 0; i < method->parameters_count; i++)
  87. {
  88. if (method->parameters[i]->byref)
  89. {
  90. Il2CppObject* arg;
  91. arg = (Il2CppObject*)il2cpp_array_get(msg->args, void*, i);
  92. il2cpp_array_setref(*out_args, j, arg);
  93. j++;
  94. }
  95. }
  96. return ret;
  97. }
  98. void ThreadPoolMs::Suspend()
  99. {
  100. threadpool_ms_suspend();
  101. }
  102. void ThreadPoolMs::Resume()
  103. {
  104. threadpool_ms_resume();
  105. }
  106. } /* namespace vm */
  107. } /* namespace il2cpp */