電子產(chǎn)業(yè)一站式賦能平臺(tái)

PCB聯(lián)盟網(wǎng)

搜索
查看: 87|回復(fù): 0
收起左側(cè)

嵌入式軟件,如何把編譯時(shí)間加入到bin文件,進(jìn)行版本管理?

[復(fù)制鏈接]

485

主題

485

帖子

1623

積分

三級(jí)會(huì)員

Rank: 3Rank: 3

積分
1623
跳轉(zhuǎn)到指定樓層
樓主
發(fā)表于 2024-11-27 17:50:00 | 只看該作者 |只看大圖 回帖獎(jiǎng)勵(lì) |倒序?yàn)g覽 |閱讀模式
我是老溫,一名熱愛(ài)學(xué)習(xí)的嵌入式工程師  V+ f. N9 r# {( N# H% E/ n2 i; ^
關(guān)注我,一起變得更加優(yōu)秀!
" ~$ `' ~8 c+ O1 ^! ]6 g嵌入式軟件工程師在發(fā)布固件之前,通常都需要先進(jìn)行提測(cè),提測(cè)過(guò)程中會(huì)不斷發(fā)現(xiàn)和修復(fù)bug,期間會(huì)涉及到不斷地編譯固件和更新固件版本,那么問(wèn)題來(lái)了:( s/ I# U" T" G' ^6 H
6 g- z7 U: c5 f" l" S. f$ s
如何保證發(fā)布出去的bin文件是最終測(cè)試通過(guò)的版本?' _8 [, s8 K, T- G4 ^: L- \

1 U+ i  ^2 l' g  Q一般的來(lái)講,代碼到了測(cè)試后期,master分支就不會(huì)頻繁的提交了,并且提交也會(huì)更加謹(jǐn)慎。$ f6 Z( Z: g+ S! P- F
但是人為操作總會(huì)出現(xiàn)紕漏,希望只要代碼被重新編譯過(guò),那么bin文件就包含新的時(shí)間信息,而這個(gè)信息是可以從外部通信或printf來(lái)查看的。  K' _4 f1 \$ |  _
在嵌入式開(kāi)發(fā)中,版本號(hào)一般的都是一個(gè)int變量或字符串變量。但是若修改了代碼而沒(méi)有改version變量或宏定義,那么從version上就看不出來(lái)文件的變化。6 v1 x/ j( Q$ U  t
那么最終編譯的版本到底是哪個(gè)版本,是否與測(cè)試的版本完全一致,這個(gè)問(wèn)題尤為突出。8 [7 r! D. C% Y; K' V5 Y$ W
目標(biāo)文件中帶有編譯時(shí)間可以防止代碼被改動(dòng)過(guò),只要代碼被重新編譯,那么就生成新的時(shí)間信息。
- ?8 y) E3 u. c' ~  B+ Igit能夠記錄文件修改信息,但是調(diào)試信息或工程配置等,很多文件都是ignore的,這些信息代表著最終的bin文件的運(yùn)行環(huán)境。
$ C" P& y# z- N2 W  w1 @6 e7 }某些復(fù)雜bug情況下,只有運(yùn)行環(huán)境一致,仿真器才能attach到目標(biāo)文件。
, b( t+ F2 i4 }. M& M$ {0 @! f! c! ]& t9 j; g$ e/ ?
如何獲取時(shí)間?! o% e# ^  f  g. Z& w- |9 E
: \0 V1 T" i4 C/ W$ u9 d9 K

& Y8 o* K+ `. h" d; t* F8 c2 F這兩個(gè)宏是日期和時(shí)間,格式如下。如果把這兩個(gè)宏加入到代碼,那么就得到了時(shí)間的字符串信息。
# E# E; G- s8 f/ b0 R// Example of __DATE__ string: "Dec 27 2017"
* P2 ]' ?2 ^2 a3 S& @// Example of __TIME__ string: "15:06:19"
  P) |" x1 C6 D3 H% I; H& C$ dconst char *BuildInfo = "Version: " VERSION " " __DATE__ " " __TIME__;" F& e" C6 {6 R
代碼實(shí)現(xiàn)獲取日期和時(shí)間的方法很多,比如:& t# M- q/ Y8 h3 b" y* {3 b
unsigned int mk_Build_Date(void). @+ u% e: t6 c% B, y
{
9 s, O: _2 I, t5 ]    int    year = 0, month = 0, day = 0;
9 t! Q! w; a" J" m/ v    int hour = 0, minute = 0, seconds = 0;! G* Y6 ^# o* R" J' y$ r& b
    char m[4] = {0};
* S7 i! b7 G# ]1 r    sscanf(__DATE__, "%3s %2d %4d", m, &day, &year);4 D+ P! A. P3 y" @+ f1 @
    for (month = 0; month 12; month++)$ k4 B& \( w# w! A: h8 M
    {
) C- u+ b  c6 ~        if (strcmp(m, short_char_months[month]) == 0)
1 y+ e  \- R4 z; A: t1 |        {( N0 P% H" a6 m9 G
            break;: F& \3 i. F- q+ G8 p& j$ a7 h4 S
        }
, n( b; ]# {3 d/ C4 ^& L; l    }$ G& b$ r6 y- [6 g
    sscanf(__TIME__, "%2d:%2d:%2d", &hour, &minute, &seconds);
. [: A! g7 M1 y& ?8 L4 I/ v7 i    #ifdef SHORT_DATA_CHAR__
) Q( c, g5 b! \# M+ s        printf("[null]  ** Build at:        %04u-%02u-%02us %02u:%02u:%02u
( n1 L; O. @9 `# ~* z+ W",
7 v3 }4 C, N- F' l( ?                year, month, day,4 H& u, D5 I, {9 k! g- V
                hour, minute,seconds);
