typeof、offsetof、container_of
typeof
用于获取一个对象的类型,比如:
unsigned int a = 1; // typeof (a) is unsigned int short b = 2; // typeof (b) is short
offsetof
#define offsetof(TYPE, MEMBER) ((size_t)&((TYPE *)0)->MEMBER)
获取某个成员变量(MEMBER)在其所在的结构体(TYPE)里的偏移:
- 首先将地址0强行转换为 TYPE * 类型
- 再将此对象中的MEMBER取地址后,强制转化为size_t格式返回
因为该对象的起始地址为0,那么MEMBER处的地址自然是其在此结构体中的偏移地址。
container_of
// non ISO variant from linux kernel; checks ptr type, but triggers 'ISO C forbids braced-groups within expressions [-Wpedantic]' // __extension__ is here to disable this warning #define container_of(ptr, type, member) ( __extension__ ({ \ const typeof( ((type *)0)->member ) *__mptr = (ptr); \ (type *)( (char *)__mptr - offsetof(type,member) );}))
它的作用是根据一个结构体成员的地址来获取该结构体的地址:
- typeof( ((type *)0)->member ) 就是获取type->member的数据类型,并用该类型创建一个const 的指针 __mptr
- __mptr的地址是传入的数据结构体成员的地址
- 把__mptr的地址强制转为(char *)的地址,然后再减去它在结构体中的偏移地址,不就是此结构体的地址了嘛
- 最后将此地址再强制转换为它本来的类型:type