首先,她的作用是: 如果知道了结构体中某个元素的指针,就可以利用该宏得到指向结构体首地址的指针,这样可以用指针访问结构体的所用变量。
在Linux内核中,她的定义如下:
#define container_of(ptr, type, member) ({ \
const typeof(((type *)0)->member) * __mptr = (ptr); \
(type *)((char *)__mptr - offsetof(type, member)); })
参数:
ptr 是一个已知的指针,该指针指向结构体某个成员。
type 是要计算的结构体类型。
member 是要结构体中的某个成员。(就是根据该成员计算出结构体的首地址。)
第一条语句:
const typeof(((type *)0)->member) * __mptr = (ptr);
利用typeof关键字,将__mptr指针变量转换成member成员的数据类型,并将ptr传入的指针赋值给__mptr。
第二条语句:
(type *)((char *)__mptr - offsetof(type, member));
将__mptr减去由offsetof计算member在结构体中的偏移量,这也就是container_of的最终结果。那offsetof又做了什么?
她的定义如下:
#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)
将 0 转换成 TYPE 类型的指针,也就是结构体类型的指针,改指针指向了MEMBER成员,然后取其地址,因为地址是0(可以抽象的这么看是0),所以offsetof所得到的结果就是 MEMBER 成员相对于结构体首地址的偏移量,那在计算的时候,利用MEMBER该成员的地址减去MEMBER相对于结构体首地址的偏移量,就会得到结构体的首地址。
例如:
某个结构体的首地址是 X, 而知道某个成员的地址是0x40001314,现在求 X 的地址是多少?
也就是计算这个: (type *)((char *)__mptr - offsetof(type, member));
假如由 offsetof 计算出 member 成员相对于结构体的首地址是 14 ,而又知道__mptr = 0x40001314,所以结构体的首地址是:0x40001300.