在我们使用kei c51创建一个51单片机项目时,会有如下图所示的提示:
keil创建新项目时,提示是否添加启动文件
一般情况下,需要选择“是”。当然,也可以选择不加。那么,这个启动文件的作用是什么?什么情况下需要加,什么情况下可以不加?
今天我们就来详细了解一下这个启动文件的内容,看明白这个内容后,我们就会有种恍然大悟的感觉:“哦,原来是这样啊!”
启动代码第一段
以下是启动代码原文第一段:
$nomod51;------------------------------------------------------------------------------; this file is part of the c51 compiler package; copyright (c) 1988-2005 keil elektronik gmbh and keil software, inc.; version 8.01;; *** <> ***;------------------------------------------------------------------------------; startup.a51: this code is executed after processor reset.;; to translate this file use a51 with the following invocation:;; a51 startup.a51;; to link the modified startup.obj file to your application use the following; lx51 invocation:;; lx51 your object file list, startup.obj controls;;------------------------------------------------------------------------------;; user-defined power-on initialization of memory;; with the following equ statements the initialization of memory; at processor reset can be defined:;; idatalen: idata memory size ; note: the absolute start-address of idata memory is always 0; the idata space overlaps physically the data and bit areas.idatalen equ 80h;; xdatastart: xdata memory start address ; the absolute start address of xdata memoryxdatastart equ 0 ;; xdatalen: xdata memory size ; the length of xdata memory in bytes.xdatalen equ 0 ;; pdatastart: pdata memory start address ; the absolute start address of pdata memorypdatastart equ 0h;; pdatalen: pdata memory size ; the length of pdata memory in bytes.pdatalen equ 0h;;
以下是启动代码第一段的翻译:
不使用预先定义的sfr。就是告诉汇编器不使用预定义的寄存器名,因为汇编器内部定义了51的寄存器名,但在实际使用时会用51的扩展芯片例如52之类的,如果包含了52的头文件就会出现重复定义所以要先声明一下不适用汇编器内部定义的寄存器名。
这个文件是c51编译器包的一部分
版权所有(c) 1988-2005 keil elektronik gmbh和keil software, inc。
版本8.01
*** > ***
----------------------------------------------------
startup.a51里面的代码在处理器复位后执行。
用下面的命令行语句调用a51进行编译产生目标文件,
a51 startup.a51
用下面的命令行语句调用bl51连接器把startup.obj目标文件连接到程序代码中,
lx51 invocation:
lx51调用
---------------------------------------------------
lx51 调用目标文件列表, 由startup.obj 目标文件控制
用户自定义上电后需要初始化的储存区域(初始化ram区的数据)
在处理器复位时通过下列equ伪指令来初始化内存(ram单元)
idatalen:idata存储区的大小,可以根据自己的选择修改
idata绝对的起始地址总是0
idata区涵盖data和bit区(data区(直接寻址区)以及 bit区 (位寻址区)),;至少要保证与c51编译器运行库有关的存储器的空间进行0初始化
xdata存储区的起始地址
xdata内存的绝对起始地址。
xdata存储器空间的绝对起始地址为0,
xdata空间的大小
xdata空间的长度以字节为单位
说明xdata的字节数清0,该值默认为0
pdata空间的大小
pdata存储器的空间的绝对起始地址
需用0进行初始化的pdata存储器的空间字节数
在51系列中data、idata、xdata、pdata的区别:
data:固定指前面0x00-0x7f的128个ram。
idata:固定指前面0x00-0xff的256个ram,其中前128和data的128完全相同,只是因为访问的方式不同。
xdata:外部扩展ram,一般指外部0x0000-0xffff空间。
pdata:外部扩展ram的低256个字节。
需用0进行初始化的idata存储器空间的字节数,idatalen只是一个标号(与idata不一样哦),equ只是做宏一样的替换,类似于c语言中的#define uint (unsigned int),以上的代码使得程序以后在碰到idatalen时替换成80h。
idatalen可以定义为你自己喜欢的名字如mydatalen等。之所以用idatalen,一是为了好记,二是为了表明和idata有关。
各种常数名及其含义
启动代码第二段
我们继续来看看51单片机的启动代码里面都有哪些东西。
下面先列出51单片机启动代码第二部分的原文:
;------------------------------------------------------------------------------;; reentrant stack initialization;; the following equ statements define the stack pointer for reentrant; functions and initialized it:;; stack space for reentrant functions in the small model.; ibpstack: enable small model reentrant stack; stack space for reentrant functions in the small model.ibpstack equ 0 ; set to 1 if small reentrant is used.; ibpstacktop: end address of small model stack ; set the top of the stack to the highest location.ibpstacktop equ 0xff +1 ; default 0ffh+1 ; ;; stack space for reentrant functions in the large model. ; xbpstack: enable large model reentrant stack; stack space for reentrant functions in the large model.xbpstack equ 0 ; set to 1 if large reentrant is used.; xbpstacktop: end address of large model stack ; set the top of the stack to the highest location.xbpstacktop equ 0xffff +1 ; default 0ffffh+1 ; ;; stack space for reentrant functions in the compact model. ; pbpstack: enable compact model reentrant stack; stack space for reentrant functions in the compact model.pbpstack equ 0 ; set to 1 if compact reentrant is used.;; pbpstacktop: end address of compact model stack ; set the top of the stack to the highest location.pbpstacktop equ 0xff +1 ; default 0ffh+1 ; ;
原文全是伪指令、宏定义这些东东,看起来确实很头疼啊。我们简单的翻译一下吧。
以下是第二段启动代码翻译:
再入函数模拟初始化;
以下用equ指令定义了再入函数模拟堆栈指针的初始化;
使用small存储器模式时再入函数的堆栈空间;
ibpstack equ 0 ; 使用small存储器模式再入函数时将其设置成1;
ibpstacktop equ 0ffh+1 ; 将堆栈顶设置为最高地址+1;
使用large存储器模式时再入函数的堆栈空间;
xbpstack equ 0 ; 使用large存储器模式再入函数时将其设置成1;
xbpstacktop equ 0ffffh+1; 将堆栈顶设置为最高地址+1;
使用compact存储器模式时再入函数的堆栈空间;
pbpstack equ 0 ; 使用compact存储器模式再入函数时将其设置成1;
pbpstacktop equ 0ffffh+1; 将堆栈顶设置为最高地址+1。
三种模式解析
这里提到了small,large,compact三种模式。这三种模式究竟有什么含义呢?我们下面就来了解一下。
不同内存模式下的堆栈。keil 编译器中有三种模式设置。这是由51处理器繁多的寻址模式导致的,不同的寻址模式有不同的效率。
small模式:在small模式中,所有默认变量均装入单片机内部的ram中,51单片机默认内部ram只有128b;52单片机默认256b。该模式下的优点是访问速度快,缺点是空间有限。
compact模式:在compact模式中,所有默认变量均位于单片机的256b ram中,和在small模式中使用关键字 pdata来定义数据变量的效果一样,在该模式下程序总变量空间不能超过256b。
large模式:在large模式中,所有默认变量可放在多达64kb的ram中,包括内部ram和外部ram,这和使用关键字xdata 来定义变量的效果一样。
small:变量存储在内部ram里。
compact:变量存储在外部ram里,使用页8位间接寻址。
large:变量存储在外部ram里,使用16位间接寻址。
我们一般使用small来存储变量,就是说单片机优先把变量存储在内部ram里,如果内部ram不够了,才会存到外部去。
compact的方式要自己通过程序来指定页的高位地址,编程比较复杂,如果外部ram很少,只有256个字节,那么对该256个字节的读取就比较快。如果超过256字节,那么要不断地进行切换的话,就比较麻烦。compact模式适用于比较少的外部ram的情况。
large模式,是指变量会优先分配到外部ram里。
3种存储方式都支持内部256字节和外部64k字节的ram。区别是变量的优先(或默认)存储在哪里的区别。除非你不想把变量存储在内部ram,才使用后面的compact、large模式。因为变量存储在内部ram里,运算速度比存储在外部ram要快的多,大部分的应用都是选择small的模式。
直播通道开启,OpenHarmony技术峰会邀您线上见
英伟达为其Holodeck平台增加了新工具,专门用于工程、建筑和施工
综合能源服务的定义是什么?知识体系架构是怎样的?
屏下指纹哪家强:安卓抢滩,三星领跑
iPhone12和三星S21,哪款手机更好?
单片机启动文件.s有什么作用?
ALSA高级Linux声音架构浅析
广西与百度智能交通合作共同发布 “高速公路车道级事件预警”功能
工业4.0如何使生活更美好
罗德与施瓦茨已连续参加四届进博会
Arm推出图像处理器Mali多媒体套件 为用户提供更酣畅的视觉体验
力矩电机控制器故障排除——三相输出不平衡怎么办(I)
人工智能芯片及其运算方法专利解析
荣耀9i玩游戏怎么样?华为荣耀9i体验分享
《毁灭战士3》登录steam 售价68人民币
高阶LCD技术加速应用到平板等
7000名员工无一感染,逆势增长25.3%!这家企业究竟做对了什么?
论小米、OPPO和vivo谁是耐用机皇
乐视发工资了!每天准时去乐视“睡觉上班”的讨债人看到了希望
这年代,刷抖音也能刷出逃犯,还有什么不可以吗?