linux内核链表中会使用到这两个宏
1.offsetof
#define offsetof(type, member) (size_t)(&((type*)0)->member)
这个宏定义表示求成员member相对结构体type的地址偏移,(type*)0 ,对0地址强制转为type*,取其成员地址,结果自然就是该成员相对结构体首地址的偏移量了。
2.container_of
#define container_of(ptr, type, member) ({ \
const typeof( ((type *)0)->member ) *__mptr = (ptr); \
(type *)( (char *)__mptr - offsetof(type,member) );})
这个宏定义有些啰嗦,表示:根据“ptr”这个实例化的menber指针,求出ptr所在结构体(type)变量的地址。
如果理解了offsetof,这个自然也没什么难度。
只是,你看这个宏,外面有个括号,显然是作为右值使用,但是其中又包括两个单独语句,两个分号,恐怕在Visual Studio C中无法通过编译。因为我要在Visual Studio C中验证,所以更改了这个宏:
//根据ptr的地址获取其所在typex的起始地址
#define container_of(ptr, typex, member) (typex *) ( (size_t)ptr- offsetof(typex, member) )
这里我直接用指针ptr(也就是地址)减去其偏移量,得到的当然是其所在结构体变量的首地址,道理与上面是一样的,只是免去中间变量__mptr,也避免了使用typeof这种非ISO C标准关键字。使用起来似乎也没什么问题。