新聞中心

        EEPW首頁 > 嵌入式系統(tǒng) > 設(shè)計應用 > 理解(*(volatile unsigned char *)0x5F

        理解(*(volatile unsigned char *)0x5F

        作者: 時間:2016-11-20 來源:網(wǎng)絡(luò) 收藏
        理解#define SREG (*(volatile unsigned char *)0x5F)

        這樣的定義,總是感覺很奇怪,不知道為什么,今天終于有了一點點心得,請大蝦們多多批磚~~~

        本文引用地址:http://www.104case.com/article/201611/318876.htm

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

        類似的,如果使用一個32位處理器,要對一個32位的內(nèi)存地址進行訪問,可以這樣定義#define RAM_ADDR (*(volatile unsigned long *)0x0000555F)
        然后就可以用C語言對這個內(nèi)存地址進行讀寫操作了
        讀:tmp = RAM_ADDR;
        寫:RAM_ADDR = 0x55;

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



        關(guān)鍵詞: volatileunsignedcha

        評論


        技術(shù)專區(qū)

        關(guān)閉
        主站蜘蛛池模板: 元朗区| 文登市| 江口县| 泽州县| 碌曲县| 藁城市| 汝城县| 江川县| 霍林郭勒市| 堆龙德庆县| 于都县| 平罗县| 日土县| 天柱县| 格尔木市| 宜良县| 青铜峡市| 自治县| 安乡县| 寿光市| 襄汾县| 尼木县| 进贤县| 藁城市| 义马市| 伊春市| 六枝特区| 达拉特旗| 白银市| 沛县| 延长县| 健康| 通许县| 惠东县| 德令哈市| 五台县| 明水县| 海城市| 宁都县| 七台河市| 罗田县|