C中的#和##

本文主要讲述c语言的一点基础语法和在内核的应用中其中的一点例子。
    
    #,##分别在c语言中是怎么作用?
    
    文章代码编译的环境:
    
    桌面环境: Ubuntu  10.04
    
    内核: linux  2.6.32
    
    编译器:gcc 4.4.3
    
    一、基本的用法
    
    1、#.参数名以#作为前缀则结果将被扩展为由实际参数的带引号的字符串。
    
    如:
    
    #define  dprint(expr)  printf(#expr " = %d\n", expr);
    
    int main()
    
    {
    
    int a = 20, b = 10;
    
    dprint(a/b);
    
    return 0;
    
    }
    
    上面的例子会打印出:
    
    a/b = 2
    
    2、##.预处理器运算符##为宏提供了一种连接实际参数的手段。如果替换文本中的参数与##相邻,则该参数将被实际参数替换,##与前后的空白将被删除,并对替换后的结果重新扫描。
    
    形成一个新的标号,如果这样产生的记号无效,或者结果依赖于##运算顺序,则结果没有定义。
    
    如:
    
    #define paste(front, back)  front ## back
    
    因此,宏调用paste(name, 1)的结果为name1.
    
    如:
    
    #define createfun(name1, name2)                     \
    
    void name1 ## name2()                                       \
    
    {                                                                                \
    
    printf("%s called\n", __FUNCTION__);     \
    
    }
    
    createfun(the, function);
    
    int main()
    
    {
    
    thefunction();
    
    return 0;
    
    }
    
    输出的结果是:thefunction called
    
    二、##可以嵌套吗?
    
    看下面的例子:
    
    #define cat(x, y)  x##y
    
    宏调用cat(var, 123)讲生成var 123.
    
    但是,宏调用cat(cat(1,2),3)没有定义:##阻止了外层调用的参数的扩展。因此,它将生成下列的记号串:
    
    cat (1,2)3.
    
    如果要再引入第二层的宏定义,如下定义:
    
    #define xcat(x, y)   cat(x,y)
    
    那么xcat(xcat(1, 2), 3)将生成123,这是因为xcat自身的扩展不包含##运算符。
    
    三、 linux 内核中例子
    
    因为是做mips架构的,所以以mips为例子。
    
    Linux 2.6.25 内核, include/asm-mips/io.h文件。拷贝一部分的代码出来。
    
    #define __BUILD_MEMORY_SINGLE(pfx, bwlq, type, irq)                                     \
    
    \
    
    static inline void pfx##write##bwlq(type val,                                                              \
    
    volatile void __iomem *mem)                                                   \
    
    {                                                                                                                                         \
    
    volatile type *__mem;                                                                                            \
    
    type __val;                                                                                                               \
    
    \
    
    __mem = (void *)__swizzle_addr_##bwlq((unsigned long)(mem));              \
    
    \
    
    __val = pfx##ioswab##bwlq(__mem, val);                                                        \
    
    \
    
    if (sizeof(type) != sizeof(u64) || sizeof(u64) == sizeof(long))                           \
    
    *__mem = __val;                                                                                           \
    
    /*在这里省略了一些代码*/
    
    }
    
    #define __BUILD_MEMORY_PFX(bus, bwlq, type)                                 \
    
    \
    
    __BUILD_MEMORY_SINGLE(bus, bwlq, type, 1)
    
    #define BUILDIO_MEM(bwlq, type)                                                             \
    
    \
    
    __BUILD_MEMORY_PFX(__raw_, bwlq, type)                                        \
    
    __BUILD_MEMORY_PFX(, bwlq, type)                                                     \
    
    __BUILD_MEMORY_PFX(__mem_, bwlq, type)                                     \
    
    BUILDIO_MEM(b, u8)
    
    BUILDIO_MEM(w, u16)
    
    BUILDIO_MEM(l, u32)
    
    BUILDIO_MEM(q, u64)
    
    跟踪宏的展开。
    
    BUILDIO_MEM(b, u8)
    
    BUILDIO_MEM(w, u16)
    
    BUILDIO_MEM(l, u32)
    
    BUILDIO_MEM(q, u64)
    
    就会生成了如下四个函数:
    
    static inline void writeb(u8 val, volatile void __iomem *mem)  {……}
    
    static inline void writew(u16 val, volatile void __iomem *mem)  {……}
    
    static inline void writel(u32 val, volatile void __iomem *mem)  {……}
    
    static inline void writeq(u64 val, volatile void __iomem *mem)  {……}
    
    同时,如果当我们用函数类似writeb之类的出现了问题,一般情况下用编辑工具是找不到函数定义的,于是乎跟踪不是去了,其实不然,可以针对里面的关键字,譬如:write.是可以找到的(在linux下面用find,grep,或者是vim的配置,都可以找到)。
    
    是的,内核代码有时候这样的调试法……
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值