首页 > 编程语言 >廖雪峰博客汇编函数压栈的解析

廖雪峰博客汇编函数压栈的解析

时间:2023-05-30 15:11:39浏览次数:35  
标签:汇编 esp 压栈 雪峰 ret eax add call ebx

int add_a_and_b(int a, int b) {
   return a + b;
}

int main() {
   return add_a_and_b(2, 3);
}
_add_a_and_b:
   push   %ebx
   mov    %eax, [%esp+8] 
   mov    %ebx, [%esp+12]
   add    %eax, %ebx 
   pop    %ebx 
   ret  

_main:
   push   3
   push   2
   call   _add_a_and_b 
   add    %esp, 8
   ret

首先我们看call和ret对于栈部分的操作等价于:

call和ret的理解:
  ret 指令相当于 pop eip; esp = esp + 4
  call 指令相当于 push eip; esp = esp - 4

记住这个公式下面就好理解了. 其实就是把当前ip入栈和出站.

栈是一个|  |  高地址

    |  |

    |  |  低地址

栈开口往下. 从高地址作为栈开始的位置, 越压栈栈顶越往小地址走. (记忆的方式就是这样可以让栈最后干到0,保证他有界限,否则内存都被他沾满崩溃了).

下面我们写一下整个汇编对于栈的操作.

add_a_and_b(2, 3) 这行命令对应于:  栈压入3. 压入2 (注意这里面入栈顺序是跟参数顺序相反)

然后call _add_a_and_b函数. 这里面汇编自动每个函数前面加下划线了.

然后我们把call替换成 push eip; esp = esp - 4. 也就是压入ip是call的下一行的记做ip_1

所以当前栈里面为 [3,2,ip_1] .  然后我们进入汇编第一行. _add_a_and_b运行.

继续压入%ebx, 然后   mov    %eax, [%esp+8]   这些操作之后. 栈成为了  [3,2,ip_1,%ebx] ,注意esp

始终指向当前栈当前可以新加入的元素的首地址,也就是当前最后一个元素的最后地址. 

所以mov    %eax, [%esp+8]  指的是 2的首地址.也就是2本身占用4个字节. 他这4个字节里面的最小值

就表示2的地址. 所以这行命令mov    %eax, [%esp+8] 把2的值给eax了. 同理后面

mov    %ebx, [%esp+12] 把3给ebx了. add    %eax, %ebx 把 5给eax了. 然后我们pop %ebx. 又恢复了

栈之前ebx的值. (这样就保证了函数add_a_and_b调用之前的除了ax之外的寄存器跟函数调用之后一样)

(所以我们可以总结一条一个函数调用最开始会把使用的寄存器除了ax以外都入栈)

最后我们ret:

ret 指令相当于 pop eip; esp = esp + 4

所以我们ip=ip_1了.并且当前栈里面是[3,2], 所以我们就继续跑add    %esp, 8 这一行指令.

所以我们的栈为空了. 

这样我们就完美的进行了函数调用也不存在任何栈空间的浪费了!!!!!!!!!

 

标签:汇编,esp,压栈,雪峰,ret,eax,add,call,ebx
From: https://www.cnblogs.com/zhangbo2008/p/17443260.html

相关文章

  • 压栈思想计算Java运算表达式
         栈的规则是先进后出。利用压栈的思想来计算四则运算表达式是这样的:我们给定两个栈,一个用来存放数字、一个用来存放对应的操作符。假定我们有一个给定的四则运算表达式a+b+c/d*(e+f)-d*a,那我们先把这个表达式拆分成一个个的数字或者是运算符、或者就是括号了。然后我们......
  • 汇编DOS-BOX的安装
           ......
  • 汇编指令
    movax,18    将18送入ax     ax=18movax,bx    将bx中的数据放入axaddax,8     将ax中的数值加上8    ax=ax+8addax,bx    将ax,bx中的数据相加放入ax       ......
  • MASM汇编语言知识
    遇到的坑 注意在写$时,一定要加上‘’否则会爆constantexpected的错误,同时也会附带一些其他神奇的错误 MOVAX,BX与MOVAX,[BX]的区别是什么?前者是将BX中保存的值直接给AX后者是通过BX中保存的值寻址后将找的地址中的内容给AX那么也就能够理解MOV......
  • 汇编-debug命令
    r   查看cpu寄存器内容     ......
  • 汇编-寄存器
    8086CPU有14个寄存器:AX  BX  CX   DX  SI  DI   SP   BP   IP   CS   SS   DS   ES   PSW8086CPU所有的寄存器都是16位的,可以存放两个字节       ......
  • ARM64启动汇编和内存初始化(上)
    文章代码分析基于linux-5.19.13,架构基于aarch64(ARM64)。涉及页表代码分析部分:(1)假设页表映射层级是4,即配置CONFIG_ARM64_PGTABLE_LEVELS=4;(2)虚拟地址宽度是48,即配置CONFIG_ARM64_VA_BITS=48;(3)物理地址宽度是48,即配置CONFIG_ARM64_PA_BITS=48;1.入口分析1.1链接脚本arch/a......
  • 汇编-检测题1
     1.某机子的寻址能力是8K,问:地址总线宽度是多少?地址总线宽度是13               ......
  • 汇编-总线
    总线:连接CPU和其他芯片的导线,统称为总线逻辑上分为:地址总线一个CPU有N根地址总线,就说CPU的地址总线的宽度为N这样的CPU最多可以寻址2的N次方个内存单元一根总线就是1Byte=8bit,64位机子有8根地址总线数据总线控制总线         ......
  • 专访高雪峰:从GPT3.5到4,超强推理能力的实现与“图”密不可分
    “符号”与“向量”,AGI的两条腿。作者|王与桐整理|Ricky2023年3月15日,GPT4亮相。尽管以GPT3.5为基础的ChatGPT更具里程碑意义,毕竟引发了全球C端用户的使用,但是在更多AI从业者看来,GPT4的意义远高于3.5,这是因为,GPT4具备了令人惊艳的“逻辑推理”能力。但为什么能够实现“推......