2017-07-20 57 views
-2

我在使用X宏中的offsetof()时出现明显错误。下面的代码显示了一个相当简单的结构的例子,一个是明确定义的,另一个是使用X宏定义的。然后将两个结构中每个结构域的偏移量打印到屏幕上。正如您在下面的注释中看到的那样,使用X宏定义的结构中的字段显示不正确的偏移量。x宏打印错误offsetof()信息

我认为这是一个打印问题。有人可以解释一下这个谜团吗?

#include <stdio.h> 
#include <stdlib.h> 

const char fmt[] = "%-5s 0x%04x(%05d)\n"; 

#define i(s, f) \ 
    #f, \ 
    offsetof(s, f), \ 
    offsetof(s, f) 

int main(void) 
{ 
    /********************************** 
    METHOD1 - CORRECTLY PRINTS 

     a  0x0000(00000) 
     b  0x0004(00004) 
     c  0x0008(00008) 
     d  0x0030(00048) 

    **********************************/ 
    typedef struct { 
     uint32_t a; 
     uint32_t b; 
     uint32_t c[10]; 
     uint32_t d; 
    } struct1; 

    printf(fmt, i(struct1, a)); 
    printf(fmt, i(struct1, b)); 
    printf(fmt, i(struct1, c)); 
    printf(fmt, i(struct1, d)); 

    printf("\n"); 

    /********************************** 
    METHOD2 - PRINTS WRONG INFO 

     a  0x0000(00000) 
     b  0x0004(00004) 
     c[10] 0x0030(00048) <== WRONG 
     d  0x0030(00048) 

    **********************************/ 

    #define FIELDS \ 
     X(uint32_t, a,  "") \ 
     X(uint32_t, b,  "") \ 
     X(uint32_t, c[10], "") \ 
     X(uint32_t, d,  "") \ 

    typedef struct { 
    #define X(type, name, descr) type name; 
     FIELDS 
    #undef X 
    } struct2; 

    #define X(type, name, descr) printf(fmt, i(struct2, name)); 
     FIELDS 
    #undef X 

    return 0; 
} 
+0

您使用错误的格式说明符,导致未定义的行为。 '%x'用于unsigned int,'%d'用于'int',但'offsetof'则产生'size_t'。 –

+0

你的X-macro最终输出了'offsetof(struct2,c [10])',这与第一个例子不同,它的结果是'offsetof(struct1,c)' –

+0

谢谢@MM澄清问题发生的位置。你能想出一种方法来重新定义我的x宏以产生所需的结果吗? –

回答

3

在第二X宏:

#define X(type, name, descr) printf(fmt, i(struct2, name)); 
    FIELDS 

此扩展为:

printf(fmt, "a", offsetof(struct2, a), offsetof(struct2, a)); 
printf(fmt, "b", offsetof(struct2, b), offsetof(struct2, b)); 
printf(fmt, "c[10]", offsetof(struct2, c[10]), offsetof(struct2, c[10])); 
printf(fmt, "d", offsetof(struct2, d), offsetof(struct2, d)); 

offsetof(struct2, c[10])显然是不一样offsetof(struct2, c)

-1

在上述评论的帮助下,我想出了答案。对于数组的结构元素,我需要有一个不同的X宏定义。

#include <stdio.h> 
#include <stdlib.h> 

const char fmt[] = "%-5s 0x%04x(%05d)\n"; 

#define i(s, f) \ 
    #f, \ 
    offsetof(s, f), \ 
    offsetof(s, f) 

int main(void) 
{ 
    /********************************** 
     METHOD1 - CORRECTLY PRINTS 

      a  0x0000(00000) 
      b  0x0004(00004) 
      c  0x0008(00008) 
      d  0x0030(00048) 

    **********************************/ 
    typedef struct { 
     uint32_t a; 
     uint32_t b; 
     uint32_t c[10]; 
     uint32_t d; 
    } struct1; 

    printf(fmt, i(struct1, a)); 
    printf(fmt, i(struct1, b)); 
    printf(fmt, i(struct1, c)); 
    printf(fmt, i(struct1, d)); 

    printf("\n"); 

    /********************************** 
     METHOD2 - PRINTS WRONG INFO 

      a  0x0000(00000) 
      b  0x0004(00004) 
      c  0x0008(00008) <== CORRECT 
      d  0x0030(00048) 

    **********************************/ 

    #define FIELDS \ 
     X(uint32_t, a,  "") \ 
     X(uint32_t, b,  "") \ 
     AX(uint32_t, c, 10, "") \ 
     X(uint32_t, d,  "") \ 

    typedef struct { 
    #define X(type, name, descr)  type name; 
    #define AX(type, name, size, descr) type name[size]; 
     FIELDS 
    #undef X 
    #undef AX 
    } struct2; 

    #define X(type, name, descr)  printf(fmt, i(struct2, name)); 
    #define AX(type, name, size, descr) printf(fmt, i(struct2, name)); 
     FIELDS 
    #undef X 
    #undef AX 

    return 0; 
}