當(dāng)前位置:首頁 > 嵌入式 > 嵌入式教程
[導(dǎo)讀]本節(jié)提供一些示例,顯示如何從C++調(diào)用C和匯編語言代碼,以及從C和匯編語言調(diào)用 C++ 代碼。其中包括調(diào)用約定和數(shù)據(jù)類型。主要包括下面內(nèi)容:

12.4C' target='_blank' style='cursor:pointer;color:#D05C38;text-decoration:underline;'>C、C++和ARM匯編語言之間的調(diào)用

本節(jié)提供一些示例,顯示如何從C++調(diào)用C和匯編語言代碼,以及從C和匯編語言調(diào)用C++代碼。其中包括調(diào)用約定和數(shù)據(jù)類型。主要包括下面內(nèi)容:

·相互調(diào)用的一般規(guī)則;

·C++語言的特定信息;

·調(diào)用示例。

只要遵循正確的過程調(diào)用標(biāo)準(zhǔn)AAPCS,就可以混合調(diào)用C、C++和匯編語言例程。有關(guān)AAPCS的更多信息,請參閱ARM相關(guān)文檔。

12.4.1相互調(diào)用的一般規(guī)則

以下一般規(guī)則適用于C、C++和匯編語言之間的調(diào)用。有關(guān)的詳細(xì)信息,請參閱ARM開發(fā)相關(guān)文檔。

嵌入式匯編程序以及其與ARM嵌入式應(yīng)用程序二進(jìn)制接口(BSABI,ApplicationBinaryInterfacefortheARMArchitecture)的兼容使得混合語言編程更易于實(shí)現(xiàn)。它們可提供以下功能:

·使用__cpp關(guān)鍵字進(jìn)行名稱延伸;

·傳遞隱含this參數(shù)的方式;

·調(diào)用虛函數(shù)的方式;

·引用的表示;

·具有基類或虛成員函數(shù)的C++類的類型布局;

·非POD(PlainOldData)結(jié)構(gòu)的類對象傳遞。

以下一般規(guī)則適用于混合語言編程:

·使用C調(diào)用約定。

·在C++中,非成員函數(shù)可以聲明為extern"C",以指定它們有C鏈接。帶有C鏈接意味著定義函數(shù)的符號未延伸。C鏈接可以用于以一種語言實(shí)現(xiàn)函數(shù),然后用另一種語言調(diào)用它。

·匯編語言模塊所必須符合的AAPCS調(diào)用標(biāo)準(zhǔn),應(yīng)當(dāng)適合于應(yīng)用程序所使用的存儲器模型。

以下規(guī)則適用于從C和匯編語言調(diào)用C++函數(shù):

·要調(diào)用全局(非成員)C++函數(shù),應(yīng)將它聲明為extern"C",以提供C鏈接。

·成員函數(shù)(靜態(tài)和非靜態(tài))總是有已延伸的名稱。使用嵌入式匯編程序的__cpp關(guān)鍵字,可以不必手工尋找已延伸的名稱。

·不能從C調(diào)用C++內(nèi)聯(lián)函數(shù),除非確保C++編譯器生成了函數(shù)的外聯(lián)副本。例如,取得函數(shù)地址將導(dǎo)致生成外聯(lián)副本。

·非靜態(tài)成員函數(shù)接受隱含this參數(shù)作為r0中的第一個(gè)自變量,或作為r1中第二個(gè)自變量(如果函數(shù)返回非int類結(jié)構(gòu))。靜態(tài)成員函數(shù)不接受隱含this參數(shù)。

12.4.2C++的特定信息

本節(jié)主要介紹一些專門適用于C++的內(nèi)容。

(1)C++調(diào)用約定

ARMC++使用與ARMC相同的調(diào)用約定,但在下面的情況下,調(diào)用規(guī)則有所不同:

