从汇编角度看函数调用过程

news/2024/7/2 0:58:15 标签: 汇编, c++, 算法, 函数调用过程

本文以一个简单的程序为例,通过汇编代码查看函数调用过程,涉及如何开辟栈帧,函数如何返回等

#include <iostream>
using namespace std;

int sum(int a, int b)
{
	int temp = 0;
	temp = a + b;
	return temp;
}

int main()
{
	int a = 10;
	int b = 20;
	int ret = sum(a, b);
	cout << "ret: " << ret << endl;
	return 0;
}

代码非常简单,调用一个sum函数计算两数之和。

下面通过vs2017调试代码,看看代码编译成汇编指令后,是如何开辟栈帧及函数调用的。

(通过vs调试时记得把编译优化选项关了)

这是main函数部分汇编代码:

int main()
{
00271630  push        ebp  
00271631  mov         ebp,esp  
00271633  sub         esp,0Ch  
	int a = 10;
00271636  mov         dword ptr [ebp-8],0Ah  
	int b = 20;
0027163D  mov         dword ptr [ebp-4],14h  
	int ret = sum(a, b);
00271644  mov         eax,dword ptr [ebp-4]  
00271647  push        eax  
00271648  mov         ecx,dword ptr [ebp-8]  
0027164B  push        ecx  
0027164C  call        00271610  
00271651  add         esp,8  
00271654  mov         dword ptr [ebp-0Ch],eax 
	cout << "ret: " << ret << endl;
    //.....
}

(vs可能会用变量名代替地址,比如mov dword ptr [a],0Ah,只需要右键取消勾选显示符号名即可看到源地址)

可以看到进入main函数做的第一件事就是开辟栈帧,push ebp将栈底指针压栈,mov ebp esp将·esp复制给ebp,即将栈底指针指向栈顶,sub esp,0Ch栈顶指针减去0Ch,实则就是开辟栈空间。(栈向低地址增长)
在这里插入图片描述
随后将0Ah赋值给dword ptr[ebp-8]0Ah是10,也就是a变量,b同理。
在这里插入图片描述
接着调用sum函数,在调用函数之前这里进行了参数压栈操作,先将b的值给到eax,再进行压栈,a同理。
在这里插入图片描述
看到这里我们可以得出一个结论,参数是在函数调用方压栈的。

下面的call指令,调用sum函数,值得注意的是,调用call函数之前会将call后面的指令地址(00271651)压栈,为了在函数返回时继续向下执行。
在这里插入图片描述
进入到sum函数,这是sum函数汇编代码:

int sum(int a, int b)
{
00271610  push        ebp  
00271611  mov         ebp,esp  
00271613  push        ecx  
	int temp = 0;
00271614  mov         dword ptr [ebp-4],0  
	temp = a + b;
0027161B  mov         eax,dword ptr [ebp+8]  
0027161E  add         eax,dword ptr [ebp+0Ch]  
00271621  mov         dword ptr [ebp-4],eax  
	return temp;
00271624  mov         eax,dword ptr [ebp-4]  
}
00271627  mov         esp,ebp  
00271629  pop         ebp  
0027162A  ret  

在sum函数中,先将ebp压栈,再让ebp指向esp的位置。后将ecx压栈:
在这里插入图片描述
再往下mov dword ptr [ebp-4],0 ecx的位置赋为0。mov eax,dword ptr [ebp+8]ebp + 8刚好指向a的位置,将a的值移到eax寄存器中。add eax,dword ptr [ebp+0Ch]连加上b的值,存放在eax中。mov dword ptr [ebp-4],eaxeax的值移到ebp - 4的位置,最后在函数返回时,mov eax,dword ptr [ebp-4]将结果存放在eax寄存器中。之后进行函数返回的操作,mov esp,ebp
在这里插入图片描述
pop ebp,弹出值给到ebp,而现在栈顶的值刚好是之前保存的ebp的值:
在这里插入图片描述
ret指令首先弹出栈顶元素,并把弹出的内容放到PC寄存器中:
在这里插入图片描述
PC寄存器中存放的是下一条要执行的指令的地址,一个神奇的事情是,刚刚弹出的地址(00271651)刚好是call指令的下一条指令,也就是执行完sum函数后的下一条指令。这也就解释了函数调用完是怎么接着往后执行的。

回到main函数中,add esp,8将栈顶指针加8,回退栈顶指针,“回收”临时的函数参数。
在这里插入图片描述
这时回到最初的起点,mov dword ptr [ebp-0Ch],eax将计算所得的结果给到dword ptr [ebp-0Ch]。至此函数执行完成。可见,其实并不复杂,当然示例比较简单,但道理都一样。清楚了整个函数调用过程,或许就能更好理解为什么不要返回局部变量的地址?

