上次学习了下堆喷漏洞的原理,虽说之前有学习过缓冲区溢出的原理,但还没了解过堆喷这个概念,于是趁此机会学习了,顺便复习了缓冲区溢出这块知识,之前由于各种原因对Shellcode的编写只是了解个大概,并没有真正动手写过一个Shellcode。眼前遇到个堆喷漏洞找Shellcode时就下决定自己写个Shellcode,考虑到时间和精力的有限就写个计算器简单的练练手。

注:以下在XP SP3+VC6.0编译成功

一、首先写个简单的调用计算器的程序。

    1. 注:以下在XP SP3+VC6.0编译成功

      一、首先写个简单的调用计算器的程序。

      1. #include "windows.h"
      2. int main()
      3. {
      4. LoadLibraryA("kernel32.dll");//4c801d7b
      5. WinExec("calc.exe",SW_SHOW);
      6. return 0;
      7. }

      二、将WinExec("calc.exe",SW_SHOW);转化为汇编模样。

      在WinExec("calc.exe",SW_SHOW);处下断点,点F5进行调试,运行到此处时程序会暂停下来,程序暂停后按Alt+8即可查看到对应的汇编代码,经整理后如下:

      点击(此处)折叠或打开

      1. #include "windows.h"
      2. int main()
      3. {
      4. LoadLibraryA("kernel32.dll");//4c801d7b
      5. WinExec("calc.exe",SW_SHOW);
      6. __asm{
      7. mov esi,esp
      8. push 5
      9. push offset string "calc.exe" (0042201c)
      10. call dword ptr [__imp__WinExec@8 (0042a14c)]
      11. cmp esi,esp
      12. call __chkesp (00401090)
      13. }
      14. return 0;
      15. }

      稍微懂那么一丢丢汇编的童鞋都知道0042a14c处放着WinExec的地址,这里要注意WinExec的地址不是现在看到的0042a14c,要在地址为0042a14c放着的东东才是WinExec的地址。打个比方0042a14c是个指针,指针所指的地方才是真正需要的东东,所以我们要取出地址为0042a14c存放的数据。在VC6.0下按Alt+6可调出内存窗口,输入0042a14c即可看到。

    2. 因此WinExec真正的地址是7C8623AD,注意,这里是要反过来读取。

      三、现在有了汇编模样的语句和WinExec的地址,接下来就是要转化为具有Shellcode的汇编代码。在转化汇编时先了解汇编下面是如何完成一个函数调用的:

      1、父函数将函数的实参按照从右至左顺序压入堆栈;

      2、CPU将父函数中函数调用指令Call XXXXXXXX的下一条指令地址EIP压入堆栈;

      3、父函数通过Push Ebp将基地指针Ebp值东方钽业堆栈,并通过Mov Ebp,Esp指令将当前堆栈指针Esp值传给Ebp;

      4、通过Sub Esp,m(m是字节数)指令可以为存放函数中的局部变量开辟内存。函数在执行的时候如果需要访问实参或局部变量,都可以通过EBP指针来指引完成。

      根据汇编调用函数特点,并使用压栈的方法将参数传递进行,便可得到如下代码:

      点击(此处)折叠或打开

      1. #include "windows.h"
      2. int main()
      3. {
      4. LoadLibraryA("kernel32.dll");//4c801d7b
      5. //WinExec("calc.exe",SW_SHOW);
      6. __asm
      7. {
      8. push    ebp;
      9. mov        ebp,esp;
      10. xor eax,eax;
      11. push eax;
      12. sub esp,08h;
      13. mov byte ptr [ebp-0Ch],63h; //c
      14. mov byte ptr [ebp-0Bh],61h; //a
      15. mov byte ptr [ebp-0Ah],6Ch; //l
      16. mov byte ptr [ebp-09h],63h; //c
      17. mov byte ptr [ebp-08h],2Eh; //.
      18. mov byte ptr [ebp-07h],65h; //e
      19. mov byte ptr [ebp-06h],78h; //x
      20. mov byte ptr [ebp-05h],65h; //e
      21. lea eax,[ebp-0ch];
      22. push eax;                    //将calc.exe压入栈内
      23. mov        eax,0x7C8623AD;
      24. call    eax;                    //调用WinExec
      25. mov esp,ebp;
      26. pop    ebp;
      27. }
      28. return 0;
      29. }

      注意,字符串要以00H结束的哦,编译运行OK~~

      四、到这里已经完成最难的部分了,接下来的工作即是将汇编在内存中的代码,即是Shellcode拷出来就是了。同样,在汇编代码任意一处下断点,让程序在断点处停下来,按Alt+8即可看到程序所在的内存地址,再按Alt+6调出内存窗口即可。

      1. 将汇编代码范围内的东东全拷出来即得到传说中的Shellcode,这就是程序运行在内存中的模样了。一翻苦工后即可得到有Shellcode模样的Shellcode,同理将LoadLibraryA同样进行转化即可得到一个完整的Shellcode。

        点击(此处)折叠或打开

        1. //LoadLibraryA("kernel32.dll");
        2. //WinExec("calc.exe",SW_SHOW);
        3. #include "windows.h"
        4. unsigned char shellcode[]=
        5. "x55x8BxECx33xC0x50x83"
        6. "xECx09xC6x45xF3x6BxC6"
        7. "x45xF4x65xC6x45xF5x72"
        8. "xC6x45xF6x6ExC6x45xF7"
        9. "x65xC6x45xF8x6CxC6x45"
        10. "xF9x33xC6x45xFAx32xC6"
        11. "x45xFBx2ExC6x45xFCx64"
        12. "xC6x45xFDx6CxC6x45xFE"
        13. "x6Cx8Dx45xF3x50xB8x7B"
        14. "x1Dx80x7CxFFxD0x8BxE5"
        15. "x33xC0x50x83xECx08xC6"
        16. "x45xF4x63xC6x45xF5x61"
        17. "xC6x45xF6x6CxC6x45xF7"
        18. "x63xC6x45xF8x2ExC6x45"
        19. "xF9x65xC6x45xFAx78xC6"
        20. "x45xFBx65x8Dx45xF4x50"
        21. "xB8xADx23x86x7CxFFxD0"
        22. "x8BxE5x5D";
        23. main()
        24. {
        25. __asm
        26. {
        27. lea     eax,shellcode;
        28. call    eax;
        29. }
        30. }