·調(diào)用非靜態(tài)成員函數(shù)時(shí),隱含的this參數(shù)是第一個(gè)自變量,或者是第二個(gè)自變量(如果被調(diào)用函數(shù)返回非int類的struct)。這可能在將來的版本中有所變化。

(2)C++數(shù)據(jù)類型

ARMC++使用與ARMC相同的數(shù)據(jù)類型,但在以下幾種情況下,情況有所不同:

·如果struct或class類型的C++對象沒有基類或虛函數(shù),則它們的布局與ARMC相同。如果這樣的struct沒有用戶定義的復(fù)制賦值運(yùn)算符或用戶定義的析構(gòu)函數(shù),則它是POD結(jié)構(gòu)。

·引用表示為指針。

·C函數(shù)指針和C++(非成員)函數(shù)指針沒有區(qū)別。

(3)符號名稱延伸

鏈接程序?qū)⑷∠畔⒅蟹柮Q的延伸。

在C++程序中,C名稱必須聲明為extern"C"。ARMISOC頭文件已經(jīng)完成此操作。詳細(xì)信息請參閱ARM相關(guān)文檔。

12.4.3混合編程調(diào)用舉例

匯編程序、C程序以及C++程序相互調(diào)用時(shí),要特別注意遵守相應(yīng)的AAPCS。下面一些例子具體說明了在這些混合調(diào)用中應(yīng)注意遵守的AAPCS規(guī)則。這些示例程序默認(rèn)為使用非軟件棧檢查的ATPCS規(guī)則,因?yàn)樗鼈儓?zhí)行棧操作時(shí)不檢查棧溢出。

(1)從C調(diào)用匯編語言

下面的程序顯示如何在C程序中調(diào)用匯編語言子程序,該段代碼實(shí)現(xiàn)了將一個(gè)字符串復(fù)制到另一個(gè)字符串。

#include<stdio.h>

externvoidstrcopy(char*d,constchar*s);

intmain()

{constchar*srcstr="Firststring-source";

chardststr[]="Secondstring-destination";

/*下面將dststr作為數(shù)組進(jìn)行操作*/

printf("Beforecopying:\n");

printf("%s\n%s\n",srcstr,dststr);

strcopy(dststr,srcstr);

printf("Aftercopying:\n");

printf("%s\n%s\n",srcstr,dststr);

return(0);

}

下面為調(diào)用的匯編程序。

PRESERVE8

AREASCopy,CODE,READONLY

EXPORTstrcopy

Strcopy ;r0指向目的字符串

;r1指向源字符串

LDRBr2,[r1],#1 ;加載字節(jié)并更新源字符串指針地址

STRBr2,[r0],#1 ;存儲字節(jié)并更新目的字符串指針地址

CMPr2,#0 ;判斷是否為字符串結(jié)尾

BNEstrcopy ;如果不是,程序跳轉(zhuǎn)到strcopy繼續(xù)拷貝

MOVpc,lr ;程序返回

END

按以下步驟從命令行編譯該示例:

①輸入armasm-gscopy.s編譯匯編語言源代碼。

②輸入armcc-c-gstrtest.c編譯C源代碼。

③輸入armlinkstrtest.oscopy.o-ostrtest鏈接目標(biāo)文件。

④將ELF/DWARF2兼容調(diào)試器與相應(yīng)調(diào)試目標(biāo)配合使用,運(yùn)行映像。

(2)匯編語言調(diào)用C程序

下面的例子顯示了如何從匯編語言調(diào)用C程序。

下面的子程序段定義了C語言函數(shù)。

intg(inta,intb,intc,intd,inte)

{

returna+b+c+d+e;

}

下面的程序段顯示了匯編語言調(diào)用。假設(shè)程序進(jìn)入f時(shí),r0中的值為i。

;intf(inti){returng(i,2*i,3*i,4*i,5*i);}

PRESERVE8

EXPORTf

AREAf,CODE,READONLY

IMPORTg //聲明C程序g()

