调用约定(Calling Convention) 是计算机编程中一个比较底层的设计,它主要涉及:
- 函数参数通过寄存器传递还是栈?
- 函数参数从左到右还是从右到左压栈?
- 是否支持可变参数函数(vararg function or variadic function)。
- 是否需要函数原型?
- 调用者(caller)还是被调用者(called or callee)清理堆栈?
在C和C++中有几种调用约定:__cdecl, __stdcall, __fastcall, __thiscall, __clrcall, __vectorcall。下面对比下几种调用约定。
__cdecl
C Declaration Calling Convention,C声明调用约定。它是C和C++默认的调用约定。特点:
- 堆栈由调用者清除(手动清除)。
- 参数从右到左压栈。
- 支持可变参数(函数自己并不知道自己有多少个参数,因此需要调用者来清除)。
- 编译后函数名改编为:“_函数名”。如_funcname。
__stdcall
Standard Calling Convention,标准调用约定。特点:
- 调用者清除堆栈。
- 参数从右到坐压栈(和__cdecl一样),如果调用类的成员函数,最后压入this指针。
- 需要一个函数原型,不支持变参函数。
- 函数名改编:“_函数名@参数字节大小十进制”。如[email protected]。
__fastcall
Fast Calling Convention,快速调用约定。通过使用寄存器解决效率问题。特点:
- 函数参数部分通过寄存器传递,函数中最左的两个DWORD(寄存器大小是双字)或者更小的参数,通过寄存器传递。剩下的从右到左堆栈传递。
- 函数名改编:“@函数名@函数参数字节大小十进制”。
- 返回方式同__stdcall。
__thiscall
主要用于解决this指针问题,使用寄存器传递this指针。返回方式同__stdcall.
__clrcall
__clrcall是C++ .Net里面的。
__vectorcall
要求尽可能在寄存器中传递参数。函数名改编为”@@函数名@参数字节数十进制”。这是微软自己添加的标准。
总结
除了__cdecl(以及__clrcall),其他的都是被调用者清除堆栈。
时间: 2024-10-08 10:44:16