0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

非常实用的技巧,将编译时间加入到目标文件中

小麦大叔 来源:小麦大叔 2024-01-12 16:32 次阅读

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

一般的来讲,代码到了测试后期,master分支就不会频繁的提交了,并且提交也会更加谨慎。

但是人为操作总会出现纰漏,希望只要代码被重新编译过,那么bin文件就包含新的时间信息,而这个信息是可以从外部通信或printf来查看的。

嵌入式开发中,版本号一般的都是一个int变量或字符串变量。但是若修改了代码而没有改version变量或宏定义,那么从version上就看不出来文件的变化。

那么最终编译的版本到底是哪个版本,是否与测试的版本完全一致,这个问题尤为突出。

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

git能够记录文件修改信息,但是调试信息或工程配置等,很多文件都是ignore的,这些信息代表着最终的bin文件的运行环境。

某些复杂bug情况下,只有运行环境一致,仿真器才能attach到目标文件。

2. 如何获取时间:__DATA__ , __ TIME__

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

//Exampleof__DATE__string:"Dec272017"
//Exampleof__TIME__string:"1519"
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:	%04u-%02u-%02us%02u:%02u:%02u
",
year,month,day,
hour,minute,seconds);

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

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

return0;
}

把上面的函数加入到代码中,就能获取工程编译的时间。

但是如果该代码所在的文件没有被修改,在非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环境。如果安装过这个环境的话,那就大功告成了。

Cygwintouchcommand

Youcanenter"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.bat
Filepre-build.bat:

touch/cygdrive/d/test.c

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

Analternativetothe"touch"commandistohaveapre-buildactionthatdeletestheobjectfile,forexamplethePre-buildcommandline:

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

在pre-build中加入上面的命令,就会在编译前删除test.o文件。

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

虽有些曲线救国的味道,但还是很顺利的实现了目标。

只要工程的任何地方有改动,生成新的目标文件,那么目标文件中就会带有最新的编译时间。

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

时隔一年半再次来这里,发现当时自己简直是小白,还洋洋得意曲线救国,实际上舍近求远罢了。

如果对工具多一些了解,万万是不会用上面的方法的,当然上面的方法也是通用想法,是通用型知识点,容易想到,也能达到目标。

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

14b2868e-b100-11ee-8b88-92fbcf53809c.png

下面是测试的效果:

14c9c4d4-b100-11ee-8b88-92fbcf53809c.png

其它资料

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


声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 嵌入式开发
    +关注

    关注

    18

    文章

    977

    浏览量

    46958
  • 编译器
    +关注

    关注

    1

    文章

    1577

    浏览量

    48617
  • 编译
    +关注

    关注

    0

    文章

    615

    浏览量

    32392

原文标题:非常实用的技巧,将编译时间加入到目标文件中

文章出处:【微信号:knifewheat,微信公众号:小麦大叔】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    怎样把数组里的元素依次加入到下拉列表

    `获得一个数组之后怎样把各个元素依次加入到下拉列表?`
    发表于 08-10 17:11

    Linux加入到Windows域

    Linux加入到Windows域1.对windows域的理解 域既是 Windows 网络操作系统的逻辑组织单元,也是Internet的逻辑组织单元,在 Windows 网络操作系统,域是安全
    发表于 04-10 20:56

    请问如何TDS510的驱动加入到CCS4软件?程序配置在RAM和flash的区别是什么?

    此请教下,如何TDS510驱动加入到CCS4的驱动文件。程序配置在RAM和flash的区别是什么?谢谢!
    发表于 06-11 01:08

    请问ZLL协议栈touchlink可以加入到bridge

    TI工程师:ZLL协议栈touchlink可以加入到bridge么,我light加入到touchlink
    发表于 08-18 07:50

    请问在CCES环境怎么把UCOSii生成.DLB文件并如何它加入到新的工程里面?

    Hi,在开发过程,总会遇到每改动一些小问题就要重新编译,而ucosII文件比较多,所以想怎么能把UCOSii生成.DLB文件,就不用每次编译
    发表于 10-29 09:29

    如何Android手机和BLE软件狗加入到BLE模块的白名单

    你好,我使用PSoC 4 BLE(CY8CKIT-042-BLE先锋套件)。有人能告诉我如何Android手机和BLE软件狗加入到BLE模块的白名单吗? 以上来自于百度翻译 以下为原文Hi
    发表于 07-02 07:50

    F23sensor驱动加入到Hi3518ev300的LITEOS系统过程

    f23加入进去。5、修改完成应该去isp去编译sensor驱动,然后编译sample。出现了个问题,如图所示。没有找到这个接口。6、在sample
    发表于 02-24 18:00

    怎样Simulink生成出的代码加入到STM32工程编译

    怎样Simulink生成出的代码加入到STM32工程编译呢?其过程有哪些?
    发表于 11-22 06:22

    如何所需元器件加入到对象选择器

    一、电路设计1.1 所需元器件加入到对象选择器(1)添加AT89C51弹出“Pick Devices”页面,在“Keywords”输入AT89C51,结果显示在“Result”,在“Result
    发表于 12-06 06:14

    请问如何加入自己的代码源码

    新手上路自己写的程序,如何加入到源码中进行编译,生成新的镜像?请问在哪里加入?makefile文件如何更改添加?有什么推荐的资料吗?谢谢解答{:2_25:}{:2_25:}
    发表于 12-31 07:19

    请问开发板给的ipc_build例子如何加入到Qt

    您好,请问开发板给的ipc_build例子如何加入到Qt,或者QT程序如何调用IPC的系统API?我看到例程里的build.sh中使用了很多动态库文件,尝试这些动态库
    发表于 01-06 07:39

    怎样通过keil运用stm32芯片进行程序编译以及仿真呢

    通过keil运用stm32芯片进行程序编译以及仿真一、创建工程目标文件二、程序加入到工程中三、程序的
    发表于 02-21 06:57

    RV1126开发板怎么把自己的脚本加入到开机启动

    问题描述及复现步骤:怎么把自己的脚本加入到开机启动,目前在开发板上没有发现rc.local,rcS等类似的文件,尝试加入各种sh文件
    发表于 09-01 16:30

    把Processor Expert(PEx) driver加入到应用的方法

    如何把Processor Expert(PEx) driver加入到应用
    发表于 12-08 06:25

    用Keil编程时如何文件加入到工程?

    用Keil编程时如何文件加入到工程
    发表于 10-25 06:19