Re-factor Metadata and file format code
This commit is contained in:
@@ -20,11 +20,11 @@ namespace Il2CppInspector
|
|||||||
|
|
||||||
for (int imageIndex = 0; imageIndex < metadata.Images.Length; imageIndex++) {
|
for (int imageIndex = 0; imageIndex < metadata.Images.Length; imageIndex++) {
|
||||||
var imageDef = metadata.Images[imageIndex];
|
var imageDef = metadata.Images[imageIndex];
|
||||||
writer.Write($"// Image {imageIndex}: {metadata.GetImageName(imageDef)} - {imageDef.typeStart}\n");
|
writer.Write($"// Image {imageIndex}: {metadata.Strings[imageDef.nameIndex]} - {imageDef.typeStart}\n");
|
||||||
}
|
}
|
||||||
for (int idx = 0; idx < metadata.Types.Length; ++idx) {
|
for (int idx = 0; idx < metadata.Types.Length; ++idx) {
|
||||||
var typeDef = metadata.Types[idx];
|
var typeDef = metadata.Types[idx];
|
||||||
writer.Write($"// Namespace: {metadata.GetTypeNamespace(typeDef)}\n");
|
writer.Write($"// Namespace: {metadata.Strings[typeDef.namespaceIndex]}\n");
|
||||||
if ((typeDef.flags & DefineConstants.TYPE_ATTRIBUTE_SERIALIZABLE) != 0)
|
if ((typeDef.flags & DefineConstants.TYPE_ATTRIBUTE_SERIALIZABLE) != 0)
|
||||||
writer.Write("[Serializable]\n");
|
writer.Write("[Serializable]\n");
|
||||||
if ((typeDef.flags & DefineConstants.TYPE_ATTRIBUTE_VISIBILITY_MASK) ==
|
if ((typeDef.flags & DefineConstants.TYPE_ATTRIBUTE_VISIBILITY_MASK) ==
|
||||||
@@ -38,13 +38,12 @@ namespace Il2CppInspector
|
|||||||
writer.Write("interface ");
|
writer.Write("interface ");
|
||||||
else
|
else
|
||||||
writer.Write("class ");
|
writer.Write("class ");
|
||||||
writer.Write($"{metadata.GetTypeName(typeDef)} // TypeDefIndex: {idx}\n{{\n");
|
writer.Write($"{metadata.Strings[typeDef.nameIndex]} // TypeDefIndex: {idx}\n{{\n");
|
||||||
writer.Write("\t// Fields\n");
|
writer.Write("\t// Fields\n");
|
||||||
var fieldEnd = typeDef.fieldStart + typeDef.field_count;
|
var fieldEnd = typeDef.fieldStart + typeDef.field_count;
|
||||||
for (int i = typeDef.fieldStart; i < fieldEnd; ++i) {
|
for (int i = typeDef.fieldStart; i < fieldEnd; ++i) {
|
||||||
var pField = metadata.Fields[i];
|
var pField = metadata.Fields[i];
|
||||||
var pType = il2cpp.GetTypeFromTypeIndex(pField.typeIndex);
|
var pType = il2cpp.GetTypeFromTypeIndex(pField.typeIndex);
|
||||||
var pDefault = metadata.GetFieldDefaultFromIndex(i);
|
|
||||||
writer.Write("\t");
|
writer.Write("\t");
|
||||||
if ((pType.attrs & DefineConstants.FIELD_ATTRIBUTE_PRIVATE) ==
|
if ((pType.attrs & DefineConstants.FIELD_ATTRIBUTE_PRIVATE) ==
|
||||||
DefineConstants.FIELD_ATTRIBUTE_PRIVATE)
|
DefineConstants.FIELD_ATTRIBUTE_PRIVATE)
|
||||||
@@ -56,59 +55,12 @@ namespace Il2CppInspector
|
|||||||
writer.Write("static ");
|
writer.Write("static ");
|
||||||
if ((pType.attrs & DefineConstants.FIELD_ATTRIBUTE_INIT_ONLY) != 0)
|
if ((pType.attrs & DefineConstants.FIELD_ATTRIBUTE_INIT_ONLY) != 0)
|
||||||
writer.Write("readonly ");
|
writer.Write("readonly ");
|
||||||
writer.Write($"{il2cpp.GetTypeName(pType)} {metadata.GetString(pField.nameIndex)}");
|
writer.Write($"{il2cpp.GetTypeName(pType)} {metadata.Strings[pField.nameIndex]}");
|
||||||
if (pDefault != null && pDefault.dataIndex != -1) {
|
object multi = il2cpp.GetDefaultValueForField(i);
|
||||||
var pointer = metadata.GetDefaultValueFromIndex(pDefault.dataIndex);
|
if (multi is string)
|
||||||
Il2CppType pTypeToUse = il2cpp.GetTypeFromTypeIndex(pDefault.typeIndex);
|
writer.Write($" = \"{multi}\"");
|
||||||
if (pointer > 0) {
|
else if (multi != null)
|
||||||
metadata.Position = pointer;
|
writer.Write($" = {multi}");
|
||||||
object multi = null;
|
|
||||||
switch (pTypeToUse.type) {
|
|
||||||
case Il2CppTypeEnum.IL2CPP_TYPE_BOOLEAN:
|
|
||||||
multi = metadata.ReadBoolean();
|
|
||||||
break;
|
|
||||||
case Il2CppTypeEnum.IL2CPP_TYPE_U1:
|
|
||||||
case Il2CppTypeEnum.IL2CPP_TYPE_I1:
|
|
||||||
multi = metadata.ReadByte();
|
|
||||||
break;
|
|
||||||
case Il2CppTypeEnum.IL2CPP_TYPE_CHAR:
|
|
||||||
multi = metadata.ReadChar();
|
|
||||||
break;
|
|
||||||
case Il2CppTypeEnum.IL2CPP_TYPE_U2:
|
|
||||||
multi = metadata.ReadUInt16();
|
|
||||||
break;
|
|
||||||
case Il2CppTypeEnum.IL2CPP_TYPE_I2:
|
|
||||||
multi = metadata.ReadInt16();
|
|
||||||
break;
|
|
||||||
case Il2CppTypeEnum.IL2CPP_TYPE_U4:
|
|
||||||
multi = metadata.ReadUInt32();
|
|
||||||
break;
|
|
||||||
case Il2CppTypeEnum.IL2CPP_TYPE_I4:
|
|
||||||
multi = metadata.ReadInt32();
|
|
||||||
break;
|
|
||||||
case Il2CppTypeEnum.IL2CPP_TYPE_U8:
|
|
||||||
multi = metadata.ReadUInt64();
|
|
||||||
break;
|
|
||||||
case Il2CppTypeEnum.IL2CPP_TYPE_I8:
|
|
||||||
multi = metadata.ReadInt64();
|
|
||||||
break;
|
|
||||||
case Il2CppTypeEnum.IL2CPP_TYPE_R4:
|
|
||||||
multi = metadata.ReadSingle();
|
|
||||||
break;
|
|
||||||
case Il2CppTypeEnum.IL2CPP_TYPE_R8:
|
|
||||||
multi = metadata.ReadDouble();
|
|
||||||
break;
|
|
||||||
case Il2CppTypeEnum.IL2CPP_TYPE_STRING:
|
|
||||||
var uiLen = metadata.ReadInt32();
|
|
||||||
multi = Encoding.UTF8.GetString(metadata.ReadBytes(uiLen));
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
if (multi is string)
|
|
||||||
writer.Write($" = \"{multi}\"");
|
|
||||||
else if (multi != null)
|
|
||||||
writer.Write($" = {multi}");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
writer.Write("; // 0x{0:x}\n",
|
writer.Write("; // 0x{0:x}\n",
|
||||||
il2cpp.GetFieldOffsetFromIndex(idx, i - typeDef.fieldStart));
|
il2cpp.GetFieldOffsetFromIndex(idx, i - typeDef.fieldStart));
|
||||||
}
|
}
|
||||||
@@ -129,10 +81,10 @@ namespace Il2CppInspector
|
|||||||
if ((methodDef.flags & DefineConstants.METHOD_ATTRIBUTE_STATIC) != 0)
|
if ((methodDef.flags & DefineConstants.METHOD_ATTRIBUTE_STATIC) != 0)
|
||||||
writer.Write("static ");
|
writer.Write("static ");
|
||||||
|
|
||||||
writer.Write($"{il2cpp.GetTypeName(pReturnType)} {metadata.GetString(methodDef.nameIndex)}(");
|
writer.Write($"{il2cpp.GetTypeName(pReturnType)} {metadata.Strings[methodDef.nameIndex]}(");
|
||||||
for (int j = 0; j < methodDef.parameterCount; ++j) {
|
for (int j = 0; j < methodDef.parameterCount; ++j) {
|
||||||
Il2CppParameterDefinition pParam = metadata.parameterDefs[methodDef.parameterStart + j];
|
Il2CppParameterDefinition pParam = metadata.Params[methodDef.parameterStart + j];
|
||||||
string szParamName = metadata.GetString(pParam.nameIndex);
|
string szParamName = metadata.Strings[pParam.nameIndex];
|
||||||
Il2CppType pType = il2cpp.GetTypeFromTypeIndex(pParam.typeIndex);
|
Il2CppType pType = il2cpp.GetTypeFromTypeIndex(pParam.typeIndex);
|
||||||
string szTypeName = il2cpp.GetTypeName(pType);
|
string szTypeName = il2cpp.GetTypeName(pType);
|
||||||
if ((pType.attrs & DefineConstants.PARAM_ATTRIBUTE_OPTIONAL) != 0)
|
if ((pType.attrs & DefineConstants.PARAM_ATTRIBUTE_OPTIONAL) != 0)
|
||||||
|
|||||||
@@ -7,6 +7,8 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.IO;
|
using System.IO;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Text;
|
||||||
|
|
||||||
namespace Il2CppInspector
|
namespace Il2CppInspector
|
||||||
{
|
{
|
||||||
@@ -75,12 +77,12 @@ namespace Il2CppInspector
|
|||||||
string ret;
|
string ret;
|
||||||
if (pType.type == Il2CppTypeEnum.IL2CPP_TYPE_CLASS || pType.type == Il2CppTypeEnum.IL2CPP_TYPE_VALUETYPE) {
|
if (pType.type == Il2CppTypeEnum.IL2CPP_TYPE_CLASS || pType.type == Il2CppTypeEnum.IL2CPP_TYPE_VALUETYPE) {
|
||||||
Il2CppTypeDefinition klass = Metadata.Types[pType.data.klassIndex];
|
Il2CppTypeDefinition klass = Metadata.Types[pType.data.klassIndex];
|
||||||
ret = Metadata.GetString(klass.nameIndex);
|
ret = Metadata.Strings[klass.nameIndex];
|
||||||
}
|
}
|
||||||
else if (pType.type == Il2CppTypeEnum.IL2CPP_TYPE_GENERICINST) {
|
else if (pType.type == Il2CppTypeEnum.IL2CPP_TYPE_GENERICINST) {
|
||||||
Il2CppGenericClass generic_class = Code.Image.ReadMappedObject<Il2CppGenericClass>(pType.data.generic_class);
|
Il2CppGenericClass generic_class = Code.Image.ReadMappedObject<Il2CppGenericClass>(pType.data.generic_class);
|
||||||
Il2CppTypeDefinition pMainDef = Metadata.Types[generic_class.typeDefinitionIndex];
|
Il2CppTypeDefinition pMainDef = Metadata.Types[generic_class.typeDefinitionIndex];
|
||||||
ret = Metadata.GetString(pMainDef.nameIndex);
|
ret = Metadata.Strings[pMainDef.nameIndex];
|
||||||
var typeNames = new List<string>();
|
var typeNames = new List<string>();
|
||||||
Il2CppGenericInst pInst = Code.Image.ReadMappedObject<Il2CppGenericInst>(generic_class.context.class_inst);
|
Il2CppGenericInst pInst = Code.Image.ReadMappedObject<Il2CppGenericInst>(generic_class.context.class_inst);
|
||||||
var pointers = Code.Image.ReadMappedArray<uint>(pInst.type_argv, (int)pInst.type_argc);
|
var pointers = Code.Image.ReadMappedArray<uint>(pInst.type_argv, (int)pInst.type_argc);
|
||||||
@@ -136,6 +138,63 @@ namespace Il2CppInspector
|
|||||||
return Code.Image.Stream.ReadInt32();
|
return Code.Image.Stream.ReadInt32();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public object GetDefaultValueForField(int fieldIndex) {
|
||||||
|
var def = Metadata.FieldDefaultValues.FirstOrDefault(x => x.fieldIndex == fieldIndex);
|
||||||
|
|
||||||
|
if (def == null || def.dataIndex == -1)
|
||||||
|
return null;
|
||||||
|
|
||||||
|
var pValue = Metadata.Header.fieldAndParameterDefaultValueDataOffset + def.dataIndex;
|
||||||
|
Il2CppType type = GetTypeFromTypeIndex(def.typeIndex);
|
||||||
|
if (pValue == 0)
|
||||||
|
return null;
|
||||||
|
|
||||||
|
object value = null;
|
||||||
|
Metadata.Position = pValue;
|
||||||
|
switch (type.type) {
|
||||||
|
case Il2CppTypeEnum.IL2CPP_TYPE_BOOLEAN:
|
||||||
|
value = Metadata.ReadBoolean();
|
||||||
|
break;
|
||||||
|
case Il2CppTypeEnum.IL2CPP_TYPE_U1:
|
||||||
|
case Il2CppTypeEnum.IL2CPP_TYPE_I1:
|
||||||
|
value = Metadata.ReadByte();
|
||||||
|
break;
|
||||||
|
case Il2CppTypeEnum.IL2CPP_TYPE_CHAR:
|
||||||
|
value = Metadata.ReadChar();
|
||||||
|
break;
|
||||||
|
case Il2CppTypeEnum.IL2CPP_TYPE_U2:
|
||||||
|
value = Metadata.ReadUInt16();
|
||||||
|
break;
|
||||||
|
case Il2CppTypeEnum.IL2CPP_TYPE_I2:
|
||||||
|
value = Metadata.ReadInt16();
|
||||||
|
break;
|
||||||
|
case Il2CppTypeEnum.IL2CPP_TYPE_U4:
|
||||||
|
value = Metadata.ReadUInt32();
|
||||||
|
break;
|
||||||
|
case Il2CppTypeEnum.IL2CPP_TYPE_I4:
|
||||||
|
value = Metadata.ReadInt32();
|
||||||
|
break;
|
||||||
|
case Il2CppTypeEnum.IL2CPP_TYPE_U8:
|
||||||
|
value = Metadata.ReadUInt64();
|
||||||
|
break;
|
||||||
|
case Il2CppTypeEnum.IL2CPP_TYPE_I8:
|
||||||
|
value = Metadata.ReadInt64();
|
||||||
|
break;
|
||||||
|
case Il2CppTypeEnum.IL2CPP_TYPE_R4:
|
||||||
|
value = Metadata.ReadSingle();
|
||||||
|
break;
|
||||||
|
case Il2CppTypeEnum.IL2CPP_TYPE_R8:
|
||||||
|
value = Metadata.ReadDouble();
|
||||||
|
break;
|
||||||
|
case Il2CppTypeEnum.IL2CPP_TYPE_STRING:
|
||||||
|
var uiLen = Metadata.ReadInt32();
|
||||||
|
value = Encoding.UTF8.GetString(Metadata.ReadBytes(uiLen));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
private readonly string[] szTypeString =
|
private readonly string[] szTypeString =
|
||||||
{
|
{
|
||||||
"END",
|
"END",
|
||||||
|
|||||||
@@ -6,8 +6,8 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
using System;
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
using System.IO;
|
using System.IO;
|
||||||
using System.Linq;
|
|
||||||
using System.Reflection;
|
using System.Reflection;
|
||||||
using NoisyCowStudios.Bin2Object;
|
using NoisyCowStudios.Bin2Object;
|
||||||
|
|
||||||
@@ -15,18 +15,15 @@ namespace Il2CppInspector
|
|||||||
{
|
{
|
||||||
public class Metadata : BinaryObjectReader
|
public class Metadata : BinaryObjectReader
|
||||||
{
|
{
|
||||||
private Il2CppGlobalMetadataHeader pMetadataHdr;
|
public Il2CppGlobalMetadataHeader Header;
|
||||||
|
|
||||||
public Il2CppImageDefinition[] Images { get; }
|
public Il2CppImageDefinition[] Images { get; }
|
||||||
public Il2CppTypeDefinition[] Types { get; }
|
public Il2CppTypeDefinition[] Types { get; }
|
||||||
public Il2CppMethodDefinition[] Methods { get; }
|
public Il2CppMethodDefinition[] Methods { get; }
|
||||||
public Il2CppParameterDefinition[] parameterDefs;
|
public Il2CppParameterDefinition[] Params { get; }
|
||||||
public Il2CppFieldDefinition[] Fields { get; }
|
public Il2CppFieldDefinition[] Fields { get; }
|
||||||
public Il2CppFieldDefaultValue[] fieldDefaultValues;
|
public Il2CppFieldDefaultValue[] FieldDefaultValues { get; }
|
||||||
|
public Dictionary<int, string> Strings { get; } = new Dictionary<int, string>();
|
||||||
public string GetImageName(Il2CppImageDefinition image) => GetString(image.nameIndex);
|
|
||||||
public string GetTypeNamespace(Il2CppTypeDefinition type) => GetString(type.namespaceIndex);
|
|
||||||
public string GetTypeName(Il2CppTypeDefinition type) => GetString(type.nameIndex);
|
|
||||||
|
|
||||||
public Metadata(Stream stream) : base(stream)
|
public Metadata(Stream stream) : base(stream)
|
||||||
{
|
{
|
||||||
@@ -40,40 +37,24 @@ namespace Il2CppInspector
|
|||||||
|
|
||||||
// Rewind and read metadata header in full
|
// Rewind and read metadata header in full
|
||||||
Position -= 8;
|
Position -= 8;
|
||||||
pMetadataHdr = ReadObject<Il2CppGlobalMetadataHeader>();
|
Header = ReadObject<Il2CppGlobalMetadataHeader>();
|
||||||
if (Version != 21 && Version != 22 && Version != 23)
|
if (Version != 21 && Version != 22 && Version != 23)
|
||||||
{
|
{
|
||||||
throw new Exception($"ERROR: Metadata file supplied is not a supported version ({pMetadataHdr.version}).");
|
throw new Exception($"ERROR: Metadata file supplied is not a supported version ({Header.version}).");
|
||||||
}
|
}
|
||||||
|
|
||||||
var uiImageCount = pMetadataHdr.imagesCount / Sizeof(typeof(Il2CppImageDefinition));
|
// Load all the relevant metadata using offsets provided in the header
|
||||||
var uiNumTypes = pMetadataHdr.typeDefinitionsCount / Sizeof(typeof(Il2CppTypeDefinition));
|
Images = ReadArray<Il2CppImageDefinition>(Header.imagesOffset, Header.imagesCount / Sizeof(typeof(Il2CppImageDefinition)));
|
||||||
Images = ReadArray<Il2CppImageDefinition>(pMetadataHdr.imagesOffset, uiImageCount);
|
Types = ReadArray<Il2CppTypeDefinition>(Header.typeDefinitionsOffset, Header.typeDefinitionsCount / Sizeof(typeof(Il2CppTypeDefinition)));
|
||||||
//GetTypeDefFromIndex
|
Methods = ReadArray<Il2CppMethodDefinition>(Header.methodsOffset, Header.methodsCount / Sizeof(typeof(Il2CppMethodDefinition)));
|
||||||
Types = ReadArray<Il2CppTypeDefinition>(pMetadataHdr.typeDefinitionsOffset, uiNumTypes);
|
Params = ReadArray<Il2CppParameterDefinition>(Header.parametersOffset, Header.parametersCount / Sizeof(typeof(Il2CppParameterDefinition)));
|
||||||
//GetMethodDefinition
|
Fields = ReadArray<Il2CppFieldDefinition>(Header.fieldsOffset, Header.fieldsCount / Sizeof(typeof(Il2CppFieldDefinition)));
|
||||||
Methods = ReadArray<Il2CppMethodDefinition>(pMetadataHdr.methodsOffset, pMetadataHdr.methodsCount / Sizeof(typeof(Il2CppMethodDefinition)));
|
FieldDefaultValues = ReadArray<Il2CppFieldDefaultValue>(Header.fieldDefaultValuesOffset, Header.fieldDefaultValuesCount / Sizeof(typeof(Il2CppFieldDefaultValue)));
|
||||||
//GetParameterFromIndex
|
// TODO: Events, Properties, ParameterDefaultValue, GenericParameters, ParameterConstraints, GenericContainers, Interfaces, MetadataUsage, CustomAttributes
|
||||||
parameterDefs = ReadArray<Il2CppParameterDefinition>(pMetadataHdr.parametersOffset, pMetadataHdr.parametersCount / Sizeof(typeof(Il2CppParameterDefinition)));
|
|
||||||
//GetFieldDefFromIndex
|
|
||||||
Fields = ReadArray<Il2CppFieldDefinition>(pMetadataHdr.fieldsOffset, pMetadataHdr.fieldsCount / Sizeof(typeof(Il2CppFieldDefinition)));
|
|
||||||
//GetFieldDefaultFromIndex
|
|
||||||
fieldDefaultValues = ReadArray<Il2CppFieldDefaultValue>(pMetadataHdr.fieldDefaultValuesOffset, pMetadataHdr.fieldDefaultValuesCount / Sizeof(typeof(Il2CppFieldDefaultValue)));
|
|
||||||
}
|
|
||||||
|
|
||||||
public Il2CppFieldDefaultValue GetFieldDefaultFromIndex(int idx)
|
Position = Header.stringOffset;
|
||||||
{
|
while (Position < Header.stringOffset + Header.stringCount)
|
||||||
return fieldDefaultValues.FirstOrDefault(x => x.fieldIndex == idx);
|
Strings.Add((int)Position - Header.stringOffset, ReadNullTerminatedString());
|
||||||
}
|
|
||||||
|
|
||||||
public int GetDefaultValueFromIndex(int idx)
|
|
||||||
{
|
|
||||||
return pMetadataHdr.fieldAndParameterDefaultValueDataOffset + idx;
|
|
||||||
}
|
|
||||||
|
|
||||||
public string GetString(int idx)
|
|
||||||
{
|
|
||||||
return ReadNullTerminatedString(pMetadataHdr.stringOffset + idx);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private int Sizeof(Type type)
|
private int Sizeof(Type type)
|
||||||
|
|||||||
Reference in New Issue
Block a user