在之前学习RT-Thread操作系统时,我发现一个比较有趣的现象:
串口打印的日志中竟然包含着当前时间!并且,我每天做实验时,这个日期都会变化,还能保持和当前时间一致!
我的好奇心被引发了,系统会不会偷偷配置了RTC,不然它怎么知道现在几点了?
怀揣着问题,我决定要去探索一下。
2. 系统打印出的当前时间这是RT-Thread刚上电时控制台默认打印的内容,可以看到日期在今天:
再来看看LiteOS的,不仅能打印出当前日期,还能精确到时分秒:
其实,这些系统之所以准确的打印出当前时间,和板子硬件没有任何关系,更不会使用的RTC,只是在代码里巧妙的利用了C语言的一个不常用知识点 —— 编译器内置宏定义。
C语言编译器中内置了一些宏定义,这些内置宏定义可以巧妙地帮我们输出非常有用的调试信息,比如打印时间就用到了下面这两个宏定义:
-
__DATE__
:在源文件中插入当前的编译日期; -
__TIME__
:在源文件中插入当前编译时间;
编写一个简单的C程序测试一下:
#include <stdio.h>
int main(void)
{
printf("_DATE_ is:%s\r\n", __DATE__);
printf("_TIME_ is:%s\r\n", __TIME__);
return 0;
}
编译运行,测试结果如下:
RT-Thread的源码实现如下,是不是用__DATE__
这个宏定义实现:
void rt_show_version(void)
{
rt_kprintf("\n \\ | /\n");
rt_kprintf("- RT - Thread Operating System\n");
rt_kprintf(" / | \\ %d.%d.%d build %s\n",
RT_VERSION, RT_SUBVERSION, RT_REVISION, __DATE__);
rt_kprintf(" 2006 - 2019 Copyright by rt-thread team\n");
}
再来看看LiteOS的源码实现,是不是也用__TIME__
和__DATE__
这两个宏定义实现:
static char s_link_mainversion[64];
const char *linkmain_version()
{
snprintf(s_link_mainversion,64,"V%d.%d.%d AT %s ON %s",CN_LINK_VERSION_MAJOR,\
CN_LINK_VERSION_MINOR,CN_LINK_VERSION_FEATURE,__TIME__,__DATE__);
return s_link_mainversion;
}
需要注意的是,这两个编译器内置宏定义只有在编译程序的时候被赋值,表示的是程序编译时间。
接收更多精彩文章及资源推送,欢迎订阅我的微信公众号:『mculover666』。