C语言函数调用详细过程

函数调用是步骤如下:

  1. 按照调用约定传参

    • 调用约定是调用方(Caller)和被调方(Callee)之间按相关标准

      对函数的某些行为做出是商议,其中包括下面内容:

      传参顺序:是从左往右传还是从右往左

      传参方式:是用寄存器传还是使用内存传

      平栈方式:是调用方平栈还是被调方平栈

      返回值的传递方式:是用寄存器传还是使用内存传

    • 什么是堆桟?

      一个程序运行的时候,它的进程的地址空间一般可以分为四块:

      代码区,数据区,堆,栈,每块功能如下:

      区域 功能
      代码区 存放函数被编译后的二进制可执行代码
      数据区 只读区:存放常量,例如:常量字符串,const修饰的全局变量等
      可读写区:存放全局变量和静态变量
      除去其他三个区域,剩下的都是堆,不连续
      存放函数运行时所需的参数,寄存器环境,返回值,局部变量

    以下面代码为例:

        int  TestFunction(char szBuff[],int nSize)
    {
    for (int iIndex = 0; iIndex < nSize; iIndex++)
    {
    szBuff[iIndex] = 'x';
    }
    return 3;
    } int main()
    {
    char szBuff[32] = { "sfjdlskfjl" };
    int nRet = TestFunction(szBuff, 32);
    return 0;
    }

    函数参数参数传递:



    从上图中可以看出函数参数入栈

  2. 保存返回地址(紧挨着被调用函数的下一行可执行代码的内存地址)

    从上图中可以看出函数调用完成后,紧挨着的第一条指令为:

    00EB175B add esp,8

    所以,参数传递完成后就是返回值入栈:

  3. 程序流程转移到被调用函数地址处

  4. 保存调用方栈底

  5. 切换到当前函数(被调用函数)的栈底

    调用方栈底保存完成后,当前的栈顶(ESP记录的地址)就成为被调用函数的栈底

  6. 为局部变量分配空间



    这里程序为调试版本,所以为局部变量分配的空间比较大,在Release版本中

    会根据局部变量实际所需空间来分配大小

  7. 保存寄存器环境

    这里一共保存了3个寄存器,共12字节,在Release版本下,只保存两个



    在Debug版程序中(有/Zi(带有调试信息)和/Od(禁止优化)编译命令),除了为

    局部变量分配较大的内存空间外,还会将分配的局部变量空间全部置为0xCC:



    这种填充方式比较直观,能够让我们在调试时直观的观察到是否发生越界等错误

  8. 开始执行函数体代码

    此时当前函数的栈的内存布局如下:

  9. 恢复寄存器环境

  10. 释放分配的局部变量空间

    只是将当前的栈底指针(EBP)的值赋值给栈顶指针(ESP)就完成了:

  11. 恢复调用方栈底

  12. 平栈或者返回

    • 如果是_fastcall,_stdcall调用约定,那么被调用函数平栈后,取出返回地址

      函数流程转移到调用方
    • 其他调用约定则是直接取出保存的返回地址,函数流程返回到调用方,又调用方

      平栈

