Keil C51中变量和函数的绝对地址定位问题

时间:2023-03-08 23:51:08
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 文件中可以找到下面的内容:

.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