-
存在缺陷的程序设计 -
尤其是C语言,不像其他一些高级语言会自动进行数组或者指针的堆栈区块边界检查,增加溢出风险。 -
C语言中的C标准库还具有一些非常危险的操作函数,使用不当也为溢出创造条件。
什么是缓冲区溢出攻击
如果攻击者知道程序的内存布局,他们可以故意提供缓冲区无法存储的输入,并覆盖保存可执行代码的区域,用他们自己的代码替换它。例如,攻击者可以覆盖指针(指向内存中另一个区域的对象)并将其指向漏洞利用负载,从而获得对程序的控制权。
缓冲区溢出攻击的类型
哪些编程语言更容易受到攻击?
如何防止缓冲区溢出
-
地址空间随机化 (ASLR) — 随机移动数据区域的地址空间位置。通常,缓冲区溢出攻击需要知道可执行代码的位置,而随机化地址空间使得这几乎不可能。 -
数据执行预防 ——将某些内存区域标记为不可执行或可执行,从而阻止攻击在不可执行区域中运行代码。 -
结构化异常处理程序覆盖保护 (SEHOP) — 帮助阻止恶意代码攻击结构化异常处理 (SEH),这是一种用于管理硬件和软件异常的内置系统。因此,它可以防止攻击者利用 SEH 覆盖利用技术。在功能级别上,使用基于堆栈的缓冲区溢出来覆盖存储在线程堆栈中的异常注册记录来实现 SEH 覆盖。
代码和操作系统保护中的安全措施是不够的。当组织发现缓冲区溢出漏洞时,它必须迅速做出反应以修补受影响的软件,并确保软件用户可以访问补丁。
示例代码展示
int main(int argc, char **argv) {
MessageBoxA((HWND)-0, (LPCSTR) "缓冲区溢出测试n", (LPCSTR)"功能", (UINT)0);
int cookie;
char buf[2];
int *a = &cookie;
char * b = buf;
printf("buf: %08x cookie: %08xn", b, a);
u_int64 p =(u_int64)a-(u_int64)b;
printf("两变量内存地址之差=%dn",p);
gets(buf);
if (cookie == 0x41424344)
printf("缓冲区溢出成功!n");
}
运行效果展示 使用MessageBoxA函数检测程序是否正常运行,点击确定开始测试
使用printf()函数输出提示信息,使用gets()函数获取用户输入信息 任意输入两个数值,不满足条件,程序运行完毕
代码分析漏洞成因 诱因:char buf[2]; 代码部分解析---使用char 将变量buf声明成了一个拥有2个元素数组其中元素类型为字符.buf有了两个自己长度,
提示:u_int64 p =(u_int64)a-(u_int64)b; 代码部分对程序涉及变量了内存地址进行了一个减分计算并对赋值给变量 p,(为使其运算成立还对其进行了类型转义),结果可告知两个变量内存地址距离,方便溢出利用
隐患:使用gets()函数获取输入数据,因gets()函数无限读取数据并不检查缓冲区的大小限制,会将超出缓冲区的数据继续写入堆栈,导致存在溢出隐患。
为方便理解此处代码演示下:int main() {
char test[] = "test1";
printf("test1初始值为%sn清输入st值:",test);
char st[2];
gets(st);
printf("输出test:%sn",test);
printf("输出st:%sn",st);
}
在运行效果上可以看到超出堆栈空间的值继续写入堆栈导致覆盖了test在堆栈内对应的值,导致其数值进行了改变:test1-3456
反汇编分析其运行过程堆栈是如何变化的
有运行得知(外加源代码)程序初始关键词:test1初始值为test1可通过此关键词,在反汇编程序快速定位到程序相关函数运行区域 在入口指令处下断点方便分析 并在实际运行发现运行到此处为显示相关特征字符信息,初步判断正确 将此区域字符串进行反编译与源码对照进一步验证 没汇编指令对照不太直观换个插件与工具,进行展示 未输入st值时test对应数据堆栈情况 000000000061FDE8 000000000061FE0A "test1" 输入后查看 000000000061FDE8 000000000061FE08 "123456" 000000000061FDF0 000000000061FE0A "3456" //之前为test1 根据此思路分析之前的示例程序
反汇编分析 根据之前的思路定位关闭区域 定位特征代码: 根据特征代码搜索 在入口指令处下断点 运行看到一个变量的对应地址 继续运行看到另一个变量的地址 因剩下的只是运行用就不展示了 输入数值后可看到由之前空白数据已被输入的数据覆盖 扩展知识:发现栈中对应的值与内存对应的值是相反的,因为堆栈中的值遵循先进后出原则故是相反的 扩展分析 看到此区域有个je跳转指令,其上方有个cmp指令(功能是用来比较),其中有一值被固定为:41424344 可以看到明显不满足跳转故跳转不会执行,不显示隐藏的信息 根据此思路:输入两值将缓冲区堆满,之后数据溢出覆盖思路,并且根据提示计算出两变量相距离为2与堆栈数据先进后出原则,故输入如下条件即可使跳转成立,进而输出隐藏信息 12DCBA 跳转成立 可以看到隐藏信息已显示
扩展知识RSP+2C = rsp对应的地址+2c
即000000000061FDE0+2C=61FE0C将16进制转10进制计算
IDA调试分析使其跟清晰地去查看变化为方便分析可在关键函数处下断点单步执行看执行效果 由之前可知MesssageBoxA后才开始正式运行,故可在此call后下断点 运行,直到到断点暂停 运行后发现rsi对应数值无变化,正常因除了初始赋值后无其它赋值给rsi的指令 gets函数执行前rsi对应数据仍未变化 gets函数执行后rsi对应数据发生改变 ush rsi
.text:00000000004078D1 push rbx
.text:00000000004078D2 sub rsp, 38h
.text:00000000004078D6 call __main
.text:00000000004078DB lea rsi, [rsp+48h+var_1E] 初始rsi值
.text:00000000004078E0 xor r9d, r9d ; uType
.text:00000000004078E3 xor ecx, ecx ; hWnd
.text:00000000004078E5 lea rbx, [rsp+48h+var_1C]
.text:00000000004078EA lea r8, Caption ; lpCaption
.text:00000000004078F1 lea rdx, Text ; lpText
.text:00000000004078F8 call cs:__imp_MessageBoxA
.text:00000000004078FE mov r8, rbx
.text:0000000000407901 mov rdx, rsi
.text:0000000000407904 sub rbx, rsi
.text:0000000000407907 lea rcx, aBuf08xCookie08 ; "buf: %08x cookie: %08xn"
.text:000000000040790E call _Z6printfPKcz ; printf(char const*,...)
.text:0000000000407913 lea rcx, byte_40902D ; char *
.text:000000000040791A mov rdx, rbx
.text:000000000040791D call _Z6printfPKcz ; printf(char const*,...)
.text:0000000000407922 mov rcx, rsi
.text:0000000000407925 call gets
.text:000000000040792A cmp [rsp+48h+var_1C], 41424344h
.text:0000000000407932 jz short loc_40793D
根据此区域汇编代码分析得知,除了初始运行对rsi进行赋值后均无对其赋值的指令,故可得知存在溢出漏洞,覆盖了rsi原始数据,导致其值改变,根据单步跟踪执行流程可值执行完gets函数后其值发生变化,故可判断使用gets函数获取的数据导致堆栈数据溢出。 修复 可使用fgets或gets_s函数替换gets函数。 函数解析: fgets()函数的第2个参数指明了读入字符的最大数量。如果该参数的值是n,那么fgets()将读入n-1个字符,或者读到遇到的第一个换行符为止;
这里为3,故获取的数值为12,共输入6个数读取到第2个结束
IDA附加进程并单步执行查看执行流程 fgets执行前rsi对应数据 fgets执行后rsi对应数据 可以看到数据未溢出,分析到此告一段落 往期推荐 原创 | Windows下基础免杀技术 原创 | GitHub Java CodeQL CTF 原创 | CodeQL与AST之间联系 原文始发于微信公众号(SecIN技术平台):原创 | 缓冲区溢出漏洞那些事:C -gets函数
免责声明:文章中涉及的程序(方法)可能带有攻击性,仅供安全研究与教学之用,读者将其信息做其他用途,由读者承担全部法律及连带责任,本站不承担任何法律及连带责任;如有问题可邮件联系(建议使用企业邮箱或有效邮箱,避免邮件被拦截,联系方式见首页),望知悉。
- 左青龙
- 微信扫一扫
-
- 右白虎
- 微信扫一扫
-
评论