函数调用
函数之间的调用关系是用栈维护的,每 一次函数调用都是一个过程,该过程称为函数的调用过程。该过程需要为函数开辟空间,用于本次函数的调用中临时变量的保存、现场保护,开辟的空间就称为函数栈帧,也叫函数运行时堆栈。接下来我就简要介绍一下函数调用过程中,栈帧的创建与销毁。栈帧的创建需要用到两个寄存器:ebp和esp,在函数调用过程中这两个寄存器存放了维护这个栈的栈底和栈顶指针。ebp存放了指向函数栈帧栈底的地址,称为栈底寄存器或者基址寄存器;esp存放了指向函数栈帧栈顶的指针,称为栈顶寄存器。还有一个较为重要的寄存器,eip寄存器,称为程序计数器,在微机原理中eip也称为PC(point code/program count)指针。eip寄存器是用于存放CPU当前正在执行的指令的下一条指令。
当我们要详细研究函数调用过程,必须得对应汇编代码。以下面这个程序为例,简要介绍一下函数调用过程栈帧的创建与销毁。
#include <stdio.h>
int Add(int x,int y)
{
int z = 0;
z = x+y;
return z;
}
int main()
{
int a = 10;
int b = 20;
int ret = 0;
ret = Add(a,b);
printf("%d\n",ret);
return 0;
}
程序的地址空间主要分为这几部分:栈区(stack)、堆区(heap)、全局区(也称静态区static)、文字常量区、程序代码区(code)。其中全局区可分为初始化区和未初始化区。
注:程序的地址空间指的不是物理地址而是虚拟地址。