Linux内核中的常用宏container_of其实很简单【转】

转自:http://blog.csdn.net/npy_lp/article/details/7010752

开发平台:Ubuntu11.04

    编 译器:gcc version 4.5.2 (Ubuntu/Linaro4.5.2-8ubuntu4)

 

    Container_of在Linux内核中是一个常用的宏,用于从包含在某个结构中的指针获得结构本身的指针,通俗地讲就是通过结构体变量中某个成员的首地址进而获得整个结构体变量的首地址。

    Container_of的定义如下: 

[cpp] view plain copy
  1. #define container_of(ptr, type, member) ({      \  
  2.     const typeof( ((type *)0)->member ) *__mptr = (ptr);    \  
  3.     (type *)( (char *)__mptr - offsetof(type,member) );})  

    其实它的语法很简单,只是一些指针的灵活应用,它分两步:

    第一步,首先定义一个临时的数据类型(通过typeof( ((type *)0)->member )获得)与ptr相同的指针变量__mptr,然后用它来保存ptr的值。

    第二步,用(char *)__mptr减去member在结构体中的偏移量,得到的值就是整个结构体变量的首地址(整个宏的返回值就是这个首地址)。

    其中的语法难点就是如何得出成员相对结构体的偏移量?

    通过例子说明,如清单1: 

[cpp] view plain copy
  1. /* linux-2.6.38.8/include/linux/compiler-gcc4.h */  
  2. #define __compiler_offsetof(a,b) __builtin_offsetof(a,b)  
  3.   
  4. /* linux-2.6.38.8/include/linux/stddef.h */  
  5. #undef offsetof  
  6. #ifdef __compiler_offsetof  
  7. #define offsetof(TYPE,MEMBER) __compiler_offsetof(TYPE,MEMBER)  
  8. #else  
  9. #define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)  
  10. #endif  
  11.   
  12. #include <stdio.h>  
  13.   
  14. struct test_struct {  
  15.     int num;  
  16.     char ch;  
  17.     float fl;  
  18. };  
  19.   
  20. int main(void)  
  21. {  
  22.     printf("offsetof(struct test_struct, num) = %d\n",   
  23.             offsetof(struct test_struct, num));  
  24.       
  25.     printf("offsetof(struct test_struct,  ch) = %d\n",   
  26.             offsetof(struct test_struct, ch));  
  27.       
  28.     printf("offsetof(struct test_struct,  fl) = %d\n",   
  29.             offsetof(struct test_struct, fl));  
  30.       
  31.     return 0;  
  32. }  

    说明,__builtin_offsetof(a,b)是GCC的内置函数,可认为它的实现与((size_t) &((TYPE *)0)->MEMBER)这段代码是一致的。

    例子输出结果: 

[cpp] view plain copy
  1. offsetof(struct test_struct, num) = 0  
  2. offsetof(struct test_struct,  ch) = 4  
  3. offsetof(struct test_struct,  fl) = 8  

    其中代码难以理解的地方就是它灵活地运用了0地址。如果觉得&( (struct test_struct *)0 )->ch这样的代码不好理解,那么我们可以假设在0地址分配了一个结构体变量struct test_struct a,然后定义结构体指针变量p并指向a(struct test_struct *p = &a),如此我们就可以通过&p->ch获得成员ch的地址。由于a的首地址为0x0,所以成员ch的首地址为0x4。

 

    最后通过强制类型转换(size_t)把一个地址值转换为一个整数。

    分析完container_of的定义,接下来举两个例子来体会一下它的使用方法。

    正确的例子,如清单2: 

