引言说到预编译,大家立刻就能想到#define、#if、#ifdef和#ifndef等熟悉的预编译命令。其实#include,我们通常放在源文件用来包含头文件,它也是预编译命令。当然这不是这篇文章的重点。这篇文章主要讲解下#error预编译命令,对于什么是预编译,预编译的作用等基础知识点不再讲解,
引言
说到预编译,大家立刻就能想到#define、#if、#ifdef和#ifndef等熟悉的预编译命令。其实#include,我们通常放在源文件用来包含头文件,它也是预编译命令。当然这不是这篇文章的重点。
这篇文章主要讲解下#error预编译命令,对于什么是预编译,预编译的作用等基础知识点不再讲解,不懂的同学请自行百度。
在MCU的开发中,我们经常需要用到Flash存储一些参数,且通常情况下有很多参数需要存储。
一般采用一个扇区存储一种类型的参数(可能造成Flash空间浪费),但对于一些内部Flash容量很大的MCU,这样做也是可行的。
示例demo如下:
复制 #define PARA_BATTERY_ADDR 0x08019000 // 存储电池参数#define PARA_ETH_ADDR 0x0801B800 // 存储网络参数..........#define PARA_USER_CONFIG_ADDR 0x0801C400 // 存储用户参数
很多开发工程师喜欢这样做,如果参数少时还好,当有很多参数时,这将是一个噩梦。如果这些参数分散在各个文件中,阅读这份源码将是恶心他妈给恶心开门,恶心到家了。
当想增加一个参数存储时,如果不很熟悉代码,根本不知道哪个扇区是空闲的。
优化版本如下,将这这些宏定义统一定义在flashmap.h中。
复制 #define PARA_STARADDR ( 0x08000 + ( 100 * 1024 ) ) // 前100K用于存储固件,参数存储开始地址。#define FLASH_SECTOR_SIZE ( 2 * 1024 ) // Flash每个扇区的大小
#define PARA_BATTERY_ADDR PARA_STARADDR // 存储电池参数#define PARA_BATTERY_SIZE ( 1 * FLASH_SECTOR_SIZE) // 电池参数所占空间大小
#define PARA_ETH_ADDR ( PARA_BATTERY_ADDR+ PARA_BATTERY_SIZE) // 存储网络参数#define PARA_ETH_SIZE ( 2 * FLASH_SECTOR_SIZE) // 网络参数所占空间大小
相信大家也看出来了,这样管理参数地址,就显得很合理,不会想增加一个参数时,不知道哪个扇区是空闲的,找到了一个扇区,还是担心受怕,害怕和别的参数存储地址冲突将其覆盖。
当然它也有弊端,如果问你PARA_ETH_ADDR存储地址,并不能立刻从代码中看出,这个问题,我一般使用printf打印出来。
还有个问题,如果加了很多参数,最后一个参数地址大于Flash空间怎么办,或者软件设计一段Flash空间作为参数区,不被允许超过这个地址。
在这个代码结构中,无法从代码中直接获取,有不行使用printf肉眼判断,那么可以使用#error 解决这个问题。
#error
#error 是一种预编译器指示字,用于生成一个编译错误消息 。
用法:#error [message] //message为用户自定义的错误提示信息,可缺省。
#error 可用于提示编译条件是否满足。编译过程中的任何错误意味着无法生成最终的可执行程序。
上面的程序优化为
复制 #define PARA_STARADDR ( 0x08000 + ( 100 * 1024 ) ) // 前100K用于存储固件,参数存储开始地址。#define FLASH_SECTOR_SIZE ( 2 * 1024 ) // Flash每个扇区的大小
#define PARA_BATTERY_ADDR PARA_STARADDR // 存储电池参数#define PARA_BATTERY_SIZE ( 1 * FLASH_SECTOR_SIZE) // 电池参数所占空间大小
#define PARA_ETH_ADDR ( PARA_BATTERY_ADDR+ PARA_BATTERY_SIZE) // 存储网络参数#define PARA_ETH_SIZE ( 2 * FLASH_SECTOR_SIZE) // 网络参数所占空间大小
#define PARA_END_ADDR ( PARA_ETH_ADDR+ PARA_ETH_SIZE) // 参数的结束地址#if #error#endif
这样当你的参数最后地址大于0x0801E000,编译器机会报错,根本编译不过,如下图
类似的,#warning 用于生成编译警告消息。warning可用来提示一些非致命错误。
其他用法
限定宏定义的数值范围,下面SIZE被限制在0到100内。
复制 #define#if SIZE < 0 || SIZE > 100 #error#endifuint8_t buffer[ SIZE] ;
指定使用VS编译器
判断是否定义了某个宏,比如FreeRTOS源码中
复制 #ifndef configMINIMAL_STACK_SIZE #error Missing definition: configMINIMAL_STACK_SIZE must be defined in FreeRTOSConfig.h . configMINIMAL_STACK_SIZE defines the size ( in words) of the stack allocated to the idle task. Refer to the demo project provided for your port for a suitable value.#endif