-
Notifications
You must be signed in to change notification settings - Fork 25
/
Il2CppMemoryDumper.sh
241 lines (200 loc) · 7.76 KB
/
Il2CppMemoryDumper.sh
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
229
230
231
232
233
234
235
236
237
238
239
240
241
#!/system/bin/sh
echo "########################"
echo "# Il2Cpp Memory Dumper #"
echo "# by NekoYuzu neko.ink #"
echo "########################"
if [[ $1 == "" ]]; then
echo "* Usage: $0 <package> [output]"
exit
fi
package=$1
if [[ $2 == "" ]]; then
out=/sdcard/dump
else
out=$2
fi
echo "- Target package: $package"
echo "- Output directory: $out"
mkdir -p "$out"
user=$(am get-current-user)
pid=$(ps -ef | grep $package | grep u$user | awk '{print $2}')
if [[ $pid == "" ]]; then
echo "! Target package of current user ($user) not found, is process running?"
exit
fi
echo "- Found target process: $pid"
targets="/dev/zero global-metadata.dat libil2cpp.so "$(getprop ro.product.cpu.abilist | awk -F',' '{for (i = 1; i <= NF; i++) {gsub(/-/, "_"); print "split_config."$i".apk"}}')
for target in $targets; do
local maps=$(grep $target /proc/$pid/maps | awk -v OFS='|' '{for (i = 1; i <= NF; i+=6) {print $i,$(i+1),$(i+2),$(i+3),$(i+4),$(i+5)}}')
if [[ $maps != "" ]]; then
if [[ $mem_list != "" ]]; then
mem_list="${mem_list} ${maps}"
else
mem_list=$maps
fi
fi
done
cp /proc/$pid/maps "$out/${package}_maps.txt"
SYS_PAGESIZE=$(getconf PAGESIZE)
HEX_PAGESIZE=$(printf "%x" $SYS_PAGESIZE)
echo "- Starting dump process..."
lastFile=
lastEnd=
metadataOffset=
for memory in $mem_list; do
local range=$(echo $memory | awk -F'|' '{print $1}')
if [[ $range == "(deleted)" ]]; then
continue
fi
local offset=$(echo $range | awk -F'-' '{print toupper($1)}')
local end=$(echo $range | awk -F'-' '{print toupper($2)}')
local memIndicator=$(echo $memory | awk -v OFS=',' -F'|' '{print $4,$5}')
local memPath=$(echo $memory | awk -F'|' '{print $6}')
local memName=$(echo $memPath | awk -F'/' '{print $NF}')
local fileExt="bin"
if [[ $memName == "zero" ]]; then
memName="global-metadata.dat"
fi
dd if="/proc/$pid/mem" bs=1 skip=$(echo "ibase=16;$offset" | bc) count=4 of="${out}/tmp" 2>/dev/null
if [[ $memName == "global-metadata.dat" ]]; then
if [[ $(cat "${out}/tmp") == $(echo -ne "\xAF\x1B\xB1\xFA") ]]; then
METADATA_SANITY_MATCHED=true
fi
dd if="/proc/$pid/mem" bs=1 skip=$(echo "ibase=16;${offset}+4" | bc) count=4 of="${out}/tmp" 2>/dev/null
local METADATA_VERSION=$(echo "ibase=16;$(cat "${out}/tmp" | xxd -e | awk -F' ' '{print $2}')" | bc)
if [[ $METADATA_VERSION -lt 100 ]]; then
METADATA_VERSION_VAILDATED=true
fi
if [[ $METADATA_VERSION_VAILDATED == "true" ]]; then
fileExt="dat"
if [[ $METADATA_SANITY_MATCHED == "true" ]]; then
echo "- Il2Cpp metadata version ${METADATA_VERSION} was found at ${offset}, starting dump..."
else
echo "- Il2Cpp metadata version ${METADATA_VERSION} at offset ${offset} sanity mismatch, trying to dump..."
fi
else
if [[ $memPath == "/dev/zero" ]]; then
memName="zero"
fi
echo "- Illegal version of Il2Cpp (${METADATA_VERSION}) at offset ${offset}, trying to dump as binary file..."
fi
else
if [[ $(cat "${out}/tmp") == $(echo -ne "\x7F\x45\x4C\x46") ]]; then
fileExt="so"
local EI_CLASS="None"
dd if="/proc/$pid/mem" bs=1 skip=$(echo "ibase=16;${offset}+4" | bc) count=1 of="${out}/tmp" 2>/dev/null
tmp=$(cat "${out}/tmp")
if [[ $tmp == $(echo -ne "\x01") ]]; then
EI_CLASS="32"
elif [[ $tmp == $(echo -ne "\x02") ]]; then
EI_CLASS="64"
fi
local EI_DATA="None"
dd if="/proc/$pid/mem" bs=1 skip=$(echo "ibase=16;${offset}+5" | bc) count=1 of="${out}/tmp" 2>/dev/null
tmp=$(cat "${out}/tmp")
if [[ $tmp == $(echo -ne "\x01") ]]; then
EI_DATA="LSB"
elif [[ $tmp == $(echo -ne "\x02") ]]; then
EI_DATA="MSB"
fi
local E_TYPE="None"
dd if="/proc/$pid/mem" bs=1 skip=$(echo "ibase=16;${offset}+F" | bc) count=2 of="${out}/tmp" 2>/dev/null
tmp=$(cat "${out}/tmp")
if [[ $tmp == $(echo -ne "\x01\x00") ]]; then
E_TYPE="Relocatable"
elif [[ $tmp == $(echo -ne "\x02\x00") ]]; then
E_TYPE="Executable"
elif [[ $tmp == $(echo -ne "\x03\x00") ]]; then
E_TYPE="Shared object"
elif [[ $tmp == $(echo -ne "\x04\x00") ]]; then
E_TYPE="Core"
else
E_TYPE="Processor-specific"
fi
echo "- ELF ${EI_CLASS}-Bit $E_TYPE (${EI_DATA} Encoding) was found at ${offset}, starting dump..."
fi
fi
local fileOut="${out}/${offset}_${package}_${memName}.${fileExt}"
if [[ $memName == "global-metadata.dat" ]] || [[ $memName == "libil2cpp.so" && $fileExt == "so" ]]; then
fileOut="${out}/${offset}_${package}_${memName}"
fi
if [[ $metadataOffset != "" ]] && [[ $(echo "ibase=16;(${metadataOffset}-${offset})<0" | bc) -ne 0 ]]; then
echo "- Dumping [$memName] $range... <- This might be the correct libil2cpp.so"
metadataOffset=
elif [[ $memPath == "/dev/zero" ]] && [[ $memName == "global-metadata.dat" ]]; then
echo "- Dumping [$memName] $range... <- This metadata might be the decrypted"
else
echo "- Dumping [$memName] $range..."
fi
dd if="/proc/$pid/mem" bs=$SYS_PAGESIZE skip=$(echo "ibase=16;${offset}/$HEX_PAGESIZE" | bc) count=$(echo "ibase=16;(${end}-${offset})/$HEX_PAGESIZE" | bc) of="$fileOut" 2>/dev/null
if [[ $memName == "global-metadata.dat" ]]; then
if [[ $METADATA_SANITY_MATCHED != "true" ]] && [[ $METADATA_VERSION_VAILDATED == "true" ]]; then
echo "- Trying to fix metadata sanity..."
dd if="$fileOut" of="${out}/tmp" bs=1 count=4 2>/dev/null
echo "- Original header: $(cat "${out}/tmp" | xxd -p)"
echo -ne "\xAF\x1B\xB1\xFA" | dd of="$fileOut" count=4 conv=notrunc 2>/dev/null
fi
metadataOffset=$offset
continue
fi
if [[ $? -ne 0 ]]; then
echo "* Failed to dump memory $range, skipping..."
continue
fi
memory=$(grep -i "${end}-" "/proc/$pid/maps" | grep "\[anon:.bss]")
if [[ $memory != "" ]]; then
range=$(echo $memory | awk '{print $1}')
offset=$end
end=$(echo $range | awk -F'-' '{print toupper($2)}')
bss_block=$(echo "ibase=16;(${end}-${offset})/${HEX_PAGESIZE}" | bc)
echo "- Adding [anonymous:.bss] $range..."
dd if="/proc/$pid/mem" bs=$SYS_PAGESIZE skip=$(echo "ibase=16;${lastEnd}/$HEX_PAGESIZE" | bc) count=$bss_block of="$out/tmp" 2>/dev/null
cat "$out/tmp">>"$fileOut"
fi
if [[ $fileExt == "so" ]]; then
lastFile=$fileOut
else
if [[ $lastFile != "" ]]; then
echo "- Merging memory..."
skipMerge=false
if [[ $lastEnd != $offset ]]; then
local gap_block=$(echo "ibase=16;(${offset}-${lastEnd})/${HEX_PAGESIZE}" | bc)
if [[ $gap_block -gt $SYS_PAGESIZE ]]; then
echo "- Gap blocks $gap_block is too large, skipping merge..."
skipMerge=true
lastFile=$fileOut
echo "- Patching last dump..."
memory=$(grep -i "${lastEnd}-" "/proc/$pid/maps")
offset=$(echo $memory | awk '{print $1}' | awk -F'-' '{print toupper($2)}')
local patchIndicator=$(echo $memory | awk -v OFS=',' '{print $4,$5}')
if [[ $patchIndicator != $memIndicator ]] && [[ $patchIndicator != "00:00,0" ]]; then
echo "- Inconsistent memory files found, skipping patch..."
else
gap_block=$(echo "ibase=16;(${offset}-${lastEnd})/${HEX_PAGESIZE}" | bc)
if [[ $gap_block -gt $SYS_PAGESIZE ]]; then
echo "- Next region $gap_block is too large, skipping patch..."
else
echo "- Adding $gap_block blocks..."
dd if="/proc/$pid/mem" bs=$SYS_PAGESIZE skip=$(echo "ibase=16;${lastEnd}/$HEX_PAGESIZE" | bc) count=$gap_block of="$out/tmp" 2>/dev/null
cat "$out/tmp">>"$lastFile"
fi
fi
else
echo "- Adding $gap_block gap blocks..."
dd if="/proc/$pid/mem" bs=$SYS_PAGESIZE skip=$(echo "ibase=16;${lastEnd}/$HEX_PAGESIZE" | bc) count=$gap_block of="$out/tmp" 2>/dev/null
cat "$out/tmp">>"$lastFile"
fi
fi
if [[ $skipMerge == "false" ]]; then
cat "$fileOut">>"$lastFile"
rm -f "$fileOut"
fi
else
echo "- No ELF header found, but nothing to merge. Treating as a raw dump..."
lastFile=$fileOut
fi
fi
lastEnd=$end
rm -f "${out}/tmp"
done
echo "- Done!"