[cpp] view plain copy
  1. /* linux-2.6.38.8/include/linux/compiler-gcc4.h */  
  2. #define __compiler_offsetof(a,b) __builtin_offsetof(a,b)  
  3.   
  4. /* linux-2.6.38.8/include/linux/stddef.h */  
  5. #undef offsetof  
  6. #ifdef __compiler_offsetof  
  7. #define offsetof(TYPE,MEMBER) __compiler_offsetof(TYPE,MEMBER)  
  8. #else  
  9. #define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)  
  10. #endif  
  11.   
  12. /* linux-2.6.38.8/include/linux/kernel.h * 
  13.  * container_of - cast a member of a structure out to the containing structure 
  14.  * @ptr: the pointer to the member. 
  15.  * @type:   the type of the container struct this is embedded in. 
  16.  * @member:    the name of the member within the struct. 
  17.  * 
  18.  */  
  19. #define container_of(ptr, type, member) ({      \  
  20.     const typeof( ((type *)0)->member ) *__mptr = (ptr);    \  
  21.     (type *)( (char *)__mptr - offsetof(type,member) );})  
  22.   
  23. #include <stdio.h>  
  24.   
  25. struct test_struct {  
  26.     int num;  
  27.     char ch;  
  28.     float fl;  
  29. };  
  30.   
  31. int main(void)  
  32. {  
  33.     struct test_struct init_test_struct = { 99, 'C', 59.12 };  
  34.   
  35.     char *char_ptr = &init_test_struct.ch;  
  36.   
  37.     struct test_struct *test_struct = container_of(char_ptr, struct test_struct, ch);  
  38.       
  39.     printf(" test_struct->num = %d\n test_struct->ch = %c\n test_struct->fl = %f\n",   
  40.         test_struct->num, test_struct->ch, test_struct->fl);  
  41.       
  42.     return 0;  
  43. }  

    例子输出结果: 

[cpp] view plain copy
  1. test_struct->num = 99  
  2. test_struct->ch = C  
  3. test_struct->fl = 59.119999  

    不适当的例子,如清单3: 

[cpp] view plain copy
  1. /* linux-2.6.38.8/include/linux/compiler-gcc4.h */  
  2. #define __compiler_offsetof(a,b) __builtin_offsetof(a,b)  
  3.   
  4. /* linux-2.6.38.8/include/linux/stddef.h */  
  5. #undef offsetof  
  6. #ifdef __compiler_offsetof  
  7. #define offsetof(TYPE,MEMBER) __compiler_offsetof(TYPE,MEMBER)  
  8. #else  
  9. #define offsetof(TYPE, MEMBER) ((size_t) &((TYPE *)0)->MEMBER)  
  10. #endif  
  11.   
  12. /* linux-2.6.38.8/include/linux/kernel.h * 
  13.  * container_of - cast a member of a structure out to the containing structure 
  14.  * @ptr: the pointer to the member. 
  15.  * @type:   the type of the container struct this is embedded in. 
  16.  * @member:    the name of the member within the struct. 
  17.  * 
  18.  */  
  19. #define container_of(ptr, type, member) ({      \  
  20.     const typeof( ((type *)0)->member ) *__mptr = (ptr);    \  
  21.     (type *)( (char *)__mptr - offsetof(type,member) );})  
  22.   
  23. #include <stdio.h>  
  24.   
  25. struct test_struct {  
  26.     int num;  
  27.     char ch;  
  28.     float fl;  
  29. };  
  30.   
  31. int main(void)  
  32. {  
  33.     char real_ch = 'A';  
  34.     char *char_ptr = &real_ch;  
  35.   
  36.     struct test_struct *test_struct = container_of(char_ptr, struct test_struct, ch);  
  37.   
  38.     printf(" char_ptr = %p  test_struct = %p\n\n", char_ptr, test_struct);  
  39.   
  40.     printf(" test_struct->num = %d\n test_struct->ch = %c\n test_struct->fl = %f\n",   
  41.         test_struct->num, test_struct->ch, test_struct->fl);  
  42.       
  43.     return 0;  
  44. }  

    例子输出结果: 

[cpp] view plain copy
  1. char_ptr = 0xbfb72d7f  test_struct = 0xbfb72d7b  
  2.   
  3. test_struct->num = -1511000897  
  4. test_struct->ch = A  
  5. test_struct->fl = 0.000000  

    注意,由于这里并没有一个具体的结构体变量,所以成员num和fl的值是不确定的。

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

