keil c51中C程序的启动过程

发布者:平和的心态最新更新时间:2016-12-16 来源: eefocus关键字:keil  c51  C程序  启动过程 手机看文章 扫描二维码
随时随地手机看文章

汇编是从org 0000h开始启动,那么keil c51是如何启动main()函数的?keil c51有一个启动程序startup.a51,它总是和c程序一起编译和链接。下面看看它和main()函数是如何编译的;
//主函数如下;
void main(void)
{
    while (1)    这是个无条件空循环。    
    {
    }
}

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

?C_C51STARTUP       SEGMENT   CODE
?PR?main?TESTMAIN       SEGMENT CODE 

?STACK          SEGMENT   IDATA

                RSEG    ?STACK
                DS      1

                CSEG    AT      0
?C_STARTUP:     LJMP    STARTUP1 
C:0x0000    020003   LJMP     STARTUP1(C:0003)

          RSEG    ?C_C51STARTUP 
STARTUP1:                         ;该段程序把内存清零
;          MOV     R0,#IDATALEN - 1 
C:0x0003    787F     MOV      R0,#0x7F
;          CLR     A 
C:0x0005    E4       CLR      A
;          MOV     @R0,A 
IDATALOOP: 
C:0x0006    F6       MOV      @R0,A
;          DJNZ    R0,IDATALOOP 
C:0x0007    D8FD     DJNZ     R0,IDATALOOP(C:0006)
;          MOV     SP,#?STACK-1             ;设制CPU的堆栈起始地址
C:0x0009    758107   MOV      SP(0x81),#0x07
;          LJMP    ?C_START 
C:0x000C    02000F   LJMP     main(C:000F)

    RSEG  ?PR?main?TESTMAIN
