Linux第一宏:container_of

这段代码展示了 Linux 内核编程中最经典的两个宏:offsetofcontainer_of。它们是 C 语言中实现"逆向寻找结构体入口"的核心工具。

下面我为你逐行拆解这两个宏的工作原理。


1. offsetof 宏:计算成员偏移量

c 复制代码
#define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)
  • 目的 :计算结构体成员 MEMBER 距离结构体 TYPE 起始地址的字节数。
  • 拆解
    1. (TYPE *)0:将地址 0 强制转换为指向 TYPE 结构体的指针。这并不会导致崩溃,因为我们只是在进行地址运算,并没有真正去读取(解引用)这个地址里的内容。
    2. ((TYPE *)0)->MEMBER:假设结构体从 0 开始,我们要访问其中的 MEMBER
    3. &(...):获取这个成员的地址。
    4. 由于起始点是 0,那么成员的地址值(例如 8)刚好就是它相对于起始点的偏移量
    5. (size_t):最后强制转换为无符号整型。

2. container_of 宏:根据成员指针找结构体首地址

这是内核开发的"黑科技",它能让你在只知道某个成员指针的情况下,反向推导出整个结构体的首地址。

c 复制代码
#define container_of(ptr, type, member) ({ \
    const typeof( ((type *)0)->member ) *__mptr = (ptr); \
    (type *)( (char *)__mptr - offsetof(type,member) );})
  • 参数说明

    • ptr:指向结构体内部成员的指针。
    • type:容器结构体的类型。
    • member:成员在结构体中的变量名。
  • 拆解

    1. 第一行(类型安全检查)
      • typeof( ((type *)0)->member ):通过 typeof 获取成员的实际类型。
      • 它定义了一个临时指针 __mptr 指向 ptr。如果 ptr 的类型和结构体定义的成员类型不匹配,编译器会报错,保证了程序的健壮性。
    2. 第二行(核心计算)
      • (char *)__mptr:将成员指针强转为 char *,因为 char 的大小是 1 字节,指针加减运算是以字节为单位进行的。
      • - offsetof(type, member):用成员的实际地址 减去成员在结构体中的偏移量
      • (type *):最后将计算得到的地址强转为结构体类型。

3. 图解与例子

假设我们有如下结构体:

c 复制代码
struct Person {
    int age;      // 偏移量 0
    char gender;  // 偏移量 4
    int id;       // 偏移量 8
};

struct Person p1;
int *id_ptr = &p1.id; // 我们手里只有指向 id 的指针

如果我们想通过 id_ptr 找到 p1 的起始位置:

  1. offsetof(struct Person, id) 的值是 8
  2. container_of(id_ptr, struct Person, id) 的逻辑是:
    • 获取 id_ptr 的内存地址(比如 0x1008)。
    • 计算 0x1008 - 8 = 0x1000
    • 0x1000 就是结构体 p1 的起始地址。

4. 为什么要这么做?

这种设计模式被称为**"侵入式链表(Intrusive Link List)"**。

在传统的 Java 或 C++ 集合类中,是容器包裹数据。但在 Linux 内核中,为了追求极致效率,通常是将"链表节点"嵌入到具体的业务结构体中。当内核遍历链表得到一个节点指针时,它必须通过 container_of 找回包含这个节点的业务结构体,从而访问其数据。

总结:

  • offsetof:我在结构体的什么位置?
  • container_of:我是谁的成员?(找到"父亲"是谁)

linux内核源码

相关推荐
踏着七彩祥云的小丑1 天前
嵌入式——认识电子元器件——电容系列
单片机·嵌入式硬件
海绵宝宝的月光宝盒1 天前
2-非金属材料
经验分享·笔记·学习·其他·职场和发展·课程设计·制造
zzh0811 天前
GluserFS笔记
笔记
小饕1 天前
RAG学习之- RAG 数据导入完整指南
人工智能·python·学习
cyr___1 天前
Unity教程(二十六)技能系统 黑洞技能(上)基础实现
学习·游戏·unity·游戏引擎
星幻元宇VR1 天前
VR党建蛋椅|以沉浸式体验推动党建学习方式创新
科技·学习·安全·vr·虚拟现实
xiangw@GZ1 天前
EMC原理:CS传导抗扰度测试总结
单片机·嵌入式硬件
三佛科技-134163842121 天前
PL3325CS/CD/CH/CE 与PL3325BE 之间的对比与联系(应用功率与典型应用电路)
单片机·嵌入式硬件·物联网·智能家居·pcb工艺
大学生小郑1 天前
如何定义图像质量,如何评价图像质量
图像处理·学习·音视频·视频
nuoxin1141 天前
CH6001FN/BW-富利威
网络·人工智能·嵌入式硬件·fpga开发·dsp开发