新聞中心

EEPW首頁(yè) > 嵌入式系統(tǒng) > 設(shè)計(jì)應(yīng)用 > keil C語(yǔ)言與匯編語(yǔ)言混合編程

keil C語(yǔ)言與匯編語(yǔ)言混合編程

作者: 時(shí)間:2016-11-09 來(lái)源:網(wǎng)絡(luò) 收藏
1. C語(yǔ)言中嵌入?yún)R編

1、在 C 文件中要嵌入?yún)R編代碼片以如下方式加入?yún)R編代碼:

本文引用地址:http://m.butianyuan.cn/article/201611/317686.htm

#pragma ASM

; Assembler Code Here

#pragma ENDASM

2、在 Project 窗口中包含匯編代碼的 C 文件上右鍵,選擇“Options for ...”,點(diǎn)擊右邊的“Generate Assembler SRC File”

和“Assemble SRC File”,使檢查框由灰色變成黑色(有效)狀態(tài);

3、根據(jù)選擇的編譯模式,把相應(yīng)的庫(kù)文件(如 Small 模式時(shí),是 Keil/C51/Lib/C51S.Lib)加入工程中, 該文件必須作為工程的最后文件;

4、編譯,即可生成目標(biāo)代碼。

來(lái)個(gè)實(shí)例吧:

#i nclude

void main(void)

{

P2=1;

#pragma asm

MOV R7,#10

DEL:MOV R6,#20

DJNZ R6,$

DJNZ R7,DEL

#pragma endasm

P2=0;

}

2 . 無(wú)參數(shù)傳遞的函數(shù)調(diào)用

C51調(diào)用匯編函數(shù)

1.無(wú)參數(shù)傳遞的函數(shù)調(diào)用

先來(lái)個(gè)例子:其中example.c和example.a51為項(xiàng)目中的兩個(gè)文件

*example.c*

extern void delay100();

main()

{delay100;}

*example.a51*

?PR?DELAY100 SEGMENT CODE; // 在程序存儲(chǔ)區(qū)中定義段

PUBLIC DELAY100;//聲明函數(shù)

RSEG ?PR?DELAY100;//函數(shù)可被連接器放置在任何地方

DELAY100:

MOV R7,#10

DEL:

MOV R6,#20

DJNZ R6,$

DJNZ R7,DEL

RET

END

在example.c文件中,先聲明外部函數(shù),然后直接在main中調(diào)用即可。

在example.a51中,

?PR?DELAY100 SEGMENT CODE; 作用是在程序存儲(chǔ)區(qū)中定義段,DELAY100為段名,?PR?表示段位于程序存儲(chǔ)區(qū)內(nèi)

PUBLIC DELAY100; 作用是聲明函數(shù)為公共函數(shù)

RSEG ?PR?DELAY100; 表示函數(shù)可被連接器放置在任何地方,RSEG是段名的屬性

段名的開(kāi)頭為PR,是為了和C51內(nèi)部命名轉(zhuǎn)換兼容,命名轉(zhuǎn)換規(guī)律如下:

CODE -?PR?

XDATA-?XD

DATA-?DT

BIT-?BI

PDATA-?PD

3. 有參數(shù)傳遞的函數(shù)調(diào)用

記住哦,c文件和A51文件不能使用同一個(gè)文件名,不過(guò)我還不知道為什么會(huì)這樣,有高手知道得話(huà)請(qǐng)告知。

今天說(shuō)說(shuō)帶參數(shù)傳遞的函數(shù)調(diào)用,在C51和匯編之間傳遞參數(shù)的方式有兩種,一種是通過(guò)寄存器傳遞參數(shù),C51中不同類(lèi)型的實(shí)參會(huì)存入相應(yīng)的寄存器,在匯編中只需對(duì)相應(yīng)寄存器進(jìn)行操作,即達(dá)到傳遞參數(shù)的目的。

不同類(lèi)型的數(shù)據(jù)及其傳遞參數(shù)的寄存器如下表所示:

 在C和匯編混合編程的時(shí)候,存在C語(yǔ)言和匯編語(yǔ)言的變量以及函數(shù)的接口問(wèn)題。

在C程序中定義的變量,編譯為.asm文件后,都被放進(jìn)了.bss區(qū),而且變量名的前面都帶了一個(gè)下劃線(xiàn)。在C程序中定義的函數(shù),編譯后在函數(shù)名前也帶了一個(gè)下劃線(xiàn)。例如:

extern int num就會(huì)變成 .bss _num, 1

extern float nums[5]就會(huì)變成.bss _nums, 5

extern void func ( )就會(huì)變成 _func,

一  匯編和C的相互調(diào)用可以分以下幾種情況:

