亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 編程 > C > 正文

__stdcall 和 __cdecl 的區別淺析

2020-01-26 16:21:11
字體:
來源:轉載
供稿:網友

1. __cdecl
__cdecl 是C Declaration的縮寫(declaration,聲明),表示C語言默認的函數調用方法:所有參數從右到左依次入棧,由調用者負責把參數壓入棧,最后也是由調用者負責清除棧的內容,一般來說,這是 C/C++ 的默認調用函數的規則,MS VC 編譯器采用的規則則是這種規則2. __stdcall
_stdcall 是StandardCall的縮寫,是C++的標準調用方式:所有參數從右到左依次入棧,由調用者負責把參數壓入棧,最后由被調用者負責清除棧的內容,Windows API 所采用的函數調用規則則是這種規則

另外,采用 __cdecl 和 __stdcall 不同規則的函數所生成的修飾名也各為不同,相同點則是生成的函數修飾名前綴都帶有下劃線,不同的則是后綴部分,當然,這兩者最大的不同點就在于恢復棧的方式不同,而且這點亦是最為重要的。


__cdecl 規則要求調用者本身負責棧的恢復工作,在匯編的角度上說,恢復堆棧的位置是在調用函數內,考慮這樣一段 C++ 代碼(在 VC 下 Debug)

復制代碼 代碼如下:

#include <cstdio>

void __cdecl func(int param1, int param2, int param3) {
  int var1 = param1;
  int var2 = param2;
  int var3 = param3;

  printf("%ld/n", long(¶m1));
  printf("%ld/n", long(¶m2));
  printf("%ld/n", long(¶m3));
  printf("----------------/n");
  printf("%ld/n", long(&var1));
  printf("%ld/n", long(&var2));
  printf("%ld/n", long(&var3));
  return ;
}

int main() {
  func(1, 2, 3);
  return 0;
}


注意到 func 函數使用了 __cdecl 進行修飾(其實不需要,因為 VC 下默認的是 __cdecl 規則, 這里是為了更為清晰),生成匯編代碼如下:

復制代碼 代碼如下:

3:    void __cdecl func(int param1, int param2, int param3) {
00401020   push        ebp
00401021   mov         ebp,esp
00401023   sub         esp,4Ch
00401026   push        ebx
00401027   push        esi
00401028   push        edi
00401029   lea         edi,[ebp-4Ch]
0040102C   mov         ecx,13h
00401031   mov         eax,0CCCCCCCCh
00401036   rep stos    dword ptr [edi]
4:      int var1 = param1;
00401038   mov         eax,dword ptr [ebp+8]
0040103B   mov         dword ptr [ebp-4],eax           ; 注意var1,var2,var3 壓入堆棧的順序!
5:      int var2 = param2;
0040103E   mov         ecx,dword ptr [ebp+0Ch]
00401041   mov         dword ptr [ebp-8],ecx
6:      int var3 = param3;
00401044   mov         edx,dword ptr [ebp+10h]
00401047   mov         dword ptr [ebp-0Ch],edx

...............................................        ; 省略了printf的代碼

15:     return ;
16:   }
004010BD   pop         edi
004010BE   pop         esi
004010BF   pop         ebx
004010C0   add         esp,4Ch
004010C3   cmp         ebp,esp
004010C5   call        __chkesp (004011d0)
004010CA   mov         esp,ebp
004010CC   pop         ebp
004010CD   ret                                         ; 這里是 ret,由調用者(main)恢復堆棧,但如果是 __stdcall 的話,
                                                       ; 恢復堆棧就在這里進行

*******************************************************************************************************************

18:   int main() {

...............................................       ; 省略了建立堆棧的代碼

19:     func(1, 2, 3);
00401118   push        3                              ; 將 param3 壓入棧
0040111A   push        2                              ; 將 param2 壓入棧
0040111C   push        1                              ; 將 param1 壓入棧
0040111E   call        @ILT+5(func) (0040100a)        ; @ILT+5(func) 是函數func的修飾名,而0040100a則是他的地址
00401123   add         esp,0Ch                        ; 恢復堆棧,__cdecl 規則由調用者(這里是main)恢復堆棧
20:     return 0;
00401126   xor         eax,eax
21:   }
00401128   pop         edi
00401129   pop         esi
0040112A   pop         ebx
0040112B   add         esp,40h
0040112E   cmp         ebp,esp
00401130   call        __chkesp (004011d0)
00401135   mov         esp,ebp
00401137   pop         ebp
00401138   ret

結果截圖


程序中的棧結構如下圖示:

__stdcall 規則由被調用者本身去調整堆棧,在匯編的角度上說,恢復堆棧的行為發生在調用者函數內,考慮這樣一段代碼(VC 下Debug):

復制代碼 代碼如下:

#include <cstdio>

void __stdcall func(int param1, int param2, int param3) {
  int var1 = param1;
  int var2 = param2;
  int var3 = param3;

  printf("%ld/n", long(¶m1));
  printf("%ld/n", long(¶m2));
  printf("%ld/n", long(¶m3));
  printf("----------------/n");
  printf("%ld/n", long(&var1));
  printf("%ld/n", long(&var2));
  printf("%ld/n", long(&var3));
  return ;
}

int main() {
  func(1, 2, 3);
  return 0;
}


注意到 func 函數使用了 __cdecl 進行修飾(其實不需要,因為 VC 下默認的是 __cdecl 規則, 這里是為了更為清晰),生成匯編代碼如下:

復制代碼 代碼如下:

1:    #include <cstdio>
2:
3:    void __stdcall func(int param1, int param2, int param3) {
00401020   push        ebp
00401021   mov         ebp,esp
00401023   sub         esp,4Ch
00401026   push        ebx
00401027   push        esi
00401028   push        edi
00401029   lea         edi,[ebp-4Ch]
0040102C   mov         ecx,13h
00401031   mov         eax,0CCCCCCCCh
00401036   rep stos    dword ptr [edi]
4:      int var1 = param1;
00401038   mov         eax,dword ptr [ebp+8]
0040103B   mov         dword ptr [ebp-4],eax
5:      int var2 = param2;
0040103E   mov         ecx,dword ptr [ebp+0Ch]
00401041   mov         dword ptr [ebp-8],ecx
6:      int var3 = param3;
00401044   mov         edx,dword ptr [ebp+10h]
00401047   mov         dword ptr [ebp-0Ch],edx

..............................................  ; 省略 printf 代碼

15:     return ;
16:   }
004010BD   pop         edi
004010BE   pop         esi
004010BF   pop         ebx
004010C0   add         esp,4Ch
004010C3   cmp         ebp,esp
004010C5   call        __chkesp (004011d0)
004010CA   mov         esp,ebp
004010CC   pop         ebp
004010CD   ret         0Ch                       ; __stdcall 在這里(被調用函數)恢復堆棧,但如果是 __cdecl 的話,這里是 ret,
                                                 ; 堆棧的恢復由調用者(這里是 main)來負責

*******************************************************************************************************************

18:   int main() {

...........................................       ; 省略建立堆棧代碼

19:     func(1, 2, 3);
00401118   push        3                          ; param3 壓入堆棧
0040111A   push        2                          ; param2 壓入堆棧
0040111C   push        1                          ; param1 壓入堆棧
0040111E   call        @ILT+0(func) (00401005)    ; @ILT+0(func) 是函數的修飾名,而 00401005 則是調用函數func的地址
20:     return 0;
00401123   xor         eax,eax
21:   }
00401125   pop         edi
00401126   pop         esi
00401127   pop         ebx
00401128   add         esp,40h
0040112B   cmp         ebp,esp
0040112D   call        __chkesp (004011d0)
00401132   mov         esp,ebp
00401134   pop         ebp
00401135   ret


運行的結果與使用 __cdecl 規則的一樣,兩者的棧結構基本一致的,唯一的不同就是調整堆棧(恢復堆棧)的位置以及生成函數的修飾名不同,而這樣的不同正是 __stdcall 和 __cdecl 最為重要的不同點

發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表

圖片精選

亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
正在播放亚洲1区| 欧洲亚洲女同hd| 日韩男女性生活视频| 色综合天天狠天天透天天伊人| 久久精品电影网| 成人妇女免费播放久久久| 国产精品精品视频一区二区三区| 亚洲成人激情在线| 久久免费高清视频| 亚洲视频一区二区| 中国china体内裑精亚洲片| 亚洲国产私拍精品国模在线观看| 日韩精品www| 亚洲摸下面视频| 亚洲欧洲xxxx| www.欧美精品一二三区| 欧美性受xxxx白人性爽| 欧美色欧美亚洲高清在线视频| 中文综合在线观看| 国产成人精品电影久久久| 亚洲国产精品久久久久秋霞蜜臀| 在线一区二区日韩| 国产精品va在线| 欧美日韩成人网| 亚洲视屏在线播放| 2019中文字幕在线| 国产精品中文久久久久久久| 亚洲性无码av在线| 97视频免费看| 国产suv精品一区二区| 日本高清不卡的在线| 亚洲精品一二区| 91精品在线观| 尤物九九久久国产精品的分类| 57pao精品| 日韩欧美aⅴ综合网站发布| 久久久亚洲网站| 色香阁99久久精品久久久| 精品欧美激情精品一区| 精品爽片免费看久久| 国产精品99久久久久久白浆小说| 亚洲精品一区二区三区婷婷月| 国产成人精品免高潮在线观看| 大量国产精品视频| 68精品国产免费久久久久久婷婷| 国产91九色视频| 亚洲欧美日本伦理| 2019中文字幕在线观看| 成人免费淫片视频软件| 中文字幕av一区二区| 欧美午夜激情在线| 热久久这里只有精品| 亚洲男人的天堂在线播放| 久久久久久久国产精品| 日韩中文字幕欧美| 亚洲自拍另类欧美丝袜| 欧美黄色片视频| 亚洲女人天堂视频| 亚洲欧美中文日韩在线v日本| 国产不卡在线观看| 蜜臀久久99精品久久久久久宅男| 成人午夜一级二级三级| 日本韩国欧美精品大片卡二| 国产小视频国产精品| 国产欧美韩国高清| 国产99久久精品一区二区 夜夜躁日日躁| 欧美一级在线亚洲天堂| 久久精品视频导航| 国产精选久久久久久| 国产在线a不卡| 国产丝袜一区二区三区免费视频| 午夜精品一区二区三区在线播放| 日韩av在线看| 日韩欧美在线视频| 91av在线免费观看视频| 成人免费观看网址| 国产成人小视频在线观看| 亚洲欧美日韩国产精品| 久久久噜久噜久久综合| 国产精品美女视频网站| 日韩精品久久久久久久玫瑰园| 亚洲人成在线观看| 91社影院在线观看| 国产精品丝袜白浆摸在线| 欧美成人精品三级在线观看| 亚洲图片欧美日产| 亚洲国产成人在线播放| 最近2019中文字幕mv免费看| 久久视频精品在线| 欧美精品做受xxx性少妇| 国产噜噜噜噜噜久久久久久久久| 国产在线视频欧美| 精品久久久久久久久久国产| 国产91在线播放| 国产精品 欧美在线| 亚洲社区在线观看| 日韩高清电影免费观看完整| 久久久久久久久久久人体| 久久影院资源网| 欧美成人亚洲成人| 精品国产区一区二区三区在线观看| 亚洲美女喷白浆| 亚洲国产福利在线| 欧美性理论片在线观看片免费| 久久视频在线免费观看| 国产午夜精品理论片a级探花| 久久亚洲精品毛片| 日韩在线欧美在线| 欧美日韩国产精品一区二区不卡中文| 尤物九九久久国产精品的特点| 亚洲电影第1页| 国产不卡av在线免费观看| 欧美色videos| 欧美网站在线观看| 午夜精品www| 欧美一级电影免费在线观看| 亚洲国产精品免费| 57pao成人永久免费视频| 精品福利在线视频| 日韩精品视频观看| 色av中文字幕一区| 国产成人啪精品视频免费网| 日韩欧美aⅴ综合网站发布| 国产69精品久久久久9| 亚洲午夜激情免费视频| 日韩av三级在线观看| 午夜精品久久17c| 亚洲国产高清自拍| 亚洲a区在线视频| 亚洲999一在线观看www| 亚洲国产精品系列| 久久久久国色av免费观看性色| 国产在线久久久| 久久成人精品视频| 最新的欧美黄色| 久久久电影免费观看完整版| 88国产精品欧美一区二区三区| 国产精品美女主播在线观看纯欲| 国产一区二区三区毛片| 欧美视频在线视频| 久久久噜噜噜久久| 国产精品久久久久99| 国语自产在线不卡| 日韩经典第一页| 日韩欧美在线免费观看| 美女av一区二区| 91精品国产高清久久久久久91| 国产在线不卡精品| 日韩精品中文字幕视频在线| 亚洲欧美日韩直播| 国产va免费精品高清在线观看| 欧美日韩国产va另类| 欧美午夜视频一区二区| 国产欧洲精品视频| 欧美一级免费看| 国产精品美腿一区在线看| 中文字幕亚洲一区| 国产精品免费久久久久久| 欧美中文在线免费| 久久精品电影网| 欧美在线视频网站| 亚洲美女av网站| 97免费中文视频在线观看| 日本欧美精品在线|