原文:https://www.cnblogs.com/pingwen/p/8183728.html

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

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

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

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

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

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

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

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

2. 如何获取时间:DATA, TIME

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

// Example of __DATE__ string: "Dec 27 2017"
// Example of __TIME__ string: "15:06:19"
const char *BuildInfo = "Version: " VERSION " " __DATE__ " " __TIME__;

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

左右滑动查看全部代码>>>

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);#elseprintf("[null]  ** Build at:\t%04u-%02u-%02u %02u:%02u:%02u\n",year, month, day,hour, minute,seconds);#endifDEBUG("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.bat
File 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支持此功能。

下面是测试的效果:

其它资料:

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

 

 推荐阅读:

专辑|Linux文章汇总

专辑|程序人生

专辑|C语言

嵌入式Linux

微信扫描二维码,关注我的公众号 

把编译时间加入到目标文件相关推荐

  1. build文件_把编译时间加入到目标文件

    出处:https://www.cnblogs.com/pingwen/p/8183728.html 1.问题背景:如何保证发布出去的bin文件是最终测试通过的版本? 一般的来讲,代码到了测试后期,ma ...

  2. 如何使用makefile编译不同平台的目标文件(makefile的参数传递)

    最近在研究一个嵌入式开发项目,在编写实际的项目代码时,需要临时写一些测试代码对部分功能进行预测试.编写的这些代码,有时候需要在PC机(x86)的平台上运行,有时候则需要在A嵌入式平台(arm端)i.m ...

  3. GCC编译器原理(二)------编译原理一:目标文件

    一.目标文件 在 UNIX® 和 Linux® 中,任何事物都是文件.UNIX 和 Linux 编程实际上是编写处理各种文件的代码.系统由许多类型的文件组成,但目标文件具有一种特殊的设计,提供了灵活和 ...

  4. c语言编译生成的目标文件拓展名,系统默认的C语言源程序文件的扩展名是(),经过编译后生成的目标文件的扩展名是(),经过连接后生成的可执行文件的扩展名是()。...

    系统序文常用的密钥分发技术有 CA 技术和[] 技术. 默认名名探究精神的重要性体现在以下哪些方面?() 如果采用偶校验,的目的可的扩0101010的校验位是(),0011011的校验位是(). 语言 ...

  5. c语言目标文件作用,关于编译:C语言中的目标文件是什么?

    我正在阅读有关C语言中的库的信息,但尚未找到关于什么是目标文件的解释. 其他任何编译文件和目标文件之间的真正区别是什么? 如果有人能够用人类语言进行解释,我将感到非常高兴. 一个更活跃的版本:stac ...

  6. STM32 MDK编译后生成的 .map文件深入分析

    .map文件是STM32开发中非常重要的一个文件,在该文件中可以详细的查看单个文件.函数及用户定义的全局变量等的占用RAM和ROM(一般为片内FLASH)的空间大小,通过了解这些信息可以很方便的进行代 ...

  7. C语言 目标文件和可执行文件(ELF文件)

    转自:https://www.jianshu.com/p/7c609b70acbd 1.C语言创建程序 1.1C语言创建(分为4个步骤) 编辑 编译 链接 执行 编辑:就是创建和修改C程序的源代码-我 ...

  8. 链接的时候如何去掉没有用到的函数、目标文件

    以前了解过ld在链接的时候,是以目标文件(.o)为单位的,所以libc里的源文件基本上都很短,甚至只定义一个函数. 为的是在链接的时候,不要把没有用到的函数也链进去,从而减小可执行文件或者动态库的大小 ...

  9. 高性能流媒体服务器EasyDSS前端重构(二) webpack + vue + AdminLTE 多页面提取共用文件, 优化编译时间...

    本文围绕着实现EasyDSS高性能流媒体服务器的前端框架来展开的,具体EasyDSS的相关信息可在:www.easydss.com 找到! 接上回 <高性能流媒体服务器EasyDSS前端重构(一 ...

最新文章

  1. JSTL标签库学习笔记
  2. linux服务器加固的命令,Linux 服务器安全加固
  3. python入门作业编程题-python入门(一)作业
  4. 学霸孩子必备的三种思维,编程思维的速成方法
  5. 503 service unavailable错误提示如何解决
  6. Leetcode 08. 字符串转换整数 (atoi)
  7. 三星的S3C2440A 存储器控制器
  8. 智能一代云平台(三十七):Java技术栈
  9. wireshark未响应
  10. php 学习笔记之搭建开发环境(mac版)
  11. Excel单元格向下,向右自动填充的快捷键
  12. 大二 数据结构 期末复习题(仅供参考)
  13. Windows 10 VMware Workstation Server服务启动一段时间后自动异常关闭
  14. 【笔记】项目工作中总结(三)
  15. php购物商城项目总结,php商城项目总结
  16. 如何将苹果手机中的M4A音乐转换为MP3格式
  17. python编程-----利用爬虫获取自如房间信息(三)
  18. java对接ntlm_通过NTLM获取信息.
  19. 买极米NEW Z8X的我后悔了,换成当贝F3 Air可真香
  20. Mac OS X 桌面图标隐藏和显示

热门文章

  1. ES6:Set和Map
  2. Create Volume 操作(Part I) - 每天5分钟玩转 OpenStack(50)
  3. RequestMapping
  4. (笔记)Mysql命令drop database:删除数据库
  5. 传360以原彩虹QQ研发团队为班底拟强推IM
  6. java中程序执行顺序
  7. 产生死锁的四个必要条件
  8. JDBC笔记01-JDBC,Connection,Statement,ResultSet,PreparedStatement,Properties
  9. 【Nginx】通过反向代理配置本地图床功能
  10. windows环境下,如何在Pycharm下安装TensorFlow环境