新聞中心

EEPW首頁 > 嵌入式系統(tǒng) > 設計應用 > 51系列中data,idata,xdata,pdata的區(qū)別

51系列中data,idata,xdata,pdata的區(qū)別

作者: 時間:2016-11-27 來源:網絡 收藏
dATa:固定指前面0x00-0x7f的128個RAM,可以用acc直接讀寫的,速度最快,生成的代碼也最小。

idATa:固定指前面0x00-0xff的256個RAM,其中前128和dATa的128完全相同,只是因為訪問的方式不同。idATa是用類似C中的指針方式 訪問的。匯編中的語句為:mox ACC,@Rx.(不重要的補充:c中idATa做指針式的訪問效果很好)

本文引用地址:http://m.butianyuan.cn/article/201611/322141.htm

xdATa:外部擴展RAM,一般指外部0x0000-0xffff空間,用DPTR訪問。

pdATa:外部擴展RAM的低256個字節(jié),地址出現(xiàn)在A0-A7的上時讀寫,用movx ACC,@Rx讀寫。這個比較特殊,而且C51好象有對此BUG,建議少用。但也有他的優(yōu)點,具體用法屬于中級問題,這里不提。

startup.a51的作用,和匯編一樣,在C中定義的那些變量和數(shù)組的初始化就在startup.a51中進行,如果你在定義全局變量時帶有數(shù)值,如unsigned char dATa xxx="100";,那startup.a51中就會有相關的賦值。如果沒有=100,startup.a51就會把他清0。(startup.a51==變量的初始化)。 這些初始化完畢后,還會設置SP指針。對非變量區(qū)域,如堆棧區(qū),將不會有賦值或清零動作。

有人喜歡改startup.a51,為了滿足自己一些想當然的愛好,這是不必要的,有可能錯誤的。比如掉電保護的時候想保存一些變量,但改startup.a51來實現(xiàn)是很笨的方法,實際只要利用非變量區(qū)域的特性,定義一個指針變量指向堆棧低部:0xff處就可實現(xiàn)。,為什么還要去改?可以這么說:任何時候都可以不需要改startup.a51,如果你明白它的特性。

bit
是在內部數(shù)據存儲空間中20H .. 2FH區(qū)域中一個位的地址,這在DATA的20H以后以字節(jié)形式出現(xiàn),可互相參照。另外加上8051可尋址 的SFR,但剛剛試過,只是00H--7FH起作用,也就是說當數(shù)據有變化時顏色變紅,以后的從80H到--FFH就不是位尋址區(qū)了,是位尋址的特殊寄存器,如涉及到了可位尋址的那11個當然會有反應。

復位后,程序計數(shù)器PC的內容為0000H,內部RAM各單元的值不確定。各功能寄存器的復位值如下:堆棧指針SP的復位值為07H,累加器ACC、寄存器B的復位值為00H,數(shù)據指針DPTR的復位值為0000H,而p0、p1、p2、p3四個口的復位值為0FFH。其他SFR如PSW、TCON、TMOD、TL0、TH0、TL1、TH1的復位值也為00H。

wave中是低128字節(jié)和高128字節(jié)(0-7FH),低128字節(jié)是片內RAM區(qū),高128字節(jié)(80-FFH)是SFR(特殊功能寄存器)bit則是位于低128字節(jié)的20H .. 2FH區(qū)域,即data的20H .. 2FH區(qū)域

code
是在0000H .. 0FFFFH之間的一個代碼地址。

我用
ORG5000H
TAB:DB22H,3BH,43H,66H,5H,6DH,88H后,
CODE從5000H開始以后變成DB各位


data

是在0到127之間的一個數(shù)據存儲器地址,或者加128 .. 255范圍內的一個特殊功能寄存器(SFR)地址。兩者訪問的方式不同。實際上由于PSW的復位設置PSW.3=RS0和PSW.4=RS1皆為0,所以通用工作寄存器區(qū)就是第0區(qū),所以data的00--07H部分是與REG欄中的R0--R7對應的。以后的則僅代表低128字節(jié)的內部RAM。


idata
是0 to 255范圍內的一個idata存儲器地址。

idata與data重合低128字節(jié),有的地方只有DATA表示256字節(jié)的片內RAM,
xdata是0 to 65535范圍內的一個xdata存儲器地址。

指針類型和存儲區(qū)的關系詳解

一、存儲類型與存儲區(qū)關系

data--->可尋址片內ram
bdata--->可位尋址的片內ram
idata--->可尋址片內ram,允許訪問全部內部ram
pdata--->分頁尋址片外ram (MOVX @R0) (256 BYTE/頁)
xdata--->可尋址片外ram (64k地址范圍FFFFH)
code--->程序存儲區(qū)(64k地址范圍),對應MOVC @DPTR

二、指針類型和存儲區(qū)的關系

對變量進行聲明時可以指定變量的存儲類型如:
uchar data x和data uchar x相等價都是在內ram區(qū)分配一個字節(jié)的變量。

同樣對于指針變量的聲明,因涉及到指針變量本身的存儲位置和指針所指向的存儲區(qū)位置不同而進行相應的存儲區(qū)類型關鍵字的
使用如:

uchar xdata * data pstr

是指在內ram區(qū)分配一個指針變量("*"號后的data關鍵字的作用),而且這個指針本身指向xdata區(qū)("*"前xdata關鍵字的作用),
可能初學C51時有點不好懂也不好記。沒關系,我們馬上就可以看到對應“*”前后不同的關鍵字的使用在編譯時出現(xiàn)什么情況。

......
uchar xdata tmp[10];//在外ram區(qū)開辟10個字節(jié)的內存空間,地址是外ram的0x0000-0x0009
......

第1種情況:

uchar data * data pstr;
pstr="tmp";

首先要提醒大家這樣的代碼是有bug的,他不能通過這種方式正確的訪問到tmp空間。 為什么?我們把編譯后看到下面的匯編
代碼:

MOV 0x08,#tmp(0x00);0x08是指針pstr的存儲地址

看到了嗎!本來訪問外ram需要2 byte來尋址64k空間,但因為使用data關鍵字(在"*"號前的那個),所以按KeilC編譯環(huán)境來說
就把他編譯成指向內ram的指針變量了,這也是初學C51的朋友們不理解各個存儲類型的關鍵字定義而造成的bug。特別是當工程中的
默認的存儲區(qū)類為large時,又把tmp[10]聲明為uchar tmp[10]時,這樣的bug是很隱秘的不容易被發(fā)現(xiàn)。

第2種情況:

uchar xdata * data pstr;
pstr = tmp;

這種情況是沒問題的,這樣的使用方法是指在內ram分配一個指針變量("*"號后的data關鍵字的作用),而且這個指針本身指向
xdata區(qū)("*"前xdata關鍵字的作用)。編譯后的匯編代碼如下。


上一頁 1 2 下一頁

評論


技術專區(qū)

關閉