一、在Linux Kernel中关于offsetof与container_of宏的定义为:
二、offsetof宏介绍
A. offsetof宏的作用是:用宏来计算结构体中某个元素相对结构体首地址的偏移量(其实本质是通过编译器来帮我们计算)
B. offsetof宏的原理:我们虚拟一个type类型结构体变量,然后用type.member的方式来访问那个member元素,继而得到
member相对于整个变量首地址的偏移量。
C. 学习思路:
第一步:先学会用offsetof宏
第二步:再去理解这个宏的实现原理
(TYPE *)0,这是一个强制类型转换,把0地址强制类型转换成一个指针,这个指针指向一个TYPE类型的结构体变量。
(实际上这个结构体变量可能不存在,但是只要我们不要解引用这个指针就不会出错)。
((TYPE *)0)->MEMBER,(TYPE *)0是一个TYPE类型结构体变量的指针,通过指针来访问这个结构体变量的member元素。
&((TYPE *)0)->MEMBER 等效于&(((TYPE *)0)->MEMBER),意义就是得到member元素的地址,但是因为整个结构体变量的首 地址为0。
三、container_of宏介绍
A. 作用:知道一个结构体中某个元素的指针,反推这个结构体变量的指针,有了container_of宏,我们可以从一个元素的指针得到整个结构体变量的指针,继而得到结构体中其他元素的指针。
B.typeof关键字的作用是:typeof(a)时由变量a得到a的类型,typeof就是由变量名得到变量数据类型的。
C.这个宏的工作原理:先用typeof得到member元素的类型定义成一个指针,然后用这个指针减去改元素相对于整个结构体变量的偏移量(偏移量用offsetof宏得到的),减去之后得到的就是整个结构体变量的首地址了,再把这个这个地址强制类型转换为type *即可。
四、应用举例
在源文件: leds-s5pv210-platform.c中有用到为例,进行分析
解析: 由于在s5pv210_led_set(...)函数中传参是struct led_classdev *led_cdev,而在struct led_classdev结构体中不存在gpio编号,而没法直接操作硬件,故技巧性的定义了结构体s5pv210_gpio_led,在这个结构体里面包含了struct led_classdev cdev元素,所以在知道元素cdev的地址时就可以知道结构体s5pv210_gpio_led的基地址,从而可以获得pdata的地址,struct s5pv210_led_platdata 包含了成员gpio,故可以获取到对应的gpio值,顺利操作硬件。