查看原文
其他

SMT32固件中引入版本信息的方法

strongerHuang strongerHuang 2022-09-10

关注+星标公众,不错过精彩内容

编排 | strongerHuang

微信公众号 | 嵌入式专栏


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


方法其实有很多,但基本原理都是在指定存储区域(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地址。


这里其实也有一个关于Hex文件格式的内容,推荐阅读文章:轻松理解bin、hex、axf和elf文件格式


------------ END ------------


后台回复『STM32』『Keil』阅读更多相关文章。


欢迎关注我的公众号回复“加群”按规则加入技术交流群,回复“1024”查看更多内容。

欢迎关注我的视频号:


点击“阅读原文”查看更多分享,欢迎点分享、收藏、点赞、在看。

您可能也对以下帖子感兴趣

文章有问题?点此查看未经处理的缓存