-
-
[原创]第七届山东省大学生网络安全技能大赛. 安卓逆向 fake_func
-
发表于: 2024-1-8 18:49 8526
-
其实这个刚开始认为很简单,直接静态分析了安卓层的代码和so代码直接一看stcmp函数,天真的以为就是上下文中看到的base64的字符串,结果提交错误,把字符串解码提交也不对。
故认真搞了一下,学到了不少东西。
大家先不急着往下看,可以自己使用jadx和IDA分析下,看看会不会被绕进去。
后缀是apk,先拖入夜神模拟器看看。只有个输入框。随便输入些字符串,看看效果。木有什么按钮,一通tab加回车,居然可以触发逻辑,应该还有个按钮。
直接拖入jadx中,找到MainActivity,代码逻辑很清晰。
代码解读:
关键就是check.checkflag函数,双击checkflag函数,可以看到如下代码
从代码可以看出,真正的逻辑来自native层的so库。根据以往开发经验,so文件会打包在资源中。
so在资源文件->lib->armeabi-v7a文件夹下,保存libchecso.so到本地。
用IDA64分析so文件。通过导出表很快定位到函数。
checkflag函数如下
看到了strcmp函数感觉,找到关键点了。对arm的传参约定做了下功课,逻辑应该是这样strcmp(r0,r1)。十有八九字符串“c2RuaXNjc2RuaXNjYWJjZA==”就是flag,保存到txt文档,感觉很简单草草记录下过程,关闭文件夹,做下一题了。
闲暇之际,把flag提交了,把积分兑一下。其他2个顺利的拿到了积分。这个却翻了车,试了好次都不行,又分析了下,发现还有个base64的字符,又拿去试了也不行,看来不是这么简单。
把导出表又看了下,有几个和hook相关的函数,再结合下题名,应该用到了hook,开始追踪,这几个函数函数的调用,IDA中一通交叉引用,感觉都不对。
感觉IDA7.5对arm支持的不太好,没有f5的IDA看着真不好。既然IDA不行,试试其他反汇编工具,先后试了下ninja和Ghidra感觉c代码还原都有那么点意思。
Ninja
Ghidra,我标注了些函数名,还导入了些java的相关数据类型。安卓逆向之自动化JNI静态分析
继续上面的疑虑,在ninja和Ghidra中对hook相关函数进行引用追踪,ninja没有找到什么有用的线索和信息。
但是Ghidra中有了线索。
在Ghidra中查看导出函数时,无意间双击到了_INIT_0_函数,居然调用了registerInlineHook,然后gooogle了_INIT_0_,随后这一'__DT_INIT_ARRAY'关键字进入了我的视野,https://docs.oracle.com/cd/E19683-01/816-1386/6m7qcobks/index.html。
so在加载时会调用这些初始化函数。
通过代码可以猜测出,函数strcmp被劫持了,通过观察对strcmp和jstrcmp的引用,可以确定so中只有一处对strcmp的调用就是 checkflag函数。
那么这样就会改变函数流程,strcmp不是比较字符串这么简单了。
根据第二参数FUN_00010e28 + 1找到函数如下
对hook的回调进行深入分析后,感觉后面的代码看不明白了,感觉后面应该是加密算法,因为需要base64解码,不是对常规的字符串进行处理,应该就是对称算法。
几款PE的查加密算法都不好用,这里直接借助IDA8.3 findcrypt插件,找到了相关线索。
根据ADES提示,以及查阅《加密与解密》书籍,逐步确定了AES_SBOX相关数据。
根据AES_SBOX相关的引用确定应用了AES的加密运算。函数 FUN_00010f24 部分代码。
以上位4 x 4 的subBytes操作。
这样通过梳理都能说的过去了。
既然知道了结果和key,很容易解密出输入的字符串。
public
class
MainActivity
extends
AppCompatActivity {
/* JADX INFO: Access modifiers changed from: protected */
@Override
// android.support.v7.app.AppCompatActivity, android.support.v4.app.FragmentActivity, android.support.v4.app.SupportActivity, android.app.Activity
public
void
onCreate(Bundle bundle) {
super
.onCreate(bundle);
setContentView(R.layout.activity_main);
((Button) findViewById(R.id.button)).setOnClickListener(
new
View.OnClickListener() {
// from class: com.example.p7xxtmx_g.fakefunc.MainActivity.1
@Override
// android.view.View.OnClickListener
public
void
onClick(View view) {
if
(check.checkflag(((EditText) MainActivity.
this
.findViewById(R.id.editText)).getText().toString())) {
Toast.makeText(MainActivity.
this
,
"you are right~!"
,
1
).show();
}
else
{
Toast.makeText(MainActivity.
this
,
"wrong!"
,
1
).show();
}
}
});
}
}
public
class
MainActivity
extends
AppCompatActivity {
/* JADX INFO: Access modifiers changed from: protected */
@Override
// android.support.v7.app.AppCompatActivity, android.support.v4.app.FragmentActivity, android.support.v4.app.SupportActivity, android.app.Activity
public
void
onCreate(Bundle bundle) {
super
.onCreate(bundle);
setContentView(R.layout.activity_main);
((Button) findViewById(R.id.button)).setOnClickListener(
new
View.OnClickListener() {
// from class: com.example.p7xxtmx_g.fakefunc.MainActivity.1
@Override
// android.view.View.OnClickListener
public
void
onClick(View view) {
if
(check.checkflag(((EditText) MainActivity.
this
.findViewById(R.id.editText)).getText().toString())) {
Toast.makeText(MainActivity.
this
,
"you are right~!"
,
1
).show();
}
else
{
Toast.makeText(MainActivity.
this
,
"wrong!"
,
1
).show();
}
}
});
}
}
public
class
check {
public
static
native
boolean
checkflag(String str);
static
{
System.loadLibrary(
"checkso"
);
}
}
public
class
check {
public
static
native
boolean
checkflag(String str);
static
{
System.loadLibrary(
"checkso"
);
}
}
Name | Address Ordinal |
---|---|
JNI_OnLoad | 0000000000000E84 |
Java_com_example_p7xxtmx_1g_fakefunc_check_checkflag | 0000000000000E54 |
inlineHook | 0000000000001DCC |
inlineUnHook | 0000000000001A98 |
registerInlineHook | 00000000000018BC |
relocateInstruction | 0000000000001FA8 |
inlineUnHookAll | 0000000000001D60 |
inlineHookAll | 0000000000001ED8 |
00000e54
uint32_t Java_com_example_p7xxtmx_1g_fakefunc_check_checkflag(int32_t
*
arg1)
00000e62
char
*
r0
=
(
*
(
*
arg1
+
0x2a4
))()
00000e66
sub_e08()
00000e76
int32_t temp0
=
0
00000e76
int32_t i
=
strcmp(r0, data_6004)
00000e76
while
(i !
=
0
)
00000e76
i
=
i u>>
1
00000e76
temp0
=
temp0
+
1
00000e7c
return
(
0x20
-
temp0) u>>
5
00000e54
uint32_t Java_com_example_p7xxtmx_1g_fakefunc_check_checkflag(int32_t
*
arg1)
00000e62
char
*
r0
=
(
*
(
*
arg1
+
0x2a4
))()
00000e66
sub_e08()
00000e76
int32_t temp0
=
0
00000e76
int32_t i
=
strcmp(r0, data_6004)
00000e76
while
(i !
=
0
)
00000e76
i
=
i u>>
1
00000e76
temp0
=
temp0
+
1
00000e7c
return
(
0x20
-
temp0) u>>
5
void Java_com_example_p7xxtmx_1g_fakefunc_check_checkflag(JNIEnv
*
env,jclass clazz,jstring a1)
{
char
*
__s1;
int
iVar1;
__s1
=
(
*
(
*
env)
-
>GetStringUTFChars)((JNIEnv
*
)env,a1,(jboolean
*
)
0x0
);
base64De_16004();
iVar1
=
strcmp(__s1,PTR_DAT_00016004);
count_leading_zeroes(iVar1);
return
;
}
void Java_com_example_p7xxtmx_1g_fakefunc_check_checkflag(JNIEnv
*
env,jclass clazz,jstring a1)
{
char
*
__s1;
int
iVar1;
__s1
=
(
*
(
*
env)
-
>GetStringUTFChars)((JNIEnv
*
)env,a1,(jboolean
*
)
0x0
);
base64De_16004();
iVar1
=
strcmp(__s1,PTR_DAT_00016004);
count_leading_zeroes(iVar1);
return
;
}
在将控制权转移到应用程序之前,运行时链接器会处理应用程序及其依赖项中找到的任何初始化部分。.preinit_array 、.init_array和.init节是在构建动态对象时由链接编辑器创建的。这些部分分别标有.dynamic标签DT_PREINIT_ARRAY、DT_INIT_ARRAY和DT_INIT。请参阅“初始化和终止部分”。
地址包含在DT_PREINIT_ARRAY和DT_INIT_ARRAY指定的数组中的函数由运行时链接器按照其地址在数组中出现的顺序执行。如果对象同时包含DT_INIT和DT_INIT_ARRAY条目,则DT_INIT条目引用的函数将在该对象的DT_INIT_ARRAY条目引用的函数之前处理。
动态可执行文件可以在.preinit_array部分提供预初始化函数。这些函数在运行时链接器构建进程映像并执行重定位之后但在任何其他初始化函数之前执行。共享对象中不允许使用预初始化函数
在将控制权转移到应用程序之前,运行时链接器会处理应用程序及其依赖项中找到的任何初始化部分。.preinit_array 、.init_array和.init节是在构建动态对象时由链接编辑器创建的。这些部分分别标有.dynamic标签DT_PREINIT_ARRAY、DT_INIT_ARRAY和DT_INIT。请参阅“初始化和终止部分”。
地址包含在DT_PREINIT_ARRAY和DT_INIT_ARRAY指定的数组中的函数由运行时链接器按照其地址在数组中出现的顺序执行。如果对象同时包含DT_INIT和DT_INIT_ARRAY条目,则DT_INIT条目引用的函数将在该对象的DT_INIT_ARRAY条目引用的函数之前处理。
动态可执行文件可以在.preinit_array部分提供预初始化函数。这些函数在运行时链接器构建进程映像并执行重定位之后但在任何其他初始化函数之前执行。共享对象中不允许使用预初始化函数
int
_INIT_0(void)
{
int
iVar1;
iVar1
=
registerInlineHook(strcmp,FUN_00010e28
+
1
,&DAT_00016008_orgFun);
if
(iVar1 !
=
0
) {
return
0xffffffff
;
}
iVar1
=
inlineHook(strcmp);
if
(iVar1 !
=
0
) {
iVar1
=
-
1
;
}
return
iVar1;
}
int
_INIT_0(void)
{
int
iVar1;
iVar1
=
registerInlineHook(strcmp,FUN_00010e28
+
1
,&DAT_00016008_orgFun);
if
(iVar1 !
=
0
) {
return
0xffffffff
;
}
iVar1
=
inlineHook(strcmp);
if
(iVar1 !
=
0
) {
iVar1
=
-
1
;
}
return
iVar1;
}
void Proc_00010e28(undefined4 strKey)
{
undefined4 uVar1;
uVar1
=
base64De_16004();
uVar1
=
EncryptEncode(strKey,uVar1);
/
*
WARNING: Could
not
recover jumptable at
0x00010e48
. Too many branches
*
/
/
*
WARNING: Treating indirect jump as call
*
/
(
*
strcmp_jmp)(uVar1,
"K4/7/faihmk9/WEMlfuFdpgrP86ckd4oQQ/UeAiZdx8="
);
return
;
}
void Proc_00010e28(undefined4 strKey)
{
undefined4 uVar1;
uVar1
=
base64De_16004();
uVar1
=
EncryptEncode(strKey,uVar1);
/
*
WARNING: Could
not
recover jumptable at
0x00010e48
. Too many branches
*
/
/
*
WARNING: Treating indirect jump as call
*
/
(
*
strcmp_jmp)(uVar1,
"K4/7/faihmk9/WEMlfuFdpgrP86ckd4oQQ/UeAiZdx8="
);
return
;
}
Address | Rules file | Name | String | Value |
---|---|---|---|---|
.rodata:00004255 | global | RijnDael_AES_CHAR_4255 | $c0 | b'c|w{\xf2ko\xc50\x01g+\xfe\xd7\xabv\xca\x82\xc9}\xfaYG\xf0\xad\xd4\xa2\xaf\x9c\xa4r\xc0' |
.rodata:00004255 | global | RijnDael_AES_LONG_4255 | $c0 | b'c|w{\xf2ko\xc50\x01g+\xfe\xd7\xabv\xca\x82\xc9}\xfaYG\xf0\xad\xd4\xa2\xaf\x9c\xa4r\xc0' |
.rodata:00004355 | global | RijnDael_AES_LONG_inv_4355 | $c0 | b'R\tj\xd506\xa58\xbf@\xa3\x9e\x81\xf3\xd7\xfb |
.rodata:00004156 | global | RijnDael_AES_RCON_4156 | $c0 | b'\x8d\x01\x02\x04\x08\x10 @\x80\x1b6l\xd8\xabM\x9a' |
.rodata:00004189 | global | RijnDael_AES_RCON_4189 | $c0 | b'\x8d\x01\x02\x04\x08\x10 @\x80\x1b6l\xd8\xabM\x9a' |
.rodata:000041BC | global | RijnDael_AES_RCON_41BC | $c0 | b'\x8d\x01\x02\x04\x08\x10 @\x80\x1b6l\xd8\xabM\x9a' |
.rodata:000041EF | global | RijnDael_AES_RCON_41EF | $c0 | b'\x8d\x01\x02\x04\x08\x10 @\x80\x1b6l\xd8\xabM\x9a' |
.rodata:00004222 | global | RijnDael_AES_RCON_4222 | $c0 | b'\x8d\x01\x02\x04\x08\x10 @\x80\x1b6l\xd8\xabM\x9a' |
.rodata:00004455 | global | BASE64_table_4455 | $c0 | b'ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789+/' |
AES_SBOX XREF[
5
]: FUN_00010f24:
00010ff2
(R),
FUN_0001105c:
000110d2
(R),
FUN_0001105c:
000110d8
(R),
FUN_0001105c:
000110dc
(R),
FUN_0001105c:
000110e0
(R)
00014255
63
??
63h
c
00014256
7c
??
7Ch
|
00014257
77
??
77h
w
00014258
7b
??
7Bh
{
00014259
f2 ?? F2h
0001425a
6b
??
6Bh
k
0001425b
6f
??
6Fh
o
0001425c
c5 ?? C5h
0001425d
30
??
30h
0
0001425e
01
??
01h
0001425f
67
??
67h
g
00014260
2b
??
2Bh
+
00014261
fe ?? FEh
AES_SBOX XREF[
5
]: FUN_00010f24:
00010ff2
(R),
FUN_0001105c:
000110d2
(R),
FUN_0001105c:
000110d8
(R),
FUN_0001105c:
000110dc
(R),
FUN_0001105c:
000110e0
(R)
00014255
63
??
63h
c
00014256
7c
??
7Ch
|
00014257
77
??
77h
w
00014258
7b
??
7Bh
{
00014259
f2 ?? F2h
0001425a
6b
??
6Bh
k
0001425b
6f
??
6Fh
o
0001425c
c5 ?? C5h
0001425d
30
??
30h
0
0001425e
01
??
01h
0001425f
67
??
67h
g
00014260
2b
??
2Bh
+
00014261
fe ?? FEh
while
(iVar12 != 4) {
iVar16 = 0;
while
(iVar16 != 4) {
*(undefined *)(iVar15 + iVar16 * 4) = (&AES_SBOX)[*(byte *)(iVar15 + iVar16 * 4)];
iVar16 = iVar16 + 1;
}
iVar15 = iVar15 + 1;
iVar12 = iVar12 + 1;
}
while
(iVar12 != 4) {
iVar16 = 0;
while
(iVar16 != 4) {
*(undefined *)(iVar15 + iVar16 * 4) = (&AES_SBOX)[*(byte *)(iVar15 + iVar16 * 4)];
iVar16 = iVar16 + 1;
}
iVar15 = iVar15 + 1;
iVar12 = iVar12 + 1;
}
void
Proc_00010e28(undefined4 strInput)
{
undefined4 uVar1;
uVar1 = base64De_16004();
uVar1 = EncryptEncode(strInput,uVar1);
/* WARNING: Could not recover jumptable at 0x00010e48. Too many branches */
/* WARNING: Treating indirect jump as call */
(*strcmp_jmp)(uVar1,
"K4/7/faihmk9/WEMlfuFdpgrP86ckd4oQQ/UeAiZdx8="
);
return
;
}
void
Proc_00010e28(undefined4 strInput)
{
undefined4 uVar1;
uVar1 = base64De_16004();
uVar1 = EncryptEncode(strInput,uVar1);
/* WARNING: Could not recover jumptable at 0x00010e48. Too many branches */
/* WARNING: Treating indirect jump as call */
(*strcmp_jmp)(uVar1,
"K4/7/faihmk9/WEMlfuFdpgrP86ckd4oQQ/UeAiZdx8="
);
return
;
}
undefined4 EncryptEncode(
char
*param_1,undefined4 param_2)
{
size_t
sVar1;
void
*__ptr;
int
iVar2;
void
*__ptr_00;
undefined4 uVar3;
char
*pcVar4;
char
cVar5;
int
iVar6;
size_t
__size;
sVar1 =
strlen
(param_1);
if
((
int
)sVar1 < 0x10) {
__ptr =
malloc
(0x10);
iVar2 = 0;
while
(iVar2 != 0x10) {
pcVar4 = &UNK_00014156 + -sVar1;
if
(iVar2 < (
int
)sVar1) {
pcVar4 = param_1 + iVar2;
}
*(
char
*)((
int
)__ptr + iVar2) = *pcVar4;
iVar2 = iVar2 + 1;
}
__size = 0x10;
}
else
{
__size = sVar1 + 0x10 & 0xfffffff0;
__ptr =
malloc
(__size);
iVar2 = 0;
while
(iVar2 < (
int
)__size) {
if
(iVar2 < (
int
)sVar1) {
pcVar4 = param_1 + iVar2;
LAB_00011402:
cVar5 = *pcVar4;
}
else
{
pcVar4 = &DAT_00014146 + (__size - sVar1);
if
((sVar1 & 0xf) != 0)
goto
LAB_00011402;
cVar5 =
'\x10'
;
}
*(
char
*)((
int
)__ptr + iVar2) = cVar5;
iVar2 = iVar2 + 1;
}
}
__ptr_00 =
malloc
(__size);
iVar6 = 0;
iVar2 = 0;
while
(iVar2 < (
int
)(__size + ((uint)((
int
)__size >> 0x1f) >> 0x1c)) >> 4) {
AesEncrypt((
int
)__ptr + iVar6,param_2,(
int
)__ptr_00 + iVar6);
iVar6 = iVar6 + 0x10;
iVar2 = iVar2 + 1;
}
uVar3 = base64Encode(__ptr_00,__size);
free
(__ptr);
free
(__ptr_00);
return
uVar3;
}
undefined4 EncryptEncode(
char
*param_1,undefined4 param_2)
{
size_t
sVar1;
void
*__ptr;
int
iVar2;
void
*__ptr_00;
[注意]传递专业知识、拓宽行业人脉——看雪讲师团队等你加入!