C语言函数调用完整过程的更多相关文章

  1. C语言函数调用过程,汇编角度查看

    C语言函数调用过程,汇编角度查看 把函数的参数按照调用约定压栈或者存储到寄存器中 调用要使用的函数,先把调用者的地址入栈,方便回来 跳转到函数 把函数使用到的一些寄存器压栈,避免修改寄存器的值 执行函 ...

  2. C语言函数调用时候内存中栈的动态变化详细分析(彩图)

    版权声明:本文为博主原创文章,未经博主允许不得转载.欢迎联系我qq2488890051 https://blog.csdn.net/kangkanglhb88008/article/details/8 ...

  3. 字符型图片验证码识别完整过程及Python实现

    字符型图片验证码识别完整过程及Python实现 1   摘要 验证码是目前互联网上非常常见也是非常重要的一个事物,充当着很多系统的 防火墙 功能,但是随时OCR技术的发展,验证码暴露出来的安全问题也越 ...

  4. 测试c语言函数调用性能因素之测试三

    函数调用:即调用函数调用被调用函数,调用函数压栈,被调用函数执行,调用函数出栈,调用函数继续执行的一个看似简单的过程,系统底层却做了大量操作. 操作: 1,               调用函数帧指针 ...

  5. Django之model模块创建表完整过程

    Django中,与数据库相关的模块是model模块,它提供了一种简单易操作的API方式与数据库交互,它是通过ORM映射的方式来操作数据库,一个类对应数据库一张表,一个类属性,对应该表的一个字段,一个实 ...

  6. UIPickerView/UIDatePicker/程序启动的完整过程

    一.UIPickerView 1.UIPickerView的常见属性 数据源(用来告诉UIPickerView有多少列多少行) @property(nonatomic,assign) id<UI ...

  7. C语言函数调用栈

    C语言函数调用栈 栈溢出(stack overflow)是最常见的二进制漏洞,在介绍栈溢出之前,我们首先需要了解函数调用栈. 函数调用栈是一块连续的用来保存函数运行状态的内存区域,调用函数(calle ...

  8. StartSSL免费SSL证书申请和账户注册完整过程

    StartSSL算是比较早提供免费SSL证书的第三方提供商,我们可以免费申请且免费续期使用到有需要HTTPS网址的用户.关于网站使用SSL证书主要还是因为谷歌在向导说明中提到如果一个网站使用到SSL证 ...

  9. C语言的编译过程、安装gcc编译器以及设置环境变量

    以我对C语言编译过程的了解,我用了一点时间画了一个图,提供给大家参考一下,希望有些能对您的问题提上帮助. 前几天刚初步学习了C语言的编译过程,感触挺深的.在C语言中头文件其实起了一个很大的作用. 1. ...

随机推荐

  1. vue & this.$router.resolve

    vue & this.$router.resolve gotoAutoUpdate (query = {}) { const { href } = this.$router.resolve({ ...

  2. Apple 订单系统 bug

    Apple 订单系统 bug 看不到最近的购买信息 https://secure1.www.apple.com.cn/shop/order/list refs xgqfrms 2012-2020 ww ...

  3. JavaScript var, let, const difference All In One

    JavaScript var, let, const difference All In One js var, let, const 区别 All In One 是否存在 hoisting var ...

  4. SSR & 轮询登录 & Token

    SSR & 轮询登录 & Token https://yuchengkai.cn/docs/frontend 扫码登录原理 https://www.cnblogs.com/xgqfrm ...

  5. HTTP/3 protocol

    HTTP/3 protocol https://caniuse.com/#feat=http3 HTTP/3 H3 https://en.wikipedia.org/wiki/HTTP/3 QUIC ...

  6. SVG & gradient & color

    SVG & gradient & color https://developer.mozilla.org/zh-CN/docs/Web/SVG/Tutorial/Gradients & ...

  7. js字典

    传送门https://www.2cto.com/kf/201709/680989.html 点击

  8. JUC并发编程学习笔记

    JUC并发编程学习笔记 狂神JUC并发编程 总的来说还可以,学到一些新知识,但很多是学过的了,深入的部分不多. 线程与进程 进程:一个程序,程序的集合,比如一个音乐播发器,QQ程序等.一个进程往往包含 ...

  9. 剑指 Offer 32 - I. 从上到下打印二叉树 + 层次遍历二叉树

    剑指 Offer 32 - I. 从上到下打印二叉树 Offer_32_1 题目描述 解题思路 这题属于简单题,考察的是我们对二叉树以及层次遍历的方法. 这里只需要使用简单的队列即可完成二叉树的层次遍 ...

  10. HDOJ-1043 Eight(八数码问题+双向bfs+高效记录路径+康拓展开)

    bfs搜索加记录路径 HDOJ-1043 主要思路就是使用双向广度优先搜索,找最短路径.然后记录路径,找到结果是打印出来. 使用康拓序列来来实现状态的映射. 打印路径推荐使用vector最后需要使用a ...