(1) 匯編程序中訪(fǎng)問(wèn)c程序中的變量和函數(shù)。

在匯編程序中,用_XX就可以訪(fǎng)問(wèn)C中的變量XX了。訪(fǎng)問(wèn)數(shù)組時(shí),可以用_XX+偏移量來(lái)訪(fǎng)問(wèn),如_XX+3訪(fǎng)問(wèn)了數(shù)組中的XX[3]。

在匯編程序調(diào)用C函數(shù)時(shí),如果沒(méi)有參數(shù)傳遞,直接用_funcname 就可以了。如果有參數(shù)傳遞, 則函數(shù)中最左邊的一個(gè)參數(shù)由寄存器A給出,其他的參數(shù)按順序由堆棧給出。返回值是返回到A寄存器或者由A寄存器給出的地址。同時(shí)注意,為了能夠讓匯編語(yǔ)言 能訪(fǎng)問(wèn)到C語(yǔ)言中定義的變量和函數(shù),他們必須聲明為外部變量,即加extern 前綴。

(2) c程序中訪(fǎng)問(wèn)匯編程序中的變量

如果需要在c程序中訪(fǎng)問(wèn)匯編程序中的變量,則匯編程序中的變量名必須以下劃線(xiàn)為首字符,并用global使之成為全局變量。

如果需要在c程序中調(diào)用匯編程序中的過(guò)程,則過(guò)程名必須以下劃線(xiàn)為首字符,并且,要根據(jù)c程序編譯時(shí)使用的模式是stack-based model還是register argument model來(lái)正確地編寫(xiě)該過(guò)程,使之能正確地取得調(diào)用參數(shù)。

(3) 在線(xiàn)匯編

在C程序中直接插入 asm(“ * ”),內(nèi)嵌匯編語(yǔ)句,需要注意的是這種用法要慎用,在線(xiàn)匯編提供了能直接讀寫(xiě)硬件的能力,如讀寫(xiě)中斷控制允許寄存器等,但編譯器并不檢查和分析在線(xiàn)匯編語(yǔ)言,插入在線(xiàn)匯編語(yǔ)言改變匯編環(huán)境或可能改變C變量的值可能導(dǎo)致嚴(yán)重的錯(cuò)誤。

二 匯編和C接口中尋址方式的改變:

需 要注意的是,在C語(yǔ)言中,對(duì)于局部變量的建立和訪(fǎng)問(wèn),是通過(guò)堆棧實(shí)現(xiàn)的,它的尋址是通過(guò)堆棧寄存器SP實(shí)現(xiàn)的。而在匯編語(yǔ)言中,為了使程序代碼變得更為精 簡(jiǎn),TI在直接尋址方式中,地址的低7位直接包含在指令中,這低7位所能尋址的具體位置可由DP寄存器或SP寄存器決定。具體實(shí)現(xiàn)可通過(guò)設(shè)置ST1寄存器 的CPL位實(shí)現(xiàn),CPL=0,DP尋址,CPL=1,SP尋址。在DP尋址的時(shí)候,由DP提供高9位地址,與低7位組成16位地址;在SP尋址的時(shí)候, 16位地址是由SP(16位)與低7位直接相加得來(lái)。

由于在C語(yǔ)言的環(huán)境下,局部變量的尋址必須通過(guò)SP寄存器實(shí)現(xiàn),在混合編程的時(shí)候,為了使匯編語(yǔ)言不影響堆棧寄存器SP,通常的方式是在匯編環(huán)境中使用DP方式尋址,這樣可以使二者互不干擾。編程中只要注意對(duì)CPL位正確設(shè)置即可

1 .word 的意思就相當(dāng)與C語(yǔ)言里的int,char等定義一個(gè)變兩的寬度

2. 編譯錯(cuò)誤原因有2:

a.如果在匯編里面定義.global(全局符號(hào)),那么在C語(yǔ)言里面應(yīng)該用extern聲明,以引用該符號(hào)。

b.在匯編里面聲明的時(shí)候,符號(hào)前應(yīng)加下劃線(xiàn),如 FIQ_Addr: .word EXTint_FIQ 應(yīng)為: FIQ_Addr: .word _EXTint_FIQ 在C語(yǔ)言里面應(yīng)用extern聲明。另外,一中方法是,用.ref 代替.global 來(lái)聲明符號(hào),這樣就不用在C源程序里面用extern聲明了。兩種方法結(jié)果相同。 我講的是用C和匯編混編程用法,至于C++變量如何翻譯成匯編符號(hào)可以用仿真器,自己去看,原則類(lèi)似.



評(píng)論


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

關(guān)閉