单片机加入版本信息的常见方法

cathy的头像
cathy 发布于:周四, 02/01/2024 - 15:36 ,关键词:

我们公司的产品会根据客户需求和建议,不定期升级(优化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程序 

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

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

如下设置:

1.png

固件大小

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

2.png

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

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

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

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

3.png

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

4.png

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

来源:嵌入式专栏(作者 | strongerHuang)

免责声明:本文为转载文章,转载此文目的在于传递更多信息,版权归原作者所有。本文所用视频、图片、文字如涉及作品版权问题,请联系小编进行处理(联系邮箱:cathy@eetrend.com)。

围观 31