首页  >  文章  >  运维  >  Linux内核基础篇——container_of原理和实际应用

Linux内核基础篇——container_of原理和实际应用

嵌入式Linux充电站
嵌入式Linux充电站转载
2023-07-31 15:46:131093浏览

Linux内核中经常可见container_of的身影,它在实际驱动的编写中也是广泛应用。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的大小)

以一幅图说明ptrtypemember的关系:

Linux内核基础篇——container_of原理和实际应用
  • 原理简述:

container_of的妙处就在于0作为成员变量member的基址

其中定义了一个中间变量__mptr,"__"代表内部使用,“m”代表middle

以一幅图说明ptr、类型、成员的关系:🎜
Linux内核基础篇——container_of原理和实际应用
  • 原理简述:
🎜container_of的妙处就在🎜以0成员作为变量成员的基址
。🎜🎜其中定义了一个中间变量__mptr,"__"代表内部使用,“m”代表middle 。🎜
#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)

typeof( ((type *)0)->member )是获取member的类型,__mptr = (ptr)判断ptrmember是否为同一类型,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中文网其他相关文章!

声明:
本文转载于:嵌入式Linux充电站。如有侵权,请联系admin@php.cn删除