术语简介
用Keil做51的开发也4年多了,代码量基本上 维持在5~10K左右,说大不大,说小也不小,也就是个中等货色。
这段期间工作上难得有稍许的空间,潜心研究了一下keil中如何在 CODE中定位C程序的方法。
在汇编中数据段和
程序段都比较容易定位,例如程序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_
#include
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
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++)
{
;
}
}