Container_of
Container_of在Linux内核中是一个常用的宏,用于从包含在某个结构中的指针获得结构本身的指针,通俗地讲就是通过结构体变量中某个成员的首地址进而获得整个结构体变量的首地址。
这个宏定义在include/linux/kernel.h里
#define container_of(ptr, type, member) ({ \
const typeof( ((type *)0)->member ) *__mptr = (ptr); \
(type *)( (char *)__mptr - offsetof(type,member) );})
对于container_of(ptr, type, member) 通过ptr的地址可以返回结构体的首地址
- ptr:表示结构体中member的地址
- type:表示结构体类型
- member:表示结构体中的成员
第一步,首先定义一个临时的数据类型(通过typeof( ((type *)0)->member )获得)与ptr相同的指针变量__mptr,然后用它来保存ptr的值。
第二步,用(char *)__mptr减去member在结构体中的偏移量,得到的值就是整个结构体变量的首地址(整个宏的返回值就是这个首地址)。
offsetof 在内核中的定义:
#define offsetof(TYPE, MEMBER) ((size_t)&((TYPE *)0)->MEMBER)
举个例子:
首先通过offsetof得到偏移量
- 结构体s, 地址表示为 &s;
- 结构体s成员变量m,地址表示为 &m,
偏移量 可以表示为“&m - &s”:即成员变量的首地址相对于结构体首地址的偏移量。
如果 &s 等于0 ,那么 &m - &s = &m - 0 = &m :成员变量的首地址,等于偏移量
这个就是offsetof的作用
通过ptr的地址可以返回结构体的首地址
通过offsetof有了偏移量,再用成员变量的实际地址ptr,减去上面说的偏移量,就是当前结构体的首地址!