写在前面
-----------------
在前段时间的工作中,就遇见过全局变量无法初始化的问题,不过之前是在一些C文件中定义的变量能初始化,而其他C文件中不能初始化,当时将这个问题绕过去了,并没有去深究...而这一周又出现了这个问题。于是就有了这篇文章
这里不去讨论其他情况下全局变量无法初始化的问题,只是针对我所遇见的问题讨论....
原因:由于引入库文件INIT.A51和STARTUP.A51导致的, 这里之所以要引入这两个文件,是由我们的应用环境决定的,如果在实际过程中没有手动引入这两个文件到工程项目中,则不会出现我这里提到的类似问题。
简述如下:
mcu复位后 PC=0 先执行STARTUP.A51的相关文件,该文件主要是对IDATA XDATA PDATA清零以及初始化堆栈指针;附上网络上一篇文章对其的讲解之部分关键代码:
; STARTUP.A51: 用户上电初始化程序
;------------------------------------------------------------------------------
;
; 用户定义需上电初始化的内存空间
;
; 使用以下EQU命令可定义在CPU复位时需用0进行初始化的内存空间
;
;;
; IDATA 存储器的空间的绝对起始地址总是0.;
IDATALEN EQU 80H ; 需用0进行初始化的IDATA存储器空间的字节数
;
XDATASTART EQU 0H ; XDATA存储器空间的绝对起始地址
XDATALEN EQU 400H ; 需用0进行初始化的XDATA存储器的空间字节数.
;
PDATASTART EQU 0H ; PDATA存储器的空间的绝对起始地址
PDATALEN EQU 0H ; 需用0进行初始化的PDATA存储器的空间字节数.
;
; 注意: IDATA 存储器的空间在物理上包括了8051单片机的DATA和BIT存储器空间.
; 听 说 至少要保证与C51编译器运行库有关的存储器的空间进行0初始化 不知是否
;------------------------------------------------------------------------------
;
; 再入函数模拟初始化
;
; 以下用EQU指令定义了再入函数模拟堆栈指针的初始化
;
; 使用SMALL存储器模式时再入函数的堆栈空间 .
IBPSTACK EQU 0 ; 使用SMALL存储器模式再入函数时将其设置成1.
IBPSTACKTOP EQU 0FFH+1 ; 将堆栈顶设置为最高地址+1.
;
; 使用LARGE存储器模式时再入函数的堆栈空间.; 使用LARGE存储器模式时再入函数的堆栈空间.
XBPSTACK EQU 0 ; 使用LARGE存储器模式再入函数时将其设置成1.
XBPSTACKTOP EQU 0FFFFH+1; 将堆栈顶设置为最高地址+1.
;
; 使用COMPACT存储器模式时再入函数的堆栈空间.; 使用COMPACT存储器模式时再入函数的堆栈空间.
PBPSTACK EQU 0 ; 使用COMPACT存储器模式再入函数时将其设置成1.
PBPSTACKTOP EQU 0FFFFH+1; 将堆栈顶设置为最高地址+1.
;
;------------------------------------------------------------------------------
;
; 使用COMPACT存储器模式时64K字节XDATA存储器空间的分页定义
;
; 以下用EQU指令定义PDATA类型变量在XDATA存储器空间的页地址
; 使用EQU指令定义PFAGE时必须与L51连接定位器PDATA指令的控制参数一致
;
PPAGEENABLE EQU 0 ; 使用PDATA类型变量时将其设置成1.
PPAGE EQU 0 ; 定义页号.
;
;------------------------------------------------------------------------------
.....
.....
.....
;
RSEG ?STACK ; 堆栈
;
DS 1
.....
.....
.....
CSEG AT 0x0000 ; 定义用户程序的起始地址
?C_STARTUP: LJMP STARTUP1
.....
.....
.....
; 设置堆栈的起始地址
MOV SP,#?STACK-1 ; 例如 MOV SP,#4FH;
.....
.....
.....
; 跳转到ININ.A51的初始化入口?C_START
LJMP ?C_START
END
由代码可知,在执行完清零操作后,程序将跳转到
?C_START
标号处,该程序标号定义在INIT.A51文件中,文件INIT.A51完成全局变量的初始化,其中包含有如下代码:
...
RSEG ?C_C51STARTUP
INITEND: LJMP MAIN
...
...
...
?C_START:
MOV DPTR,#?C_INITSEG
Loop:
WATCHDOG
CLR A
MOV R6,#1
MOVC A,@A+DPTR
JZ INITEND
INC DPTR
MOV R7,A
ANL A,#3FH
...
...
...
...
RSEG ?C_INITSEG
DB 0
...
...
其中
#?C_INITSEG
为编译生成的初始化段,该段保存了需要初始化的变量信息,可以看出上面的代码是利用该段的信息对相应变量进行初始化,注意其中的'
JZ INITEND
'(判断初始化段的相关位置为零则结束初始化);而INIT.A51文件的最后位置有'
RSEG ?C_INITSEG
';和'DB 0',所以问题的答案就在这里了!
问题出在INIT.A51在整个工程文件中的位置,假如在整个工程文件的最开始就编译INIT.A51,那么则'
?C_INITSEG
'第一个字节就是'0',所以接下来的初始化信息都不会被执行,解决办法就是INIT.A51最后才加入整个工程中,确保在该文件中加入'?C_INITSEG'段中的'0'是加在整个'?C_INITSEG'段的最后一个位置....