STRlr,[sp,#-4]! //保存返回地址lr

ADDr1,r0,r0 //計(jì)算2*i(第2個(gè)參數(shù))

ADDr2,r1,r0 //計(jì)算3*i(第3個(gè)參數(shù))

ADDr3,r1,r2 //計(jì)算5*i

STRr3,[sp,#-4]! //第五個(gè)參數(shù)通過堆棧傳遞

ADDr3,r1,r1 //計(jì)算4*i(第4個(gè)參數(shù))

BLg //調(diào)用C程序

ADDsp,sp,#4 //從堆棧中刪除第5個(gè)參數(shù)

LDRpc,[sp],#4 //返回

END

(3)從C++調(diào)用C

下面的例子顯示了如何從C++程序中調(diào)用C函數(shù)。

下面的C++程序調(diào)用了C程序。

structS{ //本結(jié)構(gòu)沒有基類和虛函數(shù)

S(ints):i(s){}

inti;

};

extern"C"voidcfunc(S*);

//被調(diào)用的C函數(shù)使用extern“C”聲明

intf(){

Ss(2); //初始化's'

cfunc(&s); //調(diào)用C函數(shù)'cfunc'將改變's'

returnsi*3;

}

下面顯示了被調(diào)用的C程序代碼。

structS{

inti;

};

voidcfunc(structS*p){

/*定義被調(diào)用的C功能*/

p->i+=5;

}

(4)從C++中調(diào)用匯編

下面的例子顯示了如何從C++中調(diào)用匯編程序。

下面的例子為調(diào)用匯編程序的C++代碼。

structS{ //本結(jié)果沒有基類和虛擬函數(shù)

//

S(ints):i(s){}

inti;

};

extern"C"voidasmfunc(S*); //聲明被調(diào)用的匯編函數(shù)

intf(){

Ss(2); //初始化結(jié)構(gòu)體's'

asmfunc(&s); //調(diào)用匯編子程序'asmfunc'

returns.i*3;

}

下面是被調(diào)用的匯編程序。

PRESERVE8

AREAAsm,CODE

EXPORTasmfunc

asmfunc//被調(diào)用的匯編程序定義

LDRr1,[r0]

ADDr1,r1,#5

STRr1,[r0]

MOVpc,lr

END

(5)從C中調(diào)用C++

下面的例子顯示了如何從C++代碼中調(diào)用C程序。

下面的代碼顯示了被調(diào)用C++代碼。

structS{//本結(jié)構(gòu)沒有基類和虛擬函數(shù)

S(ints):i(s){}

inti;

};

extern"C"voidcppfunc(S*p){

//定義被調(diào)用的C++代碼

//連接了C功能

p->i+=5;//

}

調(diào)用了C++代碼的C函數(shù)。

structS{

inti;

};

externvoidcppfunc(structS*p);

/*聲明將會被調(diào)用的C++功能*/

intf(void){

structSs;

s.i=2;/*初始化S*/

cppfunc(&s);/*調(diào)用cppfunc函數(shù),該函數(shù)可能改變S的值*/

returns.i*3;

}

(6)從匯編中調(diào)用C++程序

下面的代碼顯示了如何從匯編中調(diào)用C++程序。

下面是被調(diào)用的C++程序。

structS{//本結(jié)構(gòu)沒有基類和虛擬函數(shù)

S(ints):i(s){}

inti;

};

extern"C"voidcppfunc(S*p){

//定義被調(diào)用的C++功能

//功能函數(shù)體

p->i+=5;

}

在匯編語言中,聲明要調(diào)用的C++功能,使用帶連接的跳轉(zhuǎn)指令調(diào)用C++功能。

AREAAsm,CODE

IMPORTcppfunc ;聲明被調(diào)用的C++函數(shù)名

EXPORTf

f

STMFDsp!,{lr}

MOVr0,#2

STRr0,[sp,#-4]! ;初始化結(jié)構(gòu)體

MOVr0,sp ;調(diào)用參數(shù)為指向結(jié)構(gòu)體的指針

BLcppfunc ;調(diào)用C++功能'cppfunc'

LDRr0,[sp],#4

ADDr0,r0,r0,LSL#1

LDMFDsp!,{pc}

END

(7)在C和C++函數(shù)間傳遞參數(shù)

下面的例子顯示了如何在C和C++函數(shù)間傳遞參數(shù)。

下面的代碼為C++函數(shù)。

extern"C"intcfunc(constint&);

//聲明被調(diào)用的C函數(shù)

extern"C"intcppfunc(constint&r){

//定義將被C調(diào)用的C++函數(shù)

return7*r;

}

intf(){

inti=3;

returncfunc(i); //相C函數(shù)傳參

}

下面為C函數(shù)。

externintcppfunc(constint*);

/*聲明將被調(diào)用的C++函數(shù)*/

intcfunc(constint*p){

/*定義被C++調(diào)用的C函數(shù)*/

intk=*p+4;

returncppfunc(&k);

}

(8)從C或匯編語言調(diào)用C++

下面的例子綜合顯示了如何從C或匯編語言中調(diào)用非靜態(tài)、非虛的C++成員函數(shù)??梢允褂镁幾g器編譯出的匯編程序查找已延伸的函數(shù)名。

下面是被調(diào)用的C++成員函數(shù)。

structT{

T(inti):t(i){}

intt;

intf(inti);

};

intT::f(inti){returni+t;}

//定義將被C調(diào)用的C++功能函數(shù)

extern"C"intcfunc(T*);

//聲明將被C++調(diào)用的C函數(shù)

intf(){

Tt(5);//createanobjectoftypeT

returncfunc(&t);

}

下面為調(diào)用C++的C語言函數(shù)。

structT;

externint_ZN1T1fEi(structT*,int);

/*被調(diào)用的C++函數(shù)名*/

intcfunc(structT*t){

/*定義被C++調(diào)用的C函數(shù)*/

return3*_ZN1T1fEi(t,2);/*實(shí)現(xiàn)3乘以t->f(2)功能*/

}

下面為調(diào)用C++的匯編函數(shù)。

EXPORTcfunc

AREAfoo,CODE

IMPORT_ZN1T1fEi

cfunc

STMFDsp!,{lr} ;此時(shí)r0已經(jīng)包含了指向?qū)ο蟮闹羔?/p>

MOVr1,#2

BL_ZN1T1fEi

ADDr0,r0,r0,LSL#1 ;r0乘以3

LDMFDsp!,{pc}

END

下面的例子顯示了如何用嵌入式匯編語言實(shí)現(xiàn)上面的例子。在此例中,使用__cpp關(guān)鍵字引用該函數(shù)。因此,用戶不必了解已延伸的函數(shù)名。

structT{

T(inti):t(i){}

intt;

intf(inti);

};

intT::f(inti){returni+t;}

//定義被C++調(diào)用的匯編功能

__asmintasm_func(T*){

STMFDsp!,{lr}

MOVr1,#2;

BL__cpp(T::f);

ADDr0,r0,r0,LSL#1;r0乘以3

LDMFDsp!,{pc}

}

intf(){

Tt(5);//創(chuàng)建T類型的對象

returnasm_func(&t);

}

聯(lián)系方

本站聲明: 本文章由作者或相關(guān)機(jī)構(gòu)授權(quán)發(fā)布,目的在于傳遞更多信息,并不代表本站贊同其觀點(diǎn),本站亦不保證或承諾內(nèi)容真實(shí)性等。需要轉(zhuǎn)載請聯(lián)系該專欄作者,如若文章內(nèi)容侵犯您的權(quán)益,請及時(shí)聯(lián)系本站刪除。
換一批
延伸閱讀

9月2日消息,不造車的華為或?qū)⒋呱龈蟮莫?dú)角獸公司,隨著阿維塔和賽力斯的入局,華為引望愈發(fā)顯得引人矚目。

