keil 代码定位

11 篇文章 0 订阅
9 篇文章 0 订阅

在汇编中数据段和程序段都比较容易定位,例如程序1:

cseg    at 0x0000
ljmp    main

    dseg    at 0x20
aa:        ds    1

    iseg    at 0xC0
bb:        ds    1

    xseg    at 0x0010
cc:        ds    1

    cseg    at 0x1000
main:
mov        aa,#0
sjmp    main

    END

这段代码就把main定位到0x1000的位置。aa 定位在data(0x20)中,bb定位在idata(0xc0)中,CC定位到xdata(0x0010)当然,cc在不超过256字节的时候还可以用 page方式访问。

在C语言程序里面定位比较复杂。

常见的数组定位有如下几个方式:

1、使用 _at_ 例如程序2:

#include <reg51.h>

unsigned char data sysTemp_aa[20] _at_ 0x20;   //在data段中0x20开始的地方定义一个20字节的数组

unsigned char idata sysTemp_bb[20] _at_ 0x80; //在idata段中0x80开始的地方定义一个20字节的数组

unsigned char pdata sysTemp_cc[20] _at_ 0x0000; //在pdata段中0x00开始的地方定义一个20字节的数组

unsigned char xdata sysTemp_dd[20] _at_ 0x0100; //在xdata段中0x0100开始的地方定义一个20字节的数组

void main()
{
sysTemp_aa[0] = 0x55;
sysTemp_bb[0] = 0x55;
sysTemp_cc[0] = 0x55;
sysTemp_dd[0] = 0x55;

    while(1)
{
;
}
}

编译后生成的汇编代码如下:

C:0x0000    020015   LJMP     C:0015
    11: void main()
