函数栈EIP、EBP、ESP寄存器的作用

article/2025/10/4 8:38:58

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

1 栈的结构

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

  1. 先进后出。
  2. 在内存中表现为从高地址往低地址增长。
  3. 栈顶:栈的最上方(低地址区)。
  4. 栈底:栈的最下方(高地址区)。

假设我们有一个函数:

void fun(int a) {int b;char s;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 <+0>:     push   ebp ;ebp压栈
=> 0x800011ba <+1>:     mov    ebp,esp ;将当前栈顶当作函数的栈底0x800011bc <+3>:     push   ebx0x800011bd <+4>:     sub    esp,0x140x800011c0 <+7>:     call   0x800010c0 <__x86.get_pc_thunk.bx>0x800011c5 <+12>:    add    ebx,0x2e3b0x800011cb <+18>:    sub    esp,0xc0x800011ce <+21>:    lea    eax,[ebp-0x9]0x800011d1 <+24>:    push   eax0x800011d2 <+25>:    call   0x80001040 <gets@plt>0x800011d7 <+30>:    add    esp,0x100x800011da <+33>:    cmp    DWORD PTR [ebp+0x8],0x12340x800011e1 <+40>:    jne    0x800011f2 <fun+57>0x800011e3 <+42>:    sub    esp,0xc0x800011e6 <+45>:    lea    eax,[ebp-0x9]0x800011e9 <+48>:    push   eax0x800011ea <+49>:    call   0x80001050 <puts@plt>0x800011ef <+54>:    add    esp,0x100x800011f2 <+57>:    nop0x800011f3 <+58>:    mov    ebx,DWORD PTR [ebp-0x4]0x800011f6 <+61>:    leave0x800011f7 <+62>:    ret ;跳转到EIP地址
End of assembler dump.
Dump of assembler code for function main:0x800011f8 <+0>:     lea    ecx,[esp+0x4]0x800011fc <+4>:     and    esp,0xfffffff00x800011ff <+7>:     push   DWORD PTR [ecx-0x4]0x80001202 <+10>:    push   ebp0x80001203 <+11>:    mov    ebp,esp0x80001205 <+13>:    push   ecx0x80001206 <+14>:    sub    esp,0x40x80001209 <+17>:    call   0x80001230 <__x86.get_pc_thunk.ax>0x8000120e <+22>:    add    eax,0x2df20x80001213 <+27>:    sub    esp,0xc0x80001216 <+30>:    push   0x1234 ;参数压栈0x8000121b <+35>:    call   0x800011b9 <fun> ;将EIP压栈,并跳转到fun函数0x80001220 <+40>:    add    esp,0x100x80001223 <+43>:    mov    eax,0x00x80001228 <+48>:    mov    ecx,DWORD PTR [ebp-0x4]0x8000122b <+51>:    leave0x8000122c <+52>:    lea    esp,[ecx-0x4]0x8000122f <+55>:    ret

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

2 EIP、EBP、ESP的作用

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

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

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

3 函数调用前后变化

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

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

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

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

mov esp, ebp
pop ebp

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

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

4 结束语

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


http://chatgpt.dhexx.cn/article/yLfj7EEB.shtml

相关文章

你了解函数调用过程吗?

函数调用是编程语言都有的概念&#xff0c;也许你听说过函数调用栈&#xff0c;但是大家都知道函数调用是如何完成的吗&#xff1f;我们为什么要了解这个过程: 对于程序运行机制中的数据结构和实现的了解&#xff0c;对自己开发程序有着启发作用碰到一些疑难杂症的时候&#x…

ebp/栈帧/call stack

一.什么是ebp&#xff1f; &#xff08;1&#xff09;ESP&#xff1a;栈指针寄存器(extended stack pointer)&#xff0c;其内存放着一个指针&#xff0c;该指针永远指向系统栈最上面一个栈帧的栈顶。 &#xff08;2&#xff09;EBP&#xff1a;基址指针寄存器(extended base …

栈帧详解

一. 理解栈帧 栈帧是什么&#xff0c;我们基本的理解是栈帧是栈帧也叫过程 活动记录&#xff0c;是 编译器用来实现过程/ 函数调用的一种数据结构。通俗来说栈帧就时C语言函数在调用的过程中的调用原理&#xff0c;就是当我们执行一个函数操作的时候&#xff0c;它的内部是如何…

EBP 和 ESP 详解

基本概念&#xff1a; &#xff08;1&#xff09;ESP&#xff1a;栈指针寄存器(extended stack pointer)&#xff0c;其内存放着一个指针&#xff0c;该指针永远指向系统栈最上面一个栈帧的栈顶。 &#xff08;2&#xff09;EBP&#xff1a;基址指针寄存器(extended base poin…

基于生命周期的开发方法——迭代开发方法

迭代开发方法 上一篇原型方法只是一种需求验证的手段&#xff0c;如果将其思想运用到整个开发过程&#xff0c;使得每个阶段的任务经过反复多次&#xff0c;或者将分析、设计、实施的周期反复多次&#xff0c;通过一次次迭代&#xff0c;不断在原来的基础上完善和修正&#xf…

2.迭代开发的过程是怎么样的

敏捷开发系列文章目录 在讨论PO如何给团队讲好故事这个问题之前&#xff0c;先给大家了解一些基本的敏捷概念&#xff0c;然后讲讲我们敏捷团队构成与整个敏捷开发的过程。 当初敏捷老师讲课的时候就跟我们所过&#xff0c;敏捷没有什么具体的形式&#xff0c;每个敏捷团队可能…

RUP之动态结构:迭代开发

迭代过程一般分为四个阶段&#xff1a;初始、细化、构造和移交&#xff0c;简称为I,E,C和T。每个阶段以一个重要的里程碑(milestone)结束。 初始(Inception)阶段 确定最终产品的构想及其业务用例、并定义项目范围 初始阶段以生命周期目标(LCO)里程碑为结束点 细化(Elaborat…

开发模式(敏捷开发,瀑布式开发,螺旋型开发,迭代开发,devOps开发)

一. 敏捷开发 以人为核心、迭代、循序渐进的开发方式 简化文档&#xff0c;提取文档重点&#xff0c;主要在于人与人之间的沟通&#xff0c; 对开发产品进行迭代&#xff0c;最终完成开发。 迭代&#xff1a;迭代是指把一个复杂且开发周期很长的开发任务&#xff0c;分解为很…

敏捷开发-快速迭代

&#xff08;转自:http://blog.csdn.net/xiaoxian8023/article/details/8883791&#xff09; 今天跟大家分享的是“敏捷开发、快速迭代”。我们大都采用的是“瀑布开发模式”&#xff0c;有了问题&#xff0c;就得返工&#xff0c;虽然最终的产品会比较齐全完善&#xff0c;但是…

一文搞定软件过程模型——瀑布模型、增量式开发/增量开发与迭代开发的区别

软件开发比较经典的过程模型有&#xff1a; 瀑布模型&#xff1a;该模型将基本的过程活动、描述、开发、有效性验证和进化&#xff0c;看成是一些界限分明的独立的过程阶段&#xff0c;例如&#xff0c;需求描述阶段、软件设计阶段、实现阶段、测试阶段等。增量式开发&#xf…

软件迭代开发流程

如果对python自动化测试、web自动化、接口自动化、移动端自动化、大型互联网架构技术、面试经验交流等等感兴趣的老铁们&#xff0c;可以关注我。我会在公众号&#xff08;程序员阿沐&#xff09;/群里&#xff08;810119819&#xff09;不定期的发放免费的资料链接&#xff0c…

迭代开发流程

转载于:https://www.cnblogs.com/gispathfinder/p/8747869.html

git 迭代开发分支流程规范

前言 Git 的名称太响了&#xff0c;相信大家都听说过&#xff0c;鼎鼎大名的同性交友网站 GitHub 就是基于 Git 作为唯一的版本库格式进行托管的平台。本篇不详细介绍 Git 的使用&#xff0c;仅介绍基于 Git 的开发分支流程规范&#xff0c;需要对 Git有一定的了解。 简述 G…

迭代开发和增量开发

&#xff08;转自&#xff1a;http://blog.csdn.net/l12345678/article/details/5642851&#xff09; “迭代”和“增量”是敏捷软件开发中的两个重要概念。弄清楚“迭代”和“增量”以及其依据&#xff0c;我们就可以在实际的操作中有章法可循。 为什么要迭代&#xff1f; 我们…

转」最佳方案:迭代式开发

前言 Fred Brooks 在 25 年前就曾写到&#xff1a;“不要指望一次成功&#xff0c;无论如何你都要这样。” 敏捷开发&#xff0c;小步快跑&#xff0c;持续迭代&#xff0c;不断改进&#xff0c;产品升级。 在用例需要之前&#xff0c;不要添加数据成员 在代码之前编写测试 …

alert意为:警告、警报。

在JavaScript代码中&#xff0c;alert的作用是弹出一个浏览器对话框&#xff0c;"JS代码"为提示对话框的内容 <body><script type"text/JavaScript"> alert("提交成功&#xff01;我会尽快与你取得联系"); </script></body…

JS中alert和alter

JS中没有alter&#xff0c;只用alert&#xff0c;alert表示输出后面的语句

js 的 alert函数问题

alert是js人非常熟悉的东西&#xff0c;可以用来调试&#xff0c;写在这里是想警醒我自己&#xff0c;以后再遇到这样的问题的时候谨记这么修改。 看一个简单例子&#xff1a; <body><div οnmοuseοveralert((function(){return"abc";}()))>abc</…

Alert 提示框的使用

1. Alert 基本的使用 1.1 实现 /// 提示框 struct AlertBootcamp: View {State var showAlert: Bool falseState var alertType: MyAlerts? nilState var alertTitle: String ""State var alertMessage: String ""State var backgroundColor: Color …

java中的alert是什么意思_javascript 中如何使用alert?

alert不行的,要变通的话还是用showModalDialog试试吧. window.showModalDialog()方法用来创建一个显示HTML内容的模态对话框&#xff0c;由于是对话框&#xff0c;因此它并没有一般用window.open()打开的窗口的所有属性。 window.showModelessDialog()方法用来创建一个显示HTML…