ebp 函数堆栈esp_函数栈&EIP、EBP、ESP寄存器的作用 | kTWO-个人博客

这第一篇文章咱们就来重新认识一下EIP、EBP、ESP这三个寄存器,寄存器又好几个,但是为什么我们要单独看这几个呢?因为在很多情况下我们在调试的时候最注意的就是这三个寄存器,其实这几个寄存器都是为“栈”而生,下面将结合图片分别谈谈这几个寄存器。

0x01 栈的结构

“栈"想必大家都很熟悉了,我们再重复一遍他的几个重要性质和概念。

1、先进后出。

2、在内存中表现为从高地址往低地址增长。

3、栈顶:栈的最上方(低地址区)。

4、栈低:栈的最下方(高地址区)。

假设我们有一个函数:

void fun(int a) {

int b;

char s;

gets(&s);

if(a == 0x1234){

puts(&s);

}

}

1

2

3

4

5

6

7

8

voidfun(inta){

intb;

chars;

gets(&s);

if(a==0x1234){

puts(&s);

}

}

下面是这个栈的结构图:

从栈的结构中我们可以看到,这个栈中周多个临时变量,数字的代表其入站顺序。其中ESP指向了var3,在栈的顶部,EBP指向了栈的底部。在EBP的下面还有一个EIP,这里其实可以理解为我们的函数返回地址。当return语句执行后,下一条指令的执行地址。那么var1是什么呢?其实这个是函数的参数,我们对应代码说明一下var1-var3。

var1:参数a

var2:变量b

var3:变量s

当函数执行之前,函数的参数(a)会首先被push进栈里面,当进入函数之前,当前EIP的值也会被push进栈,进入函数后再将EBP压栈,所以形成了上面的结构,下面是该程序的main函数和fun函数(去掉了变量b)的汇编代码。

Dump of assembler code for function fun:

0x800011b9 :push ebp ;ebp压栈

=> 0x800011ba :mov ebp,esp ;将当前栈顶当作函数的栈低

0x800011bc :push ebx

0x800011bd :sub esp,0x14

0x800011c0 :call 0x800010c0 <__x86.get_pc_thunk.bx>

0x800011c5 :add ebx,0x2e3b

0x800011cb :sub esp,0xc

0x800011ce :lea eax,[ebp-0x9]

0x800011d1 :push eax

0x800011d2 :call 0x80001040

0x800011d7 :add esp,0x10

0x800011da :cmp DWORD PTR [ebp+0x8],0x1234

0x800011e1 :jne 0x800011f2

0x800011e3 :sub esp,0xc

0x800011e6 :lea eax,[ebp-0x9]

0x800011e9 :push eax

0x800011ea :call 0x80001050

0x800011ef :add esp,0x10

0x800011f2 :nop

0x800011f3 :mov ebx,DWORD PTR [ebp-0x4]

0x800011f6 :leave

0x800011f7 :ret ;跳转到EIP地址

End of assembler dump.

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

20

21

22

23

24

Dumpofassemblercodeforfunctionfun:

0x800011b9:pushebp;ebp压栈

=>0x800011ba:movebp,esp;将当前栈顶当作函数的栈低

0x800011bc:pushebx

0x800011bd:subesp,0x14

0x800011c0:call0x800010c0<__x86.get_pc_thunk.bx>

0x800011c5:addebx,0x2e3b

0x800011cb:subesp,0xc

0x800011ce:leaeax,[ebp-0x9]

0x800011d1:pusheax

0x800011d2:call0x80001040

0x800011d7:addesp,0x10

0x800011da:cmpDWORDPTR[ebp+0x8],0x1234

0x800011e1:jne0x800011f2

0x800011e3:subesp,0xc

0x800011e6:leaeax,[ebp-0x9]

0x800011e9:pusheax

0x800011ea:call0x80001050

0x800011ef:addesp,0x10

0x800011f2:nop

0x800011f3:movebx,DWORDPTR[ebp-0x4]

0x800011f6:leave

0x800011f7:ret;跳转到EIP地址

Endofassemblerdump.

Dump of assembler code for function main:

0x800011f8 :lea ecx,[esp+0x4]

0x800011fc :and esp,0xfffffff0

0x800011ff :push DWORD PTR [ecx-0x4]

0x80001202 :push ebp

0x80001203 :mov ebp,esp

0x80001205 :push ecx

0x80001206 :sub esp,0x4

0x80001209 :call 0x80001230 <__x86.get_pc_thunk.ax>

0x8000120e :add eax,0x2df2

0x80001213 :sub esp,0xc

0x80001216 :push 0x1234 ;参数压栈

0x8000121b :call 0x800011b9 ;将EIP压栈,并跳转到fun函数

0x80001220 :add esp,0x10

0x80001223 :mov eax,0x0

0x80001228 :mov ecx,DWORD PTR [ebp-0x4]

0x8000122b :leave

0x8000122c :lea esp,[ecx-0x4]

0x8000122f :ret

1

2

3

4

5

6

7

8

9

10

11

12

13

14

15

16

17

18

19

Dumpofassemblercodeforfunctionmain:

0x800011f8:leaecx,[esp+0x4]

0x800011fc:andesp,0xfffffff0

0x800011ff:pushDWORDPTR[ecx-0x4]

0x80001202:pushebp

0x80001203:movebp,esp

0x80001205:pushecx

0x80001206:subesp,0x4

0x80001209:call0x80001230<__x86.get_pc_thunk.ax>

0x8000120e:addeax,0x2df2

0x80001213:subesp,0xc

0x80001216:push0x1234;参数压栈

0x8000121b:call0x800011b9;将EIP压栈,并跳转到fun函数

0x80001220:addesp,0x10

0x80001223:moveax,0x0

0x80001228:movecx,DWORDPTR[ebp-0x4]

0x8000122b:leave

0x8000122c:leaesp,[ecx-0x4]

0x8000122f:ret

其实只要搞清楚上面的EIP、ESP、EBP的变化即可。

0x02 EIP、EBP、ESP的作用

EIP存储着下一条指令的地址,每执行一条指令,该寄存器变化一次。

EBP存储着当前函数栈底的地址,栈低通常作为基址,我们可以通过栈底地址和偏移相加减来获取变量地址(很重要)。

ESP就是前面说的,始终指向栈顶,只要ESP指向变了,那么当前栈顶就变了。

0x03 函数调用前后变化

函数调用的栈结构图其实是下面的这种情况:

再main函数没有调用完之前其部分变量仍然是存在栈中的。函数调用前后基本EIP、EBP、ESP基本变化流程如下:

1、调用函数中push ebp,将main函数的ebp压栈,然后mov ebp, esp将当前函数的esp赋给ebp,得到当前函数的栈底地址。

2、调用函数结束之前,执行leave指令,其实该指令等于下面两条指令:

mov esp, ebp

pop ebp

1

2

movesp,ebp

popebp

此时fun相关数据全部被出栈,ebp将得重新到main函数的栈底地址,注意在执行ret指令时,将获取站内EIP数据,然后栈内的EIP也将出栈。程序跳转到函数下方。esp回到函数栈顶部,函数调用结束。

3、继续执行main函数内指令。

0x04 结束语

上面时函数的调用过程浅析,对学习栈溢出非常之重要必看必会系列。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值