ShellCode的编写入门的更多相关文章

  1. Linux下shellcode的编写

    Linux下shellcode的编写 来源  https://xz.aliyun.com/t/2052 EdvisonV / 2018-02-14 22:00:42 / 浏览数 6638 技术文章 技 ...

  2. Gulp:插件编写入门

    之前挖了个坑,准备写篇gulp插件编写入门的科普文,之后迟迟没有动笔,因为不知道该肿么讲清楚Stream这货,毕竟,gulp插件的实现不像grunt插件的实现那么直观. 好吧,于是决定单刀直入了.文中 ...

  3. 异数OS 星星之火(三)--异数OS-织梦师云 微服务编写入门

    . 异数OS 星星之火(三)–异数OS-织梦师云 微服务编写入门 本文来自异数OS社区 github: https://github.com/yds086/HereticOS 异数OS社区QQ群: 6 ...

  4. Servlet第一篇【介绍Servlet、HTTP协议、WEB目录结构、编写入门Servlet程序、Servlet生命周期】

    什么是Serlvet? Servlet其实就是一个遵循Servlet开发的java类.Serlvet是由服务器调用的,运行在服务器端. 为什么要用到Serlvet? 我们编写java程序想要在网上实现 ...

  5. python棋类游戏编写入门

    刚接触棋类游戏程序编写的朋友,往往比较迷惑,不知从何下手. 本文总结了棋类游戏的主程序流程.计算机走子策略.打分方式(以井字棋.黑白棋.五子棋为例),未使用minimax算法,比较简单,适合刚接触的朋 ...

  6. Spring Boot 编写入门程序

    1. SpringBoot 入门 快速创建独立运行的Spring项目以及与主流框架集成; 使用嵌入式的Servlet容器,应用无需打成WAR包; starters自动依赖与版本控制; 大量的自动配置, ...

  7. 缓冲区溢出分析第04课:ShellCode的编写

    前言 ShellCode究竟是什么呢,其实它就是一些编译好的机器码,将这些机器码作为数据输入,然后通过我们之前所讲的方式来执行ShellCode,这就是缓冲区溢出利用的基本原理.那么下面我们就来编写S ...

  8. linux设备驱动编写入门

    linux设备驱动是什么,我个人的理解是liunx有用户态和内核态,用户空间中是不能直接对设备的外设进行使用而内核态中却可以,这时我们需要在内核空间中将需要的外设驱动起来供用户空间使用.linux的驱 ...

  9. mysql存储过程编写-入门案例-遁地龙卷风

    (-1)写在前面 这篇文章只是简要的叙述了mysql存储过程编写的基本概念. 我使用的mysql版本是5.7.9-log. 参照<<深入浅出MySQL>>. (0) delim ...

