51中DATA区别

合集下载
  1. 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
  2. 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
  3. 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。

51系列中data,idata,xdata,pdata的区别

2011-05-28 12:58

dATa:固定指前面0x00-0x7f的128个RAM,可以用acc直接读写的,速度最快,生成的代码也最小。

idATa:固定指前面0x00-0xff的256个RAM,其中前128和dATa的128完全

相同,只是因为访问的方式不同。idATa是用类似C中的指针方式访问的。汇编

中的语句为:mox ACC,@Rx.(不重要的补充:c中idATa做指针式的访问效果

很好)

xdATa:外部扩展RAM,一般指外部0x0000-0xffff空间,用DPTR访问。

pdATa:外部扩展RAM的低256个字节,地址出现在A0-A7的上时读写,用movx ACC,@Rx读写。这个比较特殊,而且C51好象有对此BUG, 建议少用。但

也有他的优点,具体用法属于中级问题,这里不提。

startup.a51的作用,和汇编一样,在C中定义的那些变量和数组的初始化就在startup.a51中进行,如果你在定义全局变量时带有数值,如unsigned char dATa

xxx="100";,那startup.a51中就会有相关的赋值。如果没有=100,startup.a51就会

把他清0。(startup.a51==变量的初始化)。这些初始化完毕后,还会设置SP指针。对非变量区域,如堆栈区,将不会有赋值或清零动作。

有人喜欢改startup.a51,为了满足自己一些想当然的爱好,这是不必要的,有可能错

误的。比如掉电保护的时候想保存一些变量, 但改startup.a51来实现是很笨的方法,实际只要利用非变量区域的特性,定义一个指针变量指向堆栈低部:0xff处就

可实现。, 为什么还要去改? 可以这么说:任何时候都可以不需要改startup.a51,

如果你明白它的特性。

bit

是在内部数据存储空间中 20H .. 2FH 区域中一个位的地址,这在DATA的20H

以后以字节形式出现,可互相参照。另外加上8051 可寻址的SFR,但刚刚试过,只是00H--7FH起作用,也就是说当数据有变化时颜色变红,以后的从80H 到--FFH就不是位寻址区了,是位寻址的特殊寄存器,如涉及到了可位寻址的

那11个当然会有反应。

复位后,程序计数器PC的内容为0000H,内部RAM各单元的值不确定。各功

能寄存器的复位值如下:堆栈指针SP的复位值为07H,累加器ACC、寄存器

B的复位值为00H,数据指针DPTR的复位值为0000H,而p0、p1、p2、p3四

个口的复位值为0FFH。其他SFR如PSW、TCON、TMOD、TL0、TH0、TL1、TH1的复位值也为00H。

wave中是低128字节和高128字节(0-7FH),低128字节是片内RAM区,高128字节(80-FFH)是SFR(特殊功能寄存器)bit则是位于低128字节的20H .. 2FH 区域,即data的20H .. 2FH 区域

code

是在 0000H .. 0FFFFH 之间的一个代码地址。

我用

ORG 5000H

TAB: DB 22H,3BH,43H,66H,5H,6DH,88H后,

CODE从5000H开始以后变成DB各位

data

是在 0 到 127 之间的一个数据存储器地址,或者加 128 .. 255 范围内的一个特殊功能寄存器(SFR)地址。两者访问的方式不同。实际上由于PSW的复位设置PSW.3=RS0和PSW.4=RS1皆为0,所以通用工作寄存器区就是第0区,所以data的00--07H部分是与REG栏中的R0--R7对应的。以后的则仅代表低128字节的内部RAM。

idata

是 0 to 255 范围内的一个 idata 存储器地址。

idata与data重合低128字节,有的地方只有DATA表示256字节的片内RAM, xdata 是 0 to 65535 范围内的一个 xdata 存储器地址。

指针类型和存储区的关系详解

一、存储类型与存储区关系

data ---> 可寻址片内ram

bdata ---> 可位寻址的片内ram

idata ---> 可寻址片内ram,允许访问全部内部ram

pdata ---> 分页寻址片外ram (MOVX @R0) (256 BYTE/页)

xdata ---> 可寻址片外ram (64k 地址范围FFFFH)

code ---> 程序存储区 (64k 地址范围),对应MOVC @DPTR

二、指针类型和存储区的关系

对变量进行声明时可以指定变量的存储类型如:

uchar data x和data uchar x相等价都是在内ram区分配一个字节的变量。

同样对于指针变量的声明,因涉及到指针变量本身的存储位置和指针所指向的存储区位置不同而进行相应的存储区类型关键字的

使用如:

uchar xdata * data pstr

是指在内ram区分配一个指针变量("*"号后的data关键字的作用),而且这个指针本身指向xdata区("*"前xdata关键字的作用),

可能初学C51时有点不好懂也不好记。没关系,我们马上就可以看到对应“*”前后不同的关键字的使用在编译时出现什么情况。

......

uchar xdata tmp[10]; //在外ram区开辟10个字节的内存空间,地址是外ram的0x0000-0x0009

......

第1种情况:

uchar data * data pstr;

pstr="tmp";

首先要提醒大家这样的代码是有bug的, 他不能通过这种方式正确的访问到tmp空间。为什么?我们把编译后看到下面的汇编

代码:

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

看到了吗!本来访问外ram需要2 byte来寻址64k空间,但因为使用data关

键字(在"*"号前的那个),所以按KeilC编译环境来说

就把他编译成指向内ram的指针变量了,这也是初学C51的朋友们不理解各个

存储类型的关键字定义而造成的bug。特别是当工程中的

默认的存储区类为large时,又把tmp[10] 声明为uchar tmp[10] 时,这样的bug

是很隐秘的不容易被发现。

第2种情况:

uchar xdata * data pstr;

pstr = tmp;

这种情况是没问题的,这样的使用方法是指在内ram分配一个指针变量("*"

号后的data关键字的作用),而且这个指针本身指向

xdata区("*"前xdata关键字的作用)。编译后的汇编代码如下。

MOV 0x08,#tmp(0x00) ;0x08和0x09是在内ram区分配的pstr指针变量

地址空间

MOV 0x09,#tmp(0x00)

这种情况应该是在这里所有介绍各种情况中效率最高的访问外ram的方法了,请大家记住他。

第3种情况:

uchar xdata * xdata pstr;

pstr="tmp";

这中情况也是对的,但效率不如第2种情况。编译后的汇编代码如下。

MOV DPTR, #0x000A ;0x000A,0x000B是在外ram区分配的pstr指针变

相关文档
最新文档