假设有
container_of实现了根据一个结构体变量中的一个域成员变量的指针来获取指向整个结构体变量的指针的功能。
container_of说明:
#define container_of(ptr, type, member) ({ \
const typeof( ((type *)0)->member ) *__mptr = (ptr); \
(type *)( (char *)__mptr - offsetof(type,member) );})
container_of(kernel.h)在Linux Kernel中的应用非常广泛,它用于获得某结构中某成员的入口地址.
---------------------------------------------------------------------------------------------------------------------------------------------------------------
关于offsetof见stddef.h中:
#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)
---------------------------------------------------------------------------------------------------------------------------------------------------------------
关于typeof,这是gcc的C语言扩展保留字,用于声明变量类型.
const typeof( ((type *)0->member ) *__mptr = (ptr);
是定义一个__mptr指针变量,类型和member的类型一样
typeof是获得一个变量的类型,((type *)0)->member 则是tpye类型中的member 变量,一般type为结构体类型,member 则为其中的变量
这里的0只是作为一个临时的指针地址用,任何可以表示地址的数字都可以代替0
(type *)( (char *)__mptr - offsetof(type,member) );
意思是__mptr的地址减去member在该struct中的偏移量得到的地址, 再转换成type型指针.
该指针就是member的入口地址了.
---------------------------------------------------------------------------------------------------------------------------------------------------------------
container_fo举例:
struct demo_struct {
type1 member1;
type2 member2;
type3 member3;
type4 member4;
};
struct demo_struct demo;
同时,如果在另一个地方,获得了变量demo中的某一个域成员变量的指针,比如:
type3 *memp = get_member_pointer_from_somewhere();
此时,如果需要获取指向整个结构体变量的指针,而不仅仅只是其某一个域成员变量的指针,我们就可以这么做:
struct demo_struct *demop = container_of(memp, struct demo_struct, member3);