函数调用约定

时间:2022-06-11 03:36:46

在VC中,调用方式有__stdcall,__cdecl,__fastcall,thiscall几种。其中前三种是一般调用所能使用的调用方式,thiscall是类对象调用成员函数时使用的调用方式,当在一个函数前使用__declspec(naked)修饰时,编译器不会在函数的内部将不再产生保存和恢复某些寄存器值的代码。

__stdcall

对于__stdcall调用方式,所有参数通过栈传递,参数压栈顺序为从右到左,并且由被

调用的函数平衡栈:

BOOL __stdcall call(void *a, void *b, void *c, void *d, void *e)
{
return TRUE;
}
与call(0,0,0,0,0)对应的调用汇编代码为:

push    0               ; e
push 0 ; d
push 0 ; c
push 0 ; b
push 0 ; a
call ?call@@YGHPAX0000@Z ; call(void *,void *,void *,void *,void *)

__cdecl

对于__cdecl调用方式,所有参数通过栈传递,参数压栈顺序为从右到左,由调用者平衡栈:

BOOL __cdecl call(void *a, void *b, void *c, void *d, void *e, void *f, void *g)
{
return TRUE;
}
与call(0,0,0,0,0)对应的调用汇编代码为:

push    0               ; e
push 0 ; d
push 0 ; c
push 0 ; b
push 0 ; a
call ?call@@YAHPAX0000@Z ; call(void *,void *,void *,void *,void *)
add esp, 14h

__fastcall

对于__fastcall调用方式,第一个参数通过ecx传递,第二个参数通过edx传递,其它参数

通过栈传递,参数压栈顺序为从右到左,由被调用的函数平衡栈:

BOOL __fastcall call(void *a, void *b, void *c, void *d, void *e)
{
return TRUE;
}
与call(0,0,0,0,0)对应的调用汇编代码为:

push    0               ; e
push 0 ; d
push 0 ; c
xor edx, edx ; b
xor ecx, ecx ; a
call ?call@@YIHPAX0000@Z ; call(void *,void *,void *,void *,void *)

thiscall

对于thiscall调用方式,this指针通过ecx传递,其它参数通过栈传递,参数压栈顺序为

从右到左,由被调用的函数平衡栈:

class Test
{
public:
Test()
{
}
virtual ~Test()
{
}
virtual int run(void *a, void *b, void *c, void *d, void *e)
{
return 0;
}
};

class SubTest:
public Test
{
public:
SubTest()
{
;
}
virtual ~SubTest()
{
;
}
virtual int run(void *a, void *b, void *c, void *d, void *e)
{
return 0;
}
};
与SubTest.run(0,0,0,0,0)对应的调用汇编代码为:

push    0               ; e
push 0 ; d
push 0 ; c
push 0 ; b
push 0 ; a
lea ecx, [ebp+s] ; this
call ?run@SubTest@@UAEHPAX0000@Z ; SubTest::run(void *,void *,void *,void *,void *)