关于XBYTE的使用

来源:互联网 发布:周末午夜光明知乎 编辑:程序博客网 时间:2024/05/23 18:56
原文地址:关于XBYTE的使用(收集)作者:Simba

The XBYTE macro accesses individual bytes in the external data memory of the 8051. You may use this macro in your programs as follows:
#include <absacc.h>
.rval = XBYTE [0x0002];
XBYTE [0x0002] = 57;
..This example reads and writes the contents of the byte in external data memory at address 0002h.
The range of valid index values for this macro is 0-65535.

FROM: http://www.keil.com/support/man/docs/c51/c51_xbyte.htm

//////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////

在一般的读写外部RAM的程序中,经常看到这样的句子:

   XBYTE[address]=data   写数据

   data=XBYTE[address]   读数据

采用XBYTE后,就不用顾及其时序,就是说,读写数据的时候,WR和RD怎么都不用用程序去控制。

读写外部RAM的程序,不需关注WR和RD端口的控制。

   外部总线由3组总线组成,数据 地址 控制,通常一般叫他外部总线,3组不同的信号,他们协调工作

一般情况由CPU特殊的外部数据访问指令,C51的MOVX指令(在C语言中他会编译成这个指令)协调3组线运行。

例如:

    mov dptr,#1000h

     mov a,#55h

     movx @dptr,a

上面3条语句的C语言可以表示如下:

   #define W_DATA XBYTE[0x1000]

    W_DATA=0X55;

在程序要满足时序工作条件下。使用外部总线的时候,数据 地址和控制信号是直接按照规定的时序输出高、低电平进行运行。

    例如:

#include<reg51.h>
#include<absacc.h>
#define ZXK XBYTE[0xfeff]
#define ZWK XBYTE[0xfdff]
void delay()
{
    unsigned int i=40000;
    while(i--);
}
void main()
{
    unsigned char i,j,a;
    while(1)
      { ZXK=0xbf;             //P2口赋值  P2最大为1011  1111=0xbf,
           for(i=4;i<8;i++)
          {
              ZXK=1<<i;
              delay();
           }
          for(j=4;j>0;j--)
          {  a=j-1;
             ZXK=1<<a;
             delay();
          }
        }
     }

      #define ZXK XBYTE[0xfeff]
      #define ZWK XBYTE[0xfdff]
      这里就是定义了一个字节的外部RAM,其地址为0xfeff。即将外部RAM中地址为0xfeff的字节定义为了ZXK这个名字。对它赋值即是对片外RAM的地址为0xfeff的字节赋值,不需对CPU端口P0、P2口进行设置。在程序执行过程中,访问这个外部RAM字节的时候,P0口是0xff,P2口是0xfe。根据这个地址来确定RAM中的哪个单元被选中。比如ZXK=0x00bf这条赋值是不行的,因为ZXK是字节变量,ZXK=0xbf这条赋值是对的,P2最大为1011 1111=0xbf,P2作用是将片外RAM的地址为0xfeff的字节赋值为0xbf。

 

例如:  #define DAC0832 XBYTE[0x7fff]是什么意思? 0x7fff是怎么得来的?如果p0口,p1口,p2口的接线有关,那又是怎么算的呢?


     XBYTE 的作用,可以用来定义P0和P2口绝对地址,其中P2口对应的是高位,P0口对应的是地位
     如 XBYTE[0x1234] = 0x56; 则等价于
         mov dptr,#1234h
         mov @dptr,#56h
    P0口和P2口共同作用,可以作为外部存储器的地址线 


    P2接地址总线的高8位(A15~A8),P0接地址总线的低8位(A7~A0)同时接数据总线,对片外地址由P2和P0构成,当dac0832的cs'接p2^7时低电平有效,所以P2最大为01111111=0x7f(最大为P2^6~P2^0=1),P0最大为11111111=0xff,所以为XBYTE[0x7fff]。若cs'接p2^6,P2最大为1011  1111=0xbf(最大为P2^7、P2^5~P2^0=1,P2^6=0),P0最大为11111111=0xff,则为XBYTE[0xbfff]。 

FROM:百度分享

//////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////////

如何理解#define XBYTE ((unsigned char volatile xdata * 
8051 特有的内存型态

code 以 MOVC @A+DPTR 读取的程序内存
data 可以直接存取的内部数据存储器
idata 以 Mov @Rn 存取的内部数据存储器
bdata 可以位寻址(Bit Addressable)的内部存储器
xdata 以 MOVX @DPTR 存取的外部数据存储器
pdata 以 MOVX @Rn 存取的外部数据存储器

特殊资料型态

bit 一般位(bit)变量
sbit 绝对寻址的位(bit)变量
语法
sbit my_flag = location; (location 范围从 0x00 ~ 0x7F)
范例
sbit EA = 0xAF;
或是配合 bdata 宣告的位(bit)变量
char bdata my_flags;
sbit flag0 = my_flags ^ 0;
(注意 sbit 前不可以加 static)

sfr 特殊功能缓存器(Special Function Register)
语法
sfr my_sfr = location; (location 范围从 0x80 ~ 0xFF)
范例
sfr P0 = 0x80;
指定绝对地址的变量 
在单一模块内可以使用下面的语法宣告
[memory_space] type variable_name _at_ location
范例
pdata char my_pdata _at_ 0x80;
如果该变量必须为多个模块所使用(Global Variable)则以
抽象指针(Abstract Pointer)的方式在标头档(Header File)定义较为方便。

#define variable_name *((data_type *) location)
范例
#define my_pdata *((char pdata *) 0x80)
(注意 char 与 pdata 的顺序)
ABSACC.H 提供了下列方便的宏(Macro)定义。 
#define CBYTE ((unsigned char volatile code *) 0)
#define DBYTE ((unsigned char volatile data *) 0)
#define PBYTE ((unsigned char volatile pdata *) 0)
#define XBYTE ((unsigned char volatile xdata *) 0)
#define CWORD ((unsigned int volatile code *) 0)
#define DWORD ((unsigned int volatile data *) 0)
#define PWORD ((unsigned int volatile pdata *) 0)
#define XWORD ((unsigned int volatile xdata *) 0)

隐藏的初始化程序 
80C51 在电源重置后(Power On Reset)所执行的第一个程序模块并不是使用者的主程序 
main(),而是一个隐藏在 KEIL-C51 标准链接库中称为 startup.a51 的程序模块。
startup.a51 的主要工作是把包含 idata、xdata、pdata 在内的内存区块清除为 0,并
且初始化递归指针。接着 startup.a51 被执行的仍然是一个隐藏在 KEIL-C51 标准链接库
中称为 init.a51 的程序模块。而 init.a51 的主要工作则是初始化具有非零初始值设定的
变量。 
在完成上述的初始化程序之后,80C51 的控制权才会交给 main() 开始执行使用者的程序。
#define XBYTE ((unsigned char volatile xdata *) 0)
定义 XBYTE 为 指向 xdata 地址空间unsigned char 数据类型的指针,指针值为0
这样,可以直接用XBYTE[0xnnnn]或*(XBYTE+0xnnnn)访问外部RAM了