data
:固定指前面0x00-0x7f的128个RAM,可以用acc直接读写的,速度最快,生成的代码也最小。
idata
:固定指前面0x00-0xff的256个RAM,其中前128和data
的128完全相同,只是因为访问的方式不同。idata
是用类似C中的指针方式 访问的。汇编中的语句为:mox ACC,@Rx
。
xdata
:外部扩展RAM,一般指外部0x0000-0xffff空间,用DPTR访问。
pdata
:外部扩展RAM的低256个字节,地址出现在A0-A7的上时读写,用movx ACC,@Rx
读写。这个比较特殊,对于C51来说好象有BUG, 建议少用。
code
:访问程序存储器(默认仅限于标准51能够寻址的64kB空间内),生成的代码主要通过 MOVC@A+DPTR
实现。
far
:far
是Keil 编译器为了支持新出现的8051家族增强型MCU,这些增强型MCU可能拥有大于64KB的存储器。使用far
可访问扩展的RAM,使用**const far
可访问扩展的ROM**。NXP 51MX架构51单片机提供通过通用指针访问多达8MB的code和xdata存储空间。Dallas 390架构的51单片机通过24位的DPTR寄存器结合传统MOVX
、MOVC
指令来访问扩展的RAM和ROM。
注意:访问ROM只能读取不能修改。
例:将存储在RAM (20H)(21H)(22H)中的数据乘以2后存入(20H)(21H)(22H)(23H)
#include "reg52.h"
/************************************************************
Copyright (C), HAUT.
FileName: main.c
Author:JosephCooper Date:2020/6/23
Description:将存储在RAM (20H)(21H)(22H)中的数据乘以2后存入(20H)(21H)(22H)(23H)
Version:1.0
Function List:
1.void main();
History:none
***********************************************************/
void main()
{
unsigned long idata *pRam;
pRam = 0x20;
*pRam = 0x11223344;//假定RAM中的数据为0x11223344
*pRam = ((*pRam & 0xFFFFFF00)*2);
while(1);
}
//data内部128B地址空间
//idata全部256B地址空间
//xdata外扩64K地址空间
1.大端模式:BigEndian 大端模式第一个字节是最高位字节(按照从低地址到高地址的顺序存放数据的高位字节到低位字节),高字节在低地址, 低字节在高地址。
2.小端模式:LittleEndian 小端模式第一个字节是最低位字节(按照从低地址到高地址的顺序存放数据的低位字节到高位字节),高字节在高地址, 低字节在低地址。
例: 从内存地址为0x0000开始有以下数据:0x1234abcd
BigEndian
低地址 | 高地址 | ||
---|---|---|---|
0x0000 | 0x0001 | 0x0002 | 0x0003 |
0x12 | 0x34 | 0xab | 0xcd |
LittleEndian
低地址 | 高地址 | ||
---|---|---|---|
0x0000 | 0x0001 | 0x0002 | 0x0003 |
0xcd | 0xab | 0x34 | 0x12 |
使用绝对宏时,需要添加投文件“absacc.h”
,在该文件中定义的绝对宏有CBYTE、XBYTE、PWORD、DBYTE、CWORD、XWORD、PBYTE、DWORD
。
CBYTE
:对程序存储区(code)的字节地址进行访问。例如:i=CBYTE[0X000F];表示i指向程序存储区的地址为0x000F的存储单元,地址范围为0X0000~0XFFFF。
XBYTE
:对扩展RAM区的字节地址进行访问。例如:i=XBYTE[0X000F];表示i指向扩展RAM区的地址为0x000F的存储单元,地址范围为0X0000~0XFFFF。
PBYTE
:对扩展RAM区的字节地址进行访问。例如:i=PBYTE[0X000F];表示i指向扩展RAM区的地址为0x000F的存储单元,地址范围仅为一页(256字节)。
DBYTE
:对内部RAM区的字节地址进行访问。例如:i=PBYTE[0X000F];表示i指向内部RAM区的地址为0x000F的存储单元。
以WORD
表示的是为字操做,其余的跟以上相同。
代码及调试:
例:将存储在RAM (20H)(21H)(22H)中的数据乘以2后存入(20H)(21H)(22H)(23H)
#include "reg52.h"
#include
/************************************************************
Copyright (C), HAUT.
FileName: main.c
Author:JosephCooper Date:2020/6/23
Description:将存储在RAM(20H)(21H)(22H)中的数据乘以2后存入(20H)(21H)(22H)(23H)
Version:1.0
Function List:
1.void main();
History:none
***********************************************************/
void main()
{
unsigned long Res = 0;
unsigned char i = 0,j = 0,k = 0,l = 0;
unsigned long idata *pRam;
pRam = 0x20;
*pRam = 0x11223344;//假定RAM中的数据为0x11223344
i = DBYTE[0x20];j = DBYTE[0x21];k = DBYTE[0x22];l = DBYTE[0x23];
Res = (((((unsigned long)(i<<8)|j) << 16)|((unsigned long)(k<<8)|l))&0xFFFFFF00) *2;
DBYTE[0x20] = Res >> 24;
DBYTE[0x21] = Res >> 16;
DBYTE[0x22] = Res >> 8;
DBYTE[0x23] = Res;
while(1);
}
//CBYTE用来访问rom,用到movc指令。
//XBYTE用来访问扩展ram的,使用movx指令,dptr当指针。
//DBYTE用来绝对访问片内ram中的数据的。
//PBYTE也用来访问扩展ram,但只能访问开始的256字节,即movx指令,R0或R1当指针。
//这几个宏用于51单片机的绝对地址访问。上面4个是单字节访问,对应双字节访问用下面4个。
//CWORD、XWORD、DWORD、PWORD
//这些定义在头文件中,所以使用时要加入下面语句:
//#include
_at_
使用关键字“_at_
”不能对绝对变量进行初始化,位变量及函数不能用该关键字进行指定。使用方法为直接在定义的数据后边加上_at_
,在加上要指向的绝对地址即可。
例如:
unsigned char data i _at_ 0x0F;
表示i指向内部RAM区域地址为0x0f的单元;
unsigned char xdata i _at_ 0x0F;
表示i指向扩展RAM区域地址为0x0f的单元;
unsigned char xdata i[10] _at_ 0x0F;
表示数组的起始地址为扩展RAM区的0x0f单元。
注意:变量必须是全局变量
代码及调试:
例:将存储在RAM (20H)(21H)(22H)中的数据乘以2后存入(20H)(21H)(22H)(23H)
#include "reg52.h"
/************************************************************
Copyright (C), HAUT.
FileName: main.c
Author:JosephCooper Date:2020/6/23
Description:将存储在RAM(20H)(21H)(22H)中的数据乘以2后存入(20H)(21H)(22H)(23H)
Version:1.0
Function List:
1.void main();
History:none
***********************************************************/
unsigned char idata i[3] _at_ 0x20;//必须是全局变量
void main()
{
unsigned long idata *pRam;
unsigned long Res = 0;
pRam = 0x20;
*pRam = 0x11223344;//假定RAM中的数据为0x11223344
Res = (((((unsigned long)(i[0]<<8)|i[1]) << 16)|((unsigned long)(i[2]<<8)|i[3]))&0xFFFFFF00) *2;
i[0] = Res >> 24;
i[1] = Res >> 16;
i[2] = Res >> 8;
i[3] = Res;
while(1);
}