簡介
_stdcall 是StandardCall的縮寫,是C++的標準調用方式:所有參數從右到左依次入棧,如果是調用類成員的話,最後一個入棧的是this指針。這些堆疊中的參數由被調用的函式在返回後清除,使用的指令是 retnX,X表示參數占用的位元組數,CPU在ret之後自動彈出X個位元組的堆疊空間。稱為自動清棧。函式在編譯的時候就必須確定參數個數,並且調用者必須嚴格的控制參數的生成,不能多,不能少,否則返回後會出錯。
PASCAL 是Pascal語言的函式調用方式,也可以在C/C++中使用,參數壓棧順序與前兩者相反。返回時的清棧方式與_stdcall相同。
_fastcall是編譯器指定的快速調用方式。由於大多數的函式參數個數很少,使用堆疊傳遞比較費時。因此_fastcall通常規定將前兩個(或若干個)參數由暫存器傳遞,其餘參數還是通過堆疊傳遞。不同編譯器編譯的程式規定的暫存器不同。返回方式和_stdcall相當。
_thiscall 是為了解決類成員調用中this指針傳遞而規定的。_thiscall要求把this指針放在特定暫存器中,該暫存器由編譯器決定。VC使用ecx,Borland的C++編譯器使用eax。返回方式和_stdcall相當。
_fastcall 和 _thiscall涉及的暫存器由編譯器決定,因此不能用作跨編譯器的接口。所以Windows上的COM對象接口都定義為_stdcall調用方式。
C中不加說明默認函式為_cdecl方式(C中也只能用這種方式),C++也一樣,但是默認的調用方式可以在IDE環境中設定。
帶有可變參數的函式必須且只能使用_cdecl方式,例如下面的函式:
int printf(char * fmtStr, ...);
int scanf(char * fmtStr, ...);
*/幾種調用約定的區別
區別
__cdecl __fastcall與__stdcall,三者都是調用約定(Calling convention),它決定以下內容:1)函式參數的壓棧順序,2)由調用者還是被調用者把參數彈出棧,3)以及產生函式修飾名的方法。
1、__stdcall調用約定:函式的參數自右向左通過棧傳遞,被調用的函式在返回前清理傳送參數的記憶體棧。
2、__cdecl是C和C++程式的預設調用方式。每一個調用它的函式都包含清空堆疊的代碼,所以產生的執行檔大小會比調用_stdcall函式的大。函式採用從右到左的壓棧方式。注意:對於可變參數的成員函式,始終使用__cdecl的轉換方式。
3、__fastcall調用約定:它是通過暫存器來傳送參數的(實際上,它用ECX和EDX傳送前兩個雙字(DWORD)或更小的參數,剩下的參數仍舊自右向左壓棧傳送,被調用的函式在返回前清理傳送參數的記憶體棧)。
4、thiscall僅僅套用於"C++"成員函式。this指針存放於CX暫存器,參數從右到左壓。thiscall不是關鍵字,因此不能被程式設計師指定。
5、nakedcall採用1-4的調用約定時,如果必要的話,進入函式時編譯器會產生代碼來保存ESI,EDI,EBX,EBP暫存器,退出函式時則產生代碼恢復這些暫存器的內容。naked call不產生這樣的代碼。naked call不是類型修飾符,故必須和_declspec共同使用。
規則
1、修飾名(Decoration name):"C"或者"C++"函式在內部(編譯和連結)通過修飾名識別
2、C編譯時函式名修飾約定規則:
__stdcall調用約定在輸出函式名前加上一個下劃線前綴,後面加上一個"@"符號和其參數的位元組數,格式為_functionname@number,例如 :function(int a, int b),其修飾名為:_function@8
__cdecl調用約定僅在輸出函式名前加上一個下劃線前綴,格式為_functionname。
__fastcall調用約定在輸出函式名前加上一個"@"符號,後面也是一個"@"符號和其參數的位元組數,格式為@functionname@number。
設定方法
1 可以直接在代碼中寫 __cdecl 等調用約定
2 在MS-VC++6.0中,調用約定也可以通過工程設定:Setting...\C/C++ \Code Generation項進行選擇,預設狀態為__cdecl。名字修飾約定(vs2010 在c/c++ 高級中進行設定)。