6 u3 }% d% r  m8 Y    #else
) j+ P0 B5 u, U        printf("[null]  ** Build at:        %04u-%02u-%02u %02u:%02u:%02u
; {7 Q; G0 T8 }/ d",
9 v7 X7 F, p1 X8 d: k                year, month, day,; C1 @1 f, b) ?* \% |
                hour, minute,seconds);
( B7 o' z6 p2 P" Q% e    #endif
5 }) x" I! a' t9 g& w    DEBUG("buildDate: %s %s# `- p: _4 V$ o5 H) c% G
", __DATE__, __TIME__);' ?' H# _: r$ Q
    return 0;
3 y+ }. |7 ^* i, \7 H5 E. c/ U) O}
# }+ [; S7 s6 @3 X8 z. P把上面的函數(shù)加入到代碼中,就能獲取工程編譯的時(shí)間。6 g, C/ e  q6 Z/ I; H" g6 ^* B$ x
但是如果該代碼所在的文件沒(méi)有被修改,在非build-all情況下,編譯器不會(huì)再次編譯此文件,所以時(shí)間信息也就不會(huì)被更新。
0 n+ D; v7 G9 U2 ^如果每次都使用re-build all,一來(lái)繁瑣,二來(lái)也不能保證每次都會(huì)記得點(diǎn)擊build all按鈕,靠技術(shù)手段來(lái)保證每次build都更新時(shí)間信息才是正道。" r% k  A0 d. j. I" T
如何保證時(shí)間每次編譯都更新?. \# L" @, |  E/ X

: ~- ~1 H; \! r4 w, u
% [+ T8 M' r3 G! j" p' J4 y9 h
使用預(yù)編譯指令,每次更新包含時(shí)間宏的文件或?qū)?yīng)的鏈接文件。
) T2 b; ]  P" H6 q# g在IAR環(huán)境下,官方已經(jīng)給出了解決的方法(Using pre-build actions for time stamping)。
# Z$ u& D, |/ h! K% Ehttps://www.iar.com/support/tech-notes/ide/build-actions-pre-build-and-post-build/
' X7 B7 t" i/ z% @+ p  n6 m( a
方法1:修改文件的時(shí)間,引起編譯器對(duì)文件進(jìn)行重新編譯。6 X4 E5 |/ n7 }9 d9 M
cmd /c "touch /cygdrive/d/test.c": L" B" ^: Y  Y: G1 I
方法雖好,可惜IAR用戶(hù)大多數(shù)是Windows用戶(hù),包括我在內(nèi),touch是linux命令,必須Cywin環(huán)境。如果安裝過(guò)這個(gè)環(huán)境的話,那就大功告成了。, [6 c' U" @2 _
Cygwin touch command
% h: K" ^) b7 j( [, YYou 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." Z" _5 ~' k$ C% t
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.7 p* f4 K. P: v& t+ F0 l4 }+ W
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:
6 {0 U8 f: Y4 M" y& Kcmd /c "touch /cygdrive/d/test.c"
1 b9 T' I) c; [9 n: @* O9 V: iThe .bat file (located in project directory) alternative would look like:
8 R* C" i* ]% c# i/ n: LPre-build command line:" V& G; Z: g7 L7 z6 B+ c
$PROJ_DIR$\pre-build.bat1 Q5 S9 O. L/ [" L* Y3 G% C; J
File pre-build.bat:3 o  T; y! {7 V# b8 a" Z- G2 C
touch /cygdrive/d/test.c
9 x; u5 z9 s! r5 ^7 Q: I* K5 ?方法2:修改文件對(duì)應(yīng)的鏈接文件,觸發(fā)編譯器重新編譯該文件,生成新的鏈接文件,那么就會(huì)生成新的帶有時(shí)間信息的目標(biāo)文件。
3 Y* a9 P+ u4 o0 }' ^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:
% |0 l1 Q- T8 P; g6 wcmd /c "del "$OBJ_DIR$        est.o"". V+ m: L: W3 W, T% q
在pre-build中加入上面的命令,就會(huì)在編譯前刪除test.o文件。
, }' Q; B1 t& {在這種模式下,工程代碼只要任何位置發(fā)生變化,代碼重新編譯,就會(huì)觸發(fā)刪除test.o,然后鏈接過(guò)程發(fā)現(xiàn)沒(méi)有test.o文件,那么就會(huì)重新編譯一次test.c,那么新的時(shí)間信息就會(huì)記錄下來(lái)了。3 `( r- {3 D( N3 h4 ~
雖有些曲線救國(guó)的味道,但還是很順利的實(shí)現(xiàn)了目標(biāo)。+ V( L) x% y4 H/ Z5 y
只要工程的任何地方有改動(dòng),生成新的目標(biāo)文件,那么目標(biāo)文件中就會(huì)帶有最新的編譯時(shí)間。
6 e, A& I+ ?# t' h4 F* |" w方法3:直接告訴編譯器每次重新編譯某個(gè)文件更直接,MDK支持此功能。
, f  J. z# v% d) s3 L( l; h5 {時(shí)隔一年半再次來(lái)這里,發(fā)現(xiàn)當(dāng)時(shí)自己簡(jiǎn)直是小白,還洋洋得意曲線救國(guó),實(shí)際上舍近求遠(yuǎn)罷了。
/ P! _8 ^/ t. W) u- X3 ~5 T5 P如果對(duì)工具多一些了解,萬(wàn)萬(wàn)是不會(huì)用上面的方法的,當(dāng)然上面的方法也是通用想法,是通用型知識(shí)點(diǎn),容易想到,也能達(dá)到目標(biāo)。
  ~' \* w: b2 I, d, X新的方法,不需要寫(xiě)任何腳本,如果想讓代碼每次都編譯更新DATA 和 TIME兩個(gè)宏,那么讓這個(gè)文件每次都編譯一次就可以了,不需要?jiǎng)h除它的obj文件然后讓編譯器找不到文件而觸發(fā)重新編一次,其實(shí)直接告訴編譯器每次重新編譯更直接,MDK支持此功能。- x3 j+ _3 H: B( X8 {2 ?

  Y- _# N( A( T' q7 I9 |8 L) x/ e+ d4 q1 V' o. Q
