container_of宏到底做了什么???

版权声明: https://blog.csdn.net/aa1050416698/article/details/82822800

首先,她的作用是: 如果知道了结构体中某个元素的指针,就可以利用该宏得到指向结构体首地址的指针,这样可以用指针访问结构体的所用变量。

在Linux内核中,她的定义如下:

#define container_of(ptr, type, member)  ({            \
    const typeof(((type *)0)->member) * __mptr = (ptr);    \
    (type *)((char *)__mptr - offsetof(type, member)); })

 

参数:

   ptr 是一个已知的指针,该指针指向结构体某个成员。

   type 是要计算的结构体类型。

   member 是要结构体中的某个成员。(就是根据该成员计算出结构体的首地址。) 

 

  第一条语句:

    const typeof(((type *)0)->member) * __mptr = (ptr);  

   利用typeof关键字,将__mptr指针变量转换成member成员的数据类型,并将ptr传入的指针赋值给__mptr。

第二条语句:

   (type *)((char *)__mptr - offsetof(type, member));

  将__mptr减去由offsetof计算member在结构体中的偏移量,这也就是container_of的最终结果。那offsetof又做了什么?

她的定义如下:

       #define offsetof(TYPE, MEMBER)    ((size_t) &((TYPE *)0)->MEMBER)

将 0 转换成 TYPE 类型的指针,也就是结构体类型的指针,改指针指向了MEMBER成员,然后取其地址,因为地址是0(可以抽象的这么看是0),所以offsetof所得到的结果就是 MEMBER 成员相对于结构体首地址的偏移量,那在计算的时候,利用MEMBER该成员的地址减去MEMBER相对于结构体首地址的偏移量,就会得到结构体的首地址。

例如:

  某个结构体的首地址是 X, 而知道某个成员的地址是0x40001314,现在求 X 的地址是多少?

    也就是计算这个:   (type *)((char *)__mptr - offsetof(type, member));   

     假如由 offsetof 计算出 member 成员相对于结构体的首地址是 14 ,而又知道__mptr = 0x40001314,所以结构体的首地址是:0x40001300.

展开阅读全文

没有更多推荐了,返回首页