首 页 文章下载源码教程论坛博客留言私服DJQQ最新软件

热门搜索 : X-Scan 灰鸽子 非君子迷你 SQL防注入 密码解霸

软件  漏洞扫描 | 木马间谍 | 探嗅监听 | 密码破解 | 拒绝服务 | 恶作剧类 | 杀毒软件 | 防火墙类 | OICQ专区 | 系统辅助 | 大型软件
文章  热点新闻 | 黑客工具 | 安全管理 | 菜鸟文摘 | 编程图形 | 网友原创 | 脚本收藏 | 最新资讯 | 情感文摘 | 幽默笑话 | 数据安全
源码  论坛相关 | 字体素材 | 图片素材 | 酷站素材 | 网页模版 | 应用目录 | NET 源码 | PHP 源码 | 图标素材 | ASP 源码 | CGI 源码
教程  入侵教程 | 黑客书籍 | 木马病毒 | 服务器类 | 数据库类 | 图形处理 | 网页制作 | 菜鸟教程 | 安全防护 | 破解教程 | 投稿作品


 您当前的位置: 黑软基地 网友原创技术文摘 → 文章内容 退出登录 用户管理

利用格式化字符串漏洞对系统发起攻击
作者:小心黑夜  来源:本站整理  发布时间:2007-3-9 17:28:41

减小字体 增大字体

 

【编者按:一个很小的疏忽,就会酿成危机系统安全的大患!】

什么是格式化字符串攻击

格式化字符串漏洞同其他许多安全漏洞一样是由于程序员的懒惰造成的。当你正在阅读本文的时候,也许有个程序员正在编写代码,他的任务是:打印输出一个字符串或者把这个串拷贝到某缓冲区内。他可以写出如下的代码:

printf("%s", str);

但是为了节约时间和提高效率,并在源码中少输入6个字节,他会这样写:

printf(str);

为什么不呢?干嘛要和多余的printf参数打交道,干嘛要花时间分解那些愚蠢的格式?printf的第一个参数无论如何都会输出的!程序员在不知不觉中打开了一个安全漏洞,可以让攻击者控制程序的执行,这就是不能偷懒的原因所在。

为什么程序员写的是错误的呢?他传入了一个他想要逐字打印的字符串。实际上该字符串被printf函数解释为一个格式化字符串(format string)。函数在其中寻找特殊的格式字符比如"%d"。如果碰到格式字符,一个变量的参数值就从堆栈中取出。很明显,攻击者至少可以通过打印出堆栈中的这些值来偷看程序的内存。但是有些事情就不那么明显了,这个简单的错误允许向运行中程序的内存里写入任意值。

Printf中被忽略的东西

在说明如何为了自己的目的滥用printf之前,我们应该深入领会printf提供的特性。假定读者以前用过printf函数并且知道普通的格式化特性,比如如何打印整型和字符串,如何指定最大和最小字符串宽度等。除了这些普通的特性之外,还有一些深奥和鲜为人知的特性。在这些特性当中,下面介绍的对我们比较有用:

* 在格式化字符串中任何位置都可以得到输出字符的个数。当在格式化字符串中碰到"%n"的时候,在%n域之前输出的字符个数会保存到下一个参数里。例如,为了获取在两个格式化的数字之间空间的偏量:

int pos, x = 235, y = 93; 
      printf("%d %n%d\n", x, &pos, y); 
      printf("The offset was %d\n", pos);

* %n格式返回应该被输出的字符数目,而不是实际输出的字符数目。当把一个字符串格式化输出到一个定长缓冲区内时,输出字符串可能被截短。不考虑截短的影响,%n格式表示如果不被截短的偏量值(输出字符数目)。为了说明这一点,下面的代码会输出100而不是20:

char buf[20]; 
      int pos, x = 0; 
      snprintf(buf, sizeof buf, "%.100d%n", x, &pos); 
      printf("position: %d\n", pos);

简单的例子

除了讨论抽象和复杂的理论,我们将会使用一个具体的例子来说明我们刚才讨论的原理。下面这个简单的程序能满足这个要求:

/* 
     * fmtme.c 
     *       Format a value into a fixed-size buffer 
     */ 

    #include <stdio.h> 

    int 
    main(int argc, char **argv) 
    { 
        char buf[100]; 
        int x; 
        if(argc != 2) 
            exit(1); 
        x = 1; 
        snprintf(buf, sizeof buf, argv[1]); 
        buf[sizeof buf - 1] = 0; 
        printf("buffer (%d): %s\n", strlen(buf), buf); 
        printf("x is %d/%#x (@ %p)\n", x, x, &x); 
        return 0; 
    }