關(guān)鍵字: 阿維塔 塞力斯 華為

加利福尼亞州圣克拉拉縣2024年8月30日 /美通社/ -- 數(shù)字化轉(zhuǎn)型技術(shù)解決方案公司Trianz今天宣布,該公司與Amazon Web Services (AWS)簽訂了...

關(guān)鍵字: AWS AN BSP 數(shù)字化

倫敦2024年8月29日 /美通社/ -- 英國汽車技術(shù)公司SODA.Auto推出其旗艦產(chǎn)品SODA V,這是全球首款涵蓋汽車工程師從創(chuàng)意到認(rèn)證的所有需求的工具,可用于創(chuàng)建軟件定義汽車。 SODA V工具的開發(fā)耗時(shí)1.5...

關(guān)鍵字: 汽車 人工智能 智能驅(qū)動(dòng) BSP

北京2024年8月28日 /美通社/ -- 越來越多用戶希望企業(yè)業(yè)務(wù)能7×24不間斷運(yùn)行,同時(shí)企業(yè)卻面臨越來越多業(yè)務(wù)中斷的風(fēng)險(xiǎn),如企業(yè)系統(tǒng)復(fù)雜性的增加,頻繁的功能更新和發(fā)布等。如何確保業(yè)務(wù)連續(xù)性,提升韌性,成...

