247 lines
12 KiB
C
247 lines
12 KiB
C
|
#pragma once
|
||
|
|
||
|
#include <stdint.h>
|
||
|
#include "il2cpp-config.h"
|
||
|
#include "il2cpp-blob.h"
|
||
|
#include "il2cpp-class-internals.h"
|
||
|
#include "metadata/Il2CppTypeVector.h"
|
||
|
#include "utils/dynamic_array.h"
|
||
|
#include "il2cpp-class-internals.h"
|
||
|
#include "il2cpp-object-internals.h"
|
||
|
#include "Exception.h"
|
||
|
#include "Type.h"
|
||
|
|
||
|
#if NET_4_0
|
||
|
#include "vm/MetadataCache.h"
|
||
|
#include "il2cpp-tabledefs.h"
|
||
|
#endif
|
||
|
|
||
|
|
||
|
struct Il2CppClass;
|
||
|
struct EventInfo;
|
||
|
struct FieldInfo;
|
||
|
struct PropertyInfo;
|
||
|
struct MethodInfo;
|
||
|
|
||
|
struct Il2CppImage;
|
||
|
struct Il2CppReflectionType;
|
||
|
struct Il2CppType;
|
||
|
struct Il2CppGenericContainer;
|
||
|
struct Il2CppGenericContext;
|
||
|
struct Il2CppGenericParameter;
|
||
|
|
||
|
namespace il2cpp
|
||
|
{
|
||
|
namespace vm
|
||
|
{
|
||
|
class TypeNameParseInfo;
|
||
|
|
||
|
enum TypeSearchFlags
|
||
|
{
|
||
|
kTypeSearchFlagNone = 0x0,
|
||
|
kTypeSearchFlagIgnoreCase = 0x1,
|
||
|
kTypeSearchFlagThrowOnError = 0x2,
|
||
|
kTypeSearchFlagDontUseExecutingImage = 0x4
|
||
|
};
|
||
|
|
||
|
class LIBIL2CPP_CODEGEN_API Class
|
||
|
{
|
||
|
public:
|
||
|
static Il2CppClass* FromIl2CppType(const Il2CppType* type);
|
||
|
static Il2CppClass* FromName(const Il2CppImage* image, const char* namespaze, const char *name);
|
||
|
static Il2CppClass* FromSystemType(Il2CppReflectionType *type);
|
||
|
static Il2CppClass* FromGenericParameter(const Il2CppGenericParameter *param);
|
||
|
static Il2CppClass* GetElementClass(Il2CppClass *klass);
|
||
|
static const Il2CppType* GetEnumBaseType(Il2CppClass *klass);
|
||
|
static const EventInfo* GetEvents(Il2CppClass *klass, void* *iter);
|
||
|
static FieldInfo* GetFields(Il2CppClass *klass, void* *iter);
|
||
|
static FieldInfo* GetFieldFromName(Il2CppClass *klass, const char* name);
|
||
|
static const MethodInfo* GetFinalizer(Il2CppClass *klass);
|
||
|
static int32_t GetInstanceSize(const Il2CppClass *klass);
|
||
|
static Il2CppClass* GetInterfaces(Il2CppClass *klass, void* *iter);
|
||
|
static const MethodInfo* GetMethods(Il2CppClass *klass, void* *iter);
|
||
|
static const MethodInfo* GetMethodFromName(Il2CppClass *klass, const char* name, int argsCount);
|
||
|
static const MethodInfo* GetMethodFromNameFlags(Il2CppClass *klass, const char* name, int argsCount, int32_t flags);
|
||
|
static const char* GetName(Il2CppClass *klass);
|
||
|
static const char* GetNamespace(Il2CppClass *klass);
|
||
|
static Il2CppClass* GetNestedTypes(Il2CppClass *klass, void* *iter);
|
||
|
static size_t GetNumMethods(const Il2CppClass* klass);
|
||
|
static size_t GetNumProperties(const Il2CppClass* klass);
|
||
|
static size_t GetNumFields(const Il2CppClass* klass);
|
||
|
static Il2CppClass* GetParent(Il2CppClass *klass);
|
||
|
static const PropertyInfo* GetProperties(Il2CppClass *klass, void* *iter);
|
||
|
static const PropertyInfo* GetPropertyFromName(Il2CppClass *klass, const char* name);
|
||
|
static int32_t GetValueSize(Il2CppClass *klass, uint32_t *align);
|
||
|
static bool HasParent(Il2CppClass *klass, Il2CppClass *parent);
|
||
|
// we assume that the Il2CppClass's have already been initialized in this case, like in code generation
|
||
|
static bool HasParentUnsafe(const Il2CppClass* klass, const Il2CppClass* parent) { return klass->typeHierarchyDepth >= parent->typeHierarchyDepth && klass->typeHierarchy[parent->typeHierarchyDepth - 1] == parent; }
|
||
|
static bool IsAssignableFrom(Il2CppClass *klass, Il2CppClass *oklass);
|
||
|
static bool IsGeneric(const Il2CppClass *klass);
|
||
|
static bool IsInflated(const Il2CppClass *klass);
|
||
|
static bool IsSubclassOf(Il2CppClass *klass, Il2CppClass *klassc, bool check_interfaces);
|
||
|
static bool IsValuetype(const Il2CppClass *klass);
|
||
|
static bool IsBlittable(const Il2CppClass *klass);
|
||
|
static bool HasDefaultConstructor(Il2CppClass* klass);
|
||
|
static int GetFlags(const Il2CppClass *klass);
|
||
|
static bool IsAbstract(const Il2CppClass *klass);
|
||
|
static bool IsInterface(const Il2CppClass *klass);
|
||
|
static bool IsNullable(const Il2CppClass *klass);
|
||
|
static Il2CppClass* GetNullableArgument(const Il2CppClass* klass);
|
||
|
static int GetArrayElementSize(const Il2CppClass *klass);
|
||
|
static const Il2CppType* GetType(Il2CppClass *klass);
|
||
|
static const Il2CppType* GetType(Il2CppClass *klass, const TypeNameParseInfo &info);
|
||
|
static bool HasAttribute(Il2CppClass *klass, Il2CppClass *attr_class);
|
||
|
static bool IsEnum(const Il2CppClass *klass);
|
||
|
static const Il2CppImage* GetImage(Il2CppClass* klass);
|
||
|
static const char *GetAssemblyName(const Il2CppClass *klass);
|
||
|
static const char *GetAssemblyNameNoExtension(const Il2CppClass *klass);
|
||
|
|
||
|
static const int IgnoreNumberOfArguments;
|
||
|
|
||
|
public:
|
||
|
//internal
|
||
|
static FORCE_INLINE const VirtualInvokeData& GetInterfaceInvokeDataFromVTable(const Il2CppObject* obj, const Il2CppClass* itf, Il2CppMethodSlot slot)
|
||
|
{
|
||
|
const Il2CppClass* klass = obj->klass;
|
||
|
IL2CPP_ASSERT(klass->initialized);
|
||
|
IL2CPP_ASSERT(slot < itf->method_count);
|
||
|
|
||
|
for (uint16_t i = 0; i < klass->interface_offsets_count; i++)
|
||
|
{
|
||
|
if (klass->interfaceOffsets[i].interfaceType == itf)
|
||
|
{
|
||
|
int32_t offset = klass->interfaceOffsets[i].offset;
|
||
|
IL2CPP_ASSERT(offset != -1);
|
||
|
IL2CPP_ASSERT(offset + slot < klass->vtable_count);
|
||
|
return klass->vtable[offset + slot];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return GetInterfaceInvokeDataFromVTableSlowPath(obj, itf, slot);
|
||
|
}
|
||
|
|
||
|
static FORCE_INLINE const VirtualInvokeData* GetInterfaceInvokeDataFromVTable(const Il2CppClass* klass, const Il2CppClass* itf, Il2CppMethodSlot slot)
|
||
|
{
|
||
|
IL2CPP_ASSERT(klass->initialized);
|
||
|
IL2CPP_ASSERT(slot < itf->method_count);
|
||
|
|
||
|
for (uint16_t i = 0; i < klass->interface_offsets_count; i++)
|
||
|
{
|
||
|
if (klass->interfaceOffsets[i].interfaceType == itf)
|
||
|
{
|
||
|
int32_t offset = klass->interfaceOffsets[i].offset;
|
||
|
IL2CPP_ASSERT(offset != -1);
|
||
|
IL2CPP_ASSERT(offset + slot < klass->vtable_count);
|
||
|
return &klass->vtable[offset + slot];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return GetInterfaceInvokeDataFromVTableSlowPath(klass, itf, slot);
|
||
|
}
|
||
|
|
||
|
static bool Init(Il2CppClass *klass);
|
||
|
|
||
|
// This function is critical for performance, before optimization it
|
||
|
// caused up to 20% of all CPU usage in code generated by il2cpp
|
||
|
static FORCE_INLINE bool InitFromCodegen(Il2CppClass *klass)
|
||
|
{
|
||
|
if (klass->initialized_and_no_error)
|
||
|
return true;
|
||
|
return InitFromCodegenSlow(klass);
|
||
|
}
|
||
|
|
||
|
static Il2CppClass* GetArrayClass(Il2CppClass *element_class, uint32_t rank);
|
||
|
static Il2CppClass* GetBoundedArrayClass(Il2CppClass *element_class, uint32_t rank, bool bounded);
|
||
|
static Il2CppClass* GetInflatedGenericInstanceClass(Il2CppClass* klass, const metadata::Il2CppTypeVector& types);
|
||
|
static Il2CppClass* GetInflatedGenericInstanceClass(Il2CppClass* klass, const Il2CppGenericInst* genericInst);
|
||
|
static Il2CppClass* InflateGenericClass(Il2CppClass* klass, Il2CppGenericContext *context);
|
||
|
static const Il2CppType* InflateGenericType(const Il2CppType* type, Il2CppGenericContext *context);
|
||
|
|
||
|
static Il2CppClass* GetArrayClassCached(Il2CppClass *element_class, uint32_t rank, bool bounded)
|
||
|
{
|
||
|
return GetBoundedArrayClass(element_class, rank, bounded);
|
||
|
}
|
||
|
|
||
|
static const Il2CppGenericContainer* GetGenericContainer(Il2CppClass *klass);
|
||
|
static const MethodInfo* GetCCtor(Il2CppClass *klass);
|
||
|
static const char* GetFieldDefaultValue(const FieldInfo *field, const Il2CppType** type);
|
||
|
static int GetFieldMarshaledSize(const FieldInfo *field);
|
||
|
static Il2CppClass* GetPtrClass(const Il2CppType* type);
|
||
|
static Il2CppClass* GetPtrClass(Il2CppClass* elementClass);
|
||
|
static bool HasReferences(Il2CppClass *klass);
|
||
|
static void SetupEvents(Il2CppClass *klass);
|
||
|
static void SetupFields(Il2CppClass *klass);
|
||
|
static void SetupMethods(Il2CppClass *klass);
|
||
|
static void SetupNestedTypes(Il2CppClass *klass);
|
||
|
static void SetupProperties(Il2CppClass *klass);
|
||
|
static void SetupTypeHierarchy(Il2CppClass *klass);
|
||
|
static void SetupInterfaces(Il2CppClass *klass);
|
||
|
|
||
|
static const il2cpp::utils::dynamic_array<Il2CppClass*>& GetStaticFieldData();
|
||
|
|
||
|
static size_t GetBitmapSize(const Il2CppClass* klass);
|
||
|
static void GetBitmap(Il2CppClass* klass, size_t* bitmap, size_t& maxSetBit);
|
||
|
|
||
|
static const Il2CppType* il2cpp_type_from_type_info(const TypeNameParseInfo& info, TypeSearchFlags searchFlags);
|
||
|
|
||
|
static Il2CppClass* GetDeclaringType(Il2CppClass* klass);
|
||
|
|
||
|
static void UpdateInitializedAndNoError(Il2CppClass *klass);
|
||
|
|
||
|
private:
|
||
|
static IL2CPP_NO_INLINE bool InitFromCodegenSlow(Il2CppClass *klass);
|
||
|
|
||
|
#if NET_4_0
|
||
|
static FORCE_INLINE bool IsGenericClassAssignableFrom(const Il2CppClass* klass, const Il2CppClass* oklass, const Il2CppGenericContainer* genericContainer)
|
||
|
{
|
||
|
const Il2CppGenericClass* genericClass = klass->generic_class;
|
||
|
const Il2CppGenericClass* oGenericClass = oklass->generic_class;
|
||
|
|
||
|
if (oGenericClass == NULL || oGenericClass->typeDefinitionIndex != genericClass->typeDefinitionIndex)
|
||
|
return false;
|
||
|
|
||
|
const int32_t genericParameterCount = genericContainer->type_argc;
|
||
|
|
||
|
const Il2CppGenericInst* genericInst = genericClass->context.class_inst;
|
||
|
IL2CPP_ASSERT(genericInst->type_argc == genericParameterCount);
|
||
|
|
||
|
const Il2CppGenericInst* oGenericInst = oGenericClass->context.class_inst;
|
||
|
IL2CPP_ASSERT(oGenericInst->type_argc == genericParameterCount);
|
||
|
|
||
|
for (int32_t i = 0; i < genericParameterCount; ++i)
|
||
|
{
|
||
|
const Il2CppGenericParameter* genericParameter = MetadataCache::GetGenericParameterFromIndex(genericContainer->genericParameterStart + i);
|
||
|
const int32_t parameterVariance = genericParameter->flags & IL2CPP_GENERIC_PARAMETER_ATTRIBUTE_VARIANCE_MASK;
|
||
|
Il2CppClass* genericParameterType = Class::FromIl2CppType(genericInst->type_argv[i]);
|
||
|
Il2CppClass* oGenericParameterType = Class::FromIl2CppType(oGenericInst->type_argv[i]);
|
||
|
|
||
|
if (parameterVariance == IL2CPP_GENERIC_PARAMETER_ATTRIBUTE_NON_VARIANT || Class::IsValuetype(genericParameterType) || Class::IsValuetype(oGenericParameterType))
|
||
|
{
|
||
|
if (genericParameterType != oGenericParameterType)
|
||
|
return false;
|
||
|
}
|
||
|
else if (parameterVariance == IL2CPP_GENERIC_PARAMETER_ATTRIBUTE_COVARIANT)
|
||
|
{
|
||
|
if (!Class::IsAssignableFrom(genericParameterType, oGenericParameterType))
|
||
|
return false;
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
IL2CPP_ASSERT(parameterVariance == IL2CPP_GENERIC_PARAMETER_ATTRIBUTE_CONTRAVARIANT);
|
||
|
if (!Class::IsAssignableFrom(oGenericParameterType, genericParameterType))
|
||
|
return false;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
#endif
|
||
|
|
||
|
// we don't want this method to get inlined because that makes GetInterfaceInvokeDataFromVTable method itself very large and performance suffers
|
||
|
static IL2CPP_NO_INLINE const VirtualInvokeData& GetInterfaceInvokeDataFromVTableSlowPath(const Il2CppObject* obj, const Il2CppClass* itf, Il2CppMethodSlot slot);
|
||
|
static IL2CPP_NO_INLINE const VirtualInvokeData* GetInterfaceInvokeDataFromVTableSlowPath(const Il2CppClass* klass, const Il2CppClass* itf, Il2CppMethodSlot slot);
|
||
|
};
|
||
|
} /* namespace vm */
|
||
|
} /* namespace il2cpp */
|