mysql concat例子_MYSQL中CONCAT详解

concat()函数1. 功能&#xff1a;返回结果为连接参数产生的字符串。如有任何一个参数为NULL &#xff0c;则返回值为 NULL。2. 语法concat(str1, str2,...)3. 例子案例一&#xff1a;mysql> select concat(苹果,香蕉,梨子);------------------------------| CONCAT(苹果,香蕉…

mysql php7安装配置_centos7无网络下安装部署php7.1.33+mysql5.7.28+apache2.4.6-Go语言中文社区...

centos7无网络下安装部署php7.1.33mysql5.7.28apache2.4.6一、1、先ping www.baidu.com&#xff0c;root账户下&#xff0c;如果未联网&#xff0c;创建目录&#xff0c;把提前下载好的rpm包拷贝到rpm目录下如图&#xff1a;(如果没有安装包请查看我的另一篇教程下载这些安装包…

webkit渲染

2019独角兽企业重金招聘Python工程师标准>>> 参考链接 理解WebKit和Chromium 简明魔法学院 Chrome软件渲染 WebKit渲染基础 Webkit 渲染基础 Webkit不是浏览器,它是一个渲染引擎 软件渲染 硬件渲染(GPU加速) 会触发GPU加速的属性 CSS3 3D transformation, trans…

element ui中dialog相关问题

一&#xff0c;今天需要在dialog里面引入另一个页面&#xff0c;就是打开dialog显示该页面&#xff08;把页面放到dialog中&#xff09;&#xff0c;引入的语句如下&#xff1a; <iframe src"view?pathrkdj_b" ></iframe> 二&#xff0c;使用table组件时…

spark java教程_(Spark)学习进度十四(Spark之Java独立应用编程)

环境如下:(更新了林子雨教程中不可使用的部分) Hadoop 2.6.0以上 java JDK 1.7以上 Spark 3.0.0-preview2 二、java独立应用编程(在下载依赖jar包的过程中如遇到卡顿现象可以Ctrl+C停止下载,然后重新执行本条命令即可继续下载相应的依赖jar包) 1、安装maven ubuntu中没有自带…

[零基础学JAVA]Java SE应用部分-34.Java常用API类库

本季目标1、StringBuffer类 2、Runtime 类 3、包装类与JDK 1.5的新特性——泛型 4、日期的操作类 5、Math类 6、Random类1、StringBuffer&#xff08;重点&#xff09; String 类的时候说过&#xff1a;String 类的内容一旦声明则不可改变&#xff0c;改变的只是其地址。…

我所理解的机器学习

各位请移步到【http://www.cnblogs.com/cchHers/p/8945908.html】转载于:https://www.cnblogs.com/cchHers/p/8933042.html

protobuf java文档_Java中使用Protobuf

gradle依赖库&#xff1a;implementation com.google.protobuf:protobuf-java:3.4.0implementation com.google.protobuf:protobuf-java-util:3.4.00.编写.proto文件&#xff0c;编译生成对应Java源文件&#xff1a;syntax "proto2";option java_generic_services …

多租户表设计

2019独角兽企业重金招聘Python工程师标准>>> multi-tenant-databases-in-the-cloudtips-amp-tricks-to-build-multi-tenant-databases-with-sql-databases团队开发框架实战—多租户支持转载于:https://my.oschina.net/yangjiandong/blog/1612626

CSS实现树形结构 + js加载数据

看到一款树形结构&#xff0c;比较喜欢它的样式&#xff0c;就参照它的外观自己做了一个&#xff0c;练习一下CSS。 做出来的效果如下&#xff1a; 拉莫小学 一年级 一班二班二年级三年级 一班二班三班树的dom结构&#xff1a; <div class"tree"><ul><…

Django中--自定义模型管理器类

BookInfo.objects.all()->objects是一个什么东西呢&#xff1f; 答&#xff1a;objects是models.Manger类的一个对象&#xff0c;是Django帮我自动生成的管理器对象&#xff0c;通过这个管理器可以实现对数据的查询。 自定义管理器之后Django不再帮我们生成默认的objects管…

第二章 API的理解和使用

2.1.1全局命令 Key * 查看所有键&#xff0c;(慎用&#xff0c;会把所有键都遍历一次并列出) Dbsize 查看键总数&#xff0c;不会遍历所有键&#xff0c;只是从内置函数中读取一个数 Exists [key] 检查键是否存在 Del [key] 删除键 Expire [key] [seconds] 设置键过期时间 Type…

Django中--使用redis存储历史浏览记录

class UserInfoView(LoginRequiredMixin, View):用户中心-信息页def get(self, request):显示# Django会给request对象添加一个属性request.user# 如果用户未登录->user是AnonymousUser类的一个实例对象# 如果用户登录->user是User类的一个实例对象# request.user.is_aut…

3D虚拟试衣有望解决厘米级服装误差 网购服装不再蒙

还在担心网购服装对实际穿着效果没把握吗&#xff1f;随着京东App 6.6.3版本的更新&#xff0c;京东试试3D虚拟试衣功能正式上线&#xff0c;消费者可按照自己的身材比例创建专属的3D模型&#xff0c;而试穿效果则可以完全依照模型来展现。据了解&#xff0c;这个系统未来还将实…

关于idea修改当前使用的git账户的问题

1、问题描述&#xff1a; 由于前一段时间公司迁移git&#xff0c;就是将项目代码等迁移到另一个git服务器上&#xff0c;结果用idea从git上clone代码的时候发现没有指定仓库,如下提提示 2、排查原因&#xff1a; 开始怀疑是没有把自己加入到项目成员里面&#xff0c;经过检查是…

分布式文件系统FastDFS

1. 什么是FastDFS FastDFS 是用 c 语言编写的一款开源的分布式文件系统。FastDFS 为互联网量身定制&#xff0c; 充分考虑了冗余备份、负载均衡、线性扩容等机制&#xff0c;并注重高可用、高性能等指标&#xff0c;使用 FastDFS 很容易搭建一套高性能的文件服务器集群提供文件…

html5 下拉刷新(pc+移动网页源码)

本文demo下载地址&#xff1a;http://www.wisdomdd.cn/Wisdom/resource/articleDetail.htm?resourceId1071 本文实现在html5网页中使用下拉功能自动刷新显示更多内容, 使用jquery捕捉和处理相应的鼠标事件, 例如内容在顶部时&#xff0c;触发下拉事件后显示更多内容; 如内容在…

java同名变量在list中添加两次_快速解决List集合add元素,添加多个对象出现重复的问题...

首先我们在new 一个对象的时候&#xff0c;对象的id是唯一确定的&#xff1b;将对象add入list中时&#xff0c;放入list中的其实是对象的引用 &#xff1b;而每次循环只是简单的set 对象的属性&#xff0c;set新的属性值&#xff0c;而add进list中的对象还是同一个对象id&#…

python面试题总结(1)--语言特性

1. 谈谈对 Python 和其他语言的区别 答&#xff1a; Python 是一门强类型的可移植、可扩展、可嵌入的解释型编程语言&#xff0c;属于动态语言&#xff1b;其语法简洁优美、功能强大无比、应用领域非常广泛且具有强大完备的第三方库。 &#xff08;注&#xff1a;语言有无类型…

视频网站盈利模式与营销策划

在与数十家视频网站进行信息网络传播权交易过程中&#xff0c;在研究视频网站内容和盈利模式基础上&#xff0c;综合自己在传统媒体和新媒体领域十几年的策划和营销经验&#xff0c;我发现&#xff1a;视频网站的盈利模式其实早就形成多种体系&#xff0c;但是盈利之路艰难&…