ClassInlines.h 3.7 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283
  1. #pragma once
  2. //This file should not include anything from VM. This is included by both libil2cpp and the codegen headers
  3. #include "il2cpp-config.h"
  4. #include "il2cpp-class-internals.h"
  5. #include "il2cpp-object-internals.h"
  6. namespace il2cpp
  7. {
  8. namespace vm
  9. {
  10. class LIBIL2CPP_CODEGEN_API ClassInlines
  11. {
  12. public:
  13. // we assume that the Il2CppClass's have already been initialized in this case, like in code generation
  14. static inline bool HasParentUnsafe(const Il2CppClass* klass, const Il2CppClass* parent) { return klass->typeHierarchyDepth >= parent->typeHierarchyDepth && klass->typeHierarchy[parent->typeHierarchyDepth - 1] == parent; }
  15. // This function is critical for performance, before optimization it
  16. // caused up to 20% of all CPU usage in code generated by il2cpp
  17. static IL2CPP_FORCE_INLINE Il2CppClass* InitFromCodegen(Il2CppClass *klass)
  18. {
  19. if (klass->initialized_and_no_error)
  20. return klass;
  21. return InitFromCodegenSlow(klass);
  22. }
  23. static IL2CPP_FORCE_INLINE const MethodInfo* InitRgcxFromCodegen(const MethodInfo *method)
  24. {
  25. if (method->rgctx_data)
  26. return method;
  27. return InitRgctxFromCodegenSlow(method);
  28. }
  29. static IL2CPP_NO_INLINE Il2CppClass* InitFromCodegenSlow(Il2CppClass *klass);
  30. static IL2CPP_NO_INLINE Il2CppClass* InitFromCodegenSlow(Il2CppClass *klass, bool throwOnError);
  31. static IL2CPP_NO_INLINE const MethodInfo* InitRgctxFromCodegenSlow(const MethodInfo* method);
  32. //internal
  33. static IL2CPP_FORCE_INLINE const VirtualInvokeData& GetInterfaceInvokeDataFromVTable(Il2CppObject* obj, const Il2CppClass* itf, Il2CppMethodSlot slot)
  34. {
  35. const Il2CppClass* klass = obj->klass;
  36. IL2CPP_ASSERT(klass->initialized);
  37. IL2CPP_ASSERT(slot < itf->method_count);
  38. for (uint16_t i = 0; i < klass->interface_offsets_count; i++)
  39. {
  40. if (klass->interfaceOffsets[i].interfaceType == itf)
  41. {
  42. int32_t offset = klass->interfaceOffsets[i].offset;
  43. IL2CPP_ASSERT(offset != -1);
  44. IL2CPP_ASSERT(offset + slot < klass->vtable_count);
  45. return klass->vtable[offset + slot];
  46. }
  47. }
  48. return GetInterfaceInvokeDataFromVTableSlowPath(obj, itf, slot);
  49. }
  50. static IL2CPP_FORCE_INLINE const VirtualInvokeData* GetInterfaceInvokeDataFromVTable(const Il2CppClass* klass, const Il2CppClass* itf, Il2CppMethodSlot slot)
  51. {
  52. IL2CPP_ASSERT(klass->is_vtable_initialized);
  53. IL2CPP_ASSERT(slot < itf->method_count);
  54. for (uint16_t i = 0; i < klass->interface_offsets_count; i++)
  55. {
  56. if (klass->interfaceOffsets[i].interfaceType == itf)
  57. {
  58. int32_t offset = klass->interfaceOffsets[i].offset;
  59. IL2CPP_ASSERT(offset != -1);
  60. IL2CPP_ASSERT(offset + slot < klass->vtable_count);
  61. return &klass->vtable[offset + slot];
  62. }
  63. }
  64. return GetInterfaceInvokeDataFromVTableSlowPath(klass, itf, slot);
  65. }
  66. // we don't want this method to get inlined because that makes GetInterfaceInvokeDataFromVTable method itself very large and performance suffers
  67. static IL2CPP_NO_INLINE const VirtualInvokeData& GetInterfaceInvokeDataFromVTableSlowPath(Il2CppObject* obj, const Il2CppClass* itf, Il2CppMethodSlot slot);
  68. static IL2CPP_NO_INLINE const VirtualInvokeData* GetInterfaceInvokeDataFromVTableSlowPath(const Il2CppClass* klass, const Il2CppClass* itf, Il2CppMethodSlot slot);
  69. };
  70. }
  71. }