您好,欢迎进入山西润盛进出口有限公司!

咨询服务热线

15383419322

如何把编译时间加入到目标文件中

发布时间:2024-02-04 10:18人气:

01问题背景:如何保证发布出去的bin文件是最终测试通过的版本?

一般的来讲,代码到了测试后期,master分支就不会频繁的提交了,并且提交也会更加谨慎。但是人为操作总会出现纰漏,希望只要代码被重新编译过,那么bin文件就包含新的时间信息,而这个信息是可以从外部通信或printf来查看的。

在嵌入式开发中,版本号一般的都是一个int变量或字符串变量。但是若修改了代码而没有改version变量或宏定义,那么从version上就看不出来文件的变化。最终编译的版本到底是哪个版本,是否与测试的版本完全一致,这个问题尤为突出。

目标文件中带有编译时间可以防止代码被改动过,只要代码被重新编译,那么就生成新的时间信息。

git能够记录文件修改信息,但是调试信息或工程配置等,很多文件都是ignore的,这些信息代表着最终的bin文件的运行环境。某些复杂bug情况下,只有运行环境一致,仿真器才能attach到目标文件。

02如何获取时间:__DATA__ , __ TIME__

这两个宏是日期和时间,格式如下。如果把这两个宏加入到代码,那么就得到了时间的字符串信息。

//Exampleof__DATE__string:"Dec272017"//Exampleof__TIME__string:"15:06:19"constchar*BuildInfo="Version:"VERSION""__DATE__""__TIME__;

代码实现获取日期和时间的方法很多,比如:

unsignedintmk_Build_Date(void){intyear=0,month=0,day=0;inthour=0,minute=0,seconds=0;charm[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);#ifdefSHORT_DATA_CHAR__printf("[null]**Buildat:t%04u-%02u-%02us%02u:%02u:%02un",year,month,day,hour,minute,seconds);#elseprintf("[null]**Buildat:t%04u-%02u-%02u%02u:%02u:%02un",year,month,day,hour,minute,seconds);#endifDEBUG("buildDate:%s%sn",__DATE__,__TIME__);return0;}

把上面的函数加入到代码中,就能获取工程编译的时间。但是如果该代码所在的文件没有被修改,在非build-all情况下,编译器不会再次编译此文件,所以时间信息也就不会被更新。

如果每次都使用re-build all,一来繁琐,二来也不能保证每次都会记得点击build all按钮,靠技术手段来保证每次build都更新时间信息才是正道。

03如何保证时间每次编译都更新:使用预编译指令,每次更新包含时间宏的文件或对应的链接文件。

在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环境。如果安装过这个环境的话,那就大功告成了。

CygwintouchcommandYoucanenter"cygwin-application.exe"onthepre-andpost-buildcommandlines,iftheenvironmentvariablePATHincludesthedirectorywherethe"cygwin-application.exe"islocated.YoucanruntheCygwincommand"touch"onthepre-buildcommandline,butifyouaddafilepath,forexample"touchd:/test.c",thefilepathisnotacceptedbyCygwin.CygwinexpectsthePOSIXpath/cygdrive/d/test.csotheresultingcommandlinewouldbe"touch/cygdrive/d/test.c",howeverthiscommandcannotbeexecuteddirectlyonthepre-andpost-buildcommand.Insteadyouhavetorunindirectlyusing:cmd/c"touch/cygdrive/d/test.c"The.batfile(locatedinprojectdirectory)alternativewouldlooklike:Pre-buildcommandline:$PROJ_DIR$pre-build.batFilepre-build.bat:touch/cygdrive/d/test.c

方法2:修改文件对应的链接文件,触发编译器重新编译该文件,生成新的链接文件,那么就会生成新的带有时间信息的目标文件。

Analternativetothe"touch"commandistohaveapre-buildactionthatdeletestheobjectfile,forexamplethePre-buildcommandline:cmd/c"del"$OBJ_DIR$test.o""

在pre-build中加入上面的命令,就会在编译前删除test.o文件。在这种模式下,工程代码只要任何位置发生变化,代码重新编译,就会触发删除test.o,然后链接过程发现没有test.o文件,那么就会重新编译一次test.c,那么新的时间信息就会记录下来了。

虽有些曲线救国的味道,但还是很顺利的实现了目标。只要工程的任何地方有改动,生成新的目标文件,那么目标文件中就会带有最新的编译时间。

方法3:直接告诉编译器每次重新编译某个文件更直接,MDK支持此功能。

时隔一年半再次来这里,发现当时自己简直是小白,还洋洋得意曲线救国,实际上舍近求远罢了。如果对工具多一些了解,万万是不会用上面的方法的,当然上面的方法也是通用想法,是通用型知识点,容易想到,也能达到目标。

新的方法,不需要写任何脚本,如果想让代码每次都编译更新DATA 和 TIME两个宏,那么让这个文件每次都编译一次就可以了,不需要删除它的obj文件然后让编译器找不到文件而触发重新编一次,其实直接告诉编译器每次重新编译更直接,MDK支持此功能。

640-2.png

下面是测试的效果:

640-3.png

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

  • 联系方式
  • 传 真:
  • 手 机:15383419322
  • 电 话:15383419322
  • 地 址:山西太原市杏花岭区解放路 175 号万达中心 A 座 33 楼 3301 室
友情链接
plc控制器
自动化设备
自动化设备
伺服驱动器
在线咨询

咨询电话:

15383419322

  • 微信扫码 关注我们

Copyright © 2022-2024 山西润盛进出口有限公司 版权所有 晋ICP备2021008479号-14

晋公网安备 14010702070906号

扫一扫咨询微信客服
15383419322