int *fun()
{
	int temp = 5;
	return &temp;
}

int main()
{
	int *p;
	p = fun();//为什么不要这样做?
	cout << *p << endl;
	return 0;
}

因为在函数执行完成后,栈帧已经交还给了系统,虽然这时可以得到正确结果,但这只是因为系统没有对栈帧内容清空。

如果在打印*p之前调用一下函数sum(1, 2),这时结果就是不确定的了。

int main()
{
	int *p;
	p = fun();
	sum(1, 2);
	cout << *p << endl;
	return 0;
}
//输出8322272

所以不要返回局部变量地址,即便当时程序没有报错!!!

清楚了整个函数调用过程,或许就能更好理解为什么有时未初始化的数据在调试模式下显示“烫”或者“屯”,这是因为,开辟的栈空间的每一个字节默认初始化为0xCC,而0xCCCC的汉字编码就是“烫”。有时编译器还会使用0xCDCD来初始化,这时看到的就是“屯”。

清楚了整个函数调用过程,或许就能更好理解栈非法访问以及爆栈的问题(一个进程的栈空间默认为8M左右,可以修改大小,记得之前面试被问过这个问题)。

看看ChatGPT给出的解释:
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述


http://www.niftyadmin.cn/n/174418.html

相关文章

C++智能指针原理及用法总结

部分内容摘自 链接 原理 智能指针是一个类&#xff0c;用来存储指向动态分配对象的指针&#xff0c;负责自动释放动态分配的对象&#xff0c;防止堆内存泄 漏。动态分配的资源&#xff0c;交给一个类对象去管理&#xff0c;当类对象声明周期结束时&#xff0c;自动调用析构函数…

疫情结束,数字化还在继续,“无代码”趋势下谁是受益者?

“低代码”与“无代码”“无代码”与“低代码”的概念往往是关联出现。通常意义上的“无代码”&#xff0c;是一种不需要任何代码的、适用于所有人的数字开发平台。这使得不懂编程的人&#xff0c;能够像组装“积木”那样&#xff0c;轻松地开发出一个新的应用。“低代码”这个…

零售数据分析之操作篇5:近期计算与近期筛选

各位数据的朋友&#xff0c;大家好&#xff0c;我是老周道数据&#xff0c;和你一起&#xff0c;用常人思维数据分析&#xff0c;通过数据讲故事。 又有一段时间没来了&#xff0c;抱歉&#xff0c;不过&#xff0c;不管多忙&#xff0c;我都会继续。我始终相信&#xff0c;任…

助力私募行业高质量发展,华西证券“金华彩杯”年度盛会即将启幕

践行传播行业文化&#xff0c;进一步促进私募行业高质量发展&#xff0c;资本市场这场现象级行业盛会即将拉开帷幕。 3月22日&#xff0c;第三届“金华彩杯”私募实盘大赛启动仪式暨第二届颁奖典礼将在北京隆重举行&#xff0c;著名学者、券商首席分析师、优秀私募管理人、资管…

整合中非咖啡全产业链,长沙新打法为咖啡品牌实现强赋能

中国咖啡市场潜力到底有多大&#xff1f;一则数据就可说明&#xff01;据NCBD数据显示&#xff0c;2021年中国现制咖啡市场规模为924.5亿元&#xff0c;与咖啡相关的企业成立数达到2.6万家&#xff0c;门店数已突破12万家。预计到2025年&#xff0c;中国现制咖啡市场规模将突破…

Linux内核进程管理实时调度与SMP

一&#xff0c;实时调度器类实时调度类有两类进程&#xff1a;循环进程SCHED_RR&#xff1a;循环进程有时间片&#xff0c;随着进程的运行时间会减少。当时间片用完时又将其置为初值&#xff0c;并将进程置于队列末尾。先进先出SCHED_FIFO&#xff1a;没有时间片调度&#xff0…

SpringSecurity基础教程

SpringSecurity 目标 权限管理简介【了解】权限管理解决方案【掌握】初识Spring Security【了解】Spring Security 认证配置【重点】Spring Security 鉴权配置【重点】Spring Security 退出操作【掌握】 一、权限管理简介 1、什么是权限管理 基本上涉及到用户参与的系统都…

【openGauss】账本机制与闪回特性使用

账本机制 对防篡改账本表进行数据增、删、改操作&#xff0c;系统会在历史表和全局表中记录行级数据变化追溯信息&#xff0c;操作的追溯信息&#xff0c;并通过密码学算法生成校验信息。通过对校验信息的保护、比对、验证&#xff0c;起到对重要数据的完整性保护&#xff0c;…