C語(yǔ)言訪(fǎng)問(wèn)MCU寄存器的三種方式

發(fā)布時(shí)間:2019-1-22 10:37    發(fā)布者:xyd2018
關(guān)鍵詞: C語(yǔ)言、寄存器
1.對C編譯器進(jìn)行語(yǔ)法擴充
對C編譯器進(jìn)行語(yǔ)法擴充。例如MCS51系列單片機的C-51語(yǔ)法中擴充了sfr關(guān)鍵字,舉例如下:
sfr P0 = 0x80;
這樣操作0x80單元直接寫(xiě)P0即可。

又如AtmelAVR系列單片機,其ICCAVR和GCCAVR編譯器都沒(méi)有定義新的數據類(lèi)型,只能采用標準C的強制類(lèi)型轉換和指針來(lái)實(shí)現訪(fǎng)問(wèn)MCU的寄存器。而IAR和CodeVisionAVR編譯器對ANSI C進(jìn)行了擴充,定義了新的數據類(lèi)型,使C語(yǔ)言可以直接訪(fǎng)問(wèn)MCU的有關(guān)寄存器,例如在IAR中可以使用:

SFR_B(DDRB, 0x28);

CodeVisionAVR中可以使用:
sfrb DDRB = 0x28;



2.使用標準C的強制類(lèi)型轉換和指針來(lái)實(shí)現


采用標準C的強制轉換和指針的概念來(lái)實(shí)現訪(fǎng)問(wèn)MCU的寄存器,例如:
#define DDRB (*(volatile unsigned char *)0x25)

分析如下:

1.(unsigned char *)0x25中的0x25只是個(gè)值,前面加(unsigned char *)表示把這個(gè)值強制類(lèi)型轉換為unsigned char型的指針。再在前面加”*”,即*(volatile unsigned char *)0x25表示對這個(gè)指針解引用,相當于
(unsigned char *)0x25是一個(gè)指針p,而這個(gè)宏定義為#define DDRB *p。
這樣當讀/寫(xiě)以0x25為地址的寄存器時(shí),直接書(shū)寫(xiě)DDRB即可,即寫(xiě):
DDRB = 0xff;

相當于:
unsigned char *p, i; p = 0x25; i = *p;        //把地址為0x25單元中的數據讀出送入i變量*p = 0xff;     //向地址為0x25的單元中寫(xiě)入0xff

這樣經(jīng)過(guò)一層宏定義的封裝就變得直觀(guān)和方便的多了。

2.關(guān)鍵字volatile確保本指令不會(huì )以為C編譯器的優(yōu)化而被省略,且要求每次直接讀值。例如使用while(*(unsigned char *)0x25)時(shí),有時(shí)系統可能不能真正去讀0x25的值,而是用第一次讀出的值,如果這樣,這個(gè)循環(huán)可能就是個(gè)死循環(huán)。用了volatile則要求每次都去讀0x25的實(shí)際值。

GCCAVR工具鏈中就使用了這樣的方式,例如在iomx8.h文件中一個(gè)定義如下:
#define PORTB _SFR_IO8(0x25)

而在sfr_defs.h中可以找到如下兩個(gè)宏定義:

#define _SFR_IO8(io_addr)     _MMIO_BYTE((io_addr)+0x20)#define _MMIO_BYTE(mem_addr)  (*(volatile unit8_t *)(mem_addr))


實(shí)質(zhì)上與直接的強制類(lèi)型轉換和指針定義是一樣的。


3.使用結構體實(shí)現


使用指針的方式來(lái)訪(fǎng)問(wèn)特殊功能寄存器的優(yōu)勢在于完全符合標準的ANSI-C,而無(wú)需擴展語(yǔ)法,形成“方言”,擁有更好的兼容性和可移植性。

這種方式適合簡(jiǎn)單的應用程序,而當系統用到多個(gè)同種外設時(shí),就需要為每一個(gè)這種外設定義寄存器,這樣就會(huì )使程序的維護變得非常困難。而且,由于每次寄存器操作都會(huì )有對應的常量存儲在程序Flash里,為每個(gè)寄存器定義單獨的指針還會(huì )增加程序代碼。

為了簡(jiǎn)化程序代碼,可以將寄存器組定義為結構體,而將外設當做指向這個(gè)結構體的指針。例如:

typedef struct {    volatile unsigned long DATA;    //0x00    volatile unsigned long RSR;     //0x04    unsigned long RESERVED0[4];     //0x08-0x14    volatile unsigned long FLAG;    //0x18    ... }UART_TypeDef;#define Uart0 ((UART_Type *)0x40003000)#define Uart1 ((UART_Type *)0x40004000)#define Uart2 ((UART_Type *)0x40005000)int getkey(UART_TypeDef * uartptr) {    while((uartptr->FLAG & 0x40) == 0);    //無(wú)數據,等待    return uartptr->DATA;                  // 讀取字符}int main(void) {    unsigned long data;    data = getkey(Uart0); }


在這種設定下,同一個(gè)外設寄存器的結構體可以被多個(gè)外設實(shí)體共用,這樣也使得程序維護變得容易。另外,由于立即數存儲的減少,編譯出的程序代碼也會(huì )變小。

本文來(lái)自bill_20106029的博客

想要了解嵌入式、物聯(lián)網(wǎng)相關(guān)技術(shù)的可以聯(lián)系宋工企鵝號三五二四六五九零八八
Tel:173--1795--1908
免費試聽(tīng)C語(yǔ)言、電子、PCB、STM32、Linux、FPGA、JAVA、安卓等。
想學(xué)習的你和我聯(lián)系預約就可以免費聽(tīng)課了。



本文地址:http://selenalain.com/thread-559893-1-1.html     【打印本頁(yè)】

本站部分文章為轉載或網(wǎng)友發(fā)布,目的在于傳遞和分享信息,并不代表本網(wǎng)贊同其觀(guān)點(diǎn)和對其真實(shí)性負責;文章版權歸原作者及原出處所有,如涉及作品內容、版權和其它問(wèn)題,我們將根據著(zhù)作權人的要求,第一時(shí)間更正或刪除。
您需要登錄后才可以發(fā)表評論 登錄 | 立即注冊

關(guān)于我們  -  服務(wù)條款  -  使用指南  -  站點(diǎn)地圖  -  友情鏈接  -  聯(lián)系我們
電子工程網(wǎng) © 版權所有   京ICP備16069177號 | 京公網(wǎng)安備11010502021702
快速回復 返回頂部 返回列表
午夜高清国产拍精品福利|亚洲色精品88色婷婷七月丁香|91久久精品无码一区|99久久国语露脸精品|动漫卡通亚洲综合专区48页