首页
学习
活动
专区
工具
TVP
发布
精选内容/技术社群/优惠产品,尽在小程序
立即前往

C语言】汇编角度剖析函数调用的整个过程

提示:本文意在使用汇编的语言给大家介绍函数调用中栈区上的过程变化,加深我们对于代码底层的理解,由于不同的编译器使用下,可能造成一些差异,但这并不影响我们对于知识原理的掌握,所以本文不必过多纠结细节处的变化...ret 二、函数栈帧的创建与销毁过程(从汇编角度去看) 1.从下面的原码中我们也可以看出,其实我们的main函数也是被其他函数调用的。...x y开辟的栈帧空间,此时也就是销毁了形参x y 读到这里我们今天的学习就结束了,我们讲解了Add函数汇编角度下是如何被调用的?...我们会在调用函数前进行函数参数的内容,进行一个压栈操作,当进入到被调用函数内部的时候,我们会通过指针的偏移量找到函数参数,并对其进行操作。...我们会通过汇编语言中的call指令,先将其下一条指令的IP压栈到我们的栈帧空间当中,并且指向call指令,会进入到被调用函数汇编代码当中,进行被调用函数汇编指令 并且我们函数调用结束后,通过ret指令能够回到上一层函数

99910

汇编角度看函数堆栈调用

下面以主函数调用求和函数分析函数堆栈调用 带着以下一个问题来探索: (1)形参的内存空间的开辟和清理是由调用方还是由被调用方执行的? (2)主函数调用函数结束后,主函数从哪里开始执行?...= 0; ret = sum(a,b); printf("ret = %d\n",ret); return 0; } 实验环境:vc++ 6.0 和 Win10操作系统 注意:linux...mov esp,ebp 004010C2 pop ebp 004010C3 ret 可以看到在主函数和求和函数中首先出现的反汇编代码,我们以求函数举例,其实它们的功能是相同的...现在回答最开始我们提出的几个题: (1)形参的内存空间的开辟和清理是由调用方还是由被调用方执行的? (2)主函数调用函数结束后,主函数从哪里开始执行?从头开始还是从调用之后开始?...答: (1)形参的内存空间的开辟和清理是由调用方执行的。 (2)主函数调用函数后执行执行调用之后的代码,是因为调用方在进行调用的过程中,将下一行指令的地址压栈。

58820
您找到你想要的搜索结果了吗?
是的
没有找到

汇编学习(6), 外部函数调用约定

本篇介绍 本篇介绍下汇编中的外部函数调用约定。...外部函数 在前面已经多次见过使用printf了,这次我们也可以自己写一些外部函数,下面是一个例子: 首先定义2个外部函数,分别是c_area和c_circum。...调用约定 调用约定(Calling Convertions)就是调用函数时传参和返回值的约定。不同的平台约定也不一样,比如linux和windows 就都有自己的一套调用约定。...当调用函数的时候,返回地址rip也会压栈,prologue中保存rbp也会压栈一次,这样如果需要通过rsp拿到第7个参数,就需要是rsp + 16。...在调用函数时,对于寄存器的保存也有一套约定,有的寄存器值需要caller保存,有的需要callee保存,具体如下: image.png image.png 关键信息如下: 对于callee save

56820

5.5 汇编语言:函数调用约定

函数是任何一门高级语言中必须要存在的,使用函数式编程可以让程序可读性更高,充分发挥了模块化设计思想的精髓,今天我将带大家一起来探索函数的实现机理,探索编译器到底是如何对函数这个关键字进行实现的,并使用汇编语言模拟实现函数编程中的参数传递调用规范等...CDECL:C/C++默认的调用约定,调用方平栈,不定参数的函数可以使用,参数通过堆栈传递. STDCALL:被调方平栈,不定参数的函数无法使用,参数默认全部通过堆栈传递....C++中的一种默认调用约定(调用者平栈)。...FASTCALL效率最高,其他两种调用方式都是通过栈传递参数,唯独_fastcall可以利用寄存器传递参数,一般前两个或前四个参数用寄存器传递,其余参数传递则转换为栈传递,此约定不定参数函数无法使用。...ShowPrint函数以及该函数所对应的int(__stdcall *pShowPrint)(int, int)函数指针,看一下在汇编层面该如何实现这个功能。