下面是測(cè)試的效果:6 h+ \, S/ m6 H
* y/ H& T& A- ?9 s0 Q
: _: K: q, K; E% q- c6 ~5 @
8 e. r1 J  E- c( _2 ~* s3 _- v  ?
其它資料:
, S1 s% {' o, K( {https://stackoverflow.com/questions/11697820/how-to-use-date-and-time-predefined-macros-in-as-two-integers-then-stri
! g6 M: Q4 P0 q1 @
來(lái)源:Internet。& ]6 o( v4 ]; U; i% `
-END-: d' c6 ~: a* K. ]8 Z! r
往期推薦:點(diǎn)擊圖片即可跳轉(zhuǎn)閱讀
' e: _* h2 l! v8 A& r, ?, X, P& n                                                        - z& d" z* A. q6 `4 c: v, a
                                                                5 A% ~$ i* G. I" N
                                                                        & U' u' P( T& s& q/ R
                                                                               
. ^. C/ q( L( e1 E# X9 J' Y
4 K# r8 |, h9 c% n3 v                                                                               
2 g! ?0 T; e. Q% q& g3 r                                                                                        適用于嵌入式設(shè)備,用 C 語(yǔ)言編寫(xiě)的輕量級(jí)日志庫(kù)!
$ s! ~( X. U4 T7 N1 ]  L' D                                                       
0 \" a, q- {6 X7 ]& o) A' J) X* V5 k                                                               
( d9 X3 `/ E+ T9 c  a* W' n: V/ l                                                                       
# y9 b  m4 w6 O( P                                                                               
; ^( O* @& l6 {( A 5 u4 C* e; `, ?& g4 [% V1 i
                                                                                8 z" {( A4 M( R) [# v$ `
                                                                                        工業(yè)現(xiàn)場(chǎng)經(jīng)常用到的 RS485 通信,原來(lái)它的收發(fā)機(jī)制是這樣的。
) \+ g% f& {0 }" E' Y                                                               
3 F: s+ s( G$ J                                                                        * {/ L2 u5 |% f+ s
                                                                                # B( p, u  r# U0 I9 B
# M# g% J+ b4 {# `2 ~7 G( {
                                                                               
4 Y5 e# V' b7 |1 U* Q3 Z9 N# J$ [                                                                                        嵌入式應(yīng)用程序開(kāi)發(fā),經(jīng)常使用哪些數(shù)據(jù)結(jié)構(gòu)?" i( x5 [1 p2 F. X* S( k5 C
                                                                                7 q' Q3 X9 @) u6 O+ V( t7 }* u
                                                                       
" M: O5 t5 m- h; K, W3 c6 |                                                                & H- G$ d' n3 w" M( d9 G6 ~1 y3 ?1 |; G
                                                        我是老溫,一名熱愛(ài)學(xué)習(xí)的嵌入式工程師- n- U$ R, N) r3 t- q% Q9 L- {# j
關(guān)注我,一起變得更加優(yōu)秀!. \; Z! }2 z8 p; e  Q0 t$ N! ^: P

發(fā)表回復(fù)

本版積分規(guī)則


聯(lián)系客服 關(guān)注微信 下載APP 返回頂部 返回列表