免杀笔记 ----->汇编基础

免杀的第一件事,当然是学汇编啦,小编也是从0开始呢!!!一起来看看吧!

免责声明,本人所有教学内容仅供参考,无任何不良引导,内容资料来自网络,如有侵权请联系删除!!!!

**1.**数据宽度

学计算机的大家一定不会陌生,比如经常听什么一个字节等于8个比特 ....这种之类的。

其实可以分为以下这些单位

  • 比特 比如像我们平时所看见的那些 0101101001010这样子,一位就是一个bit
  • 字节 也可以叫做Byte,就好像我们上面说的,一个字节就是8个比特
  • 字 是计算机进行数据存储和数据处理的运算的单位。而一个字又是两个字节

这里就要说一下,如果你往一个32位寄存器里面存很大的数,肯定是不行的,超过了数据宽度

2.进制

我们在生活中用的最多的就是十进制了 ,但是计算机可不是用的十进制 而是二进制01010这些

其中我们在免杀中需要熟悉的,就是2进制和十进制吧,8进制的话少一点

就比如 0x08 = 8 = 0100 , 0x10 = 16 = 10000 0xC = 12 = 1100

这种简单的计算还是要会

3.通用寄存器

我们一开始先是去研究32位下的寄存器

  • EAX
  • ECX
  • EDX
  • EBX
  • ESP
  • EBP

我们去OD看一下mimikatz的就知道了 ,这个OD只能打开32位的哦,你别拿64位的mimikatz 那当然是打不开的

我们看见最右边那个就是对应的通用的寄存器

4.OD的使用

虽然在这里插入感觉有的点突兀,但是既然提到了OD,那还是要给大家去讲一下怎么使用的

  • F8 单步步过

当我们想要从0075ECF7走到0075ECFA的时候只需要F8 一下就好了!!!

  • F7 单步步入

就拿我们的c++来举例子

当我们输出完1之后我们就要进入函数,如果f8的话就直接略过函数,f7则是单步进入

那这个举例子,如果我们继续f8的话,就会直接执行这个函数的结果,然后走到0075ED0E

如果我们F7的话,就会进入到栈(可以看见前面的地址都不一样了)

  • F2 打断点

假如我现在有一个程序在 0075EF12这个位置我想让他跑去 0075EF27这个位置

我们只需要在 0075EF27 这个位置点一下,然后直接F2

他变红了,但是此时我们程序还是运行到了0075EF27这个位置而已

  • F4 运行到指定位置

继续上面,我们只要F4一下,就能让他跑到0075EF27这个位置

5.汇编指令

1.mov

其实就是move,我来操作一遍你就懂了

这个指令就是将334的值移动到eax这个寄存器里面

看,瞬间就变红了 其中eax就是寄存器,然后0x334我们称之为立即数

当然了,除了我们直接这样子赋值,也可以直接这么写

意思就是把Dword(32位数据类型)ptr 在 0x00BF5F9D4这个地址的值给拿出来,丢到EAX里面

不出什么以外的话,EAX应该是变成了00B5F9E8

当然,也是可以直接把寄存器当作一个地址来用的,但这时候寄存器的值就变成了一个地址,而不是寄存器的值 我们后买你lea会讲到

2.add sub

这个看起来都很明白了吧,我就直接用一个示例来演示一下得了

这个意思就是EAX=EAX+0x11 有没有编成里面的+=的意思捏,嘻嘻嘻

不出意外就是 00CFFC55

那么同理 SUB也是一样的!!! 我就不过多演示了

3.and or xor

这个可能对于学过离散数学的会比较好理解,我们还是来举一个例子

4 and 7 也就是离散数学上的 4 和取7 (二进制)

  • 4:0100
  • 7:0111 这两个合取结果应该是 0100 也就是4

结果也是如此

然后就是我们的XOR 也就是异或 为我们还是拿上面的来举例子

  • 4:0100
  • 7:0111 这两个异或结果应该是 0011 (异或就是只有01同时存在才为1)

那么答案不出意外的话,应该就是 3

4.lea

这个用的多,最简单的就是这样

将0x446这个地址(不是这个地址的数)直接给到EAX,所以EAX应该会变成446

但是一般没人会这么用,我们更多的是直接用的寄存器

至于为什么要这样子用,我们讲到堆栈的时候你就懂了

最常用的更是我们的立即数配和寄存器

至于为什么,我们后面再说

5.push pop

这两个都是涉及到了堆栈,我们下面再讲

6.cmp

这个我们也放在下面讲

5.内存寻址

这个其实我们在上面的add的时候就已经提到了一些了

其实也就是通过内存地址来获取或者得到一些值,这个内存地址,你可以直接写地址,或者直接写寄存器。

不过还是要说一下,我们刚才的那个 dword ptr ds:[0x2121312]这个例子

我们里面提到的ds ,在用户层只是起寻址的作用,在内核层涉及到段,页有别的作用
cs :代码段
ds :数据段
ss :堆栈段
es :附加段

6.代码是怎么执行的

我们一定用过VS或者devcpp等其他编译器写过代码,那么他的底层逻辑是什么呢???

因为我们的CPU只能看的懂0 1 ,所以一般是这样的一个过程

c/cpp代码 ---> 汇编代码 ------> 硬编码 ------> CPU

我们去VS看一下,就像这段最简单的hello world

我们右键转到反汇编

就能看见我们的汇编代码

然后我们再把他转换成硬编码

然后就是到了CPU那里了,我们的 "Hello World!" 就是这么被执行的

7.标志寄存器

我们再OD里面除了能看见我们的通用寄存器,我们还能看见这个标志寄存器

对于32位下,我们的标志寄存器叫做EFlag ,也就是我们看见上面的EFL

在64位下,我们的标志寄存器就叫做RFlag

