對C編譯器進行語法擴充。例如MCS51系列KeilC中擴充sfr關(guān)鍵字,舉例如下:
sfr P0 = 0x80;
這樣操作0x80單元直接寫P0即可。
下面對AVR的歌C編譯器對訪問MCU寄存器的方法進行簡介。
A:采用標(biāo)準(zhǔn)C的強制類型轉(zhuǎn)換和指針來實現(xiàn)訪問MCU的寄存器,每一個C編譯器都支持,原因很簡單,這是標(biāo)準(zhǔn)C。
B:ICCAVR和GCCAVR沒有定義新的數(shù)據(jù)類型,只能采用標(biāo)準(zhǔn)C的強制類型轉(zhuǎn)換和指針來實現(xiàn)訪問MCU的寄存器。而IAR和CodeVisionAVR編譯器對ANSI C進行了擴充,都定義了新的數(shù)據(jù)類型,是C語言可以直接訪問MCU的有關(guān)寄存器,例如,IAR中:
SFR_B(DDRB, 0x28)
CodeVisionAVR中:
sfrb DDRB = 0x28
這樣,PORTB=0xff;等同于(*(volatile unsigned char *)0x05) = 0xff;而0x25正好是寄存器PORTB在器件ATmega48/88/168中的地址。
GCCAVR每個AVR器件在頭文件不采用直接定義特殊功能寄存器宏,例如在iomx8.h文件中一個定義如下:
#define PORTB _SFR_IO8(0x25)
而在sfr_defs.h中可以找到如下兩個宏定義:
#define _SFR_IO8(io_addr) _MMIO_BYTE((io_addr)+0x20)
#define _MMIO_BYTE(mem_addr) (*(volatile unit8_t *)(mem_addr))
實質(zhì)上與直接的強制類型轉(zhuǎn)換和指針定義是一樣的。
另外,GCCAVR中宏_BV(bit)是操作I/O寄存器是頻繁用到的,avr-libc建議使用這一宏進行寄存器的位操作,他在文件sfr_defs.h中定義如下:
#define _BV(bit) (1<<(bit))
以下是他的使用示例;
DDRB = _BV(PB0) | _BV(PB1); //器件頭文件中已經(jīng)定義PB0代表0,PB1代表1
他等同于“DDRB=0x03;”,這樣寫的目的是為了提供程序的可讀性。不要擔(dān)心它會生成比“DDRB=0x03;”更大的代碼,編譯器會處理這種事情,更終會輸出與“DDRB=0x03;”同樣的結(jié)果。