No Description
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

Class.h 13KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264
  1. #pragma once
  2. #include <stdint.h>
  3. #include "il2cpp-config.h"
  4. #include "il2cpp-blob.h"
  5. #include "il2cpp-class-internals.h"
  6. #include "metadata/ArrayMetadata.h"
  7. #include "metadata/Il2CppTypeCompare.h"
  8. #include "utils/dynamic_array.h"
  9. #include "il2cpp-class-internals.h"
  10. #include "il2cpp-object-internals.h"
  11. #include "Exception.h"
  12. #include "GenericClass.h"
  13. #include "Type.h"
  14. #include "os/Mutex.h"
  15. #include "vm/MetadataCache.h"
  16. #include "il2cpp-tabledefs.h"
  17. struct Il2CppClass;
  18. struct EventInfo;
  19. struct FieldInfo;
  20. struct PropertyInfo;
  21. struct MethodInfo;
  22. struct Il2CppImage;
  23. struct Il2CppReflectionType;
  24. struct Il2CppType;
  25. struct Il2CppGenericContainer;
  26. struct Il2CppGenericContext;
  27. struct MonoGenericParameterInfo;
  28. namespace il2cpp
  29. {
  30. namespace vm
  31. {
  32. class TypeNameParseInfo;
  33. enum TypeSearchFlags
  34. {
  35. kTypeSearchFlagNone = 0x0,
  36. kTypeSearchFlagIgnoreCase = 0x1,
  37. kTypeSearchFlagThrowOnError = 0x2,
  38. kTypeSearchFlagDontUseExecutingImage = 0x4
  39. };
  40. class LIBIL2CPP_CODEGEN_API Class
  41. {
  42. public:
  43. static void AllocateStaticData();
  44. static void FreeStaticData();
  45. static Il2CppClass* FromIl2CppType(const Il2CppType* type, bool throwOnError = true);
  46. static Il2CppClass* FromIl2CppTypeEnum(Il2CppTypeEnum type);
  47. static Il2CppClass* FromName(const Il2CppImage* image, const char* namespaze, const char *name);
  48. static Il2CppClass* FromSystemType(Il2CppReflectionType *type);
  49. static Il2CppClass* FromGenericParameter(Il2CppMetadataGenericParameterHandle param);
  50. static Il2CppClass* GetElementClass(Il2CppClass *klass);
  51. static const Il2CppType* GetEnumBaseType(Il2CppClass *klass);
  52. static const EventInfo* GetEvents(Il2CppClass *klass, void* *iter);
  53. static FieldInfo* GetFields(Il2CppClass *klass, void* *iter);
  54. static FieldInfo* GetFieldFromName(Il2CppClass *klass, const char* name);
  55. static const MethodInfo* GetFinalizer(Il2CppClass *klass);
  56. static int32_t GetInstanceSize(const Il2CppClass *klass);
  57. static Il2CppClass* GetInterfaces(Il2CppClass *klass, void* *iter);
  58. static const MethodInfo* GetMethods(Il2CppClass *klass, void* *iter);
  59. static const MethodInfo* GetMethodFromName(Il2CppClass *klass, const char* name, int argsCount);
  60. static const MethodInfo* GetMethodFromNameFlags(Il2CppClass *klass, const char* name, int argsCount, int32_t flags);
  61. static const MethodInfo* GetMethodFromNameFlagsAndSig(Il2CppClass *klass, const char* name, int argsCount, int32_t flags, const Il2CppType** argTypes);
  62. static const MethodInfo* GetGenericInstanceMethodFromDefintion(Il2CppClass* genericInstanceClass, const MethodInfo* methodDefinition);
  63. static const char* GetName(Il2CppClass *klass);
  64. static const char* GetNamespace(Il2CppClass *klass);
  65. static Il2CppClass* GetNestedTypes(Il2CppClass *klass, void* *iter);
  66. static size_t GetNumMethods(const Il2CppClass* klass);
  67. static size_t GetNumProperties(const Il2CppClass* klass);
  68. static size_t GetNumFields(const Il2CppClass* klass);
  69. static Il2CppClass* GetParent(Il2CppClass *klass);
  70. static const PropertyInfo* GetProperties(Il2CppClass *klass, void* *iter);
  71. static const PropertyInfo* GetPropertyFromName(Il2CppClass *klass, const char* name);
  72. static int32_t GetValueSize(Il2CppClass *klass, uint32_t *align);
  73. //for Unsafe, but more performant version of HasParent, see ClassInlines.h
  74. static bool HasParent(Il2CppClass *klass, Il2CppClass *parent);
  75. static bool IsAssignableFrom(Il2CppClass *klass, Il2CppClass *oklass);
  76. static bool IsAssignableFrom(Il2CppReflectionType *klass, Il2CppReflectionType *oklass);
  77. static bool IsGeneric(const Il2CppClass *klass);
  78. static bool IsInflated(const Il2CppClass *klass);
  79. static bool IsGenericTypeDefinition(const Il2CppClass *klass);
  80. static bool IsSubclassOf(Il2CppClass *klass, Il2CppClass *klassc, bool check_interfaces);
  81. inline static bool IsValuetype(const Il2CppClass* klass)
  82. {
  83. return klass->byval_arg.valuetype;
  84. }
  85. inline static bool IsBlittable(const Il2CppClass *klass)
  86. {
  87. return klass->is_blittable;
  88. }
  89. static bool HasDefaultConstructor(Il2CppClass* klass);
  90. inline static int GetFlags(const Il2CppClass* klass)
  91. {
  92. return klass->flags;
  93. }
  94. inline static bool IsAbstract(const Il2CppClass* klass)
  95. {
  96. return (klass->flags & TYPE_ATTRIBUTE_ABSTRACT) != 0;
  97. }
  98. inline static bool IsInterface(const Il2CppClass* klass)
  99. {
  100. return (klass->flags & TYPE_ATTRIBUTE_INTERFACE) || (klass->byval_arg.type == IL2CPP_TYPE_VAR) || (klass->byval_arg.type == IL2CPP_TYPE_MVAR);
  101. }
  102. inline static bool IsNullable(const Il2CppClass* klass)
  103. {
  104. // Based on benchmarking doing the check on `klass->generic_class != NULL` makes this check faster
  105. // Likely since nullabletype is a bitfield and requires some manipulation to check
  106. return klass->generic_class != NULL && klass->nullabletype;
  107. }
  108. inline static Il2CppClass* GetNullableArgument(const Il2CppClass* klass)
  109. {
  110. IL2CPP_ASSERT(IsNullable(klass));
  111. return klass->element_class;
  112. }
  113. static int GetArrayElementSize(const Il2CppClass *klass);
  114. inline static const Il2CppType* GetByrefType(Il2CppClass* klass)
  115. {
  116. return &klass->this_arg;
  117. }
  118. inline static const Il2CppType* GetType(Il2CppClass* klass)
  119. {
  120. return &klass->byval_arg;
  121. }
  122. static const Il2CppType* GetType(Il2CppClass *klass, const TypeNameParseInfo &info);
  123. static bool HasAttribute(Il2CppClass *klass, Il2CppClass *attr_class);
  124. inline static bool IsEnum(const Il2CppClass* klass)
  125. {
  126. return klass->enumtype;
  127. }
  128. static const Il2CppImage* GetImage(Il2CppClass* klass);
  129. static const char *GetAssemblyName(const Il2CppClass *klass);
  130. static const char *GetAssemblyNameNoExtension(const Il2CppClass *klass);
  131. static Il2CppClass* GenericParamGetBaseType(Il2CppClass* klass);
  132. static MonoGenericParameterInfo* GetOrCreateMonoGenericParameterInfo(Il2CppMetadataGenericParameterHandle parameterHandle);
  133. static const int IgnoreNumberOfArguments;
  134. public:
  135. static void Init(Il2CppClass *klass);
  136. static bool InitLocked(Il2CppClass* klass, const il2cpp::os::FastAutoLock& lock);
  137. static bool InitSizeAndFieldLayoutLocked(Il2CppClass* klass, const il2cpp::os::FastAutoLock& lock);
  138. static Il2CppClass* GetArrayClass(Il2CppClass *element_class, uint32_t rank);
  139. static Il2CppClass* GetBoundedArrayClass(Il2CppClass *element_class, uint32_t rank, bool bounded);
  140. static Il2CppClass* GetInflatedGenericInstanceClass(Il2CppClass* klass, const Il2CppType** types, uint32_t typeCount);
  141. static Il2CppClass* GetInflatedGenericInstanceClass(Il2CppClass* klass, const Il2CppGenericInst* genericInst);
  142. static Il2CppClass* InflateGenericClass(Il2CppClass* klass, Il2CppGenericContext *context);
  143. static const Il2CppType* InflateGenericType(const Il2CppType* type, Il2CppGenericContext *context);
  144. static Il2CppMetadataGenericContainerHandle GetGenericContainer(Il2CppClass *klass);
  145. static const MethodInfo* GetCCtor(Il2CppClass *klass);
  146. static const char* GetFieldDefaultValue(const FieldInfo *field, const Il2CppType** type);
  147. static int GetFieldMarshaledSize(const FieldInfo *field);
  148. static int GetFieldMarshaledAlignment(const FieldInfo *field);
  149. static Il2CppClass* GetPtrClass(const Il2CppType* type);
  150. static Il2CppClass* GetPtrClass(Il2CppClass* elementClass);
  151. static bool HasReferences(Il2CppClass *klass);
  152. static void SetupEvents(Il2CppClass *klass);
  153. static void SetupFields(Il2CppClass *klass);
  154. static void SetupMethods(Il2CppClass *klass);
  155. static void SetupNestedTypes(Il2CppClass *klass);
  156. static void SetupProperties(Il2CppClass *klass);
  157. static void SetupTypeHierarchy(Il2CppClass *klass);
  158. static void SetupInterfaces(Il2CppClass *klass);
  159. static const il2cpp::utils::dynamic_array<Il2CppClass*>& GetStaticFieldData();
  160. static size_t GetBitmapSize(const Il2CppClass* klass);
  161. static void GetBitmap(Il2CppClass* klass, size_t* bitmap, size_t& maxSetBit);
  162. static const Il2CppType* il2cpp_type_from_type_info(const TypeNameParseInfo& info, TypeSearchFlags searchFlags);
  163. static Il2CppClass* GetDeclaringType(Il2CppClass* klass);
  164. static const MethodInfo* GetVirtualMethod(Il2CppClass* klass, const MethodInfo* virtualMethod);
  165. static void SetClassInitializationError(Il2CppClass* klass, Il2CppException* error);
  166. static void PublishInitialized(Il2CppClass* klass);
  167. static IL2CPP_FORCE_INLINE bool IsGenericClassAssignableFrom(const Il2CppClass* klass, const Il2CppClass* oklass, const Il2CppClass* implementingClass = il2cpp_defaults.missing_class)
  168. {
  169. return klass == oklass || IsGenericClassAssignableFromVariance(klass, oklass, implementingClass);
  170. }
  171. static IL2CPP_FORCE_INLINE bool IsGenericClassAssignableFromVariance(const Il2CppClass* klass, const Il2CppClass* oklass, const Il2CppClass* implementingClass = il2cpp_defaults.missing_class)
  172. {
  173. IL2CPP_ASSERT(klass != oklass && "IsGenericClassAssignableFromVariance is for generic variance checks - you should call IsGenericClassAssignableFrom instead");
  174. const Il2CppGenericClass* genericClass = klass->generic_class;
  175. const Il2CppGenericClass* oGenericClass = oklass->generic_class;
  176. if (oGenericClass == NULL || !GenericClass::HasSameGenericTypeDefinition(oGenericClass, genericClass))
  177. return false;
  178. const Il2CppGenericInst* genericInst = genericClass->context.class_inst;
  179. const Il2CppGenericInst* oGenericInst = oGenericClass->context.class_inst;
  180. Il2CppMetadataGenericContainerHandle genericContainer = MetadataCache::GetGenericContainerFromGenericClass(klass->image, klass->generic_class);
  181. IL2CPP_ASSERT(oGenericInst->type_argc == genericInst->type_argc);
  182. for (uint32_t i = 0; i < genericInst->type_argc; ++i)
  183. {
  184. uint16_t flags = MetadataCache::GetGenericParameterFlags(MetadataCache::GetGenericParameterFromIndex(genericContainer, i));
  185. const int32_t parameterVariance = flags & IL2CPP_GENERIC_PARAMETER_ATTRIBUTE_VARIANCE_MASK;
  186. Il2CppClass* genericParameterType = Class::FromIl2CppType(genericInst->type_argv[i]);
  187. Il2CppClass* oGenericParameterType = Class::FromIl2CppType(oGenericInst->type_argv[i]);
  188. if (parameterVariance == IL2CPP_GENERIC_PARAMETER_ATTRIBUTE_NON_VARIANT || Class::IsValuetype(genericParameterType) || Class::IsValuetype(oGenericParameterType))
  189. {
  190. if (genericParameterType != oGenericParameterType)
  191. {
  192. if (implementingClass->rank || implementingClass->declaringType == il2cpp_defaults.array_class)
  193. {
  194. // For arrays or System.Array/InternalEnumerator<T> we need to follow the array variance rules when looking for an
  195. // generic interface i.e. Int32[] should be assignable to IEnumerable<UInt32>
  196. IL2CPP_ASSERT(implementingClass->rank || strcmp(implementingClass->name, "InternalEnumerator`1") == 0);
  197. if (metadata::ArrayMetadata::GetArrayVarianceReducedType(genericParameterType) != metadata::ArrayMetadata::GetArrayVarianceReducedType(oGenericParameterType))
  198. return false;
  199. }
  200. else
  201. {
  202. return false;
  203. }
  204. }
  205. }
  206. else if (parameterVariance == IL2CPP_GENERIC_PARAMETER_ATTRIBUTE_COVARIANT)
  207. {
  208. if (!Class::IsAssignableFrom(genericParameterType, oGenericParameterType))
  209. return false;
  210. }
  211. else
  212. {
  213. IL2CPP_ASSERT(parameterVariance == IL2CPP_GENERIC_PARAMETER_ATTRIBUTE_CONTRAVARIANT);
  214. if (!Class::IsAssignableFrom(oGenericParameterType, genericParameterType))
  215. return false;
  216. }
  217. }
  218. return true;
  219. }
  220. };
  221. } /* namespace vm */
  222. } /* namespace il2cpp */