让代码中包含最新的编译时间信息

如何保证发布出去的bin文件是最终测试通过的版本?
一般的来讲,代码到了测试后期,master分支就不会频繁的提交了,并且提交也会更加谨慎。
但是人为操作总会出现纰漏,希望只要代码被重新编译过,那么bin文件就包含新的时间信息,而这个信息是可以从外部通信或printf来查看的。
在嵌入式开发中,版本号一般的都是一个int变量或字符串变量。但是若修改了代码而没有改version变量或宏定义,那么从version上就看不出来文件的变化。
那么最终编译的版本到底是哪个版本,是否与测试的版本完全一致,这个问题尤为突出。
目标文件中带有编译时间可以防止代码被改动过,只要代码被重新编译,那么就生成新的时间信息。
git能够记录文件修改信息,但是调试信息或工程配置等,很多文件都是ignore的,这些信息代表着最终的bin文件的运行环境。
某些复杂bug情况下,只有运行环境一致,仿真器才能attach到目标文件。
如何获取时间信息
这两个宏是日期和时间,格式如下。如果把这两个宏加入到代码,那么就得到了时间的字符串信息。
// example of __date__ string: dec 27 2017// example of __time__ string: 1519const 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: %04u-%02u-%02us %02u:%02u:%02u, year, month, day, hour, minute,seconds); #else printf([null] ** build at: %04u-%02u-%02u %02u:%02u:%02u, year, month, day, hour, minute,seconds); #endif debug(builddate: %s %s, __date__, __time__); return 0;}  
把上面的函数加入到代码中,就能获取工程编译的时间。
但是如果该代码所在的文件没有被修改,在非build-all情况下,编译器不会再次编译此文件,所以时间信息也就不会被更新。
如果每次都使用re-build all,一来繁琐,二来也不能保证每次都会记得点击build all按钮,靠技术手段来保证每次build都更新时间信息才是正道。
如何保证每次编译都更新时间信息
使用预编译指令,每次更新包含时间宏的文件或对应的链接文件。
在iar环境下,官方已经给出了解决的方法(using pre-build actions for time stamp
方法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.cthe .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$ est.o  
在pre-build中加入上面的命令,就会在编译前删除test.o文件。
在这种模式下,工程代码只要任何位置发生变化,代码重新编译,就会触发删除test.o,然后链接过程发现没有test.o文件,那么就会重新编译一次test.c,那么新的时间信息就会记录下来了。
虽有些曲线救国的味道,但还是很顺利的实现了目标。
只要工程的任何地方有改动,生成新的目标文件,那么目标文件中就会带有最新的编译时间。
方法3:直接告诉编译器每次重新编译某个文件更直接,mdk支持此功能。
实际上,如果对工具多一些了解,万万是不会用上面的方法的,当然上面的方法也是通用想法,是通用型知识点,容易想到,也能达到目标。
新的方法,不需要写任何脚本,如果想让代码每次都编译更新data 和 time两个宏,那么让这个文件每次都编译一次就可以了,不需要删除它的obj文件然后让编译器找不到文件而触发重新编一次,其实直接告诉编译器每次重新编译更直接,mdk支持此功能。


使用STM32 HAL库进行GPIO控制的实例
中国联通面向行业客户推出了联通云盾DDoS等多款安全产品
回顾10Gbps USB-C™ Re-Timer的性能分析和应用
英特尔砸20亿美元兴建全球首座450毫米晶圆厂
孙正义arm多少钱买的?孙正义400亿抛售只为推动计算能力增长?
让代码中包含最新的编译时间信息
虹科方案|适用于高压环境或潜在爆炸性环境的加速度计系统
基于单片机和AD574的简易电子秤设计
基于NS-2的无线网络视频QoS平台的实现
MCU是如何被破解的?(附MCU攻防技巧)
动物解剖实验VR教学系统用广州华锐互动,引领VR教育新时代
英特尔挖来晶圆代工厂格芯前CTO
智能音箱究竟有哪些用处?本文告诉你答案!
学好嵌入式Linux,需不需要全精通?专家给的建议一语中的!
基于机智云平台实现无计量4孔插座开发
传感器技术让什么领域变得智能化
2018年上半年将在全国启用新能源汽车专用号牌
单片机中高级定时器的主要功能解析
iPhone8又爆出了最新消息,运行内存为3G,支持快速充电
定氮仪的应用领域及其维护方法的简单介绍