写在前面
-----------------
在前段时间的工作中,就遇见过全局变量无法初始化的问题,不过之前是在一些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_C51STARTUPINITEND: LJMP MAIN......... ?C_START: MOV DPTR,#?C_INITSEGLoop: 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'段的最后一个位置....