随机推荐

  1. 【LOJ】 #2011. 「SCOI2015」情报传递

    题解 一写过一交A的一道数据结构水题 我们发现大于C可以转化为这条路径上有多少个在某天之前开始调查的情报员,离线全部读入,变成树上路径查询某个区间的数出现过多少次,构建一棵根缀的主席树,查询的时候用两 ...

  2. es6导入导出模块

    在JavaScript ES6中,export与export default均可用于导出常量.函数.文件.模块等,你可以在其它文件或模块中通过import+(常量 | 函数 | 文件 | 模块)名的方 ...

  3. 洛谷P3758/BZOJ4887 [TJOI2017] 可乐 [矩阵快速幂]

    洛谷传送门,BZOJ传送门 可乐 Time Limit: 10 Sec  Memory Limit: 64 MBSubmit: 299  Solved: 207 Description 加里敦星球的人 ...

  4. Failed to add VMware DC to zone due to : This DC is being managed by other CloudStack deployment.

    1.下载VMware-PowerCLI 2.安装VMware-PowerCLI 安装过程中会重复重启,请确认重启,不要设置稍后手动重启. 3.在开始,菜单中选择 vmware ->VMware ...

  5. 【BZOJ 2121】 (字符串DP,区间DP)

    2121: 字符串游戏 Description BX正在进行一个字符串游戏,他手上有一个字符串L,以及其他一些字符串的集合S,然后他可以进行以下操作:对于一个在集合S中的字符串p,如果p在L中出现,B ...

  6. Entity Framework(实体框架 EF)

    什么是Entity Framework呢(下面简称EF)? EF(实体框架)是ADO.NET中的一组支持开发面向数据的软件应用程序的技术,是微软的一个ORM框架.ORM(对象关系映射框架):指的是面向 ...

  7. Tsinsen 最长双回文串

    求最长双回文串,正反建回文树求最大. 题目链接:http://www.tsinsen.com/ViewGProblem.page?gpid=A1280 By:大奕哥 #include<bits/ ...

  8. 理解HashSet及使用

    (1) 为啥要用HahSet?    假如我们现在想要在一大堆数据中查找X数据.LinkedList的数据结构就不说了,查找效率低的可怕.ArrayList哪,如果我们不知道X的位置序号,还是一样要全 ...

  9. 【8.20校内测试】【DP】【二分+贪心】

    一开始想的贪心,可是发现贪心的问题太多了啊!只能保证当前最优,全局完全无法考虑. 所以正解是dp.预处理出前缀和,枚举每个区间,在每个点记录$now[i]$表示以$i$这个塔结尾的塔组目前的高度.$d ...

  10. bzoj 3931: [CQOI2015]网络吞吐量 -- 最短路+网络流

    3931: [CQOI2015]网络吞吐量 Time Limit: 10 Sec  Memory Limit: 512 MB Description 路由是指通过计算机网络把信息从源地址传输到目的地址 ...