知之者不如好之,好之者不如乐之
分类: 嵌入式
2015-09-20 16:38:34
$NOMOD51 ;Ax51宏汇编器控制命令:禁止预定义的8051
;------------------------------------------------------------------------------
; This file is part of the C51 Compiler package
; Copyright (c) 1988-2002 Keil Elektronik GmbH and Keil Software, Inc.
;------------------------------------------------------------------------------
; STARTUP.A51: This code is executed after processor reset.
; STARTUP.A51: STARTUP.A51文件所生成的代码将在单片机复位后被执行!
; To translate this file use A51 with the following invocation:
; 将按照下面的命令行语句调用A51编译器进行编译产生目标文件
; A51 STARTUP.A51
;
; To link the modified STARTUP.OBJ file to your application use the following
; BL51 invocation:
; 将按照下面的命令行语句调用BL51连接器把STARTUP.OBJ定位连接到您的程序代码中
; BL51
;
;------------------------------------------------------------------------------
;
; User-defined Power-On Initialization of Memory
; With the following EQU statements the initialization of memory
; at processor reset can be defined:
; 使用下列EQU伪指令定义初始化的存储区域 在单片机复位后定义生效
; ; the absolute start-address of IDATA memory is always 0
IDATALEN EQU 80H ; the length of IDATA memory in bytes.
; IDATA(间接寻址区)其起始地址固定为0;IDATALEN用于指定需要初始化的IDATA区长度(以字节为单位)*
XDATASTART EQU 0H ; the absolute start-address of XDATA memory
XDATALEN EQU 0H ; the length of XDATA memory in bytes.
;XDATA (外部直接寻址区) XDATASTART用于指定需要初始化的XDATA区起始地址
;XDATALEN 用于指定需要初始化的XDATA区长度(以字节为单位)*
PDATASTART EQU 0H ; the absolute start-address of PDATA memory
PDATALEN EQU 0H ; the length of PDATA memory in bytes.
;PDATA(页寻址区) PDATASTART用于指定需要初始化的PDATA区起始地址
;PDATALEN用于指定需要初始化的;PDATA区长度(以字节为单位)*
; Notes: The IDATA space overlaps physically the DATA and BIT areas of the
; 8051 CPU. At minimum the memory space occupied from the C51
; run-time routines must be set to zero.
;注释:8051中 IDATA 区物理上已经包括了DATA区(直接寻址区)以及 BIT区 (位寻址区)。C51(库)占用了 最小化内存空间,运行时程序需要把它设为0
;------------------------------------------------------------------------------
;
; Reentrant Stack Initilization
; 重入堆栈初始化
; The following EQU statements define the stack pointer for reentrant
; functions and initialized it:
; 下面的EQU语句定义重入函数的堆栈指针并初始化它
; Stack Space for reentrant functions in the SMALL model.
; SMALL模式下的重入函数的堆栈空间
IBPSTACK EQU 0 ; set to 1 if small reentrant is used.
;如果再SMALL模式下使用重入则设为1
IBPSTACKTOP EQU 0FFH+1 ; set top of stack to highest location+1.
;设置堆栈顶 最高位置+1
;
; Stack Space for reentrant functions in the LARGE model.
; LARGE模式下的重入函数的堆栈空间
XBPSTACK EQU 0 ; set to 1 if large reentrant is used.
;如果再LARGE模式下使用重入则设为1
XBPSTACKTOP EQU 0FFFFH+1; set top of stack to highest location+1.
;设置堆栈顶 最高位置+1
;
; Stack Space for reentrant functions in the COMPACT model.
; COMPACT模式下的重入函数的堆栈空间
PBPSTACK EQU 0 ; set to 1 if compact reentrant is used.
;如果再COMPACT模式下使用重入则设为1
PBPSTACKTOP EQU 0FFFFH+1; set top of stack to highest location+1.
;设置堆栈顶 最高位置+1
;
;------------------------------------------------------------------------------
;
; Page Definition for Using the Compact Model with 64 KByte xdata RAM
; 使用COMPACT模式时为64KB的XDATA RAM定义页
; The following EQU statements define the xdata page used for pdata
; variables. The EQU PPAGE must conform with the PPAGE control used
; in the linker invocation.
; 下面的EQU语句定义PDATA变量的使用了XDATA页
PPAGEENABLE EQU 0 ; set to 1 if pdata object are used.
;如果使用PDATA页则设为1
;
PPAGE EQU 0 ; define PPAGE number.
;定义页号
;
PPAGE_SFR DATA 0A0H ; SFR that supplies uppermost address byte
;SFR的最高地址字节
; (most 8051 variants use P2 as uppermost address byte)
; (大多数8051变量要用P2的最高地址字节)
;------------------------------------------------------------------------------
; Standard SFR Symbols
; 标准SFR符号
ACC DATA 0E0H
B DATA 0F0H
SP DATA 81H
DPL DATA 82H
DPH DATA 83H
NAME ?C_STARTUP
?C_C51STARTUP SEGMENT CODE
?STACK SEGMENT IDATA
RSEG ?STACK
DS 1
EXTRN CODE (?C_START)
;外部代码(这个标号将代表用户程序的启始地址)
PUBLIC ?C_STARTUP
;给外部使用的符号
CSEG AT 0
;在code段的0地址处放以下代码(使用AT指令进行绝对地址的定位)
?C_STARTUP: LJMP STARTUP1
RSEG ?C_C51STARTUP
STARTUP1:
IF IDATALEN <> 0
;如果长度大于1则初始化IDATA
MOV R0,#IDATALEN - 1
CLR A
IDATALOOP: MOV @R0,A
DJNZ R0,IDATALOOP
ENDIF
IF XDATALEN <> 0
;如果长度大于1则初始化XDATA
MOV DPTR,#XDATASTART
MOV R7,#LOW (XDATALEN)
IF (LOW (XDATALEN)) <> 0
;预置初始化时的外循环次数到R6
MOV R6,#(HIGH (XDATALEN)) +1
ELSE
MOV R6,#HIGH (XDATALEN)
ENDIF
CLR A
XDATALOOP: MOVX @DPTR,A
INC DPTR
DJNZ R7,XDATALOOP
DJNZ R6,XDATALOOP
ENDIF
IF PPAGEENABLE <> 0
MOV PPAGE_SFR,#PPAGE
ENDIF
IF PDATALEN <> 0
;如果长度大于1则初始化PDATA
MOV R0,#LOW (PDATASTART)
MOV R7,#LOW (PDATALEN)
CLR A
PDATALOOP: MOVX @R0,A
INC R0
DJNZ R7,PDATALOOP
ENDIF
IF IBPSTACK <> 0
;SMALL模式下使用重入函数时要设置的堆栈
EXTRN DATA (?C_IBP)
MOV ?C_IBP,#LOW IBPSTACKTOP
ENDIF
IF XBPSTACK <> 0
;COMPACT模式下使用重入函数时要设置的堆栈
EXTRN DATA (?C_XBP)
MOV ?C_XBP,#HIGH XBPSTACKTOP
MOV ?C_XBP+1,#LOW XBPSTACKTOP
ENDIF
IF PBPSTACK <> 0
;LARGE模式下使用重入函数时要设置的堆栈
EXTRN DATA (?C_PBP)
MOV ?C_PBP,#LOW PBPSTACKTOP
ENDIF
MOV SP,#?STACK-1
; This code is required if you use L51_BANK.A51 with Banking Mode 4
; 如果你的程序使用了Mode 4 程序分组技术(BANKING)请启用下面的程序代码
; EXTRN CODE (?B_SWITCH0)
; CALL ?B_SWITCH0 ; init bank mechanism to code bank 0
;程序从第一个块(bank0)开始执行
LJMP ?C_START ;从这里跳到你的程序入口
END
;*******************************************************************
原来的工程中没有加入startup.A51文件,程序有时会出现乱码和指针溢出的情况导致系统异常。 加入此文件后,针对C8051F340单片机4K Xdata,我采样如下设置:
;
; Stack Space for reentrant functions in the LARGE model.
; LARGE模式下的重入函数的堆栈空间
XBPSTACK EQU 1 ; set to 1 if large reentrant is used.
;如果再LARGE模式下使用重入则设为1
XBPSTACKTOP EQU 0FFFH+1; set top of stack to highest location+1.
;设置堆栈顶 最高位置+1
系统已可以稳定运行!
$NOMOD51 ; Ax51宏汇编器控制命令,禁止预定义的8051。使编译器不使能预定义的;8051符号,避免产生重复定义的错误。
;------------------------------------------------------------------------------
; This file is part of the C51 Compiler package
; Copyright (c) 1988-2002 Keil Elektronik GmbH and Keil Software, Inc.
;------------------------------------------------------------------------------
; 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
; BL51 invocation:
;
; BL51 <your object file list>, STARTUP.OBJ <controls>
; BL51是Keil使用的链接器(Linker),这是命令行的使用格式,一般不用,使用IDE环境,
;用project管理,有相应的按钮可以实现该功能.
;------------------------------------------------------------------------------
;
; User-defined Power-On Initialization of Memory --- 初始化RAM单元
;
; With the following EQU statements the initialization of memory---用下面的EQU声明初
;始化ram单元
; at processor reset can be defined:
;
; ; the absolute start-address of IDATA memory is always 0
IDATALEN EQU 80H ; the length of IDATA memory in bytes.--根据你选用的芯片可以适
;当 的修改这些值 。IDATALEN 只是一个标号,EQU只是做宏一样的替换,类似于C语;言中的#define uint (unsigned int) ,以上的代码使得程序以后在碰到IDATALEN时替换;成80H
XDATASTART EQU 0H ; the absolute start-address of XDATA memory--以下
;两项根据目标系统的外设配置和连接自己修改
XDATALEN EQU 0H ; the length of XDATA memory in bytes.
PDATASTART EQU 0H ; the absolute start-address of PDATA memory
PDATALEN EQU 0H ; the length of PDATA memory in bytes.
;
; Notes: The IDATA space overlaps physically the DATA and BIT areas of the
; 8051 CPU. At minimum the memory space occupied from the C51
; run-time routines must be set to zero.
;------------------------------------------------------------------------------
;
; Reentrant Stack Initilization --注意:再入堆栈的方向区别于芯片自带的堆栈的生长方
;式,自顶向下生长的!而SP是是自底向上的!
; --且再入堆栈是由编译器自己管理的,一般不必去关心,只是在有再入函数的时候,根据
;函数的存储器模式使用相应的RAM空间做为再入堆栈。
; The following EQU statements define the stack pointer for reentrant
; functions and initialized it:
;Keil C默认情况不是用堆栈来传递参数的,所以造成函数不可重入,Keil要求用户显示声
;明函数是否具有可重入属性,以便为C函数调用初始化栈。
; Stack Space for reentrant functions in the SMALL model.
IBPSTACK EQU 0 ; set to 1 if small reentrant is used.
IBPSTACKTOP EQU 0FFH+1 ; set top of stack to highest location+1.
;
; Stack Space for reentrant functions in the LARGE model.
XBPSTACK EQU 0 ; set to 1 if large reentrant is used.
XBPSTACKTOP EQU 0FFFFH+1; set top of stack to highest location+1.
;
; Stack Space for reentrant functions in the COMPACT model.
PBPSTACK EQU 0 ; set to 1 if compact reentrant is used.
PBPSTACKTOP EQU 0FFFFH+1; set top of stack to highest location+1.
;不同内存模式下的堆栈。Keil 编译器中有三种模式设置:
;Small:所有的变量都放在内部RAM区
;Compact:所有变量在默认情况下都会放在外部RAM的低256字节中(可由R0寻址)
;Large:所有变量都放在外部RAM中(DPTR寻址)
;这是由51处理器繁多的寻址模式导致的,不同的寻址模式有不同的效率
;
;------------------------------------------------------------------------------
;
; Page Definition for Using the Compact Model with 64 KByte xdata RAM
;
; The following EQU statements define the xdata page used for pdata
; variables. The EQU PPAGE must conform with the PPAGE control used
; in the linker invocation.
;
PPAGEENABLE EQU 0 ; set to 1 if pdata object are used.
;
PPAGE EQU 0 ; define PPAGE number.
;
PPAGE_SFR DATA 0A0H ; SFR that supplies uppermost address byte
; (most 8051 variants use P2 as uppermost address byte) 很多的外部页面寻址以P2
;口为高位地址的数值,有使用外部页面RAM的情况
; 对PPAGEENABLE 设置为1 ,根据硬件连接修改PPAGE的值。
;------------------------------------------------------------------------------
; Standard SFR Symbols ---标准的SFR符号
ACC DATA 0E0H;关键字DATA A51伪指令定义单片机内部数据存储器字节地址的符号
B DATA 0F0H
SP DATA 81H
DPL DATA 82H
DPH DATA 83H
NAME ?C_STARTUP ;定义当前程序模块的目标模块名
?C_C51STARTUP SEGMENT CODE ;定义一个可再定位的段符号名和段所在的
;存储空间,汇编器产生的这个段符号名在BL51/L51连接定位时用
?STACK SEGMENT IDATA ;定义一个IDATA段,段名?STACK ,符合
;C51编译器的命名规则 (SEGMENT 用于定义一个段)
RSEG ?STACK ;声明当前段是IDATA段,段中保留空间
;RSEG伪指令用于选择一个事先用SEGMENT伪指令声明的普通段
DS 1 ; DS是预留空间定义指令
EXTRN CODE (?C_START) ;声明本模块引用的外部全局符号,
;用于和C相连接在.src文件中可以看到这个符号
PUBLIC ?C_STARTUP ;声明可被其他模块使用的全局符
;号,由.src文件中可以看出这个符号的作用。
CSEG AT 0 ;结束当前的IDATA段,产生一个位于
;CODE中新段,起始地址是0000H。代码段的起始点
?C_STARTUP: LJMP STARTUP1 ;C编译器编译源程序后,芯片复位之
;后的复位代码第一个就是执行这条语句。
RSEG ?C_C51STARTUP ;选择段名为?C_C51STARTUP
;的CODE段为当前段,存储程序代码。
STARTUP1:
IF IDATALEN <> 0 ;条件汇编指令,有IDATA区的话,清IDATA区。
MOV R0,#IDATALEN – 1 ;区域为0——IDATALEN-1
CLR A
IDATALOOP: MOV @R0,A
DJNZ R0,IDATALOOP
ENDIF ;(一)如果上;面idatalen=80H,那么是对0~7FH清零;如果你的程序是改写成:
;IDATALEN EQU 0100H ;
;就是对0~FFH清零。
; (二)二、如何按你意愿加载这段程序
;一般考虑到这个往往是你的设计中要区分上电复位和程序复位。有时候当程序复位时
;你不希望一些内存单元被清零了,那么你不对startup.a51作点修改,就不行了。
;默认是自动加载这段startup.a51的。
;所以你要这样做:
;把lib目录下的原始startup.a51文件拷到你的项目所在目录下,再把你项目目录下的
;这个startup.a51加入到你的项目中
;比如改成:
;IDATALEN EQU 00H ; the length of IDATA memory in bytes.
;然后编译链接。这样你的程序中就不会包含对idata清零的内码了。
;为什么?上面提到的IF语句的作用呀!当定义IDATALEN=0时,清零代码被跳过!
IF XDATALEN <> 0 ;如果有外部数据区,则把外部数据区中从XDATASTART到
;XDATASTART+ XDATALEN的区域清零
MOV DPTR,#XDATASTART
MOV R7,#LOW (XDATALEN)
IF (LOW (XDATALEN)) <> 0
MOV R6,#(HIGH (XDATALEN)) +1 ;如果低地址是零,一个
;高地址就代表256字节
ELSE
MOV R6,#HIGH (XDATALEN)
ENDIF
CLR A
XDATALOOP: MOVX @DPTR,A
INC DPTR
DJNZ R7,XDATALOOP
DJNZ R6,XDATALOOP
ENDIF
IF PPAGEENABLE <> 0 ;清外部页RAM区域
MOV PPAGE_SFR,#PPAGE ;给P2口赋相应的值,根据用
;户自己的目标系统。
ENDIF
IF PDATALEN <> 0 ;清外部页RAM区域
MOV R0,#LOW (PDATASTART)
MOV R7,#LOW (PDATALEN)
CLR A
PDATALOOP: MOVX @R0,A
INC R0
DJNZ R7,PDATALOOP
ENDIF
IF IBPSTACK <> 0 ;使用再入堆栈的情况,用户自己在程序中定义函数的存储模式。
; C51定义了三个全局变量,?C_IBP,?C_XBP,?C_PBP来存储再入堆栈的栈顶地址
EXTRN DATA (?C_IBP) ; 声明本模块使用的外部全局符号,符号的段类型限制了符号
;的使用范围,而符号本身则代表的是一个RAM单元的地址址
MOV ?C_IBP,#LOW IBPSTACKTOP
ENDIF
IF XBPSTACK <> 0 ;函数是Large存储模式的时候,存储再入堆栈的区域。
EXTRN DATA (?C_XBP) ;
MOV ?C_XBP,#HIGH XBPSTACKTOP
MOV ?C_XBP+1,#LOW XBPSTACKTOP
ENDIF
IF PBPSTACK <> 0 ;函数是Compact模式的时候,存储再入堆栈栈顶地址的存储单元
;和栈的利用空间
EXTRN DATA (?C_PBP)
MOV ?C_PBP,#LOW PBPSTACKTOP
ENDIF
MOV SP,#?STACK-1 ;定义的硬件栈的常数。区别再入堆
;栈和硬件栈。定义的段符号代表该段的首地址
; This code is required if you use L51_BANK.A51 with Banking Mode 4
#if 0
EXTRN CODE (?B_SWITCH0)
CALL ?B_SWITCH0 ; init bank mechanism to code bank 0
#endif
LJMP ?C_START ;把执行的权力交给C主函数。也就是;说指定函数的入口点。改句话结束以后将跳入C的main函数开始执行。
END
keil 版本:uVision 4
单片机是没有上操作系统的东西,在keil中编写的代码都是裸机代码,深入编写裸机代码有助于了解硬件的特性。
若不是硬件特性已定的情况之下的其它流程都是代码作祟。忽然想到来探探51单片机的执行流程。这个念头起源于最初见到每个51程序里面的主函数里面最终都挂一个while(1);语句。为何要加一句while死循环让程序停留在main函数中呢。将while(1);语句去掉有什么影响么?
写一个很简单的程序试一下。
执行以上程序,由P1端口控制的流水灯闪了一下。程序最终进入while(1);里纠缠去了,这个到好解释。
现将while(1);语句屏蔽掉。我还以为程序不能被正确执行了呢,因为退出了main主函数,就像Render需要循环来实现一样(尽管刚刚闪灯的程序不在循环之内,但我还是不由产生了这一错觉)。程序执行的结果是:流水灯不停的闪烁!
看到这个现象后的猜想及动作^-^:
(1) 这块板坏了吧!(在带操作系统如linux字符界面下运行一个不带死循环的C语言文件完毕后就会返回到linux shell程序中)。赶紧换个板再测试一下,显然还是一样的结果。
(2) 单片机中将一直执行main函数中的最后一个(些)语句?(基于带OS平台下运行标准C语言文件的经验,可从来没有想过是main函数被多次调用或多次进入)
(3) 单片机内将C语言指令取出来加载到单片机内,单片机内自动生成一个主程序循环执行C语言中main函数的内容?(虽然很荒唐,还是想了)
(4) 赶快谷歌百度一下单片机的执行流程(虽然在谷歌百度时以“51单片机程序执行流程”搜索,没有搜到相关内容)。换朴实的搜索词:“51单片机main”。然后就出现跟我一样带有疑问的问题:为什么main函数中不加while(1);语句之后程序会反复执行呢?回答的关键词包括“程序跑飞、看门狗、复位”。
(5) 趁上嵌入式的机会将“51单片机程序执行流程”搬出来并向老师讲述了我所写程序的得到的现象,包括我怎么验证呀等等。
老师的回答:Keil C51程序自动加载了一个名为”STARTUP.A51”的文件,在这个文件里面进行了一系列的初始化操作后进入用户编写的C语言程序入口main函数中,main函数执行完毕后,STARTUP.A51文件后有一句跳转到程序入口main函数的语句,所以会再次进入C语言主程序main函数中执行相关内容。
然后我用keil软件模拟了运行一下以上那一段代码:
程序开始运行就在程序入口main函数的第一条语句之处,Disassembly窗口是c语言代码与汇编代码相对应的窗口,前面是地址,后面的是C语言对应的汇编语句。下面的窗口是相应文件的运行代码的位置,由黄色箭头指向当前正要执行的代码。然后点击单步运行工具条,指导跳出main函数为止,程序跳转到STARTUP.A51中的以下代码位置:
继续点击单步调试直到进入一个循环中:
这里是一个循环,根据DJNZ指令的功能:每执行一次DJNZ RO, IDATALOOP就将R0的值减1,若R0的值不为0则就跳到IDATALOOP地址去。很显然这是一个循环,那么RO的值是多少呢,在以下窗口显示:
可见r0的初值为0x7f,这里将要循环0x7f(128)次,具体在这里r0值的含义可查看一下子的。那么在这个循环之后程序又将去哪里呢?跳过这个循环后程序运行的地方如下:
再单步运行一次:
根据Disassembly的内容,此条语句执行了就又要回到main函数中去了,执行一下试试:
是的!
所以,在51单片机中,程序的执行流程就是会不断( 以r0的值作为延迟条件, 具体含义可继续探索 )的进入main函数中执行main函数中的代码。
为什么我们在linux等上面运行不带死循环的C语言代码后程序就会自行终止呢?这是不同的操作流程:
(1) C51单片机不带OS(操作系统),代码的执行形势在此看来就由STARTUP.A51来安排了,没有一个更大的程序来管理怎么调用main函数。
(2) 像Linux这类的平台是带了OS的,运行一个C语言程序对linux来说就是一个任务,除了运行C语言程序这个任务外还有其它的任务。当运行一个C语言程序完毕时,此次的任务也算是完成了。如在linux shell界面运行一个文件名为“hello.c”功能为输出“hello world!”的C语言程序,过程如下:
编译:gcc hello.c –o hello
运行:./hello
在运行hello可执行文件时,可以当做是shell调用了hello这个可执行程序。在hello运行完毕后,将返回值等返回给shell界面。整个C语言文件的生死全有linux shell程序管理。
归其原因,还是代码规定的机制不一样吧。
此次笔记记录完毕。