linux中的 __VA_ARGS__ 宏说明

可变参数宏 ...和_ _VA_ARGS_ _
__VA_ARGS__ 是一个可变参数的宏,很少人知道这个宏,这个可变参数的宏是新的C99规范中新增的,目前似乎只有gcc支持(VC6.0的编译器不支持)。
实现思想就是宏定义中参数列表的最后一个参数为省略号(也就是三个点)。这样预定义宏__VA_ARGS__就可以被用在替换部分中,替换省略号所代表的字符串。比如:
#define PR(...) printf(__VA_ARGS__)
int main()
{
    int wt=1,sp=2;
    PR("hello\n");
    PR("weight = %d, shipping = %d",wt,sp);
    return 0;
}
输出结果:
hello
weight = 1, shipping = 2
省略号只能代替最后面的宏参数。
#define W(x,...,y)错误!

宏定义中的#与##

1.#
假如希望在字符串中包含宏参数,ANSI C允许这样作,在类函数宏的替换部分,#符号用作一个预处理运算符,它可以把语言符号转化程字符串。例如,如果x是一个宏参量,那么#x可以把参数名转化成相应的字符串。该过程称为字符串化(stringizing).
#incldue <stdio.h>
#define PSQR(x) printf("the square of" #x "is %d.\n",(x)*(x))
int main(void)
{
    int y =4;
    PSQR(y);
    PSQR(2+4);
    return 0;
}
输出结果:
the square of y is 16.
the square of 2+4 is 36.
第一次调用宏时使用“y”代替#x;第二次调用时用“2+4"代#x。
2.##
##运算符可以用于类函数宏的替换部分。另外,##还可以用于类对象宏的替换部分。这个运算符把两个语言符号组合成单个语言符号。例如:
#define XNAME(n) x##n
这样宏调用:
XNAME(4)
展开后:
x4
程序:
#include <stdio.h>
#define XNAME(n) x##n
#define PXN(n) printf("x"#n" = %d\n",x##n)
int main(void)
{
    int XNAME(1)=12;//int x1=12;
    PXN(1);//printf("x1 = %d\n", x1);
    return 0;
}
输出结果:
x1=12

了解过后我们来看一个函数:module_i2c_driver(struct i2c_driver __i2c_driver);

源码如下:

553 #define module_i2c_driver(__i2c_driver) \                                                   
554     module_driver(__i2c_driver, i2c_add_driver, \
555             i2c_del_driver)

1203 #define module_driver(__driver, __register, __unregister, ...) \
1204 static int __init   __driver##_init(void) \
1205 { \
1206     return __register(&(__driver) , ##__VA_ARGS__); \
1207 } \
1208 module_init(__driver##_init); \
1209 static void __exit   __driver##_exit(void) \
1210 { \
1211     __unregister(&(__driver) , ##__VA_ARGS__); \
1212 } \
1213 module_exit(__driver##_exit);

两段宏函数,简单处理后如下:

。。。。。。

#define module_driver(__i2c_driver, i2c_add_driver,i2c_del_driver)
static int __init    __i2c_driver_init(void)
{
    return i2c_add_driver(&(__i2c_driver),...);//瞬间一目了然
}

。。。。。。

原文链接:#、##和__VA_ARGS__ - 方正圆 - 博客园

告辞

### Linux 中 `PRT_DBG` 定义及其使用场景 在嵌入式开发和内核调试过程中,定义用于简化代码编写并提高可维护性和灵活性。对于 `PRT_DBG` 这样的,在Linux环境中通常被用来实现不同级别的日志记录功能。 #### 定义解析 考虑一个典型的 `PRT_DBG` 实现方式: ```c #define PRT_DBG(fmt, ...) printk(KERN_DEBUG "[DEBUG] " fmt "\n", ##__VA_ARGS__) ``` 此利用了C预处理器特性来处理变参列表[^2]。具体来说,当调用带有参数的 `PRT_DBG` 时,这些参数会被传递给内部使用的 `printk()` 函数;如果没有额外参数,则通过 `##` 操作符移除多余的逗号,从而避免编译错误。 #### 使用实例 下面展示了一个具体的例子,说明如何应用该来进行调试信息输出: ```c #include <linux/kernel.h> #include <linux/module.h> static int __init my_module_init(void) { int value = 42; // 调试信息打印 PRT_DBG("The current value is %d.", value); return 0; } module_init(my_module_init); MODULE_LICENSE("GPL"); ``` 在这个案例中,每当模块初始化成功后会执行一次调试信息打印操作,显示当前变量 `value` 的数值。由于采用了 `KERN_DEBUG` 前缀,这使得消息能够在适当的日志级别下呈现出来。 #### 日志等级控制 值得注意的是,实际项目里可能会存在多种不同的日志等级(如警告、错误等),因此可以通过设置全局或局部变量的方式动态调整哪些类型的日志应该被捕获并显示。例如,如果只希望看到警告及以上级别的信息,则可以相应地配置环境变量或其他形式的开关机制[^4]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值