| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739 | #pragma once#include <unordered_map>#if HYBRIDCLR_UNITY_2021_OR_NEW#include "metadata/CustomAttributeDataReader.h"#include "CustomAttributeDataWriter.h"#endif#include "Image.h"#include "CustomAttributeDataWriter.h"namespace hybridclr{namespace metadata{	struct InterfaceOffsetInfo	{		const Il2CppType* type;		uint32_t offset;	};	struct TypeDefinitionDetail	{		uint32_t methodImplStart;		uint32_t methodImplCount;		uint32_t vtableCount;		Il2CppTypeDefinitionSizes typeSizes;		VirtualMethodImpl* vtable;	};	struct ParamDetail	{		Il2CppParameterDefinition paramDef;		uint32_t parameterIndex;		uint32_t defaultValueIndex; // -1 for invalid	};	struct FieldDetail	{		Il2CppFieldDefinition fieldDef;		uint32_t typeDefIndex;		uint32_t offset;		uint32_t defaultValueIndex; // -1 for invalid	};	struct PropertyDetail	{		const char* name;		uint16_t flags;		uint32_t signatureBlobIndex;		uint32_t getterMethodIndex; // start from 1;		uint32_t setterMethodIndex;		const Il2CppTypeDefinition* declaringType;		Il2CppPropertyDefinition il2cppDefinition;	};	struct EventDetail	{		const char* name;		uint16_t eventFlags;		uint32_t eventType; // TypeDefOrRef codedIndex		uint32_t addMethodIndex; // start from 1		uint32_t removeMethodIndex; // start from 1		uint32_t fireMethodIndex; // start from 1;#if HYBRIDCLR_UNITY_2019		const Il2CppTypeDefinition* declaringType;		Il2CppEventDefinition il2cppDefinition;#endif	};	struct CustomAttribute	{		uint32_t ctorMethodToken;		uint32_t value;	};	struct CustomAttributesInfo	{		int32_t typeRangeIndex;		bool inited;		void* dataStartPtr;		void* dataEndPtr;	};#if HYBRIDCLR_UNITY_2021_OR_NEW	enum class BlobSource	{		RAW_IMAGE = 0,		CONVERTED_IL2CPP_FORMAT = 1,	};#endif	class InterpreterImage : public Image	{	public:		static void Initialize();		static uint32_t AllocImageIndex(uint32_t dllLength);		static void RegisterImage(InterpreterImage* image);		static InterpreterImage* GetImage(uint32_t imageIndex)		{			//os::FastAutoLock lock(&s_imageLock);			IL2CPP_ASSERT(imageIndex < kMaxMetadataImageCount);			return s_images[imageIndex];		}	private:		static InterpreterImage* s_images[kMaxMetadataImageCount];	public:		InterpreterImage(uint32_t imageIndex) : _index(imageIndex), _inited(false), _il2cppImage(nullptr)#if HYBRIDCLR_UNITY_2021_OR_NEW			, _constValues(1024), _il2cppFormatCustomDataBlob(256), _tempCtorArgBlob(256), _tempFieldBlob(256), _tempPropertyBlob(256)#endif		{		}		LoadImageErrorCode Load(const void* imageData, size_t length)		{			if (_inited)			{				RaiseExecutionEngineException("image can't be inited again");			}			_inited = true;			_rawImage = new RawImage();			LoadImageErrorCode err = _rawImage->Load(imageData, length);			if (err != LoadImageErrorCode::OK)			{				delete _rawImage;				_rawImage = nullptr;				return err;			}			return LoadImageErrorCode::OK;		}		bool IsInitialized() const		{			return _inited;		}		uint32_t GetIndex() const		{			return _index;		}		const Il2CppImage* GetIl2CppImage() const		{			return _il2cppImage;		}		uint32_t EncodeWithIndex(uint32_t rawIndex) const		{			return EncodeImageAndMetadataIndex(_index, rawIndex);		}		uint32_t EncodeWithIndexExcept0(uint32_t rawIndex) const		{			return rawIndex != 0 ? EncodeImageAndMetadataIndex(_index, rawIndex) : 0;		}		MethodBody* GetMethodBody(uint32_t token) override		{			IL2CPP_ASSERT(DecodeTokenTableType(token) == TableType::METHOD);			uint32_t rowIndex = DecodeTokenRowIndex(token);			IL2CPP_ASSERT(rowIndex > 0 && rowIndex <= (uint32_t)_methodDefines.size());			const Il2CppMethodDefinition* methodDef = &_methodDefines[rowIndex - 1];			bool isGenericMethod = methodDef->genericContainerIndex != kGenericContainerIndexInvalid || _typesDefines[DecodeMetadataIndex(methodDef->declaringType)].genericContainerIndex != kGenericContainerIndexInvalid;			TbMethod methodData = _rawImage->ReadMethod(rowIndex);			MethodBody* resultMethodBody = new (HYBRIDCLR_MALLOC_ZERO(sizeof(MethodBody))) MethodBody();			ReadMethodBody(*methodDef, methodData, *resultMethodBody);			return resultMethodBody;		}		// type index start from 0, difference with table index...		Il2CppMetadataTypeHandle GetAssemblyTypeHandleFromRawIndex(AssemblyTypeIndex index) const		{			IL2CPP_ASSERT(DecodeImageIndex(index) == 0);			IL2CPP_ASSERT(index >= 0 && (size_t)index < _typesDefines.size());			return (Il2CppMetadataTypeHandle)&_typesDefines[index];		}		Il2CppMetadataTypeHandle GetAssemblyExportedTypeHandleFromRawIndex(AssemblyTypeIndex index) const		{			IL2CPP_ASSERT(DecodeImageIndex(index) == 0);			IL2CPP_ASSERT(index >= 0 && (size_t)index < _typesDefines.size());			return (Il2CppMetadataTypeHandle)&_exportedTypeDefines[index];		}		const Il2CppTypeDefinitionSizes* GetTypeDefinitionSizesFromRawIndex(TypeDefinitionIndex index)		{			IL2CPP_ASSERT((size_t)index < _typeDetails.size());			return &_typeDetails[index].typeSizes;		}		const char* GetStringFromRawIndex(StringIndex index) const		{			IL2CPP_ASSERT(DecodeImageIndex(index) == 0);			return _rawImage->GetStringFromRawIndex(index);		}		uint32_t GetTypeRawIndex(const Il2CppTypeDefinition* typeDef) const		{			return (uint32_t)(typeDef - &_typesDefines[0]);		}		Il2CppTypeDefinition* GetTypeDefinitionByTypeDetail(const TypeDefinitionDetail* typeDetail)		{			uint32_t index = (uint32_t)(typeDetail - &_typeDetails[0]);			return &_typesDefines[index];		}		uint32_t GetTypeRawIndexByEncodedIl2CppTypeIndex(int32_t il2cppTypeIndex) const		{			return GetTypeRawIndex((const Il2CppTypeDefinition*)_types[DecodeMetadataIndex(il2cppTypeIndex)]->data.typeHandle);		}		const Il2CppTypeDefinition* GetTypeFromRawIndex(uint32_t index) const		{			IL2CPP_ASSERT((size_t)index < _typesDefines.size());			return &_typesDefines[index];		}		const Il2CppType* GetIl2CppTypeFromRawIndex(uint32_t index) const		{			IL2CPP_ASSERT((size_t)index < _types.size());			return _types[index];		}		const Il2CppType* GetIl2CppTypeFromRawTypeDefIndex(uint32_t index) override		{			IL2CPP_ASSERT(index < (uint32_t)_typesDefines.size());			return _types[DecodeMetadataIndex(_typesDefines[index].byvalTypeIndex)];		}		const Il2CppFieldDefinition* GetFieldDefinitionFromRawIndex(uint32_t index)		{			IL2CPP_ASSERT(index < (uint32_t)_fieldDetails.size());			return &(_fieldDetails[index].fieldDef);		}		const FieldDetail& GetFieldDetailFromRawIndex(uint32_t index)		{			IL2CPP_ASSERT(index < (uint32_t)_fieldDetails.size());			return _fieldDetails[index];		}		const Il2CppMethodDefinition* GetMethodDefinitionFromRawIndex(uint32_t index) override		{			IL2CPP_ASSERT((size_t)index < _methodDefines.size());			return &_methodDefines[index];		}		MethodIndex GetMethodIndexFromDefinition(const Il2CppMethodDefinition* methodDefine)		{			return EncodeWithIndex((uint32_t)(methodDefine - &_methodDefines[0]));		}		const Il2CppGenericParameter* GetGenericParameterByGlobalIndex(uint32_t index)		{			IL2CPP_ASSERT(index < (uint32_t)_genericParams.size());			return &_genericParams[index];		}		const Il2CppGenericParameter* GetGenericParameterByRawIndex(const Il2CppGenericContainer* container, uint32_t index)		{			uint32_t globalIndex = DecodeMetadataIndex(container->genericParameterStart) + index;			IL2CPP_ASSERT(globalIndex < (uint32_t)_genericParams.size());			return &_genericParams[globalIndex];		}		Il2CppGenericContainer* GetGenericContainerByRawIndex(uint32_t index) override		{			if (index != kGenericContainerIndexInvalid)			{				IL2CPP_ASSERT(index < (uint32_t)_genericContainers.size());				return &_genericContainers[index];			}			return nullptr;		}		Il2CppGenericContainer* GetGenericContainerByTypeDefinition(const Il2CppTypeDefinition* typeDef)		{			GenericContainerIndex idx = DecodeMetadataIndex(typeDef->genericContainerIndex);			if (idx != kGenericContainerIndexInvalid)			{				IL2CPP_ASSERT(idx < (GenericContainerIndex)_genericContainers.size());				return &_genericContainers[idx];			}			return nullptr;		}		Il2CppGenericContainer* GetGenericContainerByTypeDefRawIndex(int32_t typeDefIndex) override		{			IL2CPP_ASSERT(typeDefIndex < (int32_t)_typeDetails.size());			return GetGenericContainerByTypeDefinition(&_typesDefines[typeDefIndex]);		}		const il2cpp::utils::dynamic_array<MethodImpl> GetTypeMethodImplByTypeDefinition(const Il2CppTypeDefinition* typeDef);		const Il2CppType* GetGenericParameterConstraintFromIndex(GenericParameterConstraintIndex index)		{			IL2CPP_ASSERT((size_t)index < _genericConstraints.size());			TypeIndex typeIndex = _genericConstraints[index];			if (typeIndex == kTypeIndexInvalid)			{				TbGenericParamConstraint data = _rawImage->ReadGenericParamConstraint(index + 1);				Il2CppGenericParameter& genericParam = _genericParams[data.owner - 1];				const Il2CppGenericContainer* klassGc;				const Il2CppGenericContainer* methodGc;				GetClassAndMethodGenericContainerFromGenericContainerIndex(genericParam.ownerIndex, klassGc, methodGc);				const Il2CppType* paramCons = ReadTypeFromToken(klassGc, methodGc, DecodeTypeDefOrRefOrSpecCodedIndexTableType(data.constraint), DecodeTypeDefOrRefOrSpecCodedIndexRowIndex(data.constraint));				_genericConstraints[index] = typeIndex = DecodeMetadataIndex(AddIl2CppTypeCache(paramCons));			}			return _types[typeIndex];		}		Il2CppClass* GetNestedTypeFromOffset(const Il2CppClass* klass, TypeNestedTypeIndex offset);		Il2CppClass* GetNestedTypeFromOffset(const Il2CppTypeDefinition* typeDef, TypeNestedTypeIndex offset);		const MethodInfo* GetMethodInfoFromMethodDefinitionRawIndex(uint32_t index);		const MethodInfo* GetMethodInfoFromMethodDefinition(const Il2CppMethodDefinition* methodDef);		const Il2CppMethodDefinition* GetMethodDefinitionFromVTableSlot(const Il2CppTypeDefinition* typeDefine, int32_t vTableSlot);		const MethodInfo* GetMethodInfoFromVTableSlot(const Il2CppClass* klass, int32_t vTableSlot);		Il2CppTypeDefinition* GetNestedTypes(Il2CppTypeDefinition* handle, void** iter);		void GetClassAndMethodGenericContainerFromGenericContainerIndex(GenericContainerIndex idx, const Il2CppGenericContainer*& klassGc, const Il2CppGenericContainer*& methodGc);		Il2CppMethodPointer GetAdjustorThunk(uint32_t token);		Il2CppMethodPointer GetMethodPointer(uint32_t token);		InvokerMethod GetMethodInvoker(uint32_t token);		const Il2CppParameterDefinition* GetParameterDefinitionFromIndex(uint32_t index)		{			IL2CPP_ASSERT((size_t)index < _params.size());			return &_params[index].paramDef;		}		const Il2CppParameterDefaultValue* GetParameterDefaultValueEntryByRawIndex(uint32_t index)		{			IL2CPP_ASSERT(index < (uint32_t)_params.size());			uint32_t defaultValueIndex = _params[index].defaultValueIndex;			return defaultValueIndex != kDefaultValueIndexNull ? &_paramDefaultValues[defaultValueIndex] : nullptr;		}		uint32_t GetFieldOffset(const Il2CppTypeDefinition* typeDef, int32_t fieldIndexInType)		{			uint32_t fieldActualIndex = DecodeMetadataIndex(typeDef->fieldStart) + fieldIndexInType;			IL2CPP_ASSERT(fieldActualIndex < (uint32_t)_fieldDetails.size());			return _fieldDetails[fieldActualIndex].offset;		}		uint32_t GetFieldOffset(TypeDefinitionIndex typeIndex, int32_t fieldIndexInType)		{			Il2CppTypeDefinition* typeDef = &_typesDefines[typeIndex];			return GetFieldOffset(typeDef, fieldIndexInType);		}		uint32_t GetFieldOffset(const Il2CppClass* klass, int32_t fieldIndexInType)		{			Il2CppTypeDefinition* typeDef = (Il2CppTypeDefinition*)(klass->typeMetadataHandle);			return GetFieldOffset(typeDef, fieldIndexInType);		}		int32_t GetPackingSize(const Il2CppTypeDefinition* typeDef) const		{			int32_t typeIndex = GetTypeRawIndex(typeDef);			auto it = _classLayouts.find(typeIndex);			return it != _classLayouts.end() ? it->second.packingSize : 0;		}		TbClassLayout GetClassLayout(const Il2CppTypeDefinition* typeDef) const		{			int32_t typeIndex = GetTypeRawIndex(typeDef);			auto it = _classLayouts.find(typeIndex);			return it != _classLayouts.end() ? it->second : TbClassLayout{};		}		const Il2CppFieldDefaultValue* GetFieldDefaultValueEntryByRawIndex(uint32_t index)		{			IL2CPP_ASSERT(index < (uint32_t)_fieldDetails.size());			uint32_t fdvIndex = _fieldDetails[index].defaultValueIndex;			IL2CPP_ASSERT(fdvIndex != kDefaultValueIndexNull);			return &_fieldDefaultValues[fdvIndex];		}#if HYBRIDCLR_UNITY_2021_OR_NEW		static uint32_t EncodeWithBlobSource(uint32_t index, BlobSource source)		{			return (index << 1) | (uint32_t)source;		}#endif		const uint8_t* GetFieldOrParameterDefalutValueByRawIndex(uint32_t index)		{#if !HYBRIDCLR_UNITY_2021_OR_NEW			return _rawImage->GetFieldOrParameterDefalutValueByRawIndex(index);#else			BlobSource source = (BlobSource)(index & 0x1);			uint32_t offset = index >> 1;			if (source == BlobSource::RAW_IMAGE)			{				return _rawImage->GetFieldOrParameterDefalutValueByRawIndex(offset);			}			else			{				return _constValues.DataAt(offset);			}#endif		}#if HYBRIDCLR_UNITY_2021_OR_NEW		DefaultValueDataIndex ConvertConstValue(CustomAttributeDataWriter& writer, uint32_t blobIndex, const Il2CppType* type);#endif		Il2CppPropertyDefinition* GetPropertyDefinitionFromIndex(PropertyIndex index)		{			IL2CPP_ASSERT(index > 0 && index <= (int32_t)_propeties.size());			PropertyDetail& pd = _propeties[(uint32_t)index - 1];			return &pd.il2cppDefinition;		}		Il2CppMetadataPropertyInfo GetPropertyInfo(const Il2CppClass* klass, TypePropertyIndex index)		{			const Il2CppTypeDefinition* typeDef = (Il2CppTypeDefinition*)klass->typeMetadataHandle;			IL2CPP_ASSERT(typeDef->propertyStart);			uint32_t rowIndex = DecodeMetadataIndex(typeDef->propertyStart) + index;			PropertyDetail& pd = _propeties[rowIndex - 1];			uint32_t baseMethodIdx = DecodeMetadataIndex(typeDef->methodStart) + 1;#if UNITY_ENGINE_TUANJIE			const MethodInfo* getter = pd.getterMethodIndex ? il2cpp::vm::Class::GetOrSetupOneMethod(const_cast<Il2CppClass*>(klass), pd.getterMethodIndex - baseMethodIdx) : nullptr;			const MethodInfo* setter = pd.setterMethodIndex ? il2cpp::vm::Class::GetOrSetupOneMethod(const_cast<Il2CppClass*>(klass), pd.setterMethodIndex - baseMethodIdx) : nullptr;#else			const MethodInfo* getter = pd.getterMethodIndex ? klass->methods[pd.getterMethodIndex - baseMethodIdx] : nullptr;			const MethodInfo* setter = pd.setterMethodIndex ? klass->methods[pd.setterMethodIndex - baseMethodIdx] : nullptr;#endif			return { pd.name, getter, setter, pd.flags, EncodeToken(TableType::PROPERTY, rowIndex) };		}#ifdef HYBRIDCLR_UNITY_2019		const Il2CppEventDefinition* GetEventDefinitionFromIndex(EventIndex index)		{			IL2CPP_ASSERT(index > 0 && index <= (int32_t)_events.size());			EventDetail& pd = _events[index - 1];			return &pd.il2cppDefinition;		}#endif		Il2CppMetadataEventInfo GetEventInfo(const Il2CppClass* klass, TypeEventIndex index)		{			const Il2CppTypeDefinition* typeDef = (Il2CppTypeDefinition*)klass->typeMetadataHandle;			IL2CPP_ASSERT(typeDef->eventStart);			uint32_t rowIndex = DecodeMetadataIndex(typeDef->eventStart) + index;			EventDetail& pd = _events[rowIndex - 1];			uint32_t baseMethodIdx = DecodeMetadataIndex(typeDef->methodStart) + 1;#if UNITY_ENGINE_TUANJIE			const MethodInfo* addOn = pd.addMethodIndex ? il2cpp::vm::Class::GetOrSetupOneMethod(const_cast<Il2CppClass*>(klass), pd.addMethodIndex - baseMethodIdx) : nullptr;			const MethodInfo* removeOn = pd.removeMethodIndex ? il2cpp::vm::Class::GetOrSetupOneMethod(const_cast<Il2CppClass*>(klass), pd.removeMethodIndex - baseMethodIdx) : nullptr;			const MethodInfo* raiseOn = pd.fireMethodIndex ? il2cpp::vm::Class::GetOrSetupOneMethod(const_cast<Il2CppClass*>(klass), pd.fireMethodIndex - baseMethodIdx) : nullptr;#else			const MethodInfo* addOn = pd.addMethodIndex ? klass->methods[pd.addMethodIndex - baseMethodIdx] : nullptr;			const MethodInfo* removeOn = pd.removeMethodIndex ? klass->methods[pd.removeMethodIndex - baseMethodIdx] : nullptr;			const MethodInfo* raiseOn = pd.fireMethodIndex ? klass->methods[pd.fireMethodIndex - baseMethodIdx] : nullptr;#endif			return { pd.name, &klass->byval_arg, addOn, removeOn, raiseOn, EncodeToken(TableType::EVENT, rowIndex) };		}		const Il2CppAssembly* GetReferencedAssembly(int32_t referencedAssemblyTableIndex, const Il2CppAssembly assembliesTable[], int assembliesCount);		Il2CppMetadataCustomAttributeHandle GetCustomAttributeTypeToken(uint32_t token)		{			auto it = _tokenCustomAttributes.find(token);			return it != _tokenCustomAttributes.end() ? (Il2CppMetadataCustomAttributeHandle)&_customAttributeHandles[DecodeMetadataIndex(it->second.typeRangeIndex)] : nullptr;		}		CustomAttributeIndex GetCustomAttributeIndex(uint32_t token)		{			auto it = _tokenCustomAttributes.find(token);			return it != _tokenCustomAttributes.end() ? it->second.typeRangeIndex : kCustomAttributeIndexInvalid;		}#if !HYBRIDCLR_UNITY_2021_OR_NEW		std::tuple<void*, void*> GetCustomAttributeDataRange(uint32_t token)		{			const Il2CppCustomAttributeTypeRange* dataRangeCur = (const Il2CppCustomAttributeTypeRange*)GetCustomAttributeTypeToken(token);			CustomAttributeIndex curIndex = DecodeMetadataIndex(GET_CUSTOM_ATTRIBUTE_TYPE_RANGE_START(*dataRangeCur));			CustomAttributeIndex nextIndex = DecodeMetadataIndex(GET_CUSTOM_ATTRIBUTE_TYPE_RANGE_START(*(dataRangeCur + 1)));			CustomAttribute& curCa = _customAttribues[curIndex];			CustomAttribute& nextCa = _customAttribues[nextIndex];			return std::make_tuple<void*, void*>((void*)_rawImage->GetBlobReaderByRawIndex(curCa.value).GetData(), (void*)_rawImage->GetBlobReaderByRawIndex(nextCa.value).GetData());		}		CustomAttributesCache* GenerateCustomAttributesCacheInternal(const Il2CppCustomAttributeTypeRange* typeRange)		{			CustomAttributeIndex index = (CustomAttributeIndex)(typeRange - (const Il2CppCustomAttributeTypeRange*)&_customAttributeHandles[0]);			IL2CPP_ASSERT(index >= 0 && index < (CustomAttributeIndex)_customAttributeHandles.size());			return GenerateCustomAttributesCacheInternal(index);		}		bool HasAttribute(CustomAttributeIndex index, Il2CppClass* attribute)		{			const Il2CppCustomAttributeTypeRange* typeRange = &_customAttributeHandles[DecodeMetadataIndex(index)];			return HasAttribute(typeRange, attribute);		}		bool HasAttribute(const Il2CppCustomAttributeTypeRange* typeRange, Il2CppClass* attribute)		{			CustomAttributesCache* attrCache = GenerateCustomAttributesCacheInternal(typeRange);			return HasAttribute(attrCache, attribute);		}		bool HasAttributeByToken(uint32_t token, Il2CppClass* attribute)		{			CustomAttributeIndex index = GetCustomAttributeIndex(token);			if (index == kCustomAttributeIndexInvalid)			{				return false;			}			CustomAttributesCache* attrCache = GenerateCustomAttributesCacheInternal(DecodeMetadataIndex(index));			return HasAttribute(attrCache, attribute);		}		bool HasAttribute(CustomAttributesCache* attrCache, Il2CppClass* attribute)		{			for (int i = 0; i < attrCache->count; i++)			{				Il2CppObject* attrObj = attrCache->attributes[i];				if (il2cpp::vm::Class::IsAssignableFrom(attribute, attrObj->klass))				{					return true;				}			}			return false;		}		CustomAttributesCache* GenerateCustomAttributesCacheInternal(CustomAttributeIndex index);#else		void InitCustomAttributeData(CustomAttributesInfo& cai, const Il2CppCustomAttributeTypeRange& dataRange);					il2cpp::metadata::CustomAttributeDataReader CreateCustomAttributeDataReader(Il2CppMetadataCustomAttributeHandle handle)		{			const Il2CppCustomAttributeTypeRange* dataRange = (const Il2CppCustomAttributeTypeRange*)handle;			IL2CPP_ASSERT(_tokenCustomAttributes.find(dataRange->token) != _tokenCustomAttributes.end());			CustomAttributesInfo& cai = _tokenCustomAttributes[dataRange->token];			if (!cai.inited)			{				InitCustomAttributeData(cai, *dataRange);			}#if HYBRIDCLR_UNITY_2022_OR_NEW			return il2cpp::metadata::CustomAttributeDataReader(_il2cppImage, cai.dataStartPtr, cai.dataEndPtr);#else			return il2cpp::metadata::CustomAttributeDataReader(cai.dataStartPtr, cai.dataEndPtr);#endif		}		std::tuple<void*, void*> CreateCustomAttributeDataTuple(const Il2CppCustomAttributeDataRange* dataRange)		{			IL2CPP_ASSERT(_tokenCustomAttributes.find(dataRange->token) != _tokenCustomAttributes.end());			CustomAttributesInfo& cai = _tokenCustomAttributes[dataRange->token];			if (!cai.inited)			{				InitCustomAttributeData(cai, *dataRange);			}			return std::tuple<void*, void*>(cai.dataStartPtr, cai.dataEndPtr);		}		std::tuple<void*, void*> CreateCustomAttributeDataTupleByToken(uint32_t token)		{			const Il2CppCustomAttributeTypeRange* dataRangeCur = (const Il2CppCustomAttributeTypeRange*)GetCustomAttributeTypeToken(token);			return dataRangeCur ? CreateCustomAttributeDataTuple(dataRangeCur) : std::tuple<void*, void*>(nullptr, nullptr);		}#if !HYBRIDCLR_UNITY_2022_OR_NEW		CustomAttributesCache* GenerateCustomAttributesCacheInternal(const Il2CppCustomAttributeTypeRange* typeRange)		{			CustomAttributeIndex index = (CustomAttributeIndex)(typeRange - (const Il2CppCustomAttributeTypeRange*)&_customAttributeHandles[0]);			IL2CPP_ASSERT(index >= 0 && index < (CustomAttributeIndex)_customAttributeHandles.size());			return GenerateCustomAttributesCacheInternal(index);		}		CustomAttributesCache* GenerateCustomAttributesCacheInternal(CustomAttributeIndex index);#endif		void BuildCustomAttributesData(CustomAttributesInfo& cai, const Il2CppCustomAttributeTypeRange& typeRange);		void ConvertILCustomAttributeData2Il2CppFormat(const MethodInfo* ctorMethod, BlobReader& reader);		void ConvertFixedArg(CustomAttributeDataWriter& writer, BlobReader& reader, const Il2CppType* type, bool writeType);		void ConvertBoxedValue(CustomAttributeDataWriter& writer, BlobReader& reader, bool writeType);		void ConvertSystemType(CustomAttributeDataWriter& writer, BlobReader& reader, bool writeType);		void WriteEncodeTypeEnum(CustomAttributeDataWriter& writer, const Il2CppType* type);		void GetFieldDeclaringTypeIndexAndFieldIndexByName(const Il2CppTypeDefinition* declaringType, const char* name, int32_t& typeIndex, int32_t& fieldIndex);		void GetPropertyDeclaringTypeIndexAndPropertyIndexByName(const Il2CppTypeDefinition* declaringType, const char* name, int32_t& typeIndex, int32_t& fieldIndex);#endif		Il2CppClass* GetTypeInfoFromTypeDefinitionRawIndex(uint32_t index);		const Il2CppType* GetInterfaceFromGlobalOffset(TypeInterfaceIndex offset);		const Il2CppType* GetInterfaceFromIndex(const Il2CppClass* klass, TypeInterfaceIndex index);		const Il2CppType* GetInterfaceFromOffset(const Il2CppClass* klass, TypeInterfaceIndex offset);		const Il2CppType* GetInterfaceFromOffset(const Il2CppTypeDefinition* typeDefine, TypeInterfaceIndex offset);		Il2CppInterfaceOffsetInfo GetInterfaceOffsetInfo(const Il2CppTypeDefinition* typeDefine, TypeInterfaceOffsetIndex index);		uint32_t AddIl2CppTypeCache(const Il2CppType* type);		uint32_t AddIl2CppGenericContainers(Il2CppGenericContainer& geneContainer);		const Il2CppType* GetModuleIl2CppType(uint32_t moduleRowIndex, uint32_t typeNamespace, uint32_t typeName, bool raiseExceptionIfNotFound) override;		void ReadFieldRefInfoFromFieldDefToken(uint32_t rowIndex, FieldRefInfo& ret) override;		void ReadMethodDefSig(BlobReader& reader, const Il2CppGenericContainer* klassGenericContainer, const Il2CppGenericContainer* methodGenericContainer, Il2CppMethodDefinition& methodDef, std::vector<ParamDetail>& paramArr);		void InitBasic(Il2CppImage* image);		void BuildIl2CppImage(Il2CppImage* image);		void BuildIl2CppAssembly(Il2CppAssembly* assembly);		void InitRuntimeMetadatas() override;	protected:		void InitTypeDefs_0();		void InitTypeDefs_1();		void InitTypeDefs_2();		void InitConsts();		void InitClass();		void InitParamDefs();		void InitGenericParamConstraintDefs();		void InitGenericParamDefs0();		void InitGenericParamDefs();		void InitFieldDefs();		void InitFieldLayouts();		void InitFieldRVAs();		void InitBlittables();		void InitMethodDefs0();		void InitMethodDefs();		void InitMethodImpls0();		void InitNestedClass();		void InitClassLayouts();		void InitCustomAttributes();		void InitProperties();		void InitEvents();		void InitMethodSemantics();		void InitInterfaces();		void InitVTables();		void ComputeBlittable(Il2CppTypeDefinition* def, std::vector<bool>& computFlags);		void ComputeVTable(TypeDefinitionDetail* tdd);		void SetIl2CppImage(Il2CppImage* image)		{			_il2cppImage = image;		}		Il2CppString* ReadSerString(BlobReader& reader);#if HYBRIDCLR_UNITY_2021_OR_NEW		bool ReadUTF8SerString(BlobReader& reader, std::string& s);#endif		Il2CppReflectionType* ReadSystemType(BlobReader& reader);		Il2CppObject* ReadBoxedValue(BlobReader& reader);		void ReadFixedArg(BlobReader& reader, const Il2CppType* argType, void* data);		void ReadCustomAttributeFieldOrPropType(BlobReader& reader, Il2CppType& type);#if !HYBRIDCLR_UNITY_2021_OR_NEW		void ConstructCustomAttribute(BlobReader& reader, Il2CppObject* obj, const MethodInfo* ctorMethod);#endif		bool _inited;		Il2CppImage* _il2cppImage;		const uint32_t _index;		std::vector<TypeDefinitionDetail> _typeDetails;		std::vector<Il2CppTypeDefinition> _typesDefines;		std::vector<Il2CppTypeDefinition> _exportedTypeDefines;		std::vector<const Il2CppType*> _types;		Il2CppHashMap<const Il2CppType*, uint32_t, Il2CppTypeHashShallow, Il2CppTypeEqualityComparerShallow> _type2Indexs;		std::vector<TypeIndex> _interfaceDefines;		std::vector<InterfaceOffsetInfo> _interfaceOffsets;		std::vector<Il2CppMethodDefinition> _methodDefines;		std::vector<ParamDetail> _params;		std::vector<int32_t>* _paramRawIndex2ActualParamIndex; // rawIindex = rowIndex - 1; because local function, param list count maybe less than actual method param count		std::vector<Il2CppParameterDefaultValue> _paramDefaultValues;		std::vector<Il2CppGenericParameter> _genericParams;		std::vector<TypeIndex> _genericConstraints; // raw TypeIndex		std::vector<Il2CppGenericContainer> _genericContainers;		std::vector<FieldDetail> _fieldDetails;		std::vector<Il2CppFieldDefaultValue> _fieldDefaultValues;		std::unordered_map<uint32_t, TbClassLayout> _classLayouts;		std::vector<uint32_t> _nestedTypeDefineIndexs;		// runtime data 		std::vector<Il2CppClass*> _classList;		Il2CppType2TypeDeclaringTreeMap _cacheTrees;#if HYBRIDCLR_UNITY_2021_OR_NEW		CustomAttributeDataWriter _constValues;#endif		std::unordered_map<uint32_t, CustomAttributesInfo> _tokenCustomAttributes;		std::vector<Il2CppCustomAttributeTypeRange> _customAttributeHandles;#if !HYBRIDCLR_UNITY_2022_OR_NEW		std::vector<CustomAttributesCache*> _customAttribtesCaches;#endif#if HYBRIDCLR_UNITY_2021_OR_NEW		CustomAttributeDataWriter _il2cppFormatCustomDataBlob;		CustomAttributeDataWriter _tempCtorArgBlob;		CustomAttributeDataWriter _tempFieldBlob;		CustomAttributeDataWriter _tempPropertyBlob;#endif		std::vector<CustomAttribute> _customAttribues;		std::vector<PropertyDetail> _propeties;		std::vector<EventDetail> _events;	};}}
 |