平时我们写程序,通常都会备注软件版本,那么,怎么在单片机中保存版本信息呢?

 

方法其实有很多,但基本原理都是在指定存储区域(Flash)中写入软件版本信息。

 

1、实现方法

下面就分享一个最常用,也是最基础的小技巧:在Keil MDK环境下,通过软件代码,直接映射到并存储到Flash指定地址。

 

包含:软件版本、编译日期、编译时间,代码如下:

 

#define VERINFO_ADDR_BASE   (0x0800FF00) //存放FLASH的地址
const char Software_Ver[] __attribute__((at(VERINFO_ADDR_BASE + 0x00)))  = "Software: 1.0.0";const char Compiler_Date[] __attribute__((at(VERINFO_ADDR_BASE + 0x40))) = "Date: "__DATE__;const char Compiler_Time[] __attribute__((at(VERINFO_ADDR_BASE + 0x60))) = "Time: "__TIME__;

 

这个代码大家能看懂么? 原理很简单,也有类似其他写入Flash地址的方法(这里暂不讲述)。
 

 

这里面包含几个重要知识点,下面给大家描述一下。

 

2、__attribute__ 语法

attribute,翻译为“属性”,在C语言中,是一个关键字,语法格式为:

__attribute__ ((attribute-list))

 

__attribute__ 可以设置函数属性(Function Attribute )、变量属性(Variable Attribute )和类型属性(Type Attribute )。

 

这部分内容,大家可以不用深入理解,知道这么用即可。要深入理解,网上也有很多学习资源。

 

3、C语言标准定义

在代码中:

const char Compiler_Date[] __attribute__((at(VERINFO_ADDR_BASE + 0x40))) = "Date: "__DATE__;const char Compiler_Time[] __attribute__((at(VERINFO_ADDR_BASE + 0x60))) = "Time: "__TIME__;

 

你会看到__DATE__ 和 __TIME__表示的日期和时间。 

 

其实,这两个是C语言特殊的标准定义。

 

__DATE__:编译时刻的日期字符串 如“Apr 13 2021”__TIME__:编译时刻的时间字符串  如”20:00:00“

 

除了这两个,其实还有很多类似的标准定义,比如:

 

__FILE__ :正在编译文件的文件名__LINE__ :正在编译文件的行号__STDC__:判断该文件是不是标准C程序

 

这部分内容,可以参看我的文章:C语言几种特殊标准定义和用法

 

4、总是编译版本文件

在Keil MDK中,默认情况下,源文件不修改,只编译一次。

 

因此,为了编译版本、日期和时间正确,需要进行设置:总是编译。

 

如下设置:

 

 

5、固件大小

生成的Hex文件会对没有使用的Falsh用0x00进行填充,比如:

 

 

填充0x00之后,这个hex就相对很大,因此,有两种方法减少hex固件大小。

 

1.存放FLASH的地址,要设置在合适的位置,如果代码量只有1K,你这只在偏移50K地址,这样偏移太多。

 

#define VERINFO_ADDR_BASE   (0x0800FF00) //存放FLASH的地址

 

2.网上还有一个方法,修改“ROM大小”:

 

 

该小之后,发现真的把0x00去掉了:

 

 

这两种方法,其实有一定风险的,如果代码量不断增加,可能会出现问题。所以,大家要主要设置Flash地址。