<meter id="pryje"><nav id="pryje"><delect id="pryje"></delect></nav></meter>
          <label id="pryje"></label>

          新聞中心

          EEPW首頁 > 設(shè)計(jì)應(yīng)用 > 如何把編譯時(shí)間加入到目標(biāo)文件中

          如何把編譯時(shí)間加入到目標(biāo)文件中

          作者: 時(shí)間:2024-02-01 來源: 收藏

          01 問題背景:如何保證發(fā)布出去的bin文件是最終測(cè)試通過的版本?

          本文引用地址:http://www.ex-cimer.com/article/202402/455311.htm

          一般的來講,到了測(cè)試后期,master分支就不會(huì)頻繁的提交了,并且提交也會(huì)更加謹(jǐn)慎。但是人為操作總會(huì)出現(xiàn)紕漏,希望只要被重新過,那么bin文件就包含新的時(shí)間信息,而這個(gè)信息是可以從外部通信或printf來查看的。

          在嵌入式開發(fā)中,版本號(hào)一般的都是一個(gè)int變量或字符串變量。但是若修改了而沒有改version變量或宏定義,那么從version上就看不出來文件的變化。最終的版本到底是哪個(gè)版本,是否與測(cè)試的版本完全一致,這個(gè)問題尤為突出。

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

          git能夠記錄文件修改信息,但是調(diào)試信息或工程配置等,很多文件都是ignore的,這些信息代表著最終的bin文件的運(yùn)行環(huán)境。某些復(fù)雜bug情況下,只有運(yùn)行環(huán)境一致,仿真器才能attach到目標(biāo)文件。

          02 如何獲取時(shí)間:__DATA__  , __ TIME__

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

          // Example of __DATE__ string: "Dec 27 2017"
          // Example of __TIME__ string: "15:06:19"
          const char *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:t%04u-%02u-%02us %02u:%02u:%02un",
                          year, month, day,
                          hour, minute,seconds);
              #else
                  printf("[null]  ** Build at:t%04u-%02u-%02u %02u:%02u:%02un",
                          year, month, day,
                          hour, minute,seconds);
              #endif
              DEBUG("buildDate: %s %sn", __DATE__, __TIME__);
              return 0;
          }

          把上面的函數(shù)加入到代碼中,就能獲取工程編譯的時(shí)間。但是如果該代碼所在的文件沒有被修改,在非build-all情況下,編譯器不會(huì)再次編譯此文件,所以時(shí)間信息也就不會(huì)被更新。

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

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

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

          https://www.iar.com/support/tech-notes/ide/build-actions-pre-build-and-post-build/

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

          cmd /c "touch /cygdrive/d/test.c"

          方法雖好,可惜IAR用戶大多數(shù)是Windows用戶,包括我在內(nèi),touch是linux命令,必須Cywin環(huán)境。如果安裝過這個(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$test.o""

          在pre-build中加入上面的命令,就會(huì)在編譯前刪除test.o文件。在這種模式下,工程代碼只要任何位置發(fā)生變化,代碼重新編譯,就會(huì)觸發(fā)刪除test.o,然后鏈接過程發(fā)現(xiàn)沒有test.o文件,那么就會(huì)重新編譯一次test.c,那么新的時(shí)間信息就會(huì)記錄下來了。

          雖有些曲線救國(guó)的味道,但還是很順利的實(shí)現(xiàn)了目標(biāo)。只要工程的任何地方有改動(dòng),生成新的目標(biāo)文件,那么目標(biāo)文件中就會(huì)帶有最新的編譯時(shí)間。

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

          時(shí)隔一年半再次來這里,發(fā)現(xiàn)當(dāng)時(shí)自己簡(jiǎn)直是小白,還洋洋得意曲線救國(guó),實(shí)際上舍近求遠(yuǎn)罷了。如果對(duì)工具多一些了解,萬萬是不會(huì)用上面的方法的,當(dāng)然上面的方法也是通用想法,是通用型知識(shí)點(diǎn),容易想到,也能達(dá)到目標(biāo)。

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

          640-2.png

          下面是測(cè)試的效果:

          640-3.png

          其它資料:https://stackoverflow.com/questions/11697820/how-to-use-date-and-time-predefined-macros-in-as-two-integers-then-stri



          關(guān)鍵詞: 編譯 代碼 C語言

          評(píng)論


          相關(guān)推薦

          技術(shù)專區(qū)

          關(guān)閉
          看屁屁www成人影院,亚洲人妻成人图片,亚洲精品成人午夜在线,日韩在线 欧美成人 (function(){ var bp = document.createElement('script'); var curProtocol = window.location.protocol.split(':')[0]; if (curProtocol === 'https') { bp.src = 'https://zz.bdstatic.com/linksubmit/push.js'; } else { bp.src = 'http://push.zhanzhang.baidu.com/push.js'; } var s = document.getElementsByTagName("script")[0]; s.parentNode.insertBefore(bp, s); })();