ARM 匯編與C調用的若干問題(一般函數調用情況)


ARM 匯編與C之間的函數調用需要符合ATPCS,建議函數的形參不超過4個,如果形參個數少於或等於4,則形參由R0,R1,R2,R3四個寄存器進行傳遞;若形參個數大於4,大於4的部分必須通過堆棧進行傳遞。

    R0 用來存放函數的第一個參數,R1用來存放第二個參數,R2用來存放第三個參數,R3用來存放第四個參數。其中R0還用來返回函數的調用結果,對應C函數里面的return value語句中的value 存放在R0中。

    ARM堆棧的是滿棧FULL STACK,SP指針指向的位置是存放有效數據的地方,若壓棧新的數據,必須先改變SP,再向SP里面壓入數據。下面結合博客http://blog.sina.com.cn/s/blog_3e5694650100tsbf.html,的內容進行分析。

                             情景(一)函數形參的個數<= 4

                                

 test_asm_args.asm
 
            IMPORT test_c_args ;聲明test_c_args函數
            AREA TEST_ASM, CODE, READONLY
            EXPORT test_asm_args
            test_asm_args
            STR lr, [sp, #-4]! 

;保存當前LR.棧是滿遞減棧FD,首先調整SP指針,然后壓入LR地址
            ldr r0,=0x10 
;參數 1
            ldr r1,=0x20 
;參數 2
            ldr r2,=0x30 
;參數 3
            ldr r3,=0x40 ;參數 4
            bl test_c_args 
;調用C函數
            LDR pc, [sp], #4 
;將LR裝進PC(返回main函數) ,PC = LR,SP = SP+4,恢復原來的棧。
                END
 
           void test_c_args(int a,int b,int c,int d)
           {
              printk("test_c_args:\n");
              printk("%0x %0x %0x %0x\n",a,b,c,d);
           }

           int main()
           {
              test_asm_args();
              for(;;);
           }

                                                                                                   情景二:函數的參數是8個

test_asm_args.asm
//--------------------------------------------------------------------------------

IMPORT test_c_args ;聲明test_c_args函數
        AREA TEST_ASM, CODE, READONLY
        EXPORT test_asm_args
test_asm_args
       STR lr, [sp, #-4]! ;保存當前lr

       ldr r0,=0x1 ;參數 1
       ldr r1,=0x2 ;參數 2
       ldr r2,=0x3 ;參數 3
       ldr r3,=0x4 ;參數 4
       ldr r4,=0x8
       str r4,[sp,#-4]! ;參數 8 入棧
       ldr r4,=0x7
       str r4,[sp,#-4]! ;參數 7 入棧
       ldr r4,=0x6
       str r4,[sp,#-4]! ;參數 6 入棧
       ldr r4,=0x5
       str r4,[sp,#-4]! ;參數 5 入棧
       bl test_c_args_lots
       ADD sp, sp, #4     ;清除棧中參數 5,本語句執行完后sp指向參數6 
       ADD sp, sp, #4     ;清除棧中參數 6,本語句執行完后sp指向參數7
       ADD sp, sp, #4     ;清除棧中參數 7,本語句執行完后sp指向參數8
       ADD sp, sp, #4     ;清除棧中參數 8,本語句執行完后sp指向 lr

       LDR pc, [sp],#4    ;將lr裝進pc(返回main函數) 
        END
test_c_args.c
//--------------------------------------------------------------------------------
void test_c_args(int a,int b,int c,int d,int e,int f,int g,int h)
{
       printk("test_c_args_lots:\n");
       printk("%0x %0x %0x %0x %0x %0x %0x %0x\n",
              a,b,c,d,e,f,g,h);
}
main.c
//--------------------------------------------------------------------------------
int main()
{
     test_asm_args();
     for(;;);
}


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM