平时我们写程序,通常都会备注软件版本,那么,怎么在单片机中保存版本信息呢?
方法其实有很多,但基本原理都是在指定存储区域(Flash)中写入软件版本信息。
实现方法
下面就分享一个最常用,也是最基础的小技巧:在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地址的方法(这里暂不讲述)。
这里面包含几个重要知识点,下面给大家描述一下。
关键词__attribute__
attribute,翻译为“属性”,在C语言中,是一个关键字,语法格式为:
__attribute__ ((attribute-list))
__attribute__ 可以设置函数属性(Function Attribute )、变量属性(Variable Attribute )和类型属性(Type Attribute )。
这部分内容,大家可以不用深入理解,知道这么用即可。要深入理解,网上也有很多学习资源。
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语言几种特殊标准定义和用法
总是编译版本文件
在Keil MDK中,默认情况下,源文件不修改,只编译一次。
因此,为了编译版本、日期和时间正确,需要进行设置:总是编译。
如下设置:
固件大小
生成的Hex文件会对没有使用的Falsh用0x00进行填充,比如:
填充0x00之后,这个hex就相对很大,因此,有两种方法减少hex固件大小。
1.存放FLASH的地址,要设置在合适的位置,如果代码量只有1K,你这只在偏移50K地址,这样偏移太多。
#define VERINFO_ADDR_BASE (0x0800FF00) //存放FLASH的地址
2.网上还有一个方法,修改“ROM大小”:
该小之后,发现真的把0x00去掉了:
这两种方法,其实有一定风险的,如果代码量不断增加,可能会出现问题。所以,大家要主要设置Flash地址。
这里其实也有一个关于Hex文件格式的内容,推荐阅读文章:轻松理解bin、hex、axf和elf文件格式