vmp简单来说就是将一些高级语言的代码通过自己实现的编译器进行编译得到字节码,这样就可以更有效的保护原有代码,而jsvmp自然就是对JS代码的编译保护,具体的可以看看H5应用加固防破解-JS虚拟机保护方案。
如何区分是不是jsvmp?看代码里面有没有很长的一串字符串或者是数组,并且代码里面没有具体的逻辑,还有就是有一个循环不断地在跑,里面的就是一些指令,比如x乎,x音,x讯滑块等等都是jsvmp,也都是基于堆栈的栈式虚拟机实现的。
样品网站:aHR0cHM6Ly90cmVuZGluc2lnaHQub2NlYW5lbmdpbmUuY29tL2FyaXRobWV0aWMtaW5kZXgvYW5hbHlzaXM/a2V5d29yZD0lRTklODAlODYlRTUlOTAlOTEmc291cmNlPW9jZWFuZW5naW5lJmFwcE5hbWU9YXdlbWU=
可以看到,逻辑清晰了很多,但是因为是if else混淆,里面还有一些中间变量,如果调试的话会需要点击多几次才能到目标语句,所以我们可以考虑将这个if else转换为switch语句,这样调试的时候直接一步到位。
上图可以看到每次都从操作码中读取两个字符,并且是十六进制的,所以_0x1383c7这个变量的范围就是0-255,这样switch的条件就有了,可以编写代码从0迭代到255,就可以得到每个指令对应的代码了。最后还原结果如下:
可以看到逻辑很清晰了,但是有个问题是0-255的case里面有些是重复的,需要删除,可以继续使用AST,在每个case的第一行插入june.push(case_num),如这样:
然后映射文件到网站上,运行一次,获取june数组里面的值,保存到代码里,继续编写AST代码,读取case条件值,将不在数组中存在的case值全部删除掉,最后就可以愉快地分析了!
这个就是在获取变量区$0的变量,$0代表的是本函数内定义的变量,$1就代表上一层,$2就继续往上推等等,但是其最外面的也会是本函数内的变量,所以需要需注意(这个如果有误,望赐教)
通过分析可知,_0x9ac2c2为栈,此处是将字符串dfp入栈,而我们要还原语句的话,入栈就不能这个了,需要改为对应的ast语句,将字符串dfp直接放上这网站astexplorer
小技巧:在每一个case前面添加一个语句:throw Error("未更改");然后直接运行代码,抛出异常的就是未处理的case,然后在浏览器调试看该case的作用并修改代码即可,运行到没有异常就代表没啥问题了!
最后就是判断循环了,循环的也会进入上面这个case,只需要在进入的时候判断当前的指令值和if语句的起始指令值是否相同,相同即为循环,就可以修改之前保存的if分支为循环,我这里修改为while循环,比较好改
我是在他定义完了之后直接调用该函数进行还原,这样可以防止他只定义没有调用到,不好的地方在于参数个数不知道,所以这个需要在代码还原之后手动优化下,问题不大,如果你有更好的方法,希望能指导下。
由于该网站的vmp只是部分vmp,而我们还原代码是在原有的解释器上修改的,如果直接整完整代码进行运行,需要补一定的环境,这个不行,所以我这里将这个_$webrt_1670312749函数扣下来修改即可
然后在这个函数里面修改传过来的函数参数,将他改为ast代码的格式,而不是一个字面量,因为真正程序调用的时候,传到这里表面就是个变量了,当然也可以在_0x207ec8这个循环函数内修改,但是不好确定哪些是函数参数,所以我在他上一个函数这里修改了
还原jsvmp代码这个第一次需要耐心点,等你熟悉起来了,后面的就会越来越快,里面主要就是一些ast代码的定义,需要传入栈的数据和存在变量区的值,最后就是函数定义和判断分支的实现,判断分支会比较难,处理好判断分支,剩下的就不是问题了。
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
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
function _0x5b7a61_vmp(args_0, args_1, args_2, args_3, args_4, args_5, args_6, args_7, args_8, args_9, args_10, args_11, args_12, args_13, args_14, args_15, args_16, args_17, args_18, args_19, args_20, args_21, args_22) {
var var_233
=
'X-Bogus'
;
var var_24
=
'_signature'
;
var var_25
=
window[
'XMLHttpRequest'
][
'prototype'
];
var var_26
=
var_25[
'open'
];
var var_27
=
var_25[
'setRequestHeader'
];
var var_28
=
var_25[
'send'
];
var var_29
=
var_25[
'overrideMimeType'
];
if
(var_25[
'_ac_intercepted'
]) {
return
;
}
var_25[
'_ac_intercepted'
]
=
!
0
;
var_25[
'setRequestHeader'
]
=
function (args_0, args_1) {
if
(!this[
'_send'
]) {
var var_9999
=
new window[
'Object'
]();
var_9999[
'func'
]
=
'setRequestHeader'
;
var_9999[
'arguments'
]
=
arguments;
this[
'_byted_intercept_list'
][
'push'
](var_9999);
if
(_0xc5dbaf(window[
'RegExp'
], _0x1a373c([
'^content-type$'
,
'i'
]))[
'test'
](args_0)) {
this[
'_byted_content'
]
=
args_1[
'toString'
]()[
'toLowerCase'
]()[
'split'
](
';'
)[
0
];
}
}
return
var_27.
apply
(this, arguments);
};
var_25[
'overrideMimeType'
]
=
function () {
this[
'_overrideMimeTypeArgs'
]
=
arguments;
return
var_29.
apply
(this, this[
'_overrideMimeTypeArgs'
]);
};
var_25[
'open'
]
=
function (args_0, args_1, args_2) {
this[
'_byted_intercept_list'
]
=
[];
var var_10000
=
new window[
'Object'
]();
var_10000[
'func'
]
=
'open'
;
var_10000[
'arguments'
]
=
arguments;
this[
'_byted_intercept_list'
][
'push'
](var_10000);
this[
'_byted_method'
]
=
args_0[
'toUpperCase'
]();
this[
'_byted_url'
]
=
args_1;
return
var_26.
apply
(this, arguments);
};
var var_30
=
[
'onabort'
,
'onerror'
,
'onload'
,
'onloadend'
,
'onloadstart'
,
'onprogress'
,
'ontimeout'
];
var var_31
=
[
'GET'
,
'POST'
];
var_25[
'send'
]
=
function fun_4(args_0) {
var var_6
=
var_31[
'indexOf'
](this[
'_byted_method'
]) !
=
=
0
-
1
;
if
(args_2(this[
'_byted_url'
])) {
if
(var_6) {
if
(this[
'_byted_url'
][
'indexOf'
](
'_signature='
) >
0
-
1
) {
return
var_28(this, arguments);
}
this[
'_byted_body'
]
=
args_0;
var var_7
=
this[
'onreadystatechange'
];
var var_8
=
this[
'onabort'
];
var var_9
=
this[
'onerror'
];
var var_10
=
this[
'onload'
];
var var_11
=
this[
'onloadend'
];
var var_12
=
this[
'onloadstart'
];
var var_13
=
this[
'onprogress'
];
var var_14
=
this[
'ontimeout'
];
var var_15
=
new window[
'Object'
]();
var var_50
=
0
;
while
(var_50 < var_30[
'length'
]) {
var_15[var_30[var_50]]
=
this[
'upload'
][var_30[var_50]];
+
+
var_50;
}
var var_16
=
args_3[
'msStatus'
];
var var_17
=
args_3[
'__ac_testid'
];
if
(var_17
=
=
'') {
var var_18
=
[
'msToken'
, args_3[
'msToken'
]];
}
else
{
var_18
=
[
'msToken'
, args_3[
'msToken'
],
'__ac_testid'
, var_17];
}
var var_19
=
args_4(args_5(this[
'_byted_url'
]), var_18);
var var_20
=
args_6(var_19);
var var_21
=
args_7(var_20, this[
'_byted_body'
]);
var var_22
=
args_4(var_19, [var_233, var_21]);
var var_23
=
'';
if
(args_8[
'v'
]) {
var_23
=
var_22;
}
else
{
var var_10002
=
new window[
'Object'
]();
var_10002[
'url'
]
=
args_9(null, var_22);
var var_100
=
var_10002;
if
(this[
'_byted_method'
]
=
=
=
'POST'
) {
if
(args_10(this[
'_byted_content'
])) {
args_11(var_100, this[
'_byted_content'
], this[
'_byted_body'
]);
var var_101
=
args_12(var_100, args_13,
'forreal'
);
var_23
=
args_4(var_22, [var_24, var_101]);
}
else
{
var_23
=
var_22;
}
}
else
{
var var_251
=
args_12(var_100, args_13,
'forreal'
);
var_23
=
args_4(var_22, [var_24, var_251]);
}
}
if
(this[
'_byted_intercept_list'
]) {
if
(this[
'_byted_intercept_list'
][
0
][
'func'
] !
=
=
'open'
) {
return
null;
}
}
var var_244
=
this[
'_byted_intercept_list'
];
var var_182
=
0
;
while
(var_182 < var_244[
'length'
]) {
if
(var_182
=
=
=
0
) {
var_244[var_182].arguments[
1
]
=
var_23;
this[
'_send'
]
=
!
0
;
var_26.
apply
(this, var_244[var_182].arguments);
}
else
{
this[var_244[var_182][
'func'
]].
apply
(this, var_244[var_182].arguments);
}
+
+
var_182;
}
if
(this[
'_overrideMimeTypeArgs'
]) {
this[
'overrideMimeType'
]();
}
delete this[
'_byted_intercept_list'
];
if
(args_8[
'sdi'
]) {
this[
'setRequestHeader'
](args_14[
'secInfoHeader'
], args_15());
}
this[
'onreadystatechange'
]
=
var_7;
this[
'onabort'
]
=
var_8;
this[
'onerror'
]
=
var_9;
this[
'onload'
]
=
function () {
var var_6
=
0
;
if
(!this[
'responseURL'
]) {
if
(!this[
'_byted_url'
]) {
var var_7
=
'';
if
(args_16(var_7)) {
var_6
=
1
;
}
if
(var_7[
'indexOf'
](window[
'location'
][
'host'
]) !
=
=
0
-
1
) {
var_6
=
2
;
}
if
(var_6 >
0
) {
var var_8
=
this[
'getResponseHeader'
](
'x-ms-token'
);
if
(var_8) {
var var_9
=
args_17(this[
'_byted_url'
]);
if
(var_9
=
=
=
args_18[
'sec'
]) {
args_3[
'msToken'
]
=
var_8;
args_3[
'msStatus'
]
=
var_9;
args_19(
'msToken'
, var_8);
args_20(var_8);
if
(var_9 > var_16) {
if
(args_3[
'msNewTokenList'
][
'length'
] >
0
) {
args_21(args_22,
2
);
}
}
}
else
if
(var_16 >
=
args_3[
'msStatus'
]) {
args_3[
'msToken'
]
=
var_8;
}
if
(var_16
=
=
=
args_18[
'init'
]) {
if
(args_3[
'msNewTokenList'
][
'length'
] <
10
) {
args_3[
'msNewTokenList'
][
'push'
](var_8);
if
(args_3[
'msNewTokenList'
][
'length'
]
=
=
=
1
) {
args_20(var_8);
args_19(
'msToken'
, var_8);
}
}
}
}
}
if
(var_10) {
var_10(args_0);
}
return
;
}
else
{
var_7
=
this[
'_byted_url'
];
if
(args_16(var_7)) {
var_6
=
1
;
}
if
(var_7[
'indexOf'
](window[
'location'
][
'host'
]) !
=
=
0
-
1
) {
var_6
=
2
;
}
if
(var_6 >
0
) {
var_8
=
this[
'getResponseHeader'
](
'x-ms-token'
);
if
(var_8) {
var_9
=
args_17(this[
'_byted_url'
]);
if
(var_9
=
=
=
args_18[
'sec'
]) {
args_3[
'msToken'
]
=
var_8;
args_3[
'msStatus'
]
=
var_9;
args_19(
'msToken'
, var_8);
args_20(var_8);
if
(var_9 > var_16) {
if
(args_3[
'msNewTokenList'
][
'length'
] >
0
) {
args_21(args_22,
2
);
}
}
}
else
if
(var_16 >
=
args_3[
'msStatus'
]) {
args_3[
'msToken'
]
=
var_8;
}
if
(var_16
=
=
=
args_18[
'init'
]) {
if
(args_3[
'msNewTokenList'
][
'length'
] <
10
) {
args_3[
'msNewTokenList'
][
'push'
](var_8);
if
(args_3[
'msNewTokenList'
][
'length'
]
=
=
=
1
) {
args_20(var_8);
args_19(
'msToken'
, var_8);
}
}
}
}
}
if
(var_10) {
var_10(args_0);
}
}
}
else
if
(!this[
'responseURL'
]) {
var_7
=
'';
if
(args_16(var_7)) {
var_6
=
1
;
}
if
(var_7[
'indexOf'
](window[
'location'
][
'host'
]) !
=
=
0
-
1
) {
var_6
=
2
;
}
if
(var_6 >
0
) {
var_8
=
this[
'getResponseHeader'
](
'x-ms-token'
);
if
(var_8) {
var_9
=
args_17(this[
'_byted_url'
]);
if
(var_9
=
=
=
args_18[
'sec'
]) {
args_3[
'msToken'
]
=
var_8;
args_3[
'msStatus'
]
=
var_9;
args_19(
'msToken'
, var_8);
args_20(var_8);
if
(var_9 > var_16) {
if
(args_3[
'msNewTokenList'
][
'length'
] >
0
) {
args_21(args_22,
2
);
}
}
}
else
if
(var_16 >
=
args_3[
'msStatus'
]) {
args_3[
'msToken'
]
=
var_8;
}
if
(var_16
=
=
=
args_18[
'init'
]) {
if
(args_3[
'msNewTokenList'
][
'length'
] <
10
) {
args_3[
'msNewTokenList'
][
'push'
](var_8);
if
(args_3[
'msNewTokenList'
][
'length'
]
=
=
=
1
) {
args_20(var_8);
args_19(
'msToken'
, var_8);
}
}
}
}
}
if
(var_10) {
var_10(args_0);
}
return
;
}
else
{
var_7
=
this[
'responseURL'
];
if
(args_16(var_7)) {
var_6
=
1
;
}
if
(var_7[
'indexOf'
](window[
'location'
][
'host'
]) !
=
=
0
-
1
) {
var_6
=
2
;
}
if
(var_6 >
0
) {
var_8
=
this[
'getResponseHeader'
](
'x-ms-token'
);
if
(var_8) {
var_9
=
args_17(this[
'_byted_url'
]);
if
(var_9
=
=
=
args_18[
'sec'
]) {
args_3[
'msToken'
]
=
var_8;
args_3[
'msStatus'
]
=
var_9;
args_19(
'msToken'
, var_8);
args_20(var_8);
if
(var_9 > var_16) {
if
(args_3[
'msNewTokenList'
][
'length'
] >
0
) {
args_21(args_22,
2
);
}
}
}
else
if
(var_16 >
=
args_3[
'msStatus'
]) {
args_3[
'msToken'
]
=
var_8;
}
if
(var_16
=
=
=
args_18[
'init'
]) {
if
(args_3[
'msNewTokenList'
][
'length'
] <
10
) {
args_3[
'msNewTokenList'
][
'push'
](var_8);
if
(args_3[
'msNewTokenList'
][
'length'
]
=
=
=
1
) {
args_20(var_8);
args_19(
'msToken'
, var_8);
}
}
}
}
}
if
(var_10) {
var_10(args_0);
}
}
};
this[
'onloadend'
]
=
var_11;
this[
'onloadstart'
]
=
var_12;
this[
'onprogress'
]
=
var_13;
this[
'ontimeout'
]
=
var_14;
var var_216
=
0
;
while
(var_216 < var_30[
'length'
]) {
this[
'upload'
][var_30[var_216]]
=
var_15[var_30[var_216]];
+
+
var_216
}
}
return
var_28.
apply
(this, arguments);
}
};
}