22120

5.5 汇编语言:函数调用约定

函数是任何一门高级语言中必须要存在的,使用函数式编程可以让程序可读性更高,充分发挥了模块化设计思想的精髓,今天我将带大家一起来探索函数的实现机理,探索编译器到底是如何对函数这个关键字进行实现的,并使用汇编语言模拟实现函数编程中的参数传递调用规范等...一般情况下在Win32环境默认遵循的就是STDCALL,而在Win64环境下使用的则是FastCALL,在Linux系统上则遵循SystemV的约定,这里我整理了他们之间的异同点.CDECL:C/C++...默认的调用约定,调用方平栈,不定参数的函数可以使用,参数通过堆栈传递.STDCALL:被调方平栈,不定参数的函数无法使用,参数默认全部通过堆栈传递.FASTCALL32:被调方平栈,不定参数的函数无法使用...C++中的一种默认调用约定(调用者平栈)。...函数以及该函数所对应的int(__stdcall *pShowPrint)(int, int)函数指针,看一下在汇编层面该如何实现这个功能。

22720

汇编程序调用c函数为什么需要设置栈?

传递参数 C语言函数调用时,会传给被调用函数一些参数,对于这些C语言级别参数,被编译器翻译成汇编语言时,要找个地方存放下来,并且让被调用函数能访问,否则没法传递。找个地方存放下来分2种情况。...文件。...该文件是包含了u-boot可执行汇编代码,从中我们可以看到相应C程序对应的汇编代码。...下面贴出两个函数汇编代码,一个是clock_init,另一个是与clock_init在同一C文件中的函数CopyCode2Ram: 33d0091c: CopyCode2Ram: 33d0091c:...(2) CopyCode2Ram对应汇编代码第一行:33d0091c: e92d4070 push {r4, r5, r6, lr} 就是我们所期望的,用push保存r4,r5,r6,lr,是因为此函数还包括其他函数调用

1.1K10

C++】泛型编程 ⑤ ( 函数模板原理 | C++ 编译器原理 | C C++ 编译器编译过程 | 分析 模板函数代码 汇编文件 | 编译 模板函数代码 汇编文件 | 模板函数汇编分析总结 )

