vlambda博客
学习文章列表

C语言中这些预定义宏的不可不会

关注、星标 嵌入式客栈 ,干货及时送达 C语言中这些预定义宏的不可不会

C语言中这些预定义宏的不可不会

[导读] 嵌入式er编程,一些常见的宏应该掌握如何去使用,会带给你很多有用的功能。关于标准定义宏的详细示例,见今天推送的第二篇转载文章,总结的很清晰。本文特别示例了一下如何利用日期时间宏自动内置编译时间信息进入固件。

标准预定义宏

C语言中这些预定义宏的不可不会

使用示例:

fprintf (stderr"Internal error: "
                 "negative string length "
                 "%d at %s, line %d.",
         length, __FILE__, __LINE__);

其他详细的示例可以阅读今日第二篇转载文章,总结的很清晰很详细,这里不重复了。

系统预定义宏

不同的编译器或不同的硬件体系结构会定义一些系统相关的预定义宏,以IAR for ARM为例:

C语言中这些预定义宏的不可不会

如何查看宏是否被预定义呢,我们可以查看编译器手册,除此之外,也可以用如下的方式进行预处理提示:

  • #error,用于生成一个编译错误消息,并停止编译
  • #warning用于生成编译警告,但不会停止编译

两者的语法一样,如下:

#error "message"

#warning "message"

如:

/*或 error 打印提示,但编译会停止*/
#ifdef __AAPCS__
#error “__AAPCS__ has been defined"
#endif

/*或 warning 打印提示,但编译会继续*/
#ifdef __AAPCS__
#warning “__AAPCS__ has been defined"

#endif

又比如GNU C,还定义了大量的扩展内置宏,比如:

C语言中这些预定义宏的不可不会

如何重定义一个宏

#undef 可以取消已定义的宏,比如:

#ifdef TRUE
#undef TRUE
#endif
#define TRUE 1

这样做的意义是什么呢?前面已定义了,这里又把它取消掉且再重定义,骚操作?这是有意义且有实用价值的。因为宏是按照包含关系层层展开的,本质是编译之前的文本替换。这样就可以明确指定宏在某些部位是确定的值,因为多人协调开发,或者使用别的开发包或者开发环境,有的宏已经被定义过了,如果不做此处理,这样宏值很可能不是你想要的值。

这里再强调一下:宏的本质是编译器之前的文本替换,这也是预编译的最重要任务之一

时间宏的工程实例

项目开发发布的固件,经常会需要将版本信息、编译时间信息内置在固件中,可能涉及的应用需求:

  • 进行界面显示
  • 上位机软件通信读出该信息
  • 固件通过bootloader升级时检测版本新旧等。

这时就可以直接使用 __DATE__, __TIME__实现将编译时间自动编译进发布的固件中。示例代码如下:

/*IAR for ARM 8.4为例*/
#include <stdio.h>

#define FW_VERSION  "1.0.0"
const char *firmwareInf[]={
  {__DATE__},
  {__TIME__},
  {FW_VERSION}
};

int main(void)
{
    printf("Compile Time:%s %s \nFw version:%s\n",firmwareInf[0],firmwareInf[1],firmwareInf[2]);

    while (1)
    {
    }
}

来看下疗效:

C语言中这些预定义宏的不可不会

延伸思考,还可以将firmwareInf常量字符串放入一个确定存储位置,这就需要利用链接器脚本来指定存储位置。

宏的使用在C语言中有极高的价值,如果阅读u-boot以及Linux内核代码,你会发现大量的复杂应用,所以有必要先将这些基础的宏应用熟练掌握。

如果觉得本文有价值,在看转发起来,也算对我的肯定支持。

—END

往期精彩推荐,点击即可阅读




长按关注

加群交流