其中EFlag寄存器各个位置有各个位置的不同作用,特别是ZF这个位置,也就是上面我们看见的Z这个位置,我们来举个栗子

我们用CMP这个汇编指令的时候,我们总不能把返回结果存储在eax中吧,这时候就要用到标志寄存器中的ZF这一个位置

我们能看见Z这个位置变成了1 就是说明了结果相同,这个到后面的JCC语句会用

同样的还有CF

CF:进位标志CF(Carry Flag),如果运算结果的最高位产生了一个进位或借位,其值为1,否则为0

OF

如果运算结果超过当前运算位数所能表示的范围,则称为溢出,OF的值被置为1,否则,OF的值被清为0。

8.堆栈

栈Stack,堆Heap这两个对于程序员来说更是经典

:用于存储函数的局部变量和函数调用的上下文信息。栈上的数据在函数执行结束时自动释放。

这个其实很好理解,就像这个代码,我们的两个a都是开在了栈上面

:用于动态分配的内存,由程序员手动分配和释放。在C语言中,使用 malloc()calloc() 等函数分配的内存位于堆上。

这个也很好理解,像下图,就是在堆区上开辟了一个长度为三的整数数组,并且用p指向了首地址

而规定了,进出函数的前后堆栈的地址不能改变

Jmp

这个其实就是Jump的缩写,就是直接跳转到其他地址

我们之际JMP一下,就能直接跳到我们对应的地方了

但是JMP不会堆堆栈的位置进行改变,怎么看是否改变呢??? 还记得我们之前的通用寄存器码

  • ESP 指向栈顶
  • EBP 指向栈底

我们可以去尝试,堆栈在JMP下是不会发生改变的

Call Ret

这两个一般都是搭配使用的

  • Call 跳转到指定地址并返回到当前指令的下一行,当我们调用这个的时候,他会先将esp提升四个字节,并将返回地址压入堆栈 (提栈)
  • ret 返回到指定地址,当我们调用这个指令的时候,会将esp降低四个字节,并且从堆栈中取出返回地址,传递给EIP(运行的地址)

这么说似乎有点抽象,我们来举个栗子就懂了

首先我们遇到了一个call,然后有以下的信息

  • 当前的地址是 EF12 下一步的地址是 EF17
  • 当前的EBP是 FB34 ESP的地址是FB28

然后我们单步F7进入一下

然后又能收集到以下的信息

  • 我们的ESP减少了4(提栈)ESP变成了FB24
  • 并且我们的返回地址EF17被压栈,压入了FB24

然后我们直接跑到retn那里

可以看见函数跑了一通之后我们的ESP还是FB24里面还是EF17,然后我们F8走一下

还是能收集到以下的信息

  • retn成功的返回了我们的地址
  • ESP是FB28 EBP是FB34 和函数一开始调用时候的栈地址是一样的!!!!!

Push Pop

这两个其实和Call 和Ret有点像的,但是也有差异

在介绍这两个命令之前,还得讲一下栈的数据存储,是从高到低的,比如我上一个是94,那么我压栈存数据之后就应该存在90这个地址!

Push 压栈,将ESP减少4,然后将数据压入。

有点类似call的功能,只是将返回的地址更换成了数据

Pop 出栈 将ESP加4,并且将原来ESP的数值取出放到指定的地址或者寄存器

有了上面的Push的基础,我们就不难理解我们的pop了,直接举例子吧

像这样,就是直接把ESP的值取出来,mov到EAX,然后将ESP的值加上4

9.JCC语句

JCC不知道能不能被翻译为Jump Condition Code

其实就是和标志寄存器实现了一个联动,一般格式如下

bash 复制代码
JCC target_address

JZ/JE jump if zero;jump if equal 其实就是检查ZF这个位是不是为1,是则跳转
举个栗子,首先我们构造一个Z标志位为1的一种情况

然后我们直接构造一个JCC语句

因为我们的Z位置是1,所以他是会直接跳到我们对应的位置的

JNZ/JNE jump if not zero;jump if not equal 如果ZF为0则跳转
这个其实和上面的那个差不多,我就不给大家演示了。

10.堆栈图

这个算是有点难吧,不过也是理解就好了,其实你把上面的指令都理解清除的话,就不会有问题。

我们之前说的栈平衡其实有点问题,就是并不是说call回来之后esp一定会与原来相同,有些则是通过 add esp,0x...... 这样来实现ESP不变 ------->这样我们也叫他是栈平衡(看他是在函数外平衡堆栈还是在函数内平衡堆栈)


具体堆栈图就不演示了,因为会非常麻烦,而且混乱(除非视频讲解),但是原理还是一样

相关推荐
雪碧透心凉_1 天前
8086汇编(16位汇编)学习笔记00.DEBUG命令使用解析及范例大全
汇编
C66668884 天前
C#多线程
开发语言·汇编·c#
傻童:CPU5 天前
汇编源程序的理解
汇编
木槿715 天前
软件包git没有可安装候选
汇编·git
ok0605 天前
各种开源汇编、反汇编引擎的非专业比较
汇编·开源
roboko_5 天前
MIPS指令集(一)基本操作
汇编
Crossoads6 天前
【汇编语言】内中断(三) —— 中断探险:从do0到特殊响应的奇妙旅程
android·开发语言·javascript·网络·汇编·单片机·机器学习
染指11106 天前
49.第二阶段x86游戏实战2-鼠标点击call深追二叉树
汇编·c++·windows·游戏安全·反游戏外挂·游戏逆向
程序leo源8 天前
深入理解指针
android·c语言·开发语言·汇编·c++·青少年编程·c#
skywalk81639 天前
好玩的汇编编译器NASM:一款基于x86架构的汇编与反汇编软件
汇编