data,bdata,idata,pdata,xdata,code存储类型与存储区

发布者:breakthrough2最新更新时间:2024-08-22 来源: cnblogs关键字:data  bdata  idata  pdata  xdata  code  存储类型  存储区 手机看文章 扫描二维码
随时随地手机看文章

bit

是在内部数据存储空间中 20H .. 2FH 区域中一个位的地址,或者 8051 位可寻址 SFR 的一个位地址。

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

data

是在 0 到 127 之间的一个数据存储器地址,或者在 128 .. 255 范围内的一个特殊功能寄存器(SFR)地址。

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

xdata 是 0 to 65535 范围内的一个 xdata 存储器地址。


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

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

    data     --->    可寻址片内ram
    bdata    --->    可位寻址的片内ram
    idata    --->    可寻址片内ram,允许访问全部内部ram
    pdata    --->    分页寻址片外ram (MOVX @R0) (256 BYTE/页)
    xdata    --->    可寻址片外ram (64k 地址范围)
    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指针变量地址空间
    MOV A, #tmp(0x00)
    MOV @DPTR, A
    INC DPTR
    MOV A, #tmp(0x00)
    MOVX @DPTR, A

    这种方式一般用在内ram资源相对紧张而且对效率要求不高的项目中。

    第4种情况:

    uchar data * xdata pstr;
    pstr=tmp;

    如果详细看了第1种情况的读者发现这种写法和第1种很相似,是的,同第1 种情况一样这样也是有bug的,但是这次是把pstr分
配到了外ram区了。编译后的汇编代码如下。

    MOV DPTR, #0x000A        ;0x000A是在外ram区分配的pstr指针变量的地址空间
    MOV A, #tmp(0x00)
    MOVX @DPTR, A

    第5种情况:

    uchar * data pstr;
    pstr=tmp;

    大家注意到'*'前的关键字声明没有了,是的这样会发生什么事呢?下面这么写呢!对了用齐豫的一首老歌名来说就是 “请跟我
来”,请跟我来看看编译后的汇编代码,有人问这不是在讲C51吗? 为什么还要给我们看汇编代码。C51要想用好就要尽可能提升C51
编译后的效率,看看编译后的汇编会帮助大家尽快成为生产高效C51代码的高手的。还是看代码吧!

    MOV 0x08, #0X01            ;0x08-0x0A是在内ram区分配的pstr指针变量的地址空间
    MOV 0x09, #tmp(0x00)
    MOV 0x0A, #tmp(0x00)

    注意:这是新介绍给大家的,大家会疑问为什么在前面的几种情况的pstr指针变量都用2 byte空间而到这里就用3 byte空间了
呢?这是KeilC的一个系统内部处理,在KeilC中一个指针变量最多占用 3 byte空间,对于没有声明指针指向存储空间类型的指针,
系统编译代码时都强制加载一个字节的指针类型分辩值。具体的对应关系可以参考KeilC的help中C51 User's Guide。

    第6种情况:

    uchar * pstr;
    pstr=tmp;

    这是最直接最简单的指针变量声明,但他的效率也最低。还是那句话,大家一起说好吗!编译后的汇编代码如下。

    MOV DPTR, #0x000A        ;0x000A-0x000C是在外ram区分配的pstr指针变量地址空间
    MOV A, #0x01
    MOV @DPTR, A
    INC DPTR
    MOV DPTR, #0x000A
    MOV A, #tmp(0x00)
    MOV @DPTR, A
    INC DPTR
    MOV A, #tmp(0x00)
    MOVX @DPTR, A

    这种情况很类似第5种和第3种情况的组合,既把pstr分配在外ram空间了又增加了指针类型的分辨值。

    小结一下:大家看到了以上的6种情况,其中效率最高的是第2种情况,既可以正确访问ram区又节约了代码,效率最差的是第 6
种,但不是说大家只使用第2种方式就可以了,还要因情况而定,一般说来应用51系列的系统架构的内部ram资源都很紧张,最好大家
在定义函数内部或程序段内部的局部变量使用内ram,而尽量不要把全局变量声明为内ram区中。所以对于全局指针变量我建议使用第
3 种情况,而对于局部的指针变量使用第2种方式。

 

 

 

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,如果你明白它的特性 

 

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 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了


