⑴ c語言調用匯編程序時,使用哪些寄存器來傳遞參數
C語言與匯編語言混合編程應遵守的規則
ARM編程中使用的C語言是標准C語言,ARM的開發環境實際上就是嵌入了一個C語言的集成開發環境,只不過這個開發環境與ARM的硬體緊密相關。
在使用C語言時,要用到和匯編語言的混合編程。若匯編代碼較為簡潔,則可使用直接內嵌匯編的方法;否則要將匯編程序以文件的形式加入到項目中,按照ATPCS(ARM/Thumb過程調用標准,ARM/Thumb Procere Call Standard)的規定與C程序相互調用與訪問。
在C程序和ARM匯編程序之間相互調用時必須遵守ATPCS規則。ATPCS規定了一些子程序間調用的基本規則,哪寄存器的使用規則,堆棧的使用規則和參數的傳遞規則等。
1)寄存器的使用規則
子程序之間通過寄存器r0~r3來傳遞參數,當參數個數多於4個時,使用堆棧來傳遞參數。此時r0~r3可記作A1~A4。
在子程序中,使用寄存器r4~r11保存局部變數。因此當進行子程序調用時要注意對這些寄存器的保存和恢復。此時r4~r11可記作V1~V8。
寄存器r12用於保存堆棧指針SP,當子程序返回時使用該寄存器出棧,記作IP。
寄存器r13用作堆棧指針,記作SP。寄存器r14稱為鏈接寄存器,記作LR。該寄存器用於保存子程序的返回地址。
寄存器r15稱為程序計數器,記作PC。
2)堆棧的使用規則
ATPCS規定堆棧採用滿遞減類型(FD,Full Descending),即堆棧通過減小存儲器地址而向下增長,堆棧指針指向內含有效數據項的最低地址。
3)參數的傳遞規則
整數參數的前4個使用r0~r3傳遞,其他參數使用堆棧傳遞;浮點參數使用編號最小且能夠滿足需要的一組連續的FP寄存器傳遞參數。
子程序的返回結果為一個32位整數時,通過r0返回;返回結果為一個64位整數時,通過r0和r1返回;依此類推。結果為浮點數時,通過浮點運算部件的寄存器F0、D0或者S0返回。
2、匯編程序調用C程序的方法
匯編程序的書寫要遵循ATPCS規則,以保證程序調用時參數正確傳遞。在匯編程序中調用C程序的方法為:首先在匯編程序中使用IMPORT偽指令事先聲明將要調用的C語言函數;然後通過BL指令來調用C函數。
例如在一個C源文件中定義了如下求和函數:
int add(int x,int y){
return(x+y);
}
調用add()函數的匯編程序結構如下:
IMPORT add ;聲明要調用的C函數
……
MOV r0,1
MOV r1,2
BL add ;調用C函數add
……
當進行函數調用時,使用r0和r1實現參數傳遞,返回結果由r0帶回。函數調用結束後,r0的值變成3。
3、C程序調用匯編程序的方法
C程序調用匯編程序時,匯編程序的書寫也要遵循ATPCS規則,以保證程序調用時參數正確傳遞。在C程序中調用匯編子程序的方法為:首先在匯編程序中使用EXPORT偽指令聲明被調用的子程序,表示該子程序將在其他文件中被調用;然後在C程序中使用extern關鍵字聲明要調用的匯編子程序為外部函數。
例如在一個匯編源文件中定義了如下求和函數:
EXPORT add ;聲明add子程序將被外部函數調用
……
add ;求和子程序add
ADD r0,r0,r1
MOV pc,lr
……
在一個C程序的main()函數中對add匯編子程序進行了調用:
extern int add (int x,int y); //聲明add為外部函數
void main(){
int a=1,b=2,c;
c=add(a,b); //調用add子程序
……
}
當main()函數調用add匯編子程序時,變數a、b的值會給了r0和r1,返回結果由r0帶回,並賦值給變數c。函數調用結束後,變數c的值變成3。
4、C程序中內嵌匯編語句
在C語言中內嵌匯編語句可以實現一些高級語言不能實現或者不容易實現的功能。對於時間緊迫的功能也可以通過在C語言中內嵌匯編語句來實現。內嵌的匯編器支持大部分ARM指令和Thumb指令,但是不支持諸如直接修改PC實現跳轉的底層功能,也不能直接引用C語言中的變數。
嵌入式匯編語句在形式上獨立定義的函數體,其語法格式為:
__asm
{
指令[;指令]
……
[指令]
}
其中「__asm」為內嵌匯編語句的關鍵字,需要特別注意的是前面有兩個下劃線。指令之間用分號分隔,如果一條指令占據多行,除最後一行外都要使用連字元「\」。
5、基於ARM的C語言與匯編語言混合編程舉例
下面給出了一個向串口不斷發送0x55的例子:
該工程的啟動代碼使用匯編語言編寫,向串口發送數據使用C語言實現,下面是啟動代碼的整體框架:
……
IMPORT Main
AREA Init,CODE,READONLY;
ENTRY
……
BL Main ;跳轉到Main()函數處的C/C++程序
……
END ;標識匯編程序結束
下面是使用C語言編寫的主函數:
#include "..\inc\config.h" //將有關硬體定義的頭文件包含進來
unsigned char data; //定義全局變數
void main(void){
Target_Init(); //對目標板的硬體初始化
Delay(10); //延時
data=0x55; //給全局變數賦值
while(1) {
Uart_Printf("%x",data); //向串口送數
Delay(10);
}
}
⑵ c語言 程序寄存器是什麼情況啊
寄存器是硬體資源,是比內存速度要快的一種存儲單元 ,集成在CPU內部和CPU同步,屬於稀缺資源,所以經常有申請不到的情況 也就是說C語言里的reg類型數據有時候不一定能放到寄存器里
⑶ C語言中如何設置寄存器的 GPFDAT[7:6]為10,而保持其他位的值不變
需要將第6位置0、第7位置1,不影響其它位,就需要通過兩次邏輯運算實現。
先將第6位置0,不影響其它位:
GPFDAT &= (~(1<<6));
再將第7位置1,不影響其它位:
GPFDAT |= (1<<7)
0X40=0100 0000
~0X40=1011 1111再與上(R2)
內聯匯編代碼可以使用 C/C++ 變數和函數,因此它能非常容易地整合到 C/C++ 代碼中。它能做一些對於單獨使用 C/C++ 來說非常笨重或不可能完成的任務。
(3)c語言程序寄存器擴展閱讀:
寄存器有串列和並行兩種數碼存取方式。將n位二進制數一次存入寄存器或從寄存器中讀出的方式稱為並行方式。將n位二進制數以每次1位,分成n次存入寄存器並從寄存器讀出,這種方式稱為串列方式。
並行方式只需一個時鍾脈沖就可以完成數據操作,工作速度快,但需要n根輸入和輸出數據線。串列方式要使用幾個時鍾脈沖完成輸入或輸出操作,工作速度慢,但只需要一根輸入或輸出數據線,傳輸線少,適用於遠距離傳輸。
⑷ C語言是如何訪問28035的寄存器
恐怕只能內嵌匯編了。但是不建議在C語言的程序中直接訪問寄存器。很容易導致程序出錯。一般 內嵌單條語句可以使用asm("");如:asm("nop");
⑸ c語言 程序寄存器是什麼情況啊
寄存器是硬體資源,是比內存速度要快的一種存儲單元
,集成在CPU內部和CPU同步,屬於稀缺資源,所以經常有申請不到的情況
也就是說C語言里的reg類型數據有時候不一定能放到寄存器里
⑹ c語言單片機寄存器是怎麼操作的
可以說是對寄存器的操作吧,不過不只是c語言,匯編語言也可以。。。。。其實單片機對寄存器的操作不外乎是採集外部信息,然後在內部處理後,再通過寄存器輸出控制信息,對外部信息作出相應的反應,其實單片機就相當於計算機。。。。
⑺ c語言讀取寄存器的值
在C語言中可以使用內聯匯編獲取寄存器的值。不同的編譯器使用內聯匯編的方法不同,在VC/VS中使用內聯匯編的關鍵為__asm,在gcc中使用內聯匯編的關鍵字為asm。
以vc6.0、IA32 cpu為例,提取通用寄存器eax、ebx、ecx、edx的值。
#include<stdio.h>
#include<string.h>
intmain()
{
unsigneeax,uebx,uecx,uedx;
__asm//使用__asm進行內聯匯編
{
//使用mov指令將eax寄存器的內容保存到ueax變數
movueax,eax
movuebx,ebx
movuecx,ecx
movuedx,edx
}
printf("eax=%x ebx=%x ecx=%x edx=%x ",ueax,uebx,uecx,uedx);
return0;
}
⑻ c語言如何讀寄存器的值
1、首先,輸入寄存器變數關鍵字【register】。
注意事項:
C語言描述問題比匯編語言迅速、工作量小、可讀性好、易於調試、修改和移植,而代碼質量與匯編語言相當。C語言一般只比匯編語言代碼生成的目標程序效率低10%~20%。因此,C語言可以編寫系統軟體。
⑼ c語言單片機寄存器是怎麼操作的
c語言對單片機寄存器的操作就是讀/寫,也很簡單的,就是直接賦值。用大寫字母表示寄存器,如
TMOD=0x20;//給寄存器賦值
x=SBUF; //讀寄存器
⑽ C語言中,如何定義單片機的寄存器
位地址在匯編語言中,只能直接定址,不支持間接定址。那麼,在 C 語言中,也就不能用指針來操作。所以不能定義單片機的寄存器。
訪問寄存器可以通過直接地址引用來完成。
例如52單片機的寄存器有4組,分別為00H-07H,08H-0FH,10H-17H,18H-1FH,各組均8位元組,要結合程序狀態字寄存器PSW(D0H)中的RS0,RS1的組合來確定是哪一組。
可以定義一個訪問Rn的函數typedefunsignedcharuchar;
uchardacRn(ucharn)//n=[0..7]
{
uchardata*p;
n+=(PSW&0x18);//PSW7:CY6:AC5:F04:RS13:RS02:OV1:F10:P
//本來要右移三位變為0-3,但由於是8個一組要乘以8(左移3位),所以直接加上寄存器編號就可以了
return*(uchardata*)n;
}