zoukankan      html  css  js  c++  java
  • (转)对于ESP、EBP寄存器的理解

    原文地址https://blog.csdn.net/yeruby/article/details/39780943

    esp是栈指针,是cpu机制决定的,push、pop指令会自动调整esp的值;

    ebp只是存取某时刻的esp,这个时刻就是进入一个函数内后,cpu会将esp的值赋给ebp,此时就可以通过ebp对栈进行操作,比如获取函数参数,局部变量等,实际上使用esp也可以;

    既然使用esp也可以,那么为什么要设定ebp呢?

    答案是为了方便程序员。

    因为esp在函数运行时会不断的变化,所以保存一个一进入某个函数的esp到ebp中会方便程序员访问参数和局部变量,而且还方便调试器分析函数调用过程中的堆栈情况。前面说了,这个ebp不是必须要有的,你非要使用esp来访问函数参数和局部变量也是可行的,只不过这样会麻烦一些。

    通过一段程序理解esp和ebp:

    main() {

    //执行test前

    print(int p1,int p2);

    //执行test后

    }

    分析下上面程序的调用原理,假设执行print前esp=Q:

    push p2; //函数参数p2入栈,esp=Q-4H

    push p1; //函数参数p1入栈,esp=Q-8H

    call print; //函数返回地址入栈,esp=Q-0CH

    //现在进入print内,做些准备工作:

    push ebp; //保护先前ebp指针,ebp入栈,esp=Q-10H

    mov esp,ebp; //设置ebp等于当前的esp

    // 此时,ebp+0CH=Q-4H,即p2的位置

    // 同样,ebp+08H=Q-8H,即p1的位置

    // 下面是print内的一些操作:

    sub esp,20H; //设置长度为10H大小的局部变量空间,esp=Q-20H

    // ... ...

    // 一系列操作

    // ... ...

    add esp,20H; //释放局部变量空间,esp=Q-10H

    pop ebp; //出栈,恢复原先的ebp的值,esp=Q-0CH

    ret 8; //ret返回,弹出先前入栈的返回地址,esp=Q-08H,后面加操作数8H为平衡堆栈

    // 之后,弹出函数参数,esp=Q,恢复执行print函数前的堆栈;

    图示,注意栈在内存中的生长方向是逆向:

    执行push p2;前,esp=Q;

    执行push p2;过程中,esp-=4H,p2入栈;

    执行push p2;后,esp=Q-4H;

    --------------------- 本文来自 qwurey 的CSDN 博客 ,全文地址请点击:https://blog.csdn.net/yeruby/article/details/39780943?utm_source=copy 

  • 相关阅读:
    org.dom4j.DocumentException: Error on line 1 of document: 前言中不允许有内容
    学习过程中的随手笔记
    IT技术团队行而有效的管理之道
    九宫格抽奖HTML+JS版
    Nginx负载均衡深入浅出
    PHP 二维数组根据某个字段排序
    MYSQL INSERT INTO SELECT 不插入重复数据
    小米2成功使用google组件的办法(为了coc游戏能登录google账户)
    PHP 数组排序方法总结
    普通标和转让标的回款和还款日期的算法。
  • 原文地址:https://www.cnblogs.com/wangshaowei/p/9718470.html
Copyright © 2011-2022 走看看