offsetof是用来判断结构体中成员的偏移位置,container_of宏用来根据成员的地址来获取结构体的地址;
offsetof宏
使用offsetof宏需要包含stddef.h头文件,实例可以参考:http://www.cplusplus.com/reference/cstddef/offsetof/。
offsetof宏的定义如下:
#define offsetof(type, member) (size_t)&(((type*)0)->member)
巧妙之处在于将地址0强制转换为type类型的指针,从而定位到member在结构体中偏移位置。编译器认为0是一个有效的地址,从而认为0是type指针的起始地址。
container_of宏
使用container_of宏需要包含linux/kernel.h头文件,container_of宏的定义如下所示:
#define container_of(ptr, type, member) ({ \
const typeof( ((type *)0)->member ) *__mptr = (ptr); \
(type *)( (char *)__mptr - offsetof(type,member) );})
container_of宏分为两部分,
第一部分:const typeof( ((type *)0)->member ) *__mptr = (ptr);
通过typeof定义一个member指针类型的指针变量__mptr,(即__mptr是指向member类型的指针),并将__mptr赋值为ptr。
第二部分: (type *)( (char *)__mptr - offsetof(type,member) ),通过offsetof宏计算出member在type中的偏移,然后用member的实际地址__mptr减去偏移,得到type的起始地址,即指向type类型的指针。
第一部分的目的是为了将统一转换为member类型指针。
__mptr - offsetof(type,member)得到的就是type类型数据结构的起始地址;
根据member成员来获取所在结构体的起始地址,需要知道三个数据,分别是:
- 成员的名字---->member
- 成员的起始地址---->ptr,用来指向member成员的指针
- 成员所在的结构体---->type
宏container_of的作用就是根据成员member的地址计算得到type结构体的地址;