Linux内核中经常可见container_of
的身影,它在实际驱动的编写中也是广泛应用。container_of
的身影,它在实际驱动的编写中也是广泛应用。
作用:通过结构体的某个成员变量地址找到该结构体的首地址。
定义如下:
/** * container_of - cast a member of a structure out to the containing structure * @ptr: the pointer to the member. * @type: the type of the container struct this is embedded in. * @member: the name of the member within the struct. * */ #define container_of(ptr, type, member) ({ \ const typeof( ((type *)0)->member ) *__mptr = (ptr); \ (type *)( (char *)__mptr - offsetof(type,member) );})
ptr
:结构体成员变量的指针type
:结构体类型member
:结构体成员变量的名字换句话说,叫:已知结构体type
的成员member
的地址ptr,求解结构体type
container_of原理
作用
#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)
ptr
:结构体成员变量的指针type
:结构体类型member
:结构体成员变量的名字type
的成员member
的地址ptr,求解结构体type
的起始地址🎜。🎜计算公式为:类型
的起始地址 = ptr
-size
(size 为成员的大小)type
的起始地址 = ptr
-size
(size为member的大小)
以一幅图说明ptr
、type
、member
的关系:
container_of
的妙处就在于以0
作为成员变量member
的基址。
其中定义了一个中间变量__mptr
,"__
"代表内部使用,“m
”代表middle
ptr
、<代码样式=“字体大小:14px;填充:2px 4px;边框半径:4px;右边距:2px;左边距:2px;背景颜色:rgba(27,31,35,0.05);font- family: "Operator Mono", Consolas, Monaco, Menlo, monospace;word-break: 全部分解;color: rgb(239, 112, 96);">类型、成员
的关系:🎜container_of
的妙处就在🎜以0
成员作为变量成员
的基址。🎜🎜其中定义了一个中间变量__mptr
,"__
"代表内部使用,“m
”代表middle
。🎜#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)
typeof( ((type *)0)->member )
是获取member
的类型,__mptr = (ptr)
判断ptr
与member
是否为同一类型,offsetof
计算成员member
的大小size
。
例如内核的pwm
驱动,通过成员变量chip
,找到结构体bcm2835_pwm
:
struct bcm2835_pwm { struct pwm_chip chip; struct device *dev; void __iomem *base; struct clk *clk; }; static inline struct bcm2835_pwm *to_bcm2835_pwm(struct pwm_chip *chip_ptr) { return container_of(chip_ptr, struct bcm2835_pwm, chip); }
使用container_of
通常都会定义一个函数,并且命名为to_xxx
或者to_find_xxx
,代表要找xxx
这个结构体,传参则传入成员变量指针,另外函数也会声明为inline
。
以上是Linux内核基础篇——container_of原理和实际应用的详细内容。更多信息请关注PHP中文网其他相关文章!