对这个程序有几点说明:第一,目的很简单:将一个通过命令行传递值格式化输出到一个定长的缓冲区里。并确保缓冲区的大小限制不被突破。在缓冲区格式化后,把它输出。除了把参数格式化,还设置了一个整型值随后输出。这个变量是随后我们攻击的目标。现在值得我们注意的是这个值应该始终为1。

本文中所有的例子都是在x86 BSD/OS 4.1机器上完成。如果你到莫桑比克执行任务超过20年时间可能会对x86不熟悉,这是一个little-endian机器。这决定在例子中多精度数字的表示方法。在这里使用的具体数值会因为系统的差异而不同,这些差异表现在不同体系结构、操作系统、环境甚至是命令行长度。经过简单调整,这些例子可以在其他x86平台上工作。通过努力也可以在其他体系结构的平台上工作。

用Format攻击

现在是我们戴上黑帽子开始以攻击者方式思考问题的时候了。我们现在手头有一个测试程序。知道这个程序有一个漏洞并且了解程序员是在哪里犯错误的(直接把用户输入的命令行参数作为snprintf的格式化参数)。我们还拥有关于printf函数深入的知识,知道如何运用这些知识。让我们开始修补我们的程序吧。

从简单的开始,我们通过简单的参数调用程序。看这儿:

% ./fmtme "hello world" 
    buffer (11): hello world 
x is 1/0x1 (@ 0x804745c)

现在这儿还没有什么特别的事情发生。程序把我们输入的字符串格式化输出到缓冲区里,然后打印出它的长度和数值。程序还告诉我们变量x的值是1(以十进制和十六进制分别显示),x的存储地址是0x804745c。接下来我们试着使用一些格式指令。在下面的例子中我们打印出在格式化字符串之上栈堆中的整型数值:

% ./fmtme "%x %x %x %x" 
    buffer (15): 1 f31 1031 3133 
    x is 1/0x1 (@ 0x804745c)

对这个程序的快速分析可以揭示在调用snprintf函数时程序堆栈的规划:

Address  Contents       Description 
    fp+8     Buffer         pointer 4-byte address 
    fp+12    Buffer         length 4-byte integer 
    fp+16   &n
 
[ ] [返回上一页] [打 印]
上一篇文章:用密码验证来保护动感首页更新的实现
下一篇文章:BIOS控制:从底层保障计算机系统安全

∷相关文章评论∷    (评论内容只代表网友观点,与本站立场无关!)[更多评论...]

    热门文章

· 400秒远程攻破你的QQ密码 OICQ存在..
· QQ丢了怎么办?教你找回被盗的QQ号码..
· 远程盗QQ,不用木马
· 网站漏洞百出 QQ密码“大白天下”
· 黑客基地被黑!
· QQ固定群免费任你开
· 目前最实用的入侵方法以及工具下载..
· 全球首个QQ木马 通过QQ即可控制电脑..
· 26万至尊QQ号的神秘功能
· 华夏会员区今日刚发布?-价值25万人..

    推荐文章

· [图文]手把手教你从XP升级到Vista
· [图文]Windows Vista简体中文版安装..
· [图文]新年将至动易官方被黑
· Microsoft Windows Vista 下载汇总..
· [图文]黑客X档案2006配套光盘
· [图文]黑鹰基地被黑
· 安全配置NT/2000 SERVER(IIS)
· [组图]日本网站被挂上中国国旗
· [组图]迅雷网站被黑
· 黑鹰基地再次被黑!!

    最新文章
· 黑客翻新DDOS攻击 DNS服务器成帮凶..
· Rinbot 和 Froot 利用已知漏洞攻击..
· [图文]东方卫士被挂马 安全网站安全..
· 危险无处不在 账号防盗防骗的小经..
· BIOS控制:从底层保障计算机系统安..
· 利用格式化字符串漏洞对系统发起攻..
· 用密码验证来保护动感首页更新的实..
· 黑客案例——浏览器执行exe文件的探..
· 双击盘符要小心 “麦戈兽”病毒在捣..
· Google桌面搜索软件仍然暗藏安全漏..