在前面几篇博客 【C++】泛型编程 ③ ( 函数模板 与 普通函数 调用规则 | 类型匹配 | 显式指定函数模板泛型类型 ) 【C++】泛型编程 ④ ( 函数模板 与 普通函数 调用规则 | 类型自动转换...直接从 Test.c 源码生成 汇编文件 : gcc -S Test.c -o Test.S ③ 汇编 Assembling ( 汇编器 ) 汇编 Assembling : 将 汇编文件 编译成 二进制机器码文件...1、编译 模板函数代码 汇编文件 在 Test.c 中定义一个简单 函数模板 , 然后再 main 函数调用函数模板 , #include "iostream" using namespace..._S0_ 调用的是 函数模板 , 下面看函数模板的 汇编内容 : 函数模板 的 函数声明 对应的汇编如下 : LC1: .ascii "\350\260\203\347\224\250\345\207...汇编文件 分析总结 ( 重要 ) C++ 编译器 将 函数模板 编译成了 汇编函数 call __Z3addIiET_S0_S0_ ; 如果 向 函数模板 中传入不同的函数 , 会生成 多个不同的 汇编函数

28820

CC++ 反汇编:关于函数调用约定

函数是任何一门高级语言中必须要存在的,使用函数式编程可以让程序可读性更高,充分发挥了模块化设计思想的精髓,今天我将带大家一起来探索函数的实现机理,探索编译器到底是如何对函数这个关键字进行实现的,并使用汇编语言模拟实现函数编程中的参数传递调用规范等...一般编译器实现调用调用约定无外乎以下这几种: CDECL:C/C++默认的调用约定,调用方平栈,不定参数的函数可以使用,参数通过堆栈传递....cdecl 调用者平栈: cdecl是C/C++默认调用约定,该调用方式在函数内不进行任何平衡参数操作,而是在退出函数后对esp执行加4操作,从而实现栈平衡。...cdecl调用方式的函数在同一作用域内多次被调用,会在效率上比stdcall高一些,因为它可以使用复写传播优化,而stdcall在函数内平衡栈,无法使用复写传播优化。...但每次访问都需要计算,如果在函数执行过程中esp发生了变化,再次访问变量就需要重新计算偏移了。 参考文献:《C++反汇编与逆向分析技术揭秘》

58710

C++调用C函数

C++调用其它语言的函数,由于编译器生成函数的机制不一样,所以需要经过特殊处理,才可以调用调用C语言的函数,需要在函数声明的地方语句extern "C"。...Test.obj : error LNK2019: 无法解析的外部符号 "void __cdecl DeleteStack(struct _Node *)" (?...因为C++源文件已经引入了C的头文件,在头文件里,声明该函数时没有extern修饰,而这里有extern修饰,所以冲突了。解决的办法有两个。 一。在C文件中加上extern修饰符。 直接加,也不行。...因为C文件也包含了这个头文件,当编译C文件时,就会出现错误。所以,需要一种机制来区分是编译C还是C++文件。...编写一个C++风格的头文件,在这里添加extern修饰符。 使用方法一,很简单。但是如果该头文件是别人写好,你无法修改。这个时候就要使用其它方法了。

2.8K40

静态逆向反汇编获取函数调用关系链

本文提出一种通过逆向二进制文件的方式,通过反汇编的指令获取函数之间的调用关系。...从C/C++语言的角度来看,这个函数可以是一个纯C函数或者类成员非虚函数(补充:对于宏,在编译时就已将其替换为其所代表的项,所以在逆向的角度而言,若要获取宏的调用关系还需要进一步的将替换者变为宏,这个....从PE文件的角度考虑这个函数可能存储在.text的代码区,导入表,导出表三个地方中。...对于普通函数而言,在汇编层面直接调用的使其所在的函数地址,ida所在的加载器会将这个调用的实际函数地址替换成对应的函数名称,如下图1所示: 图2 通过对逆向汇编的分析,C/C++代码中的函数调用在编译成二进制之后...由于非成员函数没有this指针,因此无法获得虚函数表指针,进而无法获取虚表,也就无法访问虚函数

4.9K00

C语言 | 函数调用

C语言函数调用的形式 一般形式 函数名(实参表列) 函数调用语句 把函数调用单独作为一个语句。 函数参数 函数调用作为另一个函数调用时的实参。 调用函数并不一定要求包括分号。...只有作为函数调用语句才需要有分号。如果作为函数表达式或函数参数,函数调用本身是不必有分号的。 C语言函数调用时的数据传递 在调用有参函数时,主调函数和被调函数之间有数据传递关系。...C语言函数调用的过程 在定义函数中指定的形参,在未出现函数调用时,它们并不是占内存中的存储单元。 将实参对应的值传递给形参。 通过return语句将函数值带回到主调函数调用结束,形参单元被释放。...C语言函数调用案例 #include//头文件 int maxNumber(int num1,int num2);//函数声明 int main()//主函数 {   int num1...100道C语言源码案例请去公众号:C语言入门到精通

3.3K10

lua调用C函数

Lua采取的是利用栈进行交互,利用各种Lua_pushXXX将不同的值压入栈中,然后调用Lua脚本时自然会退栈取出参数运行,对于Lua的虚拟机来说,就像是发生了一次正常的函数调用。...(这里采用的栈是Lua栈,因为若是C栈的话调用Lua的C API就会出错了。)...typedef int (*lua_CFunction) (lua_State *L); 能够被Lua调用C函数都必须是这种规则。函数的返回的int值表示C函数返回值的个数。...gcc,需要编译成.so文件 -fPIC,是Position Independent Code的意思,具体的含义可以参考这篇,主要用来避免同一份代码因为重定位位置不同而在内存中存在多个实例 lua代码...require("power") print(square(1.44)) 参考 从lua调用C函数 Lua初学者(四)–Lua调用原理展示(lua的堆栈) Lua中调用C函数

2.8K31
领券