ARM匯編中調用C函數的參數傳遞方式
不同于x86的參數傳遞規則,ATPCS建議函數的形參不超過(guò)4個(gè),如果形參個(gè)數少于或等于4,則形參由R0,R1,R2,R3四個(gè)寄存器進(jìn)行傳遞;若形參個(gè)數大于4,大于4的部分必須通過(guò)堆棧進(jìn)行傳遞。
本文引用地址:http://dyxdggzs.com/article/201611/319404.htm我們先討論一下形參個(gè)數為4的情況:
實(shí)例1
s//>>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]! ;保存當前l(fā)r 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裝進(jìn)pc(返回main函數) 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函數開(kāi)始執行,main調用了test_asm_args,test_asm_args調用了test_c_args,最后從test_asm_args返回main。代碼分別使用了匯編和C定義了兩個(gè)函數,test_asm_args 和 test_c_args,test_asm_args調用了test_c_args,其參數的傳遞方式就是向R0~R3分別寫(xiě)入參數值,之后使用bl語(yǔ)句對test_c_args進(jìn)行調用。其中值得注意的地方是用紅色標記的語(yǔ)句,test_asm_args在調用test_c_args之前必須把當前的 lr入棧,調用完test_c_args之后再把剛才保存在棧中的lr寫(xiě)回pc,這樣才能返回到main函數中。
如果test_c_args的參數是8個(gè)呢?這種情況test_asm_args應該怎樣傳遞參數呢?
實(shí)例2
s//>>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]! ;保存當前l(fā)r 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,本語(yǔ)句執行完后sp指向 參數6 ADD sp, sp, #4 ;清除棧中參數 6,本語(yǔ)句執行完后sp指向 參數7 ADD sp, sp, #4 ;清除棧中參數 7,本語(yǔ)句執行完后sp指向 參數8 ADD sp, sp, #4 ;清除棧中參數 8,本語(yǔ)句執行完后sp指向 lr LDR pc, [sp],#4 ;將lr裝進(jìn)pc(返回main函數) END
c//>>test_c_args.c ... 略 ...
后記
這段代碼另外學(xué)到的是str作為入棧操作的新用法。
STR lr, [sp, #-4]! ;保存當前l(fā)r
這條匯編指令所作的操作分別是:
將lr寄存器的內容 存入到 sp-4 所表示的內存空間中,然后執行 sp <- sp-4
作用是將lr入棧,保存當前l(fā)r的內容。
同理
LDR pc, [sp],#4 ;將lr裝進(jìn)pc(返回main函數)
也是類(lèi)似含義
評論