main:
;      void main(void) 
C:0x000F    80FE     SJMP     main(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:0x0000    020027   LJMP     STARTUP1(C:0027)

     4: void main(void) 
     5: {unsigned int a,b,c; 
     6:         a=0x5500; 
C:0x0003    750855   MOV      0x08,#0x55    ;ram地址0x08和0x09存放变量a=0x5500。
C:0x0006    750900   MOV      0x09,#0x00
     7:         b=0xaa; 
C:0x0009    750A00   MOV      0x0A,#0x00    ;ram地址0x0A和0x0B存放变量b=0x00AA。
C:0x000C    750BAA   MOV      0x0B,#0xAA


     8:         while (1) 
     9:         { 
    10:                 c=SumXY(a,b); 
C:0x000F    AD0B     MOV      R5,0x0B        ;寄存器R4和R5传递变量a的值。
C:0x0011    AC0A     MOV      R4,0x0A
C:0x0013    AF09     MOV      R7,0x09        ;寄存器R6和R7传递变量b的值。
C:0x0015    AE08     MOV      R6,0x08
C:0x0017    120020   LCALL    SumXY(C:0020)    ;调用函数SumXY(a,b)求c=a+b
C:0x001A    8E0C     MOV      0x0C,R6        ;函数SumXY(a,b)返回的整型值存在R6和R7里,
C:0x001C    8F0D     MOV      0x0D,R7        ;把返回值存入变量c,ram地址0x0C和0x0D存放变量c
    11:         } 
    12: } 
    13:  
C:0x001E    80EF     SJMP     C:000F


    14: unsigned int SumXY(unsigned int X,Y) 
    15: {unsigned int Z; 
    16:         Z=X+Y; 
C:0x0020    EF       MOV      A,R7        ;参数变量X放在寄存器R6和R7里
C:0x0021    2D       ADD      A,R5        ;参数变量Y放在寄存器R4和R5里
C:0x0022    FF       MOV      R7,A
C:0x0023    EE       MOV      A,R6
C:0x0024    3C       ADDC     A,R4        ;计算Z=X+Y; 
C:0x0025    FE       MOV      R6,A        ;局部变量Z也放在寄存器R6和R7里
    17:         return Z;             ;由寄存器R6和R7里返回函数的值
C:0x0026    22       RET    

  
   151:                 MOV     SP,#?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:0x0027    75810D   MOV      SP(0x81),#0x0D
   155:                 LJMP    ?C_START 
C:0x002A    020003   LJMP     main(C:0003)


函数的入口地址,如何调用汇编函数,c和汇编的混合编程

/*******************************************************************/
c函数的函数名是一个指向函数的指针常量,它的值就是函数的入口地址。
从汇编程序上看,函数名也是该汇编函数代码段的入口地址标号。
调用汇编函数就是调用汇编函数的入口地址标号,但是要注意c函数名和汇编函数标号之间的转换规则。
1. 不带参数的汇编函数标号和c函数名相同.
2. 带参数的汇编函数标号在c函数名前加字符_,例如;如果c函数名是SumXY,汇编函数标号是_SumXY
3. 再入函数的汇编函数标号在c函数名前加_?,例如;如果c函数名是DoTask,汇编函数标号是_?DoTask


程序示例,该例有两个文件,一个文件是exam1.c,一个文件是funcasm.c
主程序文件: exam1.c
extern unsigned int SumXY(unsigned int X,Y);    //声明外部汇编语言函数,和声明c函数方法相同。
extern void Delay(unsigned char T);

void main(void)
{unsigned int a,b,c;
    a=0x5500;
    b=0x00aa;
    while (1)
    {
        Delay(100);
        c=SumXY(a,b);
    }
}

混合编程文件: funcasm.c

//c和汇编的混合编程演示.
//注意要把汇编语言函数放在文件前面。

//求Z=X+Y的汇编语言函数,Z,X,Y是整型数。
#pragma ASM
        PUBLIC  _SumXY
?PR?_SumXY?FUNCASM    SEGMENT CODE
        RSEG     ?PR?_SumXY?FUNCASM
_SumXY:        ;求Z=X+Y
       MOV      A,R7        ;参数X放在寄存器R6和R7里
       ADD      A,R5        ;参数Y放在寄存器R4和R5里
       MOV      R7,A
       MOV      A,R6
       ADDC     A,R4        ;扑鉠=X+Y; 
       MOV      R6,A        ;局部变量Z也放在寄存器R6和R7里
       RET 
#pragma ENDASM


//c语言函数,延时函数。
void Delay(unsigned char T)
{unsigned char i;
    for (i=0;i    {
        for (i=0;i    }
}

/**********************************************************************/
上面程序编译后的反汇编代码列表;
                 C_STARTUP:
C:0x0000    020041   LJMP     STARTUP1(C:0041)

                 main:
//给变量a和b赋值 a=0x5500;    b=0x00aa;
C:0x0003    750855   MOV      0x08,#0x55
C:0x0006    750900   MOV      0x09,#0x00
C:0x0009    750A00   MOV      0x0A,#0x00
C:0x000C    750BAA   MOV      0x0B,#0xAA

//调用延时函数Delay(100); 
C:0x000F    7F64     MOV      R7,#0x64
C:0x0011    120025   LCALL    DELAY(C:0025)

//求c=SumXY(a,b);
C:0x0014    AD0B     MOV      R5,0x0B
C:0x0016    AC0A     MOV      R4,0x0A
C:0x0018    AF09     MOV      R7,0x09
C:0x001A    AE08     MOV      R6,0x08
C:0x001C    12003A   LCALL    SUMXY(C:003A)    ;调用汇编语言函数SumXY(unsigned int X,Y)
C:0x001F    8E0C     MOV      0x0C,R6
C:0x0021    8F0D     MOV      0x0D,R7
C:0x0023    80EA     SJMP     C:000F


//c语言延时函数的反汇编代码
//void Delay(unsigned char T)
                 DELAY:
C:0x0025    E4       CLR      A
C:0x0026    FE       MOV      R6,A
                 C0001:
C:0x0027    EE       MOV      A,R6
C:0x0028    C3       CLR      C
C:0x0029    9F       SUBB     A,R7
C:0x002A    500D     JNC      C0007(C:0039)
C:0x002C    E4       CLR      A
C:0x002D    FE       MOV      R6,A
                 C0004:
C:0x002E    EE       MOV      A,R6
C:0x002F    C3       CLR      C
C:0x0030    9F       SUBB     A,R7
C:0x0031    5003     JNC      C0003(C:0036)
C:0x0033    0E       INC      R6
C:0x0034    80F8     SJMP     C0004(C:002E)
                 C0003:
C:0x0036    0E       INC      R6
C:0x0037    80EE     SJMP     C0001(C:0027)
                 C0007:
C:0x0039    22       RET     


//汇编语言函数SumXY(unsigned int X,Y)的反汇编代码,求Z=X+Y
                 SUMXY:
C:0x003A    EF       MOV      A,R7
C:0x003B    2D       ADD      A,R5
C:0x003C    FF       MOV      R7,A
C:0x003D    EE       MOV      A,R6
C:0x003E    3C       ADDC     A,R4
C:0x003F    FE       MOV      R6,A
C:0x0040    22       RET   

//程序启动代码;   
                 STARTUP1:
C:0x0041    75810D   MOV      SP(0x81),#0x0D
C:0x0044    020003   LJMP     main(C:0003)


关键字:keil  c51  C程序  启动过程 引用地址:keil c51中C程序的启动过程

上一篇:基于RS485总线的温湿度监控系统设计
下一篇:keil C51 指针总结

推荐阅读最新更新时间:2024-03-16 15:25

C51单片机学习——密码锁的实现
说在前面:因为我的板子是液晶屏和数码管不能同时用,所以下面我使用的是液晶屏,下面的代码通俗易懂也有注释,如果有的控件不知到如何使用的可以查看我之前的几篇C51文章,这个密码锁就是基于之前的笔记所写的内容 #include reg52.h //#include intrins.h #define uint unsigned int #define uchar unsigned char #define data1602 P0 #define LED P2 /*******引脚定义*******/ //按键的控制引脚 sbit S1 = P3^7; sbit S2 = P3^6; sbit S3 = P3^5; sbit S4
[单片机]
各种花样的流水灯c51程序
/*----------------------------------------------- 功能:流水灯对称移动闪烁(双闪烁) 作者:fei yu 日期:2010-6-14 ------------------------------------------------*/ #include REG52.H #define uint unsigned int void delay(uint); main() { uint comp1=0xfe; uint comp2=0x80; P1=0x7e; delay(30000); while(1) { P1=0xff;
[单片机]
Keil(MDK-ARM-STM32)系列教程(五)Configuration(Ⅰ)
Ⅰ、写在前面 本文带来的内容正如标题“Configuration”,只是标题包含的内容较多,我计划将其分为:Configuration(Ⅰ)和Configuration(Ⅱ)两篇文章来讲述。 我们常见的字体大小、颜色、关键字、快捷键等这些都是在“Configuration”中进行设置。本文讲述Configuration中的前面3项Editor、Colors & Fonts、User KeyWords。 本文讲述的Configuration配置和前面文章Options for Target目标选项在保存上有一个明显的区别:Configuration修改过后并保存的配置是保存在你Keil(电脑)软件上;而Options for
[单片机]
<font color='red'>Keil</font>(MDK-ARM-STM32)系列教程(五)Configuration(Ⅰ)
keil_提示出错 cannot load flash programming algorithm !
用STM32开发板调试例程时(用的是JLINK),提示出错 cannot load flash programming algorithm ! 遇到这种问题一般都是,Programming algorithm没有添加,或者添加错误, 添加步骤如下: 首先确认,下载器是stlink还是jlink,然后根据步骤1,2,3,4 选择合适的algorithm 就可以。 官方给出ARM调试提示解决方法: http://www.keil.com/support/docs/2747.htm RESOLUTION The programming algorithm depends on the device. For ex
[单片机]
<font color='red'>keil</font>_提示出错 cannot load flash programming algorithm !
STM32 IAR工程->Keil MDK转换详解
简介:我在STM32的学习中发现,大部分的STM32示例程序都是基于IAR开发环境的,但我认为使用Keil MDK开发环境更加方便,可以利用RVMDK强大的外设仿真功能加速STM32的开发。我在以前的Blog文章里介绍过如何在RVMDK中建立 ... 关键字: STM32 我在STM32的学习中发现,大部分的STM32示例程序都是基于IAR开发环境的,但我认为使用Keil MDK开发环境更加方便,可以利用RVMDK强大的外设仿真功能加速STM32的开发。我在以前的Blog文章里介绍过如何在RVMDK中建立STM32工程,以及如何使用RVMDK的软件仿真功能,下面我将详细说明怎样将已有的IAR工程移植到RVMDK。
[单片机]
用STM32Cube+FreeRTOS+Keil5对STM32F0编程之通过USART发送数据
STM32CubeMX的设置变化 Pinout 中使能 FreeRTOS Configuration 中配置 FreeRTOS 添加任务 生成代码 Keil5 中编辑代码 main 函数 int main(void) { /* USER CODE BEGIN 1 */ /* USER CODE END 1 */ /* MCU Configuration----------------------------------------------------------*/ /* Reset of all peripherals, Initializes the Flash interface and
[单片机]
用STM32Cube+FreeRTOS+<font color='red'>Keil</font>5对STM32F0编程之通过USART发送数据
C51单片机发光二极管实例程序集
1、 程序实现功能:让8个排列的LED来回流动着亮1—》...——》8 1《——...《——8 箭头代表LED流动方向,先从1-8然后从8到1依次循环。程序已经测试过,运行正常//大家如果用此代码测试的话注意你的LED接在单片机的那个端口,本代码用的是P2口 #include reg52.h #define uint unsigned int #define uchar unsigned char uchar a,b; uchar code led_array ={ 0xfe,0xfd,0xfb,0xf7,0xef,0xdf,0xbf,0x7f }; //led显示编码数组,从0xfe——0x7f为从第一个led管到第8个
[单片机]
8051体系中Keil C51中对双数据指针的支持情况及代码生成
在8051体系中,数据指针DPTR作为一个特殊的16位寄存器,用于寻址64 KB的XDATA或CODE空间,通常它被当作一个16位指针,指向一个常数表。双数据指针可以改善同时有两个16位指针使用时的性能。作为一种增强特性,有许多8051派生型器件支持双数据指针。以宏晶科技STC89系列的产品为例,DPTR被增强为DPTR0和DPTR1两个,仍然使用原来的地址,用另外一个SFR AUXR1的0位DPS来切换。当DPS位为0时,所有对DPTR的操作使用DPTR0;当DPS位为1时,所有对DPTR的操作使用DPTR1。这样,通过一个简单的INC AUXR1指令,就可以来回切换两个数据指针。 1 Keil C51对双数据指针的支持情况 作为
[单片机]
8051体系中<font color='red'>Keil</font> <font color='red'>C51</font>中对双数据指针的支持情况及代码生成
小广播
添点儿料...
无论热点新闻、行业分析、技术干货……
设计资源 培训 开发板 精华推荐

最新单片机文章
何立民专栏 单片机及嵌入式宝典

北京航空航天大学教授,20余年来致力于单片机与嵌入式系统推广工作。

换一换 更多 相关热搜器件
电子工程世界版权所有 京B2-20211791 京ICP备10001474号-1 电信业务审批[2006]字第258号函 京公网安备 11010802033920号 Copyright © 2005-2024 EEWORLD.com.cn, Inc. All rights reserved