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