新聞中心

EEPW首頁 > 嵌入式系統(tǒng) > 設(shè)計應(yīng)用 > ARM匯編中調(diào)用C函數(shù)的參數(shù)傳遞方式

ARM匯編中調(diào)用C函數(shù)的參數(shù)傳遞方式

作者: 時間:2016-11-21 來源:網(wǎng)絡(luò) 收藏
對于ARM體系來說,不同語言撰寫的函數(shù)之間相互調(diào)用(mix calls)遵循的是 ATPCS(ARM-Thumb Procedure Call Standard)標(biāo)準(zhǔn),ATPCS主要是定義了函數(shù)呼叫時參數(shù)的傳遞規(guī)則以及如何從函數(shù)返回。

不同于x86的參數(shù)傳遞規(guī)則,ATPCS建議函數(shù)的形參不超過4個,如果形參個數(shù)少于或等于4,則形參由R0,R1,R2,R3四個寄存器進(jìn)行傳遞;若形參個數(shù)大于4,大于4的部分必須通過堆棧進(jìn)行傳遞。

本文引用地址:http://2s4d.com/article/201611/319404.htm

我們先討論一下形參個數(shù)為4的情況:

實例1

s//>>test_asm_args.asm //-------------------------------------------------------------------------------- IMPORT test_c_args ;聲明test_c_args函數(shù) AREA TEST_ASM, CODE, READONLY EXPORT test_asm_args test_asm_args STR lr, [sp, #-4]! ;保存當(dāng)前l(fā)r ldr r0,=0x10 ;參數(shù) 1 ldr r1,=0x20 ;參數(shù) 2 ldr r2,=0x30 ;參數(shù) 3 ldr r3,=0x40 ;參數(shù) 4 bl test_c_args ;調(diào)用C函數(shù) LDR pc, [sp], #4 ;將lr裝進(jìn)pc(返回main函數(shù)) END  
c//>> test_c_args.c //-------------------------------------------------------------------------------- void test_c_args(int a,int b,int c,int d) { printk("test_c_args:n"); printk("%0x %0x %0x %0xn",a,b,c,d); }  
c//>> main.c //------------------------------------------------------------------------- int main() { test_asm_args(); for(;;); }  

程序從main函數(shù)開始執(zhí)行,main調(diào)用了test_asm_args,test_asm_args調(diào)用了test_c_args,最后從test_asm_args返回main。代碼分別使用了匯編和C定義了兩個函數(shù),test_asm_args 和 test_c_args,test_asm_args調(diào)用了test_c_args,其參數(shù)的傳遞方式就是向R0~R3分別寫入?yún)?shù)值,之后使用bl語句對test_c_args進(jìn)行調(diào)用。其中值得注意的地方是用紅色標(biāo)記的語句,test_asm_args在調(diào)用test_c_args之前必須把當(dāng)前的 lr入棧,調(diào)用完test_c_args之后再把剛才保存在棧中的lr寫回pc,這樣才能返回到main函數(shù)中。

如果test_c_args的參數(shù)是8個呢?這種情況test_asm_args應(yīng)該怎樣傳遞參數(shù)呢?

實例2

s//>>test_asm_args.asm //-------------------------------------------------------------------------------- IMPORT test_c_args ;聲明test_c_args函數(shù) AREA TEST_ASM, CODE, READONLY EXPORT test_asm_args test_asm_args STR lr, [sp, #-4]! ;保存當(dāng)前l(fā)r ldr r0,=0x1 ;參數(shù) 1 ldr r1,=0x2 ;參數(shù) 2 ldr r2,=0x3 ;參數(shù) 3 ldr r3,=0x4 ;參數(shù) 4 ldr r4,=0x8 str r4,[sp,#-4]! ;參數(shù) 8 入棧 ldr r4,=0x7 str r4,[sp,#-4]! ;參數(shù) 7 入棧 ldr r4,=0x6 str r4,[sp,#-4]! ;參數(shù) 6 入棧 ldr r4,=0x5 str r4,[sp,#-4]! ;參數(shù) 5 入棧 bl test_c_args_lots ADD sp, sp, #4 ;清除棧中參數(shù) 5,本語句執(zhí)行完后sp指向 參數(shù)6 ADD sp, sp, #4 ;清除棧中參數(shù) 6,本語句執(zhí)行完后sp指向 參數(shù)7 ADD sp, sp, #4 ;清除棧中參數(shù) 7,本語句執(zhí)行完后sp指向 參數(shù)8 ADD sp, sp, #4 ;清除棧中參數(shù) 8,本語句執(zhí)行完后sp指向 lr LDR pc, [sp],#4 ;將lr裝進(jìn)pc(返回main函數(shù)) END  
c//>>test_c_args.c ... 略 ...  

后記

這段代碼另外學(xué)到的是str作為入棧操作的新用法。

STR lr, [sp, #-4]! ;保存當(dāng)前l(fā)r  

這條匯編指令所作的操作分別是:
將lr寄存器的內(nèi)容 存入到 sp-4 所表示的內(nèi)存空間中,然后執(zhí)行 sp <- sp-4
作用是將lr入棧,保存當(dāng)前l(fā)r的內(nèi)容。
同理

LDR pc, [sp],#4 ;將lr裝進(jìn)pc(返回main函數(shù))  

也是類似含義



評論


技術(shù)專區(qū)

關(guān)閉