• <ins id="pjuwb"></ins>
    <blockquote id="pjuwb"><pre id="pjuwb"></pre></blockquote>
    <noscript id="pjuwb"></noscript>
          <sup id="pjuwb"><pre id="pjuwb"></pre></sup>
            <dd id="pjuwb"></dd>
            <abbr id="pjuwb"></abbr>

            Life is Good.

            Enhance Tech and English
            隨筆 - 65, 文章 - 20, 評論 - 21, 引用 - 0
            數據加載中……

            cdecl, stdcall, pascal,fastcall的區別和調用約定

            調用約定           壓參數入棧順序     把參數彈出棧者         函數修飾名 
            (Calling convention) 
            --------------------------------------------------------------------------------------------------------
              __cdecl                 右->左             調用者                                 _function    
              __fastcall             右->左              被調用者                           @function@nnn     
              __stdcall             右->左               被調用者                           _function@nnn
              __pascal             左->右               被調用者                           _function@nnn
               
            -----------------------------------------------------------------------------------------------------------

                          
                
                
              _cdecl    
               
              按至左的順序壓參數入棧調用者把參數彈出棧。對于“C”函數或者變量,修飾名是在函數名前加下劃線。對于“C++”函數,有所不同。    
               
              如函數void   test(void)的修飾名是_test;對于不屬于一個類的“C++”全局函數,修飾名是?test@@ZAXXZ。    
               
              這是缺省調用約定。由于是調用者負責把參數彈出棧,所以可以給函數定義個數不定的參數,如printf函數。    
               
               
              _stdcall    
               
              按從至左的順序壓參數入棧被調用者把參數彈出棧。對于“C”函數或者變量,修飾名以下劃線為前綴,然后是函數名,然后是符號“@”及參數的字節數,如函數int   func(int   a,   double   b)的修飾名是_func@12。對于“C++”函數,則有所不同。    
               
              所有的Win32   API函數都遵循該約定。    

              _pascal  

               按從至右的順序壓參數入棧 ...其它的與_stdcall相同; 
                
                
              _fastcall    
               
              頭兩個DWORD類型或者占更少字節的參數被放入ECXEDX寄存器其他剩下的參數按從右到左的順序壓入棧。 由被調用者把參數彈出棧,對于“C”函數或者變量,修飾名以“@”為前綴,然后是函數名,接著是符號“@”及參數的字節數,如函數int   func(int   a,   double   b)的修飾名是@func@12。對于“C++”函數,有所不同。    
               
              未來的編譯器可能使用不同的寄存器來存放參數。    
               
               
              thiscall    
               
              僅僅應用于“C++”成員函數this指針存放于CX寄存器,參數從右到左壓棧。thiscall不是關鍵詞,因此不能被程序員指定。    
               
               
              naked   call    
               
              采用1-4的調用約定時,如果必要的話,進入函數時編譯器會產生代碼來保存ESI,EDI,EBX,EBP寄存器,退出函數時則產生代碼恢復這些寄存器的內容。naked   call不產生這樣的代碼。    
               
              naked   call不是類型修飾符,故必須和_declspec共同使用,如下:    
               
              __declspec(   naked   )   int   func(   formal_parameters   )    
               
              {    
               
              //   Function   body    
               
              }     
                
                    
               

             便于更好理解, 看下面例子(函數調用的過程以匯編代碼表示):      
                
              void   cdecl       fun1(int   x,int   y);  
              void   stdcall     fun2(int   x,int   y);  
              void   pascal     fun3(int   x,int   y);   
                

                
              ****************************************  
               
              void   cdecl       fun1(int   x,int   y);  
               
              fun1(x,y);    
               
              調用   fun1   的匯編代碼  
               
              push   y  
              push   x  
              call   fun1  
              add     sp,sizeof(x)+sizeof(y)   ;跳過參數區(x,y)  
               
              fun1   的匯編代碼:  
               
              fun1   proc    
                  push   bp  
                  mov     bp,sp  
                  ……  
                  …  
                  pop     bp  
                  ret ;返回,但不跳過參數區  
              fun1   endp  
               
              ****************************************  
               
              void   stdcall   fun2(int   x,int   y);  
               
              fun2(x,y);    
               
              調用   fun2   的匯編代碼  
               
              push   y  
              push   x  
              call   fun2  
               
              fun2   的匯編代碼:  
               
              fun2   proc    
                  push   bp  
                  mov     bp,sp  
                  ……  
                  …  
                  pop     bp  
                  ret   sizeof(x)+sizeof(y)   ;返回并跳過參數區(x,y)      
              fun2   endp  
               
              *****************************************  
               
              void   pascal     fun3(int   x,int   y);  
               
              fun3(x,y);    
               
              調用   fun3   的匯編代碼  
               
              push   x  
              push   y  
              call   fun3  
               
              fun3   的匯編代碼:  
               
              fun3   proc    
                  push   bp  
                  mov     bp,sp  
                  ……  
                  …  
                  pop     bp  
                  ret   sizeof(x)+sizeof(y)   ;返回并跳過參數區(x,y)      
              fun3   endp 

            posted on 2010-10-28 16:01 Mike Song 閱讀(179) 評論(0)  編輯 收藏 引用

            一本色道久久88精品综合| 国产成人精品综合久久久久| 狠狠色狠狠色综合久久| 国产精品中文久久久久久久| 欧美午夜精品久久久久免费视| 四虎国产永久免费久久| 久久大香萑太香蕉av| 色婷婷综合久久久久中文| 国产午夜精品久久久久免费视| 九九精品久久久久久噜噜| 色婷婷综合久久久中文字幕 | 69国产成人综合久久精品| 国产午夜久久影院| 国产精品久久久99| 国内精品久久久人妻中文字幕 | 久久综合给合综合久久| 九九久久自然熟的香蕉图片| 99久久精品免费观看国产| 97久久精品午夜一区二区| 婷婷国产天堂久久综合五月| 久久久久亚洲AV无码专区桃色| 少妇久久久久久被弄高潮| 久久ww精品w免费人成| 97久久超碰国产精品2021| 亚洲AⅤ优女AV综合久久久| 免费精品久久久久久中文字幕| 69久久精品无码一区二区| 久久精品国产亚洲AV电影| 亚洲?V乱码久久精品蜜桃| 久久久这里有精品中文字幕| 久久精品免费一区二区三区| 亚洲AV无码久久精品蜜桃| 国产精品99久久久精品无码| 亚洲国产成人久久精品99| 久久久久久久91精品免费观看| 国产激情久久久久影院老熟女| 国产午夜免费高清久久影院| 精品人妻久久久久久888| 久久w5ww成w人免费| 国产精品岛国久久久久| 久久综合丁香激情久久|