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<T;i++)
{
for (i=0;i<T;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 中嵌入汇编以及C51与A51间的相互调用
如何在 KEIL C51(v6.21) 中调用汇编函数的一个示例 有关c51调用汇编的方法已经有很多帖子讲到,但是一般只讲要点,很少有对整个过程作详细描述,对于初学者是不够的,这里笔者通过一个简单例子 ...
- 关于Keil C51中using关键字的使用心得
刚才看到一位很牛的师兄写的一篇日志中提到了Keil C51中using这个关键字的用法,粗心的我本来一直都没有留意它是用来干嘛的(因为我一般看见它都是在中断服务函数的定义开头处,好像没有了它也可以中断 ...
- MFC程序的启动过程——先全局对象theApp(第一入口),后WinMain(真正入口),会引爆pApp->InitInstance从而创建窗口(程序员入口)
原文出自:http://blog.csdn.net/yuvmen/article/details/5877271 了解MFC程序的启动过程,对于初学者来讲,了学习MFC很有帮助:对于不常用VC的人来说 ...
- Info.plist和pch文件的作用,UIApplication,iOS程序的启动过程,AppDelegate 方法解释,UIWindow,生命周期方法
Info.plist常见的设置 建立一个工程后,会在Supporting files文件夹下看到一个“工程名-Info.plist”的文件,该文件对工程做一些运行期的配置,非常重要,不能删除 注:在旧 ...
- Keil C51中变量的使用
引言 8051内核单片机是一种通用单片机,在国内占有较大的市场份额.在将C语言用于51内核单片机的研究方面,Keil公司做得最为成功.由于51内核单片机的存储结构的特殊性,Keil C51中变量的使用 ...
- Keil C51中函数指针的使用
函数指针在C语言中应用较为灵活.在单片机系统中,嵌入式操作系统.文件系统和网络协议栈等一些较为复杂的应用都大量地使用了函数指针.Keil公司推出的C51编译器是事实上80C51 C编程的工业标准,它针 ...
- Android应用程序进程启动过程的源代码分析
文章转载至CSDN社区罗升阳的安卓之旅,原文地址: http://blog.csdn.net/luoshengyang/article/details/6747696 Android 应用程序框架层创 ...
- Keil C51 中指针的使用
指针是C语言中比较难的一个内容,Keil C51在指针方面有和标准C不一样的地方,今天看了一些资料学习了一下Keil C51 中指针的使用. keil51的指针,包含两种指针:普通指针,兼容标准C:内 ...
- Android应用程序进程启动过程(后篇)
前言 在前篇中我们讲到了Android应用程序进程启动过程,这一篇我们来讲遗留的知识点:在应用程序进程创建过程中会启动Binder线程池以及在应用程序进程启动后会创建消息循环. 1.Binder线程池 ...
随机推荐
- 10g中注意谓词过滤的位置
在10g中当主查询的谓词信息,被错误的放入子查询中,会导致子查询无法展开 explain plan for UPDATE DWF.F_PTY_INDIV O SET END_DT = TO_DATE( ...
- cf493D Vasya and Chess
D. Vasya and Chess time limit per test 2 seconds memory limit per test 256 megabytes input standard ...
- bzoj1681[Usaco2005 Mar]Checking an Alibi 不在场的证明
Description A crime has been comitted: a load of grain has been taken from the barn by one of FJ's c ...
- phpcms:六、频道页(category.html)
1.当前栏目的ID:{$catid}标题样式:{title_style($v[style])}(在添加内容或编辑内容的时候,标题右边 有一个选择颜色的块).{str_cut(strip_tags($v ...
- inconvertible types; cannot cast 'android.supoort.v4.app.Fragment' to 'com.example.sevenun.littledemo.fragment.NewsTitleFragment'
inconvertible types; cannot cast 'android.supoort.v4.app.Fragment' to 'com.example.sevenun.littledem ...
- hdu4055 dp
http://acm.hdu.edu.cn/showproblem.php?pid=4055 Problem Description The signature of a permutation is ...
- MongoDB 的 MapReduce 大数据统计统计挖掘
MongoDB虽然不像我们常用的mysql,sqlserver,oracle等关系型数据库有group by函数那样方便分组,但是MongoDB要实现分组也有3个办法: * Mongodb三种分组方式 ...
- wlan0 Interface doesn't support scanning : Device or resource busy
Problem: wlan0 Interface doesn't support scanning : Device or resource busy. Solved Way: sudo ifcong ...
- Java第四周学习日记(绪)
1.静态导入 作用:简化书写静态导入可以作用一个类的所有静态成员静态导入格式:import static 包名.类名静态导入要注意的事项:如果静态导入的成员与本类的成员存在同名的情况下,那么默认使用本 ...
- python - XML文件及其操作
xml文件也是实现不同语言或者程序之间进行数据交换的协议,它的特点是尖括号开头,尖括号结尾.使用范围就更为广泛了,tomcat resin kvm 等等,使用了大量的xml文件来进行相关配置.先来看一 ...