Output: IDA Python script generation

IL2CPP: Implement MetadataUsages
This commit is contained in:
Carter Bush
2020-01-27 07:31:21 +11:00
committed by Katy Coe
parent 89a0b2e97f
commit 8045f2cfd7
6 changed files with 204 additions and 2 deletions

View File

@@ -0,0 +1,125 @@
using System;
using System.Collections.Generic;
using System.Linq;
using System.IO;
using System.Text;
using Il2CppInspector.Reflection;
namespace Il2CppInspector
{
public class Il2CppIDAScriptDumper
{
private Il2CppModel model;
public Il2CppIDAScriptDumper(Il2CppModel model) => this.model = model;
#region Writing
public void WriteScriptToFile(string outputFile) {
using (var fs = new FileStream(outputFile, FileMode.Create))
using (var sw = new StreamWriter(fs, Encoding.UTF8)) {
writeSectionHeader(sw, "Preamble");
writePreamble(sw);
writeSectionHeader(sw, "Methods");
writeMethods(sw, this.model.Types);
writeSectionHeader(sw, "Usages");
writeUsages(sw, this.model);
}
}
private static void writePreamble(StreamWriter writer) {
writeLines(writer,
@"#encoding: utf-8
import idaapi
def SetString(addr, comm):
global index
name = 'StringLiteral_' + str(index)
ret = idc.set_name(addr, name, SN_NOWARN)
idc.set_cmt(addr, comm, 1)
def SetName(addr, name):
ret = idc.set_name(addr, name, SN_NOWARN | SN_NOCHECK)
if ret == 0:
new_name = name + '_' + str(addr)
ret = idc.set_name(addr, new_name, SN_NOWARN | SN_NOCHECK)
index = 1
"
);
}
private static void writeMethods(StreamWriter writer, IEnumerable<TypeInfo> types) {
foreach (var type in types.Where(t => t != null)) {
writeMethods(writer, type.Name, type.DeclaredConstructors);
writeMethods(writer, type.Name, type.DeclaredMethods);
}
}
private static void writeMethods(StreamWriter writer, string typeName, IEnumerable<MethodBase> methods) {
foreach (var method in methods.Where(m => m.VirtualAddress.HasValue)) {
writeLines(writer,
$"SetName({method.VirtualAddress.Value.Start.ToAddressString()}, '{typeName}$${method.Name}')"
);
}
}
private static void writeUsages(StreamWriter writer, Il2CppModel model) {
foreach (var usage in model.Package.MetadataUsages) {
switch (usage.Type) {
case MetadataUsageType.TypeInfo:
case MetadataUsageType.Type:
var type = model.GetTypeFromUsage(usage.SourceIndex);
writeLines(writer,
$"SetName({model.Package.BinaryMetadataUsages[usage.DestinationIndex].ToAddressString()}, 'Class${type.Name}')"
);
break;
case MetadataUsageType.MethodDef:
var method = model.MethodsByDefinitionIndex[usage.SourceIndex];
writeLines(writer,
$"SetName({model.Package.BinaryMetadataUsages[usage.DestinationIndex].ToAddressString()}, 'Method${method.DeclaringType.Name}.{method.Name}')"
);
break;
case MetadataUsageType.FieldInfo:
var field = model.Package.Fields[usage.SourceIndex];
type = model.GetTypeFromUsage(field.typeIndex);
var fieldName = model.Package.Strings[field.nameIndex];
writeLines(writer,
$"SetName({model.Package.BinaryMetadataUsages[usage.DestinationIndex].ToAddressString()}, 'Field${type.Name}.{fieldName}')"
);
break;
case MetadataUsageType.StringLiteral:
// TODO: String literals
break;
case MetadataUsageType.MethodRef:
var methodSpec = model.Package.MethodSpecs[usage.SourceIndex];
method = model.MethodsByDefinitionIndex[methodSpec.methodDefinitionIndex];
type = method.DeclaringType;
writeLines(writer,
$"SetName({model.Package.BinaryMetadataUsages[usage.DestinationIndex].ToAddressString()}, 'Method${type.Name}.{method.Name}')"
);
break;
default:
break;
}
}
}
private static void writeSectionHeader(StreamWriter writer, string sectionName) {
writeLines(writer,
$"# SECTION: {sectionName}",
$"# -----------------------------"
);
}
private static void writeLines(StreamWriter writer, params string[] lines) {
foreach (var line in lines) {
writer.WriteLine(line);
}
}
#endregion
}
}

View File

@@ -25,7 +25,7 @@ namespace Il2CppInspector
[Option('c', "cs-out", Required = false, HelpText = "C# output file (when using single-file layout) or path (when using per namespace, assembly or class layout)", Default = "types.cs")]
public string CSharpOutPath { get; set; }
[Option('p', "py-out", Required = false, Hidden = true, HelpText = "IDA Python script output file", Default = "ida.py")]
[Option('p', "py-out", Required = false, HelpText = "IDA Python script output file", Default = "ida.py")]
public string PythonOutFile { get; set; }
[Option('e', "exclude-namespaces", Required = false, Separator = ',', HelpText = "Comma-separated list of namespaces to suppress in C# output, or 'none' to include all namespaces",
@@ -208,7 +208,10 @@ namespace Il2CppInspector
}
// IDA Python script output
// TODO: IDA Python script output
using (var scriptDumperTimer = new Benchmark("IDA Python Script Dumper")) {
var idaWriter = new Il2CppIDAScriptDumper(model);
idaWriter.WriteScriptToFile(options.PythonOutFile);
}
}
// Success exit code