|
C51:
#define XBYTE ((unsigned char volatile xdata *) 0)
解
釋:XBYTE被定義成(unsigned char volatile xdata *)
0,0為一個(gè)基于寄存器的指針(可理解為:只不過(guò)這個(gè)指針用一個(gè)二進(jìn)制常量標(biāo)識(shí),且其指向的內(nèi)存單元在定義時(shí)就以固定為該二進(jìn)制常量且不可更改.),其存
儲(chǔ)內(nèi)型為2指向xdata,偏移量為0,這樣X(jué)BYTE成為指向xdata零地址的指針,通過(guò)XBYTE[0]可以讀寫(xiě)該地址處的數(shù)據(jù);同樣通過(guò)
XBYTE[8000]也可以訪問(wèn)xdata空間絕對(duì)地址為8000處的數(shù)據(jù)。
待證中:通過(guò)*XBYTE,*(XBYTE+8000)是否可以對(duì)相應(yīng)地址進(jìn)行讀寫(xiě)?
ARM_ADS:
#define BCFG0 (*((volatile unsigned int *) 0xFFE00000))
解
釋:首先0xFFE00000為一個(gè)指針沒(méi)錯(cuò),且該指針為unsigned
int型指向0xFFE00000內(nèi)存單元,BCFG0被定義為對(duì)該指針做“*”指針運(yùn)算,這樣,BCFG0可理解為對(duì)內(nèi)存單元0xFFE00000進(jìn)行
標(biāo)識(shí)的一個(gè)int32u型普通變量,同過(guò)BCFG0可是實(shí)現(xiàn)對(duì)該寄存器的讀寫(xiě)訪問(wèn)。
轉(zhuǎn)載網(wǎng)絡(luò)上的一些文章:
------------------------------------------------------------------------
理解#define SREG (*(volatile unsigned char *)0x5F)
這樣的定義,總是感覺(jué)很奇怪,不知道為什么,今天終于有了一點(diǎn)點(diǎn)心得,請(qǐng)大蝦們多多批磚~~~
嵌入式系統(tǒng)編程,要求程序員能夠利用C語(yǔ)言訪問(wèn)固定的內(nèi)存地址。既然是個(gè)地址,那么按照C語(yǔ)言的語(yǔ)法規(guī)則,這個(gè)表示地址的量應(yīng)該是指針類型。所以,知道要訪問(wèn)的內(nèi)存地址后,比如0x5F,
第一步是要把它強(qiáng)制轉(zhuǎn)換為指針類型
(unsigned char *)0x5F,AVR的SREG是八位寄存器,所以0x5F強(qiáng)制轉(zhuǎn)換為指向
unsigned char類型。
volatile(可變的)這個(gè)關(guān)鍵字說(shuō)明這變量可能會(huì)被意想不到地改變,這樣編譯器就不會(huì)去假設(shè)這個(gè)變量的值了。這種“意想不到地改變”,不是由程序去改變,而是由硬件去改變——意想不到。
第二步,對(duì)指針變量解引用,就能操作指針?biāo)赶虻牡刂返膬?nèi)容了
*(volatile unsigned char *)0x5F
第三步,小心地把#define宏中的參數(shù)用括號(hào)括起來(lái),這是一個(gè)很好的習(xí)慣,所以#define SREG (*(volatile unsigned char *)0x5F)
類似的,如果使用一個(gè)32位處理器,要對(duì)一個(gè)32位的內(nèi)存地址進(jìn)行訪問(wèn),可以這樣定義#define RAM_ADDR (*(volatile unsigned long *)0x0000555F)
然后就可以用C語(yǔ)言對(duì)這個(gè)內(nèi)存地址進(jìn)行讀寫(xiě)操作了
讀:tmp = RAM_ADDR;
寫(xiě):RAM_ADDR = 0x55;
zhiwei 發(fā)表于 2005-4-30 18:59 AVR 單片機(jī)
定義未volatile是因?yàn)樗闹悼赡軙?huì)改變,大家都知道為什么改變了;
如
果在一個(gè)循環(huán)操作中需要不停地判斷一個(gè)內(nèi)存數(shù)據(jù),例如要等待SREG的I標(biāo)志位置位,因?yàn)镾REG也是映射在SRAM空間,為了加快速度,編譯器可能會(huì)編
譯出這樣的代碼:把SREG讀取到Register中,然后不停地判斷Register相應(yīng)位。而不會(huì)再讀取SREG,這樣當(dāng)然是不行了,因?yàn)槌绦蚧蚱渌?br />
事件(中斷等)會(huì)改變SREG,結(jié)果很可能是一個(gè)死循環(huán)出不來(lái)了。如果定義成volatile型變量,編譯的代碼是這樣的:每次要操作一個(gè)變量的時(shí)候都從
內(nèi)存中讀取一次。
#define SREG (*(volatile unsigned char *)0x5F) 之后,可以進(jìn)行如下基本操作,
unsigned char temp,*ptr;
temp=SREG;把SREG值保存到temp中
SREG=temp;把temp的值賦給SREG
ptr = & SREG; 不知對(duì)否,大家試一下。
-----------------------------------------------------------------------
對(duì)于(volatile unsigned char *)0x20我們分析一下,它是由兩部分組成:
1)
(unsigned char *)0x20,0x20只是個(gè)值,前面加(unsigned char
*)表示0x20是個(gè)地址,而且這個(gè)地址類型是unsigned char ,意思是說(shuō)讀寫(xiě)這個(gè)地址時(shí),要寫(xiě)進(jìn)unsigned char
的值,讀出也是unsigned char 。
2)volatile,關(guān)鍵字volatile 確保本條指令不會(huì)因C
編譯器的優(yōu)化而被省略,且要求每次直接讀值。例如用while((unsigned char
*)0x20)時(shí),有時(shí)系統(tǒng)可能不真正去讀0x20的值,而是用第一次讀出的值,如果這樣,那這個(gè)循環(huán)可能是個(gè)死循環(huán)。用了volatile
則要求每次都去讀0x20的實(shí)際值。那么(volatile unsigned char *)0x20
是一個(gè)固定的指針,是不可變的,不是變
量。而char *u則是個(gè)指針變量。再在前面加"*":*(volatile unsigned char
*)0x20則變成了變量(普通的unsigned char變量,不是指針變量),如果#define i (*(volatile
unsigned char *)0x20),那么與unsigned char i是一樣了,只不過(guò)前面的i的地址是固定的。
=》(*(volatile unsigned char *)0x20)可看作是一個(gè)普通變量,這個(gè)變量有固定的地址,指向0x20。而0x20只是個(gè)常量,不是指針更不是變量
|
|