理解(*(volatile unsigned char *)0x5F[轉]

理解#define SREG    (*(volatile unsigned char *)0x5F)
這樣的定義,總是感覺很奇怪,不知道爲什麼,今天終於有了一點點心得,請大蝦們多多批磚~~~

   嵌入式系統編程,要求程序員能夠利用C語言訪問固定的內存地址。既然是個地址,那麼按照C語言的語法規則,這個表示地址的量應該是指針類型。所以,知道要訪問的內存地址後,比如0x5F,
   第一步是要把它強制轉換爲指針類型
(unsigned char *)0x5F,AVR的SREG是八位寄存器,所以0x5F強制轉換爲指向
unsigned char類型。
   volatile(可變的)這個關鍵字說明這變量可能會被意想不到地改變,這樣編譯器就不會去假設這個變量的值了。這種“意想不到地改變”,不是由程序去改變,而是由硬件去改變——意想不到。
   第二步,對指針變量解引用,就能操作指針所指向的地址的內容了
   *(volatile unsigned char *)0x5F
   第三步,小心地把#define宏中的參數用括號括起來,這是一個很好的習慣,所以#define SREG    (*(volatile unsigned char *)0x5F)

    類似的,如果使用一個32位處理器,要對一個32位的內存地址進行訪問,可以這樣定義#define RAM_ADDR     (*(volatile unsigned long  *)0x0000555F)
    然後就可以用C語言對這個內存地址進行讀寫操作了
    讀:tmp = RAM_ADDR;
    寫:RAM_ADDR = 0x55;
zhiwei 發表於 2005-4-30 18:59 AVR 單片機

定義未volatile是因爲它的值可能會改變,大家都知道爲什麼改變了;
如果在一個循環操作中需要不停地判斷一個內存數據,例如要等待SREG的I標誌位置位,因爲SREG也是映射在SRAM空間,爲了加快速度,編譯器可能會編譯出這樣的代碼:把SREG讀取到Register中,然後不停地判斷Register相應位。而不會再讀取SREG,這樣當然是不行了,因爲程序或其它事件(中斷等)會改變SREG,結果很可能是一個死循環出不來了。如果定義成volatile型變量,編譯的代碼是這樣的:每次要操作一個變量的時候都從內存中讀取一次。
#define SREG (*(volatile unsigned char *)0x5F) 之後,可以進行如下基本操作,
unsigned char temp,*ptr;
temp=SREG;把SREG值保存到temp中
SREG=temp;把temp的值賦給SREG
ptr = & SREG; 不知對否,大家試一下。

 
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章