微信公眾號 |?嵌入式專欄
我們公司的產(chǎn)品會根據(jù)客戶需求和建議,不定期升級(優(yōu)化bug、增刪功能),這個時候軟件版本就顯得很重要了。
不知道大家平時開發(fā)項目有沒有在軟件中加入版本信息?我看最近有小伙伴在討論相關(guān)問題,就來簡單分享一下。
方法其實有很多,但基本原理都是在指定存儲區(qū)域(Flash)中寫入軟件版本信息,這里講述其中一種比較常見的方法。
實現(xiàn)方法
本文分享一個常用,也是最基礎(chǔ)的小技巧:在Keil MDK環(huán)境下,通過軟件代碼,直接映射到并存儲到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語言中,是一個關(guān)鍵字,語法格式為:
__attribute__ ((attribute-list))
__attribute__ 可以設(shè)置函數(shù)屬性(Function Attribute )、變量屬性(Variable Attribute )和類型屬性(Type Attribute )。
這部分內(nèi)容,大家可以不用深入理解,知道這么用即可。要深入理解,網(wǎng)上也有很多學(xué)習(xí)資源。
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程序
這部分內(nèi)容,可以參看我的文章:C語言幾種特殊標準定義和用法
【總是編譯】版本文件
在Keil MDK中,默認情況下,源文件不修改,只編譯一次。
因此,為了編譯版本、日期和時間正確,需要進行設(shè)置:總是編譯。
如下設(shè)置:
固件大小
生成的Hex文件會對沒有使用的Falsh用0x00進行填充,比如:
填充0x00之后,這個hex就相對很大,因此,有兩種方法減少hex固件大小。
1.存放FLASH的地址,要設(shè)置在合適的位置,如果代碼量只有1K,你這只在偏移50K地址,這樣偏移太多。
#define VERINFO_ADDR_BASE (0x0800FF00) //存放FLASH的地址
2.網(wǎng)上還有一個方法,修改“ROM大小”:
該小之后,發(fā)現(xiàn)真的把0x00去掉了:
這兩種方法,其實有一定風(fēng)險的,如果代碼量不斷增加,可能會出現(xiàn)問題。所以,大家要主要設(shè)置Flash地址。
這里其實也有一個關(guān)于Hex文件格式的內(nèi)容,推薦閱讀文章:輕松理解bin、hex、axf和elf文件格式