12: {

    13:         sysTemp_aa[0] = 0x55;
C:0x0003    752055   MOV      sysTemp_aa(0x20),#0x55
    14:         sysTemp_bb[0] = 0x55;
C:0x0006    7880     MOV      R0,#sysTemp_bb(0x80)
C:0x0008    7655     MOV      @R0,#0x55
    15:         sysTemp_cc[0] = 0x55;
C:0x000A    7800     MOV      R0,#sysTemp_cc(0x00)
C:0x000C    7455     MOV      A,#0x55
C:0x000E    F2       MOVX     @R0,A
    16:         sysTemp_dd[0] = 0x55;
17:
 
C:0x000F    900100   MOV      DPTR,#sysTemp_dd(0x0100)
C:0x0012    F0       MOVX     @DPTR,A
    18:         while(1)
C:0x0013    80FE     SJMP     C:0013
C:0x0015    787F     MOV      R0,#0x7F
C:0x0017    E4       CLR      A
C:0x0018    F6       MOV      @R0,A
C:0x0019    D8FD     DJNZ     R0,C:0018
C:0x001B    758193   MOV      SP(0x81),#0x93
C:0x001E    020003   LJMP     main(C:0003)

请注意加横线的是汇编代码对应的C程序。

更深入一 点,如果我想把main函数定位呢?上面的汇编代码是由keil自动定位的一个地址,添加或者删除程序后,main的地址是不确定的,能否在也使用 _at_呢?答案是否定的。要想把程序段绝对定位需要在keil里面采用一定的设置。

例如程序3:

#include <reg51.h>

unsigned char data sysTemp_aa[20] _at_ 0x20;

unsigned char idata sysTemp_bb[20] _at_ 0x80;

unsigned char pdata sysTemp_cc[20] _at_ 0x0000;

unsigned char xdata sysTemp_dd[20] _at_ 0x0100;

extern void test1();

void main()
{
sysTemp_aa[0] = 0x55;
sysTemp_bb[0] = 0x55;
sysTemp_cc[0] = 0x55;
sysTemp_dd[0] = 0x55;

    while(1)
{
test1();
}
}

void test1()
{
unsigned char i;
for(i=0;i<100;i++)
{
;
}
}

我想把main函数定位到0x1000开始的地放,test1定位到0x2000开始的地方

怎么 设置呢?

点击“project”->”option for target ..."弹出如下窗口:

 

选择BL51 MISC,在Misc controls的框中填入code (?PR?MAIN?TEST (0x1000),?PR?TEST1?TEST (0x2000))。这里我解释一下

code 表示程序段

?PR?各位可参考一下keil的帮助文件

?PR? 表示 program 意思是 Executable program code(可执行程序段)

MAIN 和 TEST1 就是你要定位的程序名称

?TEST是main和test1所在的文件名,我这里只是示例一下,命名不是很明确,在实际应 用中需要看情况而定。

好,我们编译一下,看看实际的效果。

 

再看看test1汇编程序

 

o(∩_∩)o...,正如自己所期望的那 样,main和test1均定位到了所需的地址。

快下班了,明天再来更深入的介绍。

如何定位整个文件,和keil创 建工程文件中的group。

 

继续昨天讲解的如 何定位子程序的话题。当程序变大,子程序很多,我想把整个文件定位到一个固定地址怎么办呢?难不成每一个文件都在BL51 MISC中设置吗?那手脖子都累掉了,有那功夫还不如仔细研究研究keil的帮助文件,看看它里面是如何说明的。

我仔细研究了一下keil的帮助文件,发现文件定位的方式有如下3种,且听我一一道来:

方法1 :这是最基本的方法,和昨天定位子程序的方法差不多,示例代码如下:

//test.c

#include <reg51.h>

unsigned char data sysTemp_aa[20] _at_ 0x20;

unsigned char idata sysTemp_bb[20] _at_ 0x80;

unsigned char pdata sysTemp_cc[20] _at_ 0x0000;

unsigned char xdata sysTemp_dd[20] _at_ 0x0100;

extern void test1();

extern void test2();

void main()
{
sysTemp_aa[0] = 0x55;
sysTemp_bb[0] = 0x55;
sysTemp_cc[0] = 0x55;
sysTemp_dd[0] = 0x55;

    while(1)
{
test1();

        test2();
}
}

void test1()
{
unsigned char i;
for(i=0;i<100;i++)
{
;
}
} void test2()
{
unsigned char i;
for(i=0;i<00;i++)
{
;
}
}

 

现在我想把整个test.c的程序定位到0x1200开始的地方,还使用昨天的那个方法,点击“project”->”option for target ..."弹出如下窗口:

 

在Misc controls中填入code(?PR?*?TEST (0x1200)),什么意思呢,就是test文件中所有的程序就定位到0x1200,呵呵,和昨天子程序定位相似吧,用一个*字符代替了所有的子程序名 称。好看一下实际的结果。

 

MAIN函数已经定位到了0x1200了,再看下面两个调用的函数test1和test2分别定位到了0x1218和0x121f,完全符 合要求

如果想定位其他文件怎么办?假设现在有一个test2.c文件我想把test2定位到0x2200开始的地方,只要稍微修改一下 code(?PR?*?TEST (0x1200), ?PR?*?TEST2(0x2200)), 各位看官是否注意到那个逗号',',用‘ ’把各个文件隔开。实际效果请自行实践,我就不再 一一列举。

方法2 :使用USERCLASS。USERCLASS官方解释是The USERCLASS directive assigns a user defined class name to a compiler generated segment. By default, the C51 Compiler uses the basic class name for segment definitions . 呵呵我的E文不是很好,只能看懂个大概,各位自己理解吧,想了解更多请参考keil的帮助文件。

不在废话了,进入正题。首先定义USRCLASS,不过第一步要设置一下编译器,点击“project”->”option for target ..."弹出如下窗口:勾选LX51和AX51 ,其实AX51不是必须的, 本帖只是讲C的定位,汇编程序不谈。

 

选择好了以后选择“LX51 Locate ",如图:

 

在"User classes"中填写" CODE_TEST (C:0x1500 - C:0x2000)",如果你对程序的大小不确定那么只要写一个" CODE_TEST (C:0x1500 )"就可以,其他的交给KEIL自动分配。

点击”确定“回到程序编辑区,在程序添加如下代码:#pragma userclass (code = test) //code的含义应该不用我说吧

使用#pragma参数,在文件头部定位整个文件,刚刚填写的(CODE_TEST ...)到了程序中就需要写 code = test,为什么这么写,我也不知道keil的规定。总结了一下,比如说我再定一个userClass,名称为 CODE_TEST1(C:0x2000),那么到了程序中我就要这么写:

#pragma userclass (code = test1)

引申一下,同理可以使用userclass来限制一个文件使用的RAM区域 稍许修改一下那个CODE,怎么改各位参照KEIL帮助文档自己思考。

实际修改后的源码:

//test.c

#include <reg51.h>

#pragma userclass (code = test)

unsigned char data sysTemp_aa[20] _at_ 0x20;

unsigned char idata sysTemp_bb[20] _at_ 0x80;

unsigned char pdata sysTemp_cc[20] _at_ 0x0000;

unsigned char xdata sysTemp_dd[20] _at_ 0x0100;

extern void test1();
extern void test2();

void main()
{
sysTemp_aa[0] = 0x55;
sysTemp_bb[0] = 0x55;
sysTemp_cc[0] = 0x55;
sysTemp_dd[0] = 0x55;

    while(1)
{
test1();
test2();
}
}

void test1()
{
unsigned char i;
for(i=0;i<100;i++)
{
;
}
}

void test2()
{
unsigned char i;
for(i=0;i<200;i++)
{
;
}
}

看看实际的效果:

 

main、test1和test2都已经定位到了1500开始的地方。

方法3 :和方法2差不多,也是使用userclass不过方法稍许有点差别

同样定义一个userclass

 

回到程序编辑区,右键test.c选择”Option for file test.c",弹出如下窗口:

 

在Misc Controls中填写:userclass (code = test),确定后回到程序编辑区

此时test.c文件的图标会有一个变化,如图:

image

编译源程序:

//test.c

#include <reg51.h>

unsigned char data sysTemp_aa[20] _at_ 0x20;

unsigned char idata sysTemp_bb[20] _at_ 0x80;

unsigned char pdata sysTemp_cc[20] _at_ 0x0000;

unsigned char xdata sysTemp_dd[20] _at_ 0x0100;

extern void test1();
extern void test2();

void main()
{
sysTemp_aa[0] = 0x55;
sysTemp_bb[0] = 0x55;
sysTemp_cc[0] = 0x55;
sysTemp_dd[0] = 0x55;

    while(1)
{
test1();
test2();
}
}

void test1()
{
unsigned char i;
for(i=0;i<100;i++)
{
;
}
}

void test2()
{
unsigned char i;
for(i=0;i<200;i++)
{
;
}
}

调试一下,看看结果是否和方法二相同?结果我就不展示了。

 

来自:http://hi.baidu.com/jonytsx/blog/item/090b73cce8a4751b00e92804.html

 

****************************************************分割线**********************************************

如果设置Keil从C代码编译出来的hex文件地址从0x8000开始

和MON51的设置一样,这样作:
1、把Startup.a51拷贝到工程目录加入工程,修改125行的
      CSEG    AT      0  为   CSEG    AT      0X8000  //这句话会将Startup.A51中的程序 从地址0x8000开始
2、菜单中选Project--Option for Target...,“C51”页下将中断偏移“Interrupt vectors at”改为0X8000 //这句话将中断服务程序放置在0x8000之后,各中断服务程序地址为0x8003+(中断号*8),
3、“BL51 Locate”页下在"CODE"后面的输入0X8000  //这句话将所有C程序地址放置在0x8000之后

4、重新编译工程

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值