前言
借调试宏的设计,梳理下宏的用法
重定向printf打印
嵌入式设备基本会配置RS232串口作为调试IO接口,假设底层串口单字节输出函数为SERIAL_PutChar()
,利用fputc()
和fputs()
重定向printf函数
void fputc(int byte, FILE* stream)
{
(void)stream;
SERIAL_PutChar(byte);
}
void fputs(const char *pstr, FILE *stream)
{
(void)stream;
while(*pstr)
{
SERIAL_PutChar(*pstr++);
}
}
这样在代码里面利用printf()
函数输出的字符串都老老实实从调试串口出来
调试宏使用场景
某个C驱动模块,希望在调试时打印调试信息,而产品代码中不显示调试信息。
V1-单参数宏
#define DRV_DEBUG 1
#if DRV_DEBUG
#define DRV_PRINT(x) printf(x)
#else
#define DRV_PRINT(x)
#endif
这个版本的DRV_PRINT(x)
只能输出单变量——纯字符串
void foo()
{
DRV_PRINT("Driver Initialize Success!");
}
不需要打印调试信息时,更改DRV_DEBUG
宏定义
#define DRV_DEBUG 0
当然也可以直接这样定义
#define DRV_PRINT printf
但是如果宏调用了多个参数:
void foo()
{
DRV_PRINT("Driver Initialize Success: ver %d.%d !", 1, 2);
}
产品代码中的#define DRV_PRINT(x)
将编译错误!
怎么办?一种处女座
接受不了的做法,多加对括号
void foo()
{
DRV_PRINT(("Driver Initialize Success: ver %d.%d !", 1, 2));
}
不管是调试代码还是产品代码,编译都OK
v2-指定参数宏
待续
v3-参数数量可变宏
待续