container_of宏的作用是根据指向一个结构体某个成员的指针得到这个结构体的基址。
@include/linux/kernel.h
#define container_of(ptr, type, member) ({ \
const typeof( (type *)0->member) * __mptr = (ptr); \
(type *)( (char *)__mptr - offsetof(type, member) ); })
@include/linux/stddef.h
#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0->MEMBER)) // 这个&符号是取地址运算符, 一开始还以为是按位与运算符, size_t类型符号竟然可以参与与运算了。。。基础知识欠缺啊
这里的关键是得到结构体成员相对于结构体基址的偏移量, 构建了一个地址为0的结构体类型为type的对象, 然后 (TYPE *)0->MEMBER 就可以得到MEMBER这个成员了, &((TYPE *)0->MEMBER))就是这个成员的地址了,而结构体的基址为0, 这样就得到的偏移量了, 最后在用(size_t)进行强制类型转换, 两个指针相减不是按字节去算的,是按指针类型去计算的, 得到结构体的基址后在进行类型转换就可以得到指向结构体的指针了