36.【C语言】函数栈帧的创建和销毁

本文将解决以下问题

局部变量是怎么创建的?
为什么局部变量的值是随机值?
函数是怎么传参的?传参的顺序是怎样的?
形参和实参是什么关系?
函数调用是怎么做的?
函数调用是结束后怎么返回的?

本文使用VS2013查看 ,从汇编语言来理解

1.基本概念

寄存器作用:寄存数据

eax:累加器 ebx:作为指针指向数据 ecx:循环计数器 edx:eax的扩展

地址寄存器:ebp,esp(维护函数栈帧)

★注意:在C语言中每一次函数调用,都需要为本次函数调用在内存的栈区中申请一块内存空间来保存函数调用期间的各种局部变量的值,这块空间被称为运行时堆栈或者函数栈帧

画图解释f1427af9cb0a4e629c81a35c1591205b.png

#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 c = 0;
	c = add(a, b);
	printf("%d\n",c);
	getchar();
	return 0;
}

2.main函数的调用

按F11后点击调用堆栈

3058751bc9df447ca991c6847f724415.png

执行完后:

//crtexe.c的代码
#endif  /* _WINMAIN_ */
        void
        )
{
        /*
         * The /GS security cookie must be initialized before any exception
         * handling targetting the current image is registered.  No function
         * using exception handling can be called in the current image until
         * after __security_init_cookie has been called.
         */
        __security_init_cookie();

        return __tmainCRTStartup();
}
……………………
static
int
__tmainCRTStartup(
         void
         );
……………………
#else  /* WPRFLAG */
            __initenv = envp;
            mainret = main(argc, argv, envp);//main函数被调用

则调用关系为mainCRTStartup-->__tmainCRTStartup-->main

3.从反汇编上理解

按F11后右击反汇编,选择不显示符号名

#include <stdio.h>
int add(int x, int y)
{
000213C0  push        ebp  
000213C1  mov         ebp,esp  
000213C3  sub         esp,0CCh  
000213C9  push        ebx  
000213CA  push        esi  
000213CB  push        edi  
000213CC  lea         edi,[ebp+FFFFFF34h]  
000213D2  mov         ecx,33h  
000213D7  mov         eax,0CCCCCCCCh  
000213DC  rep stos    dword ptr es:[edi]  
	int z = 0;
000213DE  mov         dword ptr [ebp-8],0  
	z = x + y;
000213E5  mov         eax,dword ptr [ebp+8]  
000213E8  add         eax,dword ptr [ebp+0Ch]  
000213EB  mov         dword ptr [ebp-8],eax  
	return z;
000213EE  mov         eax,dword ptr [ebp-8]  
}
000213F1  pop         edi  
000213F2  pop         esi  
000213F3  pop         ebx  
000213F4  mov         esp,ebp  
000213F6  pop         ebp  
000213F7  ret 
--------------------------------------------------------------------
int main()
{
00021410  push        ebp  
00021411  mov         ebp,esp  
00021413  sub         esp,0E4h  
00021419  push        ebx  
0002141A  push        esi  
0002141B  push        edi  
0002141C  lea         edi,[ebp+FFFFFF1Ch]  
00021422  mov         ecx,39h  
00021427  mov         eax,0CCCCCCCCh  
0002142C  rep stos    dword ptr es:[edi]  
	int a = 10;
0002142E  mov         dword ptr [ebp-8],0Ah  
	int b = 20;
00021435  mov         dword ptr [ebp-14h],14h  
	int c = 0;
0002143C  mov         dword ptr [ebp-20h],0  
	c = add(a, b);
00021443  mov         eax,dword ptr [ebp-14h]  
00021446  push        eax  
00021447  mov         ecx,dword ptr [ebp-8]  
0002144A  push        ecx  
0002144B  call        000210E6  
00021450  add         esp,8  
00021453  mov         dword ptr [ebp-20h],eax  
	printf("%d\n",c);
00021456  mov         esi,esp  
00021458  mov         eax,dword ptr [ebp-20h]  
0002145B  push        eax  
0002145C  push        25858h  
00021461  call        dword ptr ds:[00029114h]  
00021467  add         esp,8  
0002146A  cmp         esi,esp  
0002146C  call        0002113B  
	return 0;
00021471  xor         eax,eax  
}
00021473  pop         edi  
00021474  pop         esi  
00021475  pop         ebx  
00021476  add         esp,0E4h  
0002147C  cmp         ebp,esp  
0002147E  call        0002113B  
00021483  mov         esp,ebp  
00021485  pop         ebp  
00021486  ret  

以0xCCCCCCCC填充的原因:int3指令机器码为0xCC,当发生指针栈溢出时eip会指向初始化时填充的0xCC从而引发int3断点异常,使程序中断

如果VS2022打开调试模式,会发现main函数的反汇编指令的下方有许多int 3,

图片详解

ecf01c8f537943b38dac02c2739f694a.jpeg

7887465d12ed424db742135002065ecb.jpeg

0cda392eaf6e4772964d5a1f78e6c9e9.jpeg

db6d30ffd1834f98b20c865fecb93447.png

备注:

第⑧~第⑩行类似8086汇编

8086汇编中,cx存储循环次数

每循环一次,cx自减1再判断cx是否为0,如果cz不为0,则循环

常见于两种格式

mov cx,????H

s:

loop s

;==============================

mov cx,????H

;设置源地址和目标地址

;源地址 :si

;目标地址 :di

mov si,????H

mov di,????H

cld或std

rep movsb或rep movsw

4.视频演示

36.【C语言】函数栈帧的创建和销毁文章的汇编演示

5.栈帧的结构

栈帧的定义:函数调用过程中为管理和存储函数相关信息而在上分配的一块内存区域

图来自thecandcppclub.com

6.往期推荐

35.【C语言】详解函数递归

28.【C语言】函数系列上 库函数

29.【C语言】函数系列中 自定义函数

30.【C语言】函数系列下

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

zhangcoder

赠人玫瑰手有余香,感谢支持~

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值