关键字:data  bdata  idata  pdata  xdata  code  存储类型  存储区 引用地址:data,bdata,idata,pdata,xdata,code存储类型与存储区

上一篇:51单片机串口通信原理讲解
下一篇:CS5532 C51驱动程序

推荐阅读最新更新时间:2024-11-13 01:42

TI最新的 Code Composer Studio™ IDE v6 上市
提供了全新的应用中心、简化的用户界面与智能学习工具。 德州仪器为开发人员提供了一个统一的集成开发环境 (IDE),从而能跨越其广泛的嵌入式产品组合来提高代码质量,缩短开发时间并提升客户体验 2014 年 5月 5 日,北京讯---日前,德州仪器 (TI) 推出其最新版(第 6 版)的 Code Composer Studio™ 集成开发环境 (IDE),旨在能始终如一地提供资源,使软件开发变得轻松并降低附随成本。基于广受欢迎、符合业界标准的最新版开源 Eclipse 软件框架,Code Composer Studio v6 可提供许多更新、功能和集成式工具,以便使软件开发体验更轻松。运行在 Windows 和 Linu
[嵌入式]
9500万Instagram账户是机器人?Ghost Data公布INS研究数据
据外媒报道,Ghost Data近日公布了他们对于Instagram平台用户的研究数据,他们发现,Instagram平台至少存在着9500万账户是机器人,也就是俗称的“僵尸号”。Instagram平台作为全世界风靡的社交APP,使用者已经达到了10亿人次,这次Ghost Data调查出的机器人账户占Instagram总账户的 9.5%。 本文引用地址:  http://www.21ic.com/news/ce/201807/806724.htm 为了进行研究,Ghost Data购买了20000个机器人,以分析它们的行为,之后才得出了上述结论。该公司将这些机器人的行为和约100万个账户进行了匹配,发现这些机器人十分活跃地关注In
[嵌入式]
高质低价趋势明显,主流协作机器人厂商产品性能与技术路线分析 | MIR DATABANK
协作机器人 是一种可以与人类在共享工作环境中进行物理交互的 机器人 ,它们设计灵活,安全易用,能够在保持人类与机器之间协同互动的同时提高生产效率,改变了传统生产线的固定作业模式。 近年来,随着智能制造和工业5.0的快速发展,协作机器人在汽车、电子、物流、 医疗 等行业都有着批量化应用。2021年以来,中国已成为全球最大的协作机器人市场,MIR DATABANK数据表明,2024-2026年协作机器人将迈向更快增长通道。各大机器人制造商纷纷推出自己的协作机器人产品,市场竞争日趋激烈,协作机器人在性能、功能和成本方面也在不断优化和提升。 (数据来源:MIR DATABANK) 目前,中国的协作机器人行业产品和技术呈现出哪些典型特征?
[机器人]
keil c51中定义XDATA,CODE类型的结构体方法
定义CODE型: typedef struct MenuItem { int i; unsigned char j; }code _MenuItem; //数据存放于FLASH中 定义XDATA型: typedef struct MenuItem { int i; unsigned char j; }xdata _MenuItem; //数据存放在外部RAM中 使用示例: _MenuItem m0_main ; //定义一个_MenuItem 类型的结构体数组
[单片机]
51单片机RAM 数据存储知识总结
1.RAM keil C语言编程 RAM是程序运行中存放随机变量的数据空间。在keil中编写程序,如果当前模式为small模式,如果总的变量大小未超过128B,则未初始化的变量的初值默认为0.如果所有的变量超过单片机small模式下的128B大小,则必须对变量进行初始化,否则超过RAM大小变量的值是不确定的,在small模式下超过128B大小的变量也必须在编译器中重新设定存储器的存储模式。 在keil中,可选择small,compact,large三种方式存储数据变量: 在keil中可以用“TargetOptions”来配置这一项: 图1 选择数据存储模式 2.片内数据存储区 (1) 工作寄存器区 工作
[单片机]
51单片机RAM 数据<font color='red'>存储</font><font color='red'>区</font>知识总结
小广播
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

 
EEWorld订阅号

 
EEWorld服务号

 
汽车开发圈

电子工程世界版权所有 京ICP证060456号 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved