内核中常用宏定义| container_of

文章目录

    • 前言
    • container_of函数介绍
    • container_of函数实现
    • container_of函数解析
    • offsetof的使用
    • container_of的使用
    • 结语

前言

前两篇我们写到内核中两种C语言高级语法__attribute__, __read_mostly。本篇写内核中另外一种常用宏定义之container_of

container_of函数介绍

container_of是内核中使用最为常用的一个函数了,简单来说,它的主要作用是根据结构体中的已知的成员变量的地址,来寻求该结构体的首地址,直接看图,更容易理解。

container_of函数实现

5.10内核源码定义位置:include/linux/kernel.h(不同的内核函数实现会小有不同)

 /**
    * 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) ({                              \
           void *__mptr = (void *)(ptr);                                   \
           BUILD_BUG_ON_MSG(!__same_type(*(ptr), ((type *)0)->member) &&   \
                           ¦!__same_type(*(ptr), void),                    \
                           ¦"pointer type mismatch in container_of()");    \
           ((type *)(__mptr - offsetof(type, member))); })

传入三个参数

  • ptr, 表示结构体中的某一成员变量
  • type, 表示结构体的类型
  • member, 表示该变量在结构体中的成员具体命名

container_of函数解析

该宏定义下面有三个语句:

  • void *__mptr = (void *)(ptr); \
  •       BUILD_BUG_ON_MSG(!__same_type(*(ptr), ((type *)0)->member) &&   \
                         ¦!__same_type(*(ptr), void),                    \
                         ¦"pointer type mismatch in container_of()");    \
    
  •        ((type *)(__mptr - offsetof(type, member))); })
    

从语法角度,我们可以看到,container_of 宏的实现由一个语句表达式构成。说到container_of宏就必须说下offsetof宏语句,因为语句表达式的值即为最后一个表达式的值:

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

最后一句的意义是拿到结构体成员member的地址,减去这个成员在结构体type中的偏移,结果就是结构体type的首地址。那么如何获取结构体的偏移呢?内核用定义了offsetof宏来定义实现这个功能。

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

这个宏有两个参数,一个是结构体类型TYPE,一个是结构体成员MEMBER,就是获得0地址常量指针的偏移是一样的,如果是结构体的首地址为0的话,那么获取的变量地址既是绝对地址也是结构体内的偏移。

offsetof的使用

不用内核offsetof宏,我们求地址偏移像下面这样

struct student{
    int num;
    int age;
    int math;
};

int main(void)
{
     printf("%p\n",&((struct student *)0)->num);
    printf("%p\n",&((struct student *)0)->age);
    printf("%p\n",&((struct student *)0)->math); 
    return 0;
}

在上面的程序中,我们没有直接定义结构体变量,而是将数字0,通过强制类型转换,转换为一个指向结构体类型为 student 的常量指针,然后分别打印这个常量指针指向的结构体的各成员地址。运行结果如下:

因为常量指针为0,即可以看做结构体首地址为0,所以结构体中每个成员变量的地址即为该成员相对于结构体首地址的偏移

在尝试代入用offsetof的宏

#define container_of(ptr, type, member) ({                              \
           void *__mptr = (void *)(ptr);   \  
           ((type *)(__mptr - offsetof(type, member))); })

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

struct ST
{
    int i;
    int j;
    char c;
};

int main()
{
    printf("offset i: %d\n",offsetof(struct ST,i));
    printf("offset j: %d\n",offsetof(struct ST,j));
    printf("offset c: %d\n",offsetof(struct ST,c));

    return 0;
}

运行结果如下:

这样就可以理解结构体成员是以4字节对齐进行偏移的

container_of的使用

理解了offsetof的使用,我们就可以在写个demo来理解下container_of的使用

#define container_of(ptr, type, member) ({				\
	void *__mptr = (void *)(ptr);					\
	((type *)(__mptr - offsetof(type, member))); })



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

struct ST
{
    int member1;
    int member2;
    int member3;
};

int main()
{
    struct ST stest;
    
    printf("&stest: 0x%x\n",&stest);
    printf("offsetof(struct ST,member3) : %d\n",offsetof(struct ST,member3));
    printf("&stest.member1: 0x%x\n",&stest.member1);
    printf("&stest.member2: 0x%x\n",&stest.member2);
    printf("&stest.member3: 0x%x\n",&stest.member3);
    printf("container_of member3: 0x%x\n", container_of(&stest.member3, struct ST, member3));

    return 0;
}

运行结果如下


可以看到stest和stest.member1的地址和container_of计算的首地址一样的

画个图可能大家更好理解

结语

相信大家对内核宏container_of的语法有所了解,如果我的文章对你有所收获的话,可以点赞关注转发给你小伙伴们


作者潘小帅, 是一名Linux底层爱好者,不定期写写技术原创文章和分享职场面试文章,徒步,旅游,看电影的爱好,喜欢我的文章可以点赞收藏+关注,感谢你的支持,微信公众号【Linux随笔录】

参考文章:

https://blog.csdn.net/xi_xix_i/article/details/134625972
https://zhuanlan.zhihu.com/p/672129384

本文由mdnice多平台发布

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/593287.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

高级事件.

高级事件 1. 注册事件(addEventListener)2.删除事件(removeEventListener)3.DOM事件流4.事件对象及其方法(当形参来看)5.阻止默认事件/冒泡6.事件委托7.鼠标事件(禁止右键/选中文字)8.鼠标事件对象8.常用键盘事件9.键盘…

【C++】模板初阶:泛型编程的起点

💞💞 前言 hello hello~ ,这里是大耳朵土土垚~💖💖 ,欢迎大家点赞🥳🥳关注💥💥收藏🌹🌹🌹 💥个人主页&#x…

大模型下的Agent、AIGC的商业案例集合

算是一份摘录 1 AIGC 对影楼的影响 https://mp.weixin.qq.com/s/3j-6FAxZEEvXUZ1q6by2uw 2 出海Talkie :情感智能体 https://mp.weixin.qq.com/s/KHPmfuVvywxxcI2rqoOghA Talkie 为每条消息提供 3 个免费灵感,如果用户需要更多 AI 生成的灵感选项&…

Delta lake with Java--在spark集群上运行程序

昨天写了第一篇入门,今天看见有人收藏,继续努力学习下去。今天要实现的内容是如何将昨天的HelloDetlaLake 在spark集群上运行,。具体步骤如下 1、安装spark,我使用的是 spark-3.5.1-bin-hadoop3-scala2.13,去官网下载&#xff0c…

无穷级数错题本

1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 <

2024五一赛数学建模A题B题C题完整思路+数据代码+参考论文

A题 钢板最优切割路径问题 &#xff08;完整资料在文末获取&#xff09; 1. 建立坐标系和表示方法&#xff1a; 在建模之前&#xff0c;我们需要将切割布局转换为数学表示。首先&#xff0c;我们可以将布局中的每个点表示为二维坐标系中的一个点。例如&#xff0c;B1可以表示…

Ubuntu服务器创建新用户及解决新用户登录Access denied问题

目录 Ubuntu服务器创建新用户及解决新用户登录Access denied问题创建账号步骤创建用户只创建用户添加用户到sudo组 允许账号远程连接重启ssh服务 删除账号要删除用户而不删除用户文件如果要删除并且删除用户的家目录和邮件 查询指令查看所有用户查询特定用户账户信息查看用户组…

【Java基础】Maven的生命周期(clean+site+default)

1. 前言 在 Maven 出现之前&#xff0c;项目构建的生命周期就已经存在&#xff0c;开发人员每天都在对项目进行清理&#xff0c;编译&#xff0c;测试及部署&#xff0c;但由于没有统一的规范&#xff0c;不同公司甚至不同项目之间的构建的方式都不尽相同。 Maven 从大量项目…

[C++基础学习-07]----C++结构体详解

前言 结构体&#xff08;Struct&#xff09;是C中一种用户定义的复合数据类型&#xff0c;用于存储不同类型的数据项。结构体可以包含不同类型的数据成员&#xff0c;这些数据成员可以是基本类型&#xff08;如int、float、char等&#xff09;&#xff0c;也可以是数组、指针、…

Linux编辑器——vim的基础使用

文章目录 1.vim的基本概念2.vim的基本操作3.vim命令模式命令集3.1移动光标3.2删除文字3.3复制3.4替换3.5撤销3.6更改3.7跳到指定的行 1.vim的基本概念 本文将介绍vim的三种模式&#xff0c;分别位&#xff1a;命令模式、插入模式、低行模式。他们的功能区分如下&#xff1a; 正…

2. 深度学习笔记--损失函数

在机器学习中&#xff0c;损失函数是代价函数的一部分&#xff0c;而代价函数则是目标函数的一种类型。 Loss function&#xff0c;即损失函数&#xff1a;用于定义单个训练样本与真实值之间的误差&#xff1b; Cost function&#xff0c;即代价函数&#xff1a;用于定义单个批…

学习和“劳动”相关的谚语,柯桥俄语培训

1. Бог труды́ лю́бит. 天道酬勤。 2. В ми́ре нет тру́дных дел, ну́жно лишь усе́рдие. 世上无难事,只怕有心人。 3. У́тро вечера мудренее. 一日之计在于晨。 4. Что посе́ешь,…

车载电子电器架构 —— 关于bus off汇总

车载电子电器架构 —— 关于bus off汇总 我是穿拖鞋的汉子,魔都中坚持长期主义的汽车电子工程师。 老规矩,分享一段喜欢的文字,避免自己成为高知识低文化的工程师: 屏蔽力是信息过载时代一个人的特殊竞争力,任何消耗你的人和事,多看一眼都是你的不对。非必要不费力证明…

[Java EE] 多线程(六):线程池与定时器

&#x1f338;个人主页:https://blog.csdn.net/2301_80050796?spm1000.2115.3001.5343 &#x1f3f5;️热门专栏:&#x1f355; Collection与数据结构 (90平均质量分)https://blog.csdn.net/2301_80050796/category_12621348.html?spm1001.2014.3001.5482 &#x1f9c0;Java …

语义分割——铁路轨道数据集

引言 亲爱的读者们&#xff0c;您是否在寻找某个特定数据集&#xff0c;用于研究或项目实践&#xff1f;欢迎您在评论区留言&#xff0c;或者通过公众号私信告诉我&#xff0c;您想要的数据集的类型主题。小编会竭尽全力为您寻找&#xff0c;并在找到后第一时间与您分享。 重…

NASA数据集——NOAA 气溶胶和海洋科学考察数据(AEROSE)

Saharan Dust AERosols and Ocean Science Expeditions 简介 NOAA 气溶胶和海洋科学考察&#xff08;AEROSE&#xff09;是一种基于测量的综合方法&#xff0c;用于了解热带海洋上空气溶胶长程飘移的影响&#xff08;Morris 等人&#xff0c;2006 年&#xff1b;Nalli 等人&a…

直流屏整流模块HG07A220R电源模块HG10A220R

直流屏整流模块HG07A220R电源模块HG10A220R 其他同类型监控模块PM09T电源模块HG22005/S&#xff0c;HG22010/S&#xff0c;HG11010/S&#xff0c;HG11020/S&#xff0c;HG10A220Z&#xff0c;HG10A220F&#xff0c;HG05A220Z&#xff0c;HG07A220Z&#xff0c;HG10A110Z&#x…

Electron 对 SQLite 进行加密

上一篇讲了如何在 Electron使用 SQLite&#xff0c;如果 SQLite 中存有敏感数据&#xff0c;客户端采用明文存储风险很高&#xff0c;为了保护客户数据&#xff0c;就需要对数据进行加密&#xff0c;由于 electron 对代码并不加密&#xff0c;所以这里排除通过逆向工程进行数据…

从论文中看AI绘画

个人博客:Sekyoro的博客小屋 个人网站:Proanimer的个人网站 主要看是看Diffusion Models,CLIP,ControlNet,IP-Adapter这种经典论文,尝试总结论文写作的一些方式以及图像生成模型的一些内在思想. 对于其中的数学原理和代码不过深究. DDPM 使用扩散模型得到高质量图像,证明了这…

三、Linux基础命令

章节目标 了解Linux系统注意事项掌握Linux基础命令知道vmware tools的作用 一、Linux系统使用注意 1. Linux严格区分大小写 Linux 和Windows不同&#xff0c;Linux严格区分大小写的&#xff0c;包括文件名和目录名、命令、命令选项、配置文件设置选项等。例如&#xff0c;在…