当前位置:首页 > 单片机 > 单片机
[导读]汇编是从org 0000h开始启动,那么keil c51是如何启动main()函数的?keil c51有一个启动程序startup.a51,它总是和c程序一起编译和链接。下面看看它和main()函数是如何编译的;//主函数如下;void main(void){while (

汇编是从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)

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

嵌入式开发作为一种专业且技术密集型的领域,涵盖了从硬件底层驱动、中间件到应用层软件开发等多个层面的工作,其所需的工具种类繁多,各有针对性,旨在提升开发效率、保证代码质量以及简化调试过程。

关键字: 嵌入式开发 keil

单片机内部有很多的特殊功能寄存器,每个寄存器在单片机内部都分配有唯一的地址,一般我们会根据寄存器功能的不同给寄存器赋予各自的名称,当我们需要在程序中操作这些特殊功能寄存器时,必须要在程序的最前面将这些名称加以声明,声明的...

关键字: C51 数据类型 扩充定义

数据元(Data Element),也称为数据元素,是用一组属性描述其定义、标识、表示和允许值的数据单元,在一定语境下,通常用于构建一个语义正确、独立且无歧义的特定概念语义的信息单元。数据元可以理解为数据的基本单元,将若...

关键字: C51 数据类型

之后新建新的工程,添加.a文件就可以使用了,当然也可以使用keil来添加,但是keil默认的是用.lab,需要自己配置一下文件属性,改为lib文件即可。一半release sdk的时候用这种方式很关键的,毕竟自己的核心代...

关键字: keil 文件属性 lib文件

▼点击下方名片,关注公众号▼欢迎关注【玩转单片机与嵌入式】公众号,回复关键字获取更多免费资料。回复【加群】,限时免费进入知识共享群;回复【3D封装库】,常用元器件的3D封装库;回复【电容】,获取电容、元器件选型相关的内容...

关键字: C51 MDK RealView

分享这篇文章,谈一下STM32启动流程。如果读者朋友已经有过汇编相关基础,能够够好理解本文内容。汇编语言是比C语言更接近机器底层的编程语言,能让我们更好的理解和操纵硬件底层。STM32的三种启动模式下好程序后,重启芯片时...

关键字: STM32 启动过程 汇编代码

0xFFFF:0x00000xF000:0xE05B0x0000:0x7C00在第一篇文章中,我们就提到,现代操作系统是从最古老的8086系统一步一步发展而来的。处理器厂商为了向后兼容,很多底层相关的原理都是一样的(如果...

关键字: 启动过程

在Keil C51软件中51单片机的中断服务和外设驱动程序的开发

关键字: keil5 编译 C51

Intel公司1980年推出了MCS-51系列单片机:集成 8位CPU、4K字节ROM、128字节RAM、4个8位并口、1个全双工串行口、2个16位定时/计数器。寻址范围64K,并有控制功能较强的布尔处理器。 80C5...

关键字: C51 KEIL 编程

c上标3下标5怎么算用计算机,c上标3下标5怎么算

关键字: C51 KEIL
关闭
关闭