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内核源码

相关推荐
sheeta19981 天前
LeetCode 每日一题笔记 日期:2026.05.13 题目:1674. 使数组互补的最少操作次数
笔记·算法·leetcode
z200509301 天前
【linux学习】linux的基本指令
linux·学习
叁散1 天前
实验项目1 LTE通信原理与应用
笔记·其他
AOwhisky1 天前
Docker 学习笔记:镜像分发、容器运行与资源限制
笔记·学习·docker
MediaTea1 天前
人工智能通识课:机器学习之无监督学习
人工智能·深度学习·学习·机器学习
黑白园1 天前
I2C_GPIO模拟 读取AS5600编码器数据
stm32·单片机·嵌入式硬件
羽获飞1 天前
从零开始学嵌入式之STM32——34.ADC-模数转换
stm32·单片机·嵌入式硬件
csg11071 天前
智慧养殖篇(四):猪场自动化饲喂与疫病预警
运维·单片机·嵌入式硬件·物联网·自动化
羽获飞1 天前
从零开始学嵌入式之STM32——33.直接存储器访问-DMA
stm32·单片机·嵌入式硬件
Fanfanaas1 天前
2026 年 华中杯 数学建模竞赛 A 题 问题一个人解析
学习·数学建模