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

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

3天内不再提示

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

STM32嵌入式开发 来源:STM32嵌入式开发 2023-10-25 09:58 次阅读

如何保证发布出去的bin文件是最终测试通过的版本?

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

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

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

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

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

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

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

如何获取时间信息

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



// Example of __DATE__ string: "Dec 27 2017"
// Example of __TIME__ string: "1519"
constchar*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 command


You 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$	est.o""

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

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

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

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

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

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

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

wKgaomU4duuAfWP7AABcNqdC4-s430.png

审核编辑:汤梓红

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

    关注

    14

    文章

    988

    浏览量

    82998
  • 字符串
    +关注

    关注

    1

    文章

    552

    浏览量

    20147
  • 代码
    +关注

    关注

    30

    文章

    4556

    浏览量

    66814
  • 编译
    +关注

    关注

    0

    文章

    615

    浏览量

    32397

原文标题:让代码中包含最新的编译时间信息

文章出处:【微信号:c-stm32,微信公众号:STM32嵌入式开发】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    ARM代码编译链接的工作流程

    梳理下ARM代码编译链接的工作流程,以及过程需要的相关概念信息,不具体关注编译链接的具体命令。一、编译
    发表于 06-08 10:22

    如何在源代码包含包含文件的定义?

    我在C编写代码,但需要与包含文件通信。如何在源代码包含
    发表于 09-17 11:51

    编译项目有日期和时间戳的自动生成号能插入代码

    您好,如果我编译这个项目,我想插入一个带有日期和时间戳的自动生成号。是否可以将它插入代码?谢谢
    发表于 04-22 10:14

    vscode code-2.0 window编译问题-编译时间很长

    大家有没有发现,编译当前代码时,如果当前代码有错误,vscode会把之前所有程序编译过报错的信息全部显示一遍,最后才报错当前
    发表于 11-01 19:39

    C编译器的设计文档与源代码

    C-编译器的设计文档与源代码:本压缩包包含了C-编译器的设计文档与源代码,供学习参考。  整体框架. 3ɨ
    发表于 02-09 11:13 45次下载

    编译代码与源代码的比较算法

    编译代码与源代码
    发表于 02-09 15:19 10次下载

    iftop源代码编译安装

    iftop源代码编译安装
    发表于 04-03 23:39 15次下载
    iftop源<b class='flag-5'>代码</b><b class='flag-5'>编译</b>安装

    谁能缩短大容量FPGA的编译时间?增量式编译QIC!

    增量式编译(Incremental Compilation)是ALTERA为解决大容量FPGA设计编译时间太长的问题给出的一个新式工具!在本文中我们将阐述QIC在缩短编译
    发表于 12-25 11:26 4502次阅读

    Android源代码编译

    Android源代码编译
    发表于 10-24 09:30 5次下载
    Android源<b class='flag-5'>代码</b><b class='flag-5'>编译</b>

    CubeMX版本编译代码的教程

    用的CubeMX版本是4.20.0如下图,用的Keil5编译代码编译结果0 Error(s),0 Warning(s)。
    的头像 发表于 10-10 10:32 3934次阅读
    CubeMX版本<b class='flag-5'>编译</b><b class='flag-5'>代码</b>的教程

    ARM代码编译链接的工作流程

    ARM处理器在市面上到处都是ARM7、ARM9、Cortex-M、Cortex-R、Cortex-A包含的种类繁多,今天我们就来了解一下ARM代码编译链接的工作流程,以及过程中需要的相关概念
    的头像 发表于 12-22 16:57 1903次阅读

    Keil MDK使用编译器AC5与AC6生成浏览信息的差异有哪些

    很多使用过Keil MDK的工程师都发现了一个问题,Keil MDK编译代码的时候,速度很慢。   编译速度慢有几个原因: 编译代码量大
    的头像 发表于 01-18 11:17 4142次阅读
    Keil MDK使用<b class='flag-5'>编译</b>器AC5与AC6生成浏览<b class='flag-5'>信息</b>的差异有哪些

    什么是预编译

    了程序在正式编译前,由编译器进行的操作,可以放在程序中的任何位置。 什么时候使用预编译? 1、总是使用不经常改动的大型代码体 2、程序
    发表于 01-13 13:52 0次下载
    什么是预<b class='flag-5'>编译</b>

    ARM代码编译与链接调试的工作流程梳理

    梳理下ARM代码编译链接的工作流程,以及过程中需要的相关概念信息,不具体关注编译链接的具体命令。
    发表于 02-09 10:03 1次下载
    ARM<b class='flag-5'>代码</b><b class='flag-5'>编译</b>与链接调试的工作流程梳理

    【GCC编译优化系列】实战分析C代码遇到的编译问题及解决思路

    【GCC编译优化系列】实战分析C工程代码可能遇到的编译问题及其解决思路
    的头像 发表于 07-10 23:15 966次阅读
    【GCC<b class='flag-5'>编译</b>优化系列】实战分析C<b class='flag-5'>代码</b>遇到的<b class='flag-5'>编译</b>问题及解决思路