<dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><small id="yhprb"></small><dfn id="yhprb"></dfn><small id="yhprb"><delect id="yhprb"></delect></small><small id="yhprb"></small><small id="yhprb"></small> <delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"></dfn><dfn id="yhprb"></dfn><s id="yhprb"><noframes id="yhprb"><small id="yhprb"><dfn id="yhprb"></dfn></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><small id="yhprb"></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn> <small id="yhprb"></small><delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn>

新聞中心

EEPW首頁(yè) > 嵌入式系統 > 設計應用 > ARM系統中函數調用時(shí)參數傳遞規則

ARM系統中函數調用時(shí)參數傳遞規則

作者: 時(shí)間:2016-11-21 來(lái)源:網(wǎng)絡(luò ) 收藏
嵌入式軟件編程中,經(jīng)常會(huì )用到函數調用,之前在學(xué)習如何在C語(yǔ)言中嵌入匯編時(shí)有了解到C語(yǔ)言之前的參數調用是使用寄存器R0傳遞第一個(gè)參數,R1傳遞到第二個(gè)..一直到R3傳遞第四個(gè)參數.但是實(shí)際上有時(shí)可能傳遞的參數非常多,超過(guò)8個(gè),或是參數中有浮點(diǎn)數之類(lèi),參數也會(huì )超過(guò)4個(gè)寄存器,對于超出的部份并不使用R4,而是使用堆棧的方式,但具體是如何的方式很多網(wǎng)站就沒(méi)了下文了。

對于A(yíng)RM體系來(lái)說(shuō),不同語(yǔ)言撰寫(xiě)的函數之間相互調用(mix calls)遵循的是 ATPCS(ARM-Thumb Procedure Call Standard),ATPCS主要是定義了函數呼叫時(shí)參數的傳遞規則以及如何從函數返回,關(guān)于A(yíng)TPCS的詳細內容可以查看ADS1.2 Online Books ——Developer Guide的2.1節。這篇文檔要講的是匯編代碼中對C函數調用時(shí)如何進(jìn)行參數的傳遞以及如何從C函數正確返回

本文引用地址:http://dyxdggzs.com/article/201611/319131.htm

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

我們先討論一下形參個(gè)數為4的情況.
實(shí)例1:
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,=0×10 ;參數 1
ldr r1,=0×20 ;參數 2
ldr r2,=0×30 ;參數 3
ldr r3,=0×40 ;參數 4
bl test_c_args ;調用C函數
LDR pc, [sp], #4 ;將lr裝進(jìn)pc(返回main函數)
END
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);
}
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:
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,=0×1 ;參數 1
ldr r1,=0×2 ;參數 2
ldr r2,=0×3 ;參數 3
ldr r3,=0×4 ;參數 4
ldr r4,=0×8
str r4,[sp,#-4]! ;參數 8 入棧
ldr r4,=0×7
str r4,[sp,#-4]! ;參數 7 入棧
ldr r4,=0×6
str r4,[sp,#-4]! ;參數 6 入棧
ldr r4,=0×5
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
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 %0xn”,
a,b,c,d,e,f,g,h);
}
main.c
//——————————————————————————–
int main()
{
test_asm_args();
for(;;);
}

這部分的代碼和實(shí)例1的代碼大部分是相同的,不同的地方是test_c_args的參數個(gè)數和test_asm_args的參數傳遞方式。
在test_asm_args中,參數1~參數4還是通過(guò)R0~R3進(jìn)行傳遞,而參數5~參數8則是通過(guò)把其壓入堆棧的方式進(jìn)行傳遞,不過(guò)要注意這四個(gè)入棧參數的入棧順序,是以參數8->參數7->參數6->參數5的順序入棧的。
直到調用test_c_args之前,堆棧內容如下:
sp->+———-+
| 參數5 |
+———-+
| 參數6 |
+———-+
| 參數7 |
+———-+
| 參數8 |
+———-+
| lr |
+———-+
test_c_args執行返回后,則設置sp,對之前入棧的參數進(jìn)行清除,最后將lr裝入pc返回main函數,在執行 LDR pc, [sp],#4 指令之前堆棧內容如下:
+———-+
| 參數5 |
+———-+
| 參數6 |
+———-+
| 參數7 |
+———-+
| 參數8 |
sp->+———-+
| lr |
+———-+

上面是轉自http://lionwq.spaces.eepw.com.cn/articles/article/item/17475/

但實(shí)際上可能不同的編譯器可能用著(zhù)不同的處理方式,于我們所使用的編譯器我們可以寫(xiě)一個(gè)簡(jiǎn)單的代碼,調用10個(gè)參數的函數,然后升成匯編再查看它是如何處理,這樣再根據編譯器進(jìn)行特殊的優(yōu)化.




關(guān)鍵詞: ARM系統函數調用參數傳

評論


技術(shù)專(zhuān)區

關(guān)閉
国产精品自在自线亚洲|国产精品无圣光一区二区|国产日产欧洲无码视频|久久久一本精品99久久K精品66|欧美人与动牲交片免费播放
<dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><small id="yhprb"></small><dfn id="yhprb"></dfn><small id="yhprb"><delect id="yhprb"></delect></small><small id="yhprb"></small><small id="yhprb"></small> <delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"></dfn><dfn id="yhprb"></dfn><s id="yhprb"><noframes id="yhprb"><small id="yhprb"><dfn id="yhprb"></dfn></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><small id="yhprb"></small><dfn id="yhprb"><delect id="yhprb"></delect></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn> <small id="yhprb"></small><delect id="yhprb"><strike id="yhprb"></strike></delect><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn><dfn id="yhprb"><s id="yhprb"><strike id="yhprb"></strike></s></dfn><dfn id="yhprb"><s id="yhprb"></s></dfn>