-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathProgram.cs
228 lines (214 loc) · 8.91 KB
/
Program.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
using System;
using System.IO;
using System.Linq;
using System.Runtime.InteropServices;
using System.Text.Json;
namespace Il2CppDumper
{
class Program
{
private static Config config;
static void Main(string[] args)
{
config = JsonSerializer.Deserialize<Config>(File.ReadAllText(Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "config.json")));
string il2cppPath = null;
string metadataPath = null;
string outputDir = null;
if (args.Length == 1 && (args[0] == "-h" || args[0] == "--help"))
{
ShowHelp();
return;
}
if (args.Length < 3)
{
ShowHelp();
return;
}
// Assumes args are in the order: <il2cppPath> <metadataPath> <outputDir>
il2cppPath = args[0];
metadataPath = args[1];
outputDir = args[2];
if (!File.Exists(il2cppPath) || !File.Exists(metadataPath))
{
Console.WriteLine("ERROR: File not found.");
ShowHelp();
return;
}
outputDir = Directory.Exists(outputDir) ? Path.GetFullPath(outputDir) + Path.DirectorySeparatorChar : AppDomain.CurrentDomain.BaseDirectory;
try
{
if (Init(il2cppPath, metadataPath, out var metadata, out var il2Cpp))
{
Dump(metadata, il2Cpp, outputDir);
}
}
catch (Exception e)
{
Console.WriteLine(e);
}
if (config.RequireAnyKey)
{
Console.WriteLine("Press any key to exit...");
Console.ReadKey(true);
}
}
// Keep only one ShowHelp method
static void ShowHelp()
{
Console.WriteLine($"usage: {AppDomain.CurrentDomain.FriendlyName} <il2cpp-file> <metadata-file> <output-directory>");
}
private static bool Init(string il2cppPath, string metadataPath, out Metadata metadata, out Il2Cpp il2Cpp)
{
Console.WriteLine("Initializing metadata...");
var metadataBytes = File.ReadAllBytes(metadataPath);
metadata = new Metadata(new MemoryStream(metadataBytes));
Console.WriteLine($"Metadata Version: {metadata.Version}");
Console.WriteLine("Initializing il2cpp file...");
var il2cppBytes = File.ReadAllBytes(il2cppPath);
var il2cppMagic = BitConverter.ToUInt32(il2cppBytes, 0);
var il2CppMemory = new MemoryStream(il2cppBytes);
switch (il2cppMagic)
{
default:
throw new NotSupportedException("ERROR: il2cpp file not supported.");
case 0x6D736100:
var web = new WebAssembly(il2CppMemory);
il2Cpp = web.CreateMemory();
break;
case 0x304F534E:
var nso = new NSO(il2CppMemory);
il2Cpp = nso.UnCompress();
break;
case 0x905A4D: //PE
il2Cpp = new PE(il2CppMemory);
break;
case 0x464c457f: //ELF
if (il2cppBytes[4] == 2) //ELF64
{
il2Cpp = new Elf64(il2CppMemory);
}
else
{
il2Cpp = new Elf(il2CppMemory);
}
break;
case 0xCAFEBABE: //FAT Mach-O
case 0xBEBAFECA:
var machofat = new MachoFat(new MemoryStream(il2cppBytes));
Console.Write("Select Platform: ");
for (var i = 0; i < machofat.fats.Length; i++)
{
var fat = machofat.fats[i];
Console.Write(fat.magic == 0xFEEDFACF ? $"{i + 1}.64bit " : $"{i + 1}.32bit ");
}
Console.WriteLine();
var key = Console.ReadKey(true);
var index = int.Parse(key.KeyChar.ToString()) - 1;
var magic = machofat.fats[index % 2].magic;
il2cppBytes = machofat.GetMacho(index % 2);
il2CppMemory = new MemoryStream(il2cppBytes);
if (magic == 0xFEEDFACF)
goto case 0xFEEDFACF;
else
goto case 0xFEEDFACE;
case 0xFEEDFACF: // 64bit Mach-O
il2Cpp = new Macho64(il2CppMemory);
break;
case 0xFEEDFACE: // 32bit Mach-O
il2Cpp = new Macho(il2CppMemory);
break;
}
var version = config.ForceIl2CppVersion ? config.ForceVersion : metadata.Version;
il2Cpp.SetProperties(version, metadata.metadataUsagesCount);
Console.WriteLine($"Il2Cpp Version: {il2Cpp.Version}");
if (config.ForceDump || il2Cpp.CheckDump())
{
if (il2Cpp is ElfBase elf)
{
Console.WriteLine("Detected this may be a dump file.");
Console.WriteLine("Input il2cpp dump address or input 0 to force continue:");
var DumpAddr = Convert.ToUInt64(Console.ReadLine(), 16);
if (DumpAddr != 0)
{
il2Cpp.ImageBase = DumpAddr;
il2Cpp.IsDumped = true;
if (!config.NoRedirectedPointer)
{
elf.Reload();
}
}
}
else
{
il2Cpp.IsDumped = true;
}
}
Console.WriteLine("Searching...");
try
{
var flag = il2Cpp.PlusSearch(metadata.methodDefs.Count(x => x.methodIndex >= 0), metadata.typeDefs.Length, metadata.imageDefs.Length);
if (RuntimeInformation.IsOSPlatform(OSPlatform.Windows))
{
if (!flag && il2Cpp is PE)
{
Console.WriteLine("Use custom PE loader");
il2Cpp = PELoader.Load(il2cppPath);
il2Cpp.SetProperties(version, metadata.metadataUsagesCount);
flag = il2Cpp.PlusSearch(metadata.methodDefs.Count(x => x.methodIndex >= 0), metadata.typeDefs.Length, metadata.imageDefs.Length);
}
}
if (!flag)
{
flag = il2Cpp.Search();
}
if (!flag)
{
flag = il2Cpp.SymbolSearch();
}
if (!flag)
{
Console.WriteLine("ERROR: Can't use auto mode to process file, try manual mode.");
Console.Write("Input CodeRegistration: ");
var codeRegistration = Convert.ToUInt64(Console.ReadLine(), 16);
Console.Write("Input MetadataRegistration: ");
var metadataRegistration = Convert.ToUInt64(Console.ReadLine(), 16);
il2Cpp.Init(codeRegistration, metadataRegistration);
}
if (il2Cpp.Version >= 27 && il2Cpp.IsDumped)
{
var typeDef = metadata.typeDefs[0];
var il2CppType = il2Cpp.types[typeDef.byvalTypeIndex];
metadata.ImageBase = il2CppType.data.typeHandle - metadata.header.typeDefinitionsOffset;
}
}
catch (Exception e)
{
Console.WriteLine(e);
Console.WriteLine("ERROR: An error occurred while processing.");
return false;
}
return true;
}
private static void Dump(Metadata metadata, Il2Cpp il2Cpp, string outputDir)
{
Console.WriteLine("Dumping...");
var executor = new Il2CppExecutor(metadata, il2Cpp);
var decompiler = new Il2CppDecompiler(executor);
decompiler.Decompile(config, outputDir);
Console.WriteLine("Done!");
if (config.GenerateStruct)
{
Console.WriteLine("Generate struct...");
var scriptGenerator = new StructGenerator(executor);
scriptGenerator.WriteScript(outputDir);
Console.WriteLine("Done!");
}
if (config.GenerateDummyDll)
{
Console.WriteLine("Generate dummy dll...");
DummyAssemblyExporter.Export(executor, outputDir, config.DummyDllAddToken);
Console.WriteLine("Done!");
}
}
}
}