Keil C51中变量和函数的绝对地址定位问题
扫描二维码
随时随地手机看文章
1、变量绝对地址定位
1) 在定义变量时使用 _at_ 关键字加上地址就可。
unsigned char idata myvar _at_ 0x40;
把变量 myvar 定义在 idata 的 0x40 处, 在 M51 文件中可以找到这麽一行
IDATA 0040H 0001H ABSOLUTE ;表示有变量在 idata 的 0x0040 处绝对地址定位.
2) 使用 KeilC 编译器定义绝对地址的变量, 方法待查.
2、函数绝对地址定位
1) 在程序中编写一函数 myTest
void myTest(void)
{
// Add your code here
}
2) 使用 KeilC 编译器定位绝对地址的函数,打开 Project -> Options for Target 菜单,选中 BL51 Locate 选项卡,在 Code: 中输入:?PR?myTest?MAIN(0x4000) 把函数 myTest 定位到程序区的 0x4000 处,再次编译就可以了。
3) 一次定位多个函数的方法
同样地, 在程序中再编写另外一个函数 myTest1
void myTest1(void)
{
// Add your code here
}
在 Options for Target 菜单的 BL51 Locate 选项卡的 Code: 中输入:?PR?myTest1?MAIN(0x3900), ?PR?myTest?MAIN(0x4000) 把函数 myTest1 定位在程序区的 0x3900 处, 把函数 myTest 定义在程序区的 0x4000 处,重新编译就可以了.
在 M51 文件中可以找到下面的内容:
复制代码
3.obj TO Reader RAMSIZE (256) CODE (?PR?MYTEST1?MAIN (0X3900), ?PR?MYTEST?MAIN (0X4000))
3665H 029BH *** GAP ***
CODE 3900H 0014H UNIT ?PR?MYTEST1?MAIN
3914H 06ECH *** GAP ***
CODE 4000H 0014H UNIT ?PR?MYTEST?MAIN
复制代码
4) 函数的调用
程序中直接调用函数的方式就不说明了, 这里重点讲使用函数指针调用绝对地址处的函数的方法.
(1) 定义调用的函数原形 typedef void (*CALL_MYTEST)(void); 这是一个回调函数的原形, 参数为空.
(2) 定义相应的函数指针变量 CALL_MYTEST myTestCall = NULL;
(3) 函数指针变量赋值,指向我们定位的绝对地址的函数 myTestCall = 0x3900; 指向函数 myTest1
(4) 函数指针调用
if (myTestCall != NULL)
{
myTestCall(); // 调用函数指针处的函数 myTest1, 置 PC 指针为 0x3900
}
检查编译生成的 bin 文件, 到 0x3900 处可以看到 myTest1 的内容,在 0x4000 处可以看到 myTest 的内容,
(5) 其它说明:如果在 0x3000 到 0x3900 的程序空间没有内容时,把 myTestCall 的地址指针指到 0x3800 (在 0x3000 到 0x3900 之间) 时, 会从 0x3900 处开始执行。至於在 Load 中调用 AP 中的函数的方法与此类似, 但是相应的参数传递可能要另寻方法.
段的定义
RSEG是段选择指令,要想明白它的意思就要了解段的意思。
段是程序代码或数据对象的存储单位。程序代码放到代码段,数据对象放到数据段。段分两种,一是绝对段,一是再定位段。绝对段在汇编语言中指定,在用L51联接的时候,地址不会改变。用于如访问一个固定存储器的i/o,或提供中断向量的入口地址。而再定位段的地址是浮动的,它的地址有L51对程序模块连接时决定,C51对源程序编译所产生的段都是再定位段,它都有段名和存储类型。绝对段没有段名。
说了这么多,大家可能还是不明白段是什么意思。别急,接着往下看。
例如,你写用C写了一个函数 void test_fun(void) { …},存在test.c中,用编译器编译以后.SRC FILE中看到,:
?PR?test_fun?TEST SEGMENT CODE //(函数放到代码段中)
写这个函数体的时候:
RSEG ?PR?test_fun?TEST //选择已定位的代码段为当前段 test_fun:
…… //代码
所以函数的表达模式是这样: ?PR?函数名?文件名
而函数名又分:
1:无参函数 ?PR?函数名?文件名
2:有参函数 ?PR?_函数名?文件名
3:再入函数 ?PR?_?函数名?文件名
又例如 你定义了全局变量
unsigned char data temp1,temp2;
unsigned char xdata temp3;
在test.c文件中,编译器会为每个文件分0到多个全局数据段,相同类型的全局变量被存到同一段中。所以上面会编译成如下:
复制代码
RSEG ? DT? TEST
. temp1: DS 1
. temp2: DS 1
;
RSEG ?XD? TEST
. temp3: DS 1
复制代码
复制代码
//下面是各个类型的数据全局段的表示
?CO? 文件名 //常数段
?XD? FILE_NAME //XDATA 数据段
?DT? FILE_NAME //DATA 数据段
?ID? FILE_NAME //IDATA…..
?BI? FILE_NAME //BIT …..
?BA? FILE_NAME //BDATA….
?PD? FILE_NAME //PDATA…..
复制代码
看到这里大家应该明白段的意思了吧。也许你会问,这有什么作用哪?它就是用在当你需要用汇编语言写一部份程序的时候,把汇编写的函数放在这个文件中,改名xxx.a51,按上面的规则写。编译就好。
既然知道了段的意思,现在我们回到SEG的用法上来。A51中有两种段选择指令:再定位段选择指令 和 绝对段选择指令,它们用来选择当前段是再定位段还是绝对段。使用不同的段选择指令,将使程序定位在不同的地址空间之内。
1、再定位段的选择指令是: RSEG 段名
它用来选择一个在前面已经定义过的再定位段作为当前段。用法就像我们上面的例子,先申明了一个函数段,后面写这个函数段。
2、绝对段选择指令
复制代码
CSEG [AT 绝对地址表达式] //绝对代码段
DSEG [AT 绝对地址表达式] //内部绝对数据段
XSEG [AT 绝对地址表达式] //外部绝对数据段
ISEG [AT 绝对地址表达式] //内部间接寻址绝对数据段
BSEG [AT 绝对地址表达式] //绝对位寻址段
复制代码
它们的用法,举一个例子:
例如我们写串口中断程序,起始地址是0x23.就这样写
CSEG AT 0X23
LJMP serialISR
RSEG ?PR?serialISR?TEST
. serialISR:
汇编函数使用同一个工程C文件中的变量,例如ICFLAG在C文件中定义,则汇编文件中定义方式为
EXTERN ICFLAG ;定义外部变量
定义函数,例如
复制代码
CARDATR:
...........
RET
GLOBAL CARDATR
复制代码
在同一个工程文件下调用汇编中的函数CARDATR,则应该定义函数
extern void CARDATR(void);
C18指定数据绝对地址
例如:
#pragma udata overlay RECBUFS =0x190 //200
UINT8 NUMBER;
UINT8 REC_BUF[31];
#pragma udata