關(guān)鍵字: 亞馬遜 解密 控制平面 BSP

8月30日消息,據(jù)媒體報(bào)道,騰訊和網(wǎng)易近期正在縮減他們對日本游戲市場的投資。

關(guān)鍵字: 騰訊 編碼器 CPU

8月28日消息,今天上午,2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會開幕式在貴陽舉行,華為董事、質(zhì)量流程IT總裁陶景文發(fā)表了演講。

關(guān)鍵字: 華為 12nm EDA 半導(dǎo)體

8月28日消息,在2024中國國際大數(shù)據(jù)產(chǎn)業(yè)博覽會上,華為常務(wù)董事、華為云CEO張平安發(fā)表演講稱,數(shù)字世界的話語權(quán)最終是由生態(tài)的繁榮決定的。

關(guān)鍵字: 華為 12nm 手機(jī) 衛(wèi)星通信

要點(diǎn): 有效應(yīng)對環(huán)境變化,經(jīng)營業(yè)績穩(wěn)中有升 落實(shí)提質(zhì)增效舉措,毛利潤率延續(xù)升勢 戰(zhàn)略布局成效顯著,戰(zhàn)新業(yè)務(wù)引領(lǐng)增長 以科技創(chuàng)新為引領(lǐng),提升企業(yè)核心競爭力 堅(jiān)持高質(zhì)量發(fā)展策略,塑強(qiáng)核心競爭優(yōu)勢...

關(guān)鍵字: 通信 BSP 電信運(yùn)營商 數(shù)字經(jīng)濟(jì)

北京2024年8月27日 /美通社/ -- 8月21日,由中央廣播電視總臺與中國電影電視技術(shù)學(xué)會聯(lián)合牽頭組建的NVI技術(shù)創(chuàng)新聯(lián)盟在BIRTV2024超高清全產(chǎn)業(yè)鏈發(fā)展研討會上宣布正式成立。 活動(dòng)現(xiàn)場 NVI技術(shù)創(chuàng)新聯(lián)...

關(guān)鍵字: VI 傳輸協(xié)議 音頻 BSP

北京2024年8月27日 /美通社/ -- 在8月23日舉辦的2024年長三角生態(tài)綠色一體化發(fā)展示范區(qū)聯(lián)合招商會上,軟通動(dòng)力信息技術(shù)(集團(tuán))股份有限公司(以下簡稱"軟通動(dòng)力")與長三角投資(上海)有限...

關(guān)鍵字: BSP 信息技術(shù)
關(guān)閉
關(guān)閉