0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

讓代碼中包含最新的編譯時(shí)間信息

STM32嵌入式開發(fā) ? 來(lái)源:STM32嵌入式開發(fā) ? 2023-10-25 09:58 ? 次閱讀

如何保證發(fā)布出去的bin文件是最終測(cè)試通過(guò)的版本?

一般的來(lái)講,代碼到了測(cè)試后期,master分支就不會(huì)頻繁的提交了,并且提交也會(huì)更加謹(jǐn)慎。

但是人為操作總會(huì)出現(xiàn)紕漏,希望只要代碼被重新編譯過(guò),那么bin文件就包含新的時(shí)間信息,而這個(gè)信息是可以從外部通信或printf來(lái)查看的。

嵌入式開發(fā)中,版本號(hào)一般的都是一個(gè)int變量或字符串變量。但是若修改了代碼而沒(méi)有改version變量或宏定義,那么從version上就看不出來(lái)文件的變化。

那么最終編譯的版本到底是哪個(gè)版本,是否與測(cè)試的版本完全一致,這個(gè)問(wèn)題尤為突出。

目標(biāo)文件中帶有編譯時(shí)間可以防止代碼被改動(dòng)過(guò),只要代碼被重新編譯,那么就生成新的時(shí)間信息。

git能夠記錄文件修改信息,但是調(diào)試信息或工程配置等,很多文件都是ignore的,這些信息代表著最終的bin文件的運(yùn)行環(huán)境。

某些復(fù)雜bug情況下,只有運(yùn)行環(huán)境一致,仿真器才能attach到目標(biāo)文件。

如何獲取時(shí)間信息

這兩個(gè)宏是日期和時(shí)間,格式如下。如果把這兩個(gè)宏加入到代碼,那么就得到了時(shí)間的字符串信息。



// Example of __DATE__ string: "Dec 27 2017"
// Example of __TIME__ string: "1519"
constchar*BuildInfo="Version:"VERSION""__DATE__""__TIME__;

代碼實(shí)現(xiàn)獲取日期和時(shí)間的方法很多,比如:

unsigned int mk_Build_Date(void)
{
    int    year = 0, month = 0, day = 0;
    int hour = 0, minute = 0, seconds = 0;
    char m[4] = {0};


    sscanf(__DATE__, "%3s %2d %4d", m, &day, &year);


    for (month = 0; month < 12; month++)
    {
        if (strcmp(m, short_char_months[month]) == 0)
        {
            break;
        }
    }


    sscanf(__TIME__, "%2d:%2d:%2d", &hour, &minute, &seconds);


    #ifdef SHORT_DATA_CHAR__
        printf("[null]  ** Build at:	%04u-%02u-%02us %02u:%02u:%02u
",
                year, month, day,
                hour, minute,seconds);


    #else
        printf("[null]  ** Build at:	%04u-%02u-%02u %02u:%02u:%02u
",
                year, month, day,
                hour, minute,seconds);
    #endif


    DEBUG("buildDate: %s %s
", __DATE__, __TIME__);


    return 0;
}

把上面的函數(shù)加入到代碼中,就能獲取工程編譯的時(shí)間。

但是如果該代碼所在的文件沒(méi)有被修改,在非build-all情況下,編譯器不會(huì)再次編譯此文件,所以時(shí)間信息也就不會(huì)被更新。

如果每次都使用re-build all,一來(lái)繁瑣,二來(lái)也不能保證每次都會(huì)記得點(diǎn)擊build all按鈕,靠技術(shù)手段來(lái)保證每次build都更新時(shí)間信息才是正道。

如何保證每次編譯都更新時(shí)間信息

使用預(yù)編譯指令,每次更新包含時(shí)間宏的文件或?qū)?yīng)的鏈接文件。

在IAR環(huán)境下,官方已經(jīng)給出了解決的方法(Using pre-build actions for time stamp

方法1:修改文件的時(shí)間,引起編譯器對(duì)文件進(jìn)行重新編譯。


cmd /c "touch /cygdrive/d/test.c"
方法雖好,可惜IAR用戶大多數(shù)是Windows用戶,包括我在內(nèi),touch是linux命令,必須Cywin環(huán)境。如果安裝過(guò)這個(gè)環(huán)境的話,那就大功告成了。

Cygwin touch command


You can enter "cygwin-application.exe" on the pre- and post-build command lines, if the environment variable PATH includes the directory where the "cygwin-application.exe" is located.


You can run the Cygwin command "touch" on the pre-build command line, but if you add a file path, for example "touch d:/test.c", the file path is not accepted by Cygwin.


Cygwin expects the POSIX path /cygdrive/d/test.c so the resulting command line would be "touch /cygdrive/d/test.c", however this command cannot be executed directly on the pre- and post-build command. Instead you have to run indirectly using:


cmd /c "touch /cygdrive/d/test.c"
The .bat file (located in project directory) alternative would look like:


Pre-build command line:


 $PROJ_DIR$pre-build.bat
File pre-build.bat:


 touch /cygdrive/d/test.c
方法2:修改文件對(duì)應(yīng)的鏈接文件,觸發(fā)編譯器重新編譯該文件,生成新的鏈接文件,那么就會(huì)生成新的帶有時(shí)間信息的目標(biāo)文件。

An alternative to the "touch" command is to have a pre-build action that deletes the object file, for example the Pre-build command line:


cmd /c "del "$OBJ_DIR$	est.o""

在pre-build中加入上面的命令,就會(huì)在編譯前刪除test.o文件。

在這種模式下,工程代碼只要任何位置發(fā)生變化,代碼重新編譯,就會(huì)觸發(fā)刪除test.o,然后鏈接過(guò)程發(fā)現(xiàn)沒(méi)有test.o文件,那么就會(huì)重新編譯一次test.c,那么新的時(shí)間信息就會(huì)記錄下來(lái)了。

雖有些曲線救國(guó)的味道,但還是很順利的實(shí)現(xiàn)了目標(biāo)。

只要工程的任何地方有改動(dòng),生成新的目標(biāo)文件,那么目標(biāo)文件中就會(huì)帶有最新的編譯時(shí)間。

方法3:直接告訴編譯器每次重新編譯某個(gè)文件更直接,MDK支持此功能。

實(shí)際上,如果對(duì)工具多一些了解,萬(wàn)萬(wàn)是不會(huì)用上面的方法的,當(dāng)然上面的方法也是通用想法,是通用型知識(shí)點(diǎn),容易想到,也能達(dá)到目標(biāo)。

新的方法,不需要寫任何腳本,如果想讓代碼每次都編譯更新DATA 和 TIME兩個(gè)宏,那么讓這個(gè)文件每次都編譯一次就可以了,不需要?jiǎng)h除它的obj文件然后讓編譯器找不到文件而觸發(fā)重新編一次,其實(shí)直接告訴編譯器每次重新編譯更直接,MDK支持此功能。

wKgaomU4duuAfWP7AABcNqdC4-s430.png

審核編輯:湯梓紅

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問(wèn)題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • 仿真器
    +關(guān)注

    關(guān)注

    14

    文章

    1014

    瀏覽量

    83591
  • 字符串
    +關(guān)注

    關(guān)注

    1

    文章

    575

    瀏覽量

    20470
  • 代碼
    +關(guān)注

    關(guān)注

    30

    文章

    4722

    瀏覽量

    68234
  • 編譯
    +關(guān)注

    關(guān)注

    0

    文章

    648

    瀏覽量

    32775

原文標(biāo)題:讓代碼中包含最新的編譯時(shí)間信息

文章出處:【微信號(hào):c-stm32,微信公眾號(hào):STM32嵌入式開發(fā)】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    ARM代碼編譯鏈接的工作流程

    梳理下ARM代碼編譯鏈接的工作流程,以及過(guò)程需要的相關(guān)概念信息,不具體關(guān)注編譯鏈接的具體命令。一、編譯
    發(fā)表于 06-08 10:22

    如何在源代碼包含包含文件的定義?

    我在C編寫代碼,但需要與包含文件通信。如何在源代碼包含
    發(fā)表于 09-17 11:51

    C編譯器的設(shè)計(jì)文檔與源代碼

    C-編譯器的設(shè)計(jì)文檔與源代碼:本壓縮包包含了C-編譯器的設(shè)計(jì)文檔與源代碼,供學(xué)習(xí)參考。  整體框架. 3ɨ
    發(fā)表于 02-09 11:13 ?45次下載

    編譯代碼與源代碼的比較算法

    編譯代碼與源代碼
    發(fā)表于 02-09 15:19 ?10次下載

    iftop源代碼編譯安裝

    iftop源代碼編譯安裝
    發(fā)表于 04-03 23:39 ?15次下載
    iftop源<b class='flag-5'>代碼</b><b class='flag-5'>編譯</b>安裝

    ccacheXcode編譯速度大大提高

    [iOS]ccache 你的編譯時(shí)間飛起來(lái)2017-05-31 11:36程序設(shè)計(jì)/iOS 原文地址:Using ccache for Fun and Profit 作者 Peter
    發(fā)表于 09-25 16:33 ?0次下載

    Android源代碼編譯

    Android源代碼編譯
    發(fā)表于 10-24 09:30 ?5次下載
    Android源<b class='flag-5'>代碼</b><b class='flag-5'>編譯</b>

    CubeMX版本編譯代碼的教程

    用的CubeMX版本是4.20.0如下圖,用的Keil5編譯代碼,編譯結(jié)果0 Error(s),0 Warning(s)。
    的頭像 發(fā)表于 10-10 10:32 ?4237次閱讀
    CubeMX版本<b class='flag-5'>編譯</b><b class='flag-5'>代碼</b>的教程

    ARM代碼編譯鏈接的工作流程

    ARM處理器在市面上到處都是ARM7、ARM9、Cortex-M、Cortex-R、Cortex-A包含的種類繁多,今天我們就來(lái)了解一下ARM代碼編譯鏈接的工作流程,以及過(guò)程需要的相
    的頭像 發(fā)表于 12-22 16:57 ?2106次閱讀

    Keil MDK使用編譯器AC5與AC6生成瀏覽信息的差異有哪些

    很多使用過(guò)Keil MDK的工程師都發(fā)現(xiàn)了一個(gè)問(wèn)題,Keil MDK編譯代碼的時(shí)候,速度很慢。 ? 編譯速度慢有幾個(gè)原因: 編譯代碼量大
    的頭像 發(fā)表于 01-18 11:17 ?5000次閱讀
    Keil MDK使用<b class='flag-5'>編譯</b>器AC5與AC6生成瀏覽<b class='flag-5'>信息</b>的差異有哪些

    如何使用MDK編譯STM32程序HEX文件中加入固件版本信息

    本文介紹一個(gè)小技巧: 使用MDK編譯器,STM32程序HEX文件中加入固件版本信息代碼 代碼如下
    的頭像 發(fā)表于 06-23 15:14 ?3971次閱讀
    如何使用MDK<b class='flag-5'>編譯</b>器<b class='flag-5'>讓</b>STM32程序HEX文件中加入固件版本<b class='flag-5'>信息</b>

    什么是預(yù)編譯

    程序在正式編譯前,由編譯器進(jìn)行的操作,可以放在程序的任何位置。 什么時(shí)候使用預(yù)編譯? 1、總是使用不經(jīng)常改動(dòng)的大型代碼
    發(fā)表于 01-13 13:52 ?0次下載
    什么是預(yù)<b class='flag-5'>編譯</b>

    ARM代碼編譯與鏈接調(diào)試的工作流程梳理

    梳理下ARM代碼編譯鏈接的工作流程,以及過(guò)程需要的相關(guān)概念信息,不具體關(guān)注編譯鏈接的具體命令。
    發(fā)表于 02-09 10:03 ?1次下載
    ARM<b class='flag-5'>代碼</b><b class='flag-5'>編譯</b>與鏈接調(diào)試的工作流程梳理

    ida反編譯出來(lái)代碼能直接用嗎

    IDA反編譯出來(lái)的代碼通常 不能直接使用 ,這主要基于以下幾個(gè)方面的原因: 一、代碼的不完整性 IDA反編譯生成的代碼可能缺少原始源
    的頭像 發(fā)表于 09-02 10:55 ?433次閱讀

    java反編譯代碼可以修改么

    Java反編譯是一種將編譯后的Java字節(jié)碼(.class文件)轉(zhuǎn)換回源代碼的過(guò)程。反編譯后的代碼可以進(jìn)行修改,但是需要注意,反
    的頭像 發(fā)表于 09-02 11:00 ?446次閱讀