您的位置 首页 新品

keil c51是怎么发动c程序的

keil c51是如何启动c程序的-该程序有三个代码段;

第一个代码段?C_STARTUP在0x0000地址,是CPU第一条指令的入口,它只有一条长跳转指令,直接跳到第二个代码段。

第二个代码段?C_C51STARTUP是可重定位的段,该程序把内存清零,然后再设置CPU的堆栈,最后跳转到main()函数。

第三个代码段就是main()函数,在keil c51编译器里main()的段地址名就是?C_START。

汇编是从org 0000h开端发动,那么keil c51是怎么发动main()函数的?keil c51有一个发动程序startup.a51,它总是和c程序一同编译和链接。下面看看它和main()函数是怎么编译的;

 

//主函数如下;

void main(void)

{

while (1)这是个无条件空循环。

{

}

}

把上面的main()函数编译后的汇编程序和反汇编代码收拾后对照如下;

?C_C51STARTUPSEGMENTCODE

?PR?main?TESTMAINSEGMENT CODE

?STACKSEGMENTIDATA

RSEG?STACK

DS1

CSEGAT0

?C_STARTUP:LJMPSTARTUP1

C:0x0000020003LJMPSTARTUP1(C:0003)

RSEG?C_C51STARTUP

STARTUP1:;该段程序把内存清零

;MOVR0,#IDATALEN – 1

C:0x0003787FMOVR0,#0x7F

;CLRA

C:0x0005E4CLRA

;MOV@R0,A

IDATALOOP:

C:0x0006F6MOV@R0,A

;DJNZR0,IDATALOOP

C:0x0007D8FDDJNZR0,IDATALOOP(C:0006)

;MOVSP,#?STACK-1;设制CPU的仓库开始地址

C:0x0009758107MOVSP(0x81),#0x07

;LJMP?C_START

C:0x000C02000FLJMPmain(C:000F)

RSEG?PR?main?TESTMAIN

main:

;void main(void)

C:0x000F80FESJMPmain(C:000F);main()函数

现在剖析上面的汇编程序就会理解c51程序是怎么发动的。

该程序有三个代码段;

第一个代码段?C_STARTUP在0x0000地址,是CPU第一条指令的进口,它只要一条长跳转指令,直接跳到第二个代码段。

第二个代码段?C_C51STARTUP是可重定位的段,该程序把内存清零,然后再设置CPU的仓库,最终跳转到main()函数。

第三个代码段便是main()函数,在keil c51编译器里main()的段地址名便是?C_START。

还有一个IDATA数据段?STACK便是仓库,?STACK用于设制CPU的仓库开始地址,这是由keil编译器主动完结的。

keil c51函数的回来值是存储在r0-r7中的。

多字节变量在存储器里都是低地址存高位,高地址存低位。

main()函数的局部变量都是放在存储器里的,不象其他函数先选寄存器r0-r7寄存,假如不够用再存入存储器里。

看下面的示例;

c51程序;

unsigned int SumXY(unsigned int X,Y);

void main(void)

{unsigned int a,b,c;

a=0x5500;

b=0xaa;

while (1)

{

c=SumXY(a,b);

}

}

unsigned int SumXY(unsigned int X,Y)

{unsigned int Z;

Z=X+Y;

return Z;

}

编译后的反汇编代码列表;

C:0x0000020027LJMPSTARTUP1(C:0027)

4: void main(void)

5: {unsigned int a,b,c;

6:a=0x5500;

C:0x0003750855MOV0x08,#0x55;ram地址0x08和0x09寄存变量a=0x5500。

C:0x0006750900MOV0x09,#0x00

7:b=0xaa;

C:0x0009750A00MOV0x0A,#0x00;ram地址0x0A和0x0B寄存变量b=0x00AA。

C:0x000C750BAAMOV0x0B,#0xAA

8:while (1)

9:{

10:c=SumXY(a,b);

C:0x000FAD0BMOVR5,0x0B;寄存器R4和R5传递变量a的值。

C:0x0011AC0AMOVR4,0x0A

C:0x0013AF09MOVR7,0x09;寄存器R6和R7传递变量b的值。

C:0x0015AE08MOVR6,0x08

C:0x0017120020LCALLSumXY(C:0020);调用函数SumXY(a,b)求c=a+b

C:0x001A8E0CMOV0x0C,R6;函数SumXY(a,b)回来的整型值存在R6和R7里,

C:0x001C8F0DMOV0x0D,R7;把回来值存入变量c,ram地址0x0C和0x0D寄存变量c

11:}

12: }

13:

C:0x001E80EFSJMPC:000F

14: unsigned int SumXY(unsigned int X,Y)

15: {unsigned int Z;

16:Z=X+Y;

C:0x0020EFMOVA,R7;参数变量X放在寄存器R6和R7里

C:0x00212DADDA,R5;参数变量Y放在寄存器R4和R5里

C:0x0022FFMOVR7,A

C:0x0023EEMOVA,R6

C:0x00243CADDCA,R4;核算Z=X+Y;

C:0x0025FEMOVR6,A;局部变量Z也放在寄存器R6和R7里

17:return Z;;由寄存器R6和R7里回来函数的值

C:0x002622RET

151:MOVSP,#?STACK-1

152: ; This code is required if you use L51_BANK.A51 with Banking Mode 4

153: ; EXTRN CODE (?B_SWITCH0)

154: ;CALL?B_SWITCH0; init bank mechanism to code bank 0

C:0x002775810DMOVSP(0x81),#0x0D

155:LJMP?C_START

C:0x002A020003LJMPmain(C:0003)

声明:本文内容来自网络转载或用户投稿,文章版权归原作者和原出处所有。文中观点,不代表本站立场。若有侵权请联系本站删除(kf@86ic.com)https://www.86ic.net/xinpin/346185.html

为您推荐

联系我们

联系我们

在线咨询: QQ交谈

邮箱: kf@86ic.com

关注微信
微信扫一扫关注我们

微信扫一扫关注我们

返回顶部