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 文件中可以找到下面的内容:
.obj TO Reader RAMSIZE () 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 . . ; RSEG ?XD? TEST .
//下面是各个类型的数据全局段的表示 ?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 // UINT8 NUMBER; UINT8 REC_BUF[]; #pragma udata