微信公众号 | 嵌入式专栏
我们公司的产品会根据客户需求和建议,不定期升级(优化bug、增删功能),这个时候软件版本就显得很重要了。
不知道大家平时开发项目有没有在软件中加入版本信息?我看最近有小伙伴在讨论相关问题,就来简单分享一下。
方法其实有很多,但基本原理都是在指定存储区域(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文件格式