微波EDA网,见证研发工程师的成长!
首页 > 硬件设计 > 嵌入式设计 > C51中关于指针的种种用法

C51中关于指针的种种用法

时间:11-13 来源:互联网 点击:
#i nclude

f(){}
f1(){}
f2(){}

main()
{
{
int x;
int *px;

//下面这些表示虽然很烦,但是生成的代码却及其简洁:(黑体部分实际使用过)

//将 xdata 型指针 0x4000 赋给 px
px=(int xdata *)0x4000;

//表示从 xdata 0x4000处取一个 char 给x
x=*((char xdata *)0x4000); (可以将0X4000处,改成一个整形变量,方便进行操作。)

或者*((char xdata *)0x4000)=X;//表示给存在xdata中,地址为0x4000的空间赋值。

// 表示从 code 0x4000处取一个 word 作为 xdata 型的指针 给
px
px=*((int xdata * xdata *)0x4000);

//表示从 code 0x4000处取一个 word 作为 xdata 型的指针,
//再把这个指针指向的char数据赋给x
x=**((char xdata * code *)0x4000);

//表示把函数f()入口地址当作xdata型指针,再把指向的xdata
//中的int型数据作为code型指针,把指向的code字节
//赋给x(晕,这样有意义吗?)
x=**(int code * xdata *)f;

//把f()入口地址处的ROM中两个code型字节,
//赋给堆栈指针SP指向的字节(想干什么?编操作系统?)
*(unsigned int idata *)SP=*(unsigned int code *)&f;

//表示把f()入口地址处的ROM中两个code型字节,
//作为一个xdata指针寻址,
//把指向的数据作为pdata指针寻址,
//再把把指向的数据作为idata指针寻址,
//把该地址处的一个字节赋给x (我靠,累死了)
x= ****(unsigned int data * idata * pdata * xdata * code
*)&f;

//总之,一个括号里面外面的"*"一样多就表示指向的是数据。
}

{
//数组函数
code void (*ArrFn[])(void) =
{ &f1,
&f2,
};
//可以像引用数组一样调用函数啦:
(*ArrFn[0])();
(*ArrFn[1])();
}

{
//这样将使函数调用0000H处:
void (*reset) (void);
reset=0x0;
(*reset)();
reset();

//或者直接这样,仅仅生成一条指令LCALL 1234H
((void (code *)(void))0x1234)();
}

{
//这样可以调用RETI指令:
#define INT_NUM 30 //空闲中断号
((void (code *)(void))(INT_NUM*8+3))();
//当然需要在外面声明 int_rpt()interrupt INT_NUM {}
}
{
//这样调用RETI指令太变态:
code unsigned char ret_i=0x32;
((void (code *)(void))(&ret_i))();
}
}
int_rpt()interrupt INT_NUM {}

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

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

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;

这中情况也是对的

Copyright © 2017-2020 微波EDA网 版权所有

网站地图

Top