一、問題背景:如何保證發佈出去的bin文件是最終測試經過的版本?
通常的來說,代碼到了測試後期,master分支就不會頻繁的提交了,而且提交也會更加謹慎。linux
可是人爲操做總會出現紕漏,但願只要代碼被從新編譯過,那麼bin文件就包含新的時間信息,而這個信息是能夠從外部通訊或printf來查看的。git
在嵌入式開發中,版本號通常的都是一個int變量或字符串變量。可是若修改了代碼而沒有改version變量或宏定義,那麼從version上就看不出來文件的變化。app
那麼最終編譯的版本究竟是哪一個版本,是否與測試的版本徹底一致,這個問題尤其突出。ide
目標文件中帶有編譯時間能夠防止代碼被改動過,只要代碼被從新編譯,那麼就生成新的時間信息。函數
git可以記錄文件修改信息,可是調試信息或工程配置等,不少文件都是ignore的,這些信息表明着最終的bin文件的運行環境。工具
某些複雜bug狀況下,只有運行環境一致,仿真器才能attach到目標文件。post
2. 如何獲取時間:DATA, TIME
這兩個宏是日期和時間,格式以下。若是把這兩個宏加入到代碼,那麼就獲得了時間的字符串信息。測試
// Example of __DATE__ string: "Dec 27 2017"// Example of __TIME__ string: "15:06:19"const char *BuildInfo = "Version: " VERSION " " __DATE__ " " __TIME__;
代碼實現獲取日期和時間的方法不少,好比:ui
左右滑動查看所有代碼>>>this
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:%02u\n", year, month, day, hour, minute,seconds); #else printf("[null] ** Build at:\t%04u-%02u-%02u %02u:%02u:%02u\n", year, month, day, hour, minute,seconds); #endif DEBUG("buildDate: %s %s\n", __DATE__, __TIME__); return 0;}
把上面的函數加入到代碼中,就能獲取工程編譯的時間。
可是若是該代碼所在的文件沒有被修改,在非build-all狀況下,編譯器不會再次編譯此文件,因此時間信息也就不會被更新。
若是每次都使用re-build all,一來繁瑣,二來也不能保證每次都會記得點擊build all按鈕,靠技術手段來保證每次build都更新時間信息纔是正道。
3. 如何保證時間每次編譯都更新:使用預編譯指令,每次更新包含時間宏的文件或對應的連接文件。
在IAR環境下,官方已經給出瞭解決的方法(Using pre-build actions for time stamping)。
https://www.iar.com/support/tech-notes/ide/build-actions-pre-build-and-post-build/
方法1:修改文件的時間,引發編譯器對文件進行從新編譯。
cmd /c "touch /cygdrive/d/test.c"
方法雖好,惋惜IAR用戶大多數是Windows用戶,包括我在內,touch是linux命令,必須Cywin環境。若是安裝過這個環境的話,那就大功告成了。
Cygwin touch commandYou 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.batFile pre-build.bat: touch /cygdrive/d/test.c
方法2:修改文件對應的連接文件,觸發編譯器從新編譯該文件,生成新的連接文件,那麼就會生成新的帶有時間信息的目標文件。
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中加入上面的命令,就會在編譯前刪除test.o文件。
在這種模式下,工程代碼只要任何位置發生變化,代碼從新編譯,就會觸發刪除test.o,而後連接過程發現沒有test.o文件,那麼就會從新編譯一次test.c,那麼新的時間信息就會記錄下來了。
雖有些曲線救國的味道,但仍是很順利的實現了目標。
只要工程的任何地方有改動,生成新的目標文件,那麼目標文件中就會帶有最新的編譯時間。
方法3:直接告訴編譯器每次從新編譯某個文件更直接,MDK支持此功能。
時隔一年半再次來這裏,發現當時本身簡直是小白,還洋洋得意曲線救國,實際上舍近求遠罷了。
若是對工具多一些瞭解,萬萬是不會用上面的方法的,固然上面的方法也是通用想法,是通用型知識點,容易想到,也能達到目標。
新的方法,不須要寫任何腳本,若是想讓代碼每次都編譯更新DATA 和 TIME兩個宏,那麼讓這個文件每次都編譯一次就能夠了,不須要刪除它的obj文件而後讓編譯器找不到文件而觸發從新編一次,其實直接告訴編譯器每次從新編譯更直接,MDK支持此功能。
下面是測試的效果:
以上就是良許教程網爲各位朋友分享的如何把編譯時間加入到目標文件。 以上就是良許教程網爲各位朋友分享的Linux相關知識。