Linux内核之Binder驱动container_of进阶用法(三十四)

简介: CSDN博客专家,专注Android/Linux系统,分享多mic语音方案、音视频、编解码等技术,与大家一起成长!

优质专栏:Audio工程师进阶系列原创干货持续更新中……】🚀
优质专栏:多媒体系统工程师系列原创干货持续更新中……】🚀

人生格言: 人生从来没有捷径,只有行动才是治疗恐惧和懒惰的唯一良药.

更多原创,欢迎关注:Android系统攻城狮

欢迎关注Android系统攻城狮

1.前言

本篇目的:Linux内核之Binder驱动container_of进阶用法

2.Linux内核container_of介绍

  • container_of 是Linux内核中常用的一种宏,用于从包含在结构体中的成员变量的指针,反推出该结构体本身的指针。这种技术在内核编程中非常重要,尤其是在驱动开发、内核模块编写等场景下,它使得我们能够灵活地访问结构体中的其他成员变量,即使我们手中只有结构体中一个成员的指针。
  • 在Linux内核中,结构体和指针的使用非常频繁,而container_of宏就是处理结构体和指针之间关系的利器。其定义通常可以在内核源码的include/linux/kernel.h文件中找到,其基本形式如下:
#define container_of(ptr, type, member) ({          \const typeof( ((type *)0)->member ) *__mptr = (ptr);    \(type *)( (char *)__mptr - offsetof(type,member) );})
  • 这里,ptr是结构体typemember成员的指针,type是结构体的类型,而membertype类型中的一个成员。container_of宏首先通过typeof获取member的类型,然后通过offsetof宏计算membertype中的偏移量,最后通过指针算术将ptr调整回type的起始地址。
  • offsetof宏通常也是内核提供的一个宏,用于计算结构体成员在结构体中的偏移量。其实现方式也很有趣,它利用了结构体成员的地址总是结构体首地址加上某个偏移量的原理。
  • container_of宏的一个典型应用场景是在链表操作中。Linux内核中广泛使用链表来组织数据结构,而链表节点通常是嵌入在用户自定义的结构体中的。当我们遍历链表时,通常只能获得链表节点的指针,而container_of宏可以帮助我们轻松地从链表节点指针得到包含该节点的用户结构体指针。
  • 例如,假设我们有一个struct my_struct类型的结构体,其中包含一个struct list_head类型的链表节点:
struct my_struct {int data;struct list_head list;
};
struct list_head {struct list_head *next, *prev;
};
  • 如果我们有一个指向list_head的指针list_ptr,我们可以使用container_of宏来获取指向my_struct的指针:
struct my_struct *my_ptr = container_of(list_ptr, struct my_struct, list);
  • 这样,我们就可以方便地访问my_struct中的其他成员变量了。
    container_of宏是Linux内核中一种巧妙的设计,它充分利用了C语言中指针和结构体的特性,提供了一种简洁而有效的方式来处理复杂的结构体关系。这种宏的运用,大大提高了内核代码的灵活性和可维护性,是Linux内核编程中的一颗璀璨明珠。

3.代码实例

<1>.v1.0版本

#include <iostream>
#include <stdio.h>
#include <stddef.h> //offsetof头文件.//v1.0 C++版本.
#define container_of(ptr, type, member) ({                      \const typeof( ((type *)0)->member ) *__mptr = (ptr);	\(type *)( (char *)__mptr - offsetof(type,member) );})
/*
ptr:表示结构体中成员变量(member)的地址。
type:表示结构体类型。
member:表示结构体中成员变量member。
返回值:返回结构体的首地址。
*/struct binder_object_header {int type;
};struct flat_binder_object {struct binder_object_header  boh;int pad_flags;char*		cookie;
};struct binder_object {int bo;union {struct binder_object_header boh;struct flat_binder_object fbo;};
};int main(void){struct binder_object_header *boh1;struct flat_binder_object *fbo = nullptr;struct binder_object object;object.bo = 101;boh1 = &(object.boh);printf("xxx---------------> object = %p\n",reinterpret_cast<void*>(&object));printf("xxx---------------> fbo = %p\n",reinterpret_cast<void*>(fbo));//v1.0 通过binder_object_header拿到flat_binder_object的地址.fbo = container_of(boh1, struct flat_binder_object, boh);//fbo = container_of(&(object.boh), struct flat_binder_object, boh);printf("xxx---------------> fbo = %p\n",reinterpret_cast<void*>(fbo));//v2.0 通过binder_object_header拿到binder_object的地址.struct binder_object *bo_ret;bo_ret = container_of(boh1 ,struct binder_object, boh);printf("xxx---------------> bo_ret = %p\n",reinterpret_cast<void*>(bo_ret));printf("xxx--------------->%s, %s(), line = %d, bo = %d\n",__FILE__,__FUNCTION__,__LINE__,bo_ret->bo);return 0;
}

打印:

xxx---------------> object = 0x7ffdcdc3e0b0
xxx---------------> fbo = (nil)
xxx---------------> fbo = 0x7ffdcdc3e0b8
xxx---------------> bo_ret = 0x7ffdcdc3e0b0
xxx---------------> bo = 101

<2>.v2.0 简化版本

/***********************************************************
* Author        : 公众号: Android系统攻城狮
* Create time   : 2024-03-29 12:39:52 星期五
* Filename      : container_of_03.cpp
* Description   :
************************************************************/
#include <iostream>
#include <stdio.h>
#include <stddef.h> //offsetof头文件.//v1.0 C++版本.
#define container_of(ptr, type, member) ({                      \const typeof( ((type *)0)->member ) *__mptr = (ptr);	\(type *)( (char *)__mptr - offsetof(type,member) );})
/*
ptr:表示结构体中成员变量(member)的地址。
type:表示结构体类型。
member:表示结构体中成员变量member。
返回值:返回结构体的首地址。
*/struct HEADER {int type;
};struct FLAT {struct HEADER  header;int pad_flags;
};struct OBJECT {int bo;union {struct HEADER obj_header;struct FLAT flat;};
};int main(void){struct FLAT *flat1 = nullptr;struct OBJECT object;object.bo = 101;printf("xxx----已分配地址:-----------> OBJECT_addr = %p\n",reinterpret_cast<void*>(&object));printf("xxx---container_of前------------> flat1_addr = %p\n",reinterpret_cast<void*>(flat1));//v1.0 通过HEADER拿到在OBJECT结构体中,已经实例化的FLAT结构体的地址.flat1 = container_of(&(object.obj_header), struct FLAT, header);printf("xxx---container_of后-----------> flat1 = %p\n",reinterpret_cast<void*>(flat1));//v2.0 通过HEADER拿到OBJECT结构体的地址.struct OBJECT *object_ret;object_ret = container_of(&(object.obj_header) ,struct OBJECT, obj_header);printf("xxx----container_of后-----------> object_ret = %p\n",reinterpret_cast<void*>(object_ret));printf("xxx---------------> object_ret->bo = %d\n",object_ret->bo);return 0;
}

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

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

相关文章

小小狠招:巧妙使用HANA数据库的jdbc driver

SAP旗下的HANA数据库&#xff0c;实际上是分为两个系列进行发布&#xff0c;一种是基于本地部署的称之为HANA Platform。另一种是面向Cloud平台的&#xff0c;称之为HANA Cloud。 在实际使用当用&#xff0c;因为两者基本上共用同一代码库&#xff0c;除个别地方略有差异以外&…

车载电子电器架构 —— 电气架构释放检查

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

Sql注入---基础

文章目录 前言一、pandas是什么&#xff1f;二、使用步骤 1.引入库2.读入数据总结 一.Sql注入概述 攻击者通过构造恶意的SQL查询语句&#xff0c;将其注入到应用程序的数据库查询中&#xff0c;以执行未经授权的操作或者获取敏感信息。 假设如下场景&#xff0c;当你想要知道对…

【c++】简单的日期计算器

&#x1f525;个人主页&#xff1a;Quitecoder &#x1f525;专栏&#xff1a;c笔记仓 朋友们大家好啊&#xff0c;在我们学习了默认成员函数后&#xff0c;我们本节内容来完成知识的实践&#xff0c;来实现一个简易的日期计算器 目录 头文件声明函数函数的实现1.全缺省默认构…

每日一博 - 关于日志记录的最佳实践

文章目录 概述选择合适的日志等级打印函数的入参、出参打印日志对象要做判空处理&#xff0c;避免阻断流程推荐使用 Slf4j不用e.printStackTrace()打印日志低级别的日志输出&#xff0c;必须进行日志级别开关判断不打印重复日志打印全部的异常信息&#xff0c;方便定位问题核心…

cocos2.x => node 属性修改

简介 与节点属性相关的几个核心变量_trs、_matrix、_worldMatrix、_localMatDirty、_worldMatDirty。 _trs&#xff1a;存储节点的position、rotation、scale _matrix&#xff1a;存储节点的缩放、位移、旋转三者合一的变化矩陈&#xff08;仿射矩陈&#xff09; _worldMat…

6、Cocos Creator 2D 渲染组件(一)

目录 1、Sprite 组件 2、Label 组件 3、Mask 组件 4、Graphics 组件 5、RichText 组件 1、Sprite 组件 2、Label 组件 3、Mask 组件 4、Graphics 组件 5、RichText 组件 6、BlockInputEvents 组件 BlockInputEvents 组件将拦截所属节点 bounding box 内的所有输入事件&a…

python多方式操作elasticsearch介绍

python多方式操作elasticsearch介绍 1. requests模块操作ES ​ requests 是一个 Python HTTP 库&#xff0c;它简化了发送 HTTP 请求和处理响应的过程。通过 requests 模块&#xff0c;开发人员可以轻松地与 Web 服务进行通信&#xff0c;包括获取网页内容、执行 API 请求等。…

Java初级八股文面试题

1. Java的基本数据类型有哪些&#xff1f; 答&#xff1a;Java的基本数据类型包括&#xff1a; 整型&#xff1a;byte, short, int, long浮点型&#xff1a;float, double字符型&#xff1a;char布尔型&#xff1a;boolean 2. Java中的变量作用域有哪些&#xff1f; 答&…

Qt for WebAssembly 环境搭建 - Windows新手入门

Qt for WebAssembly 环境搭建 - Windows新手入门 一、所需工具软件1、安装Python2、安装Git2.1 注册Github账号2.2 下载安装Git2.2.1配置Git&#xff1a;2.2.2 配置Git环境2.2.3解决gitgithub.com: Permission denied (publickey) 3 安装em编译器 二、Qt配置编译器三、参考链接…

怎么让ChatGPT批量写作原创文章

随着人工智能技术的不断发展&#xff0c;自然语言处理模型在文本生成领域的应用也日益广泛。ChatGPT作为其中的佼佼者之一&#xff0c;凭借其强大的文本生成能力和智能对话特性&#xff0c;为用户提供了一种高效、便捷的批量产出内容的解决方案。以下将就ChatGPT批量写作内容进…

厦门攸信技术亮相新技术研讨会,展现物流自动化解决方案新高度!

今日&#xff0c;厦门攸信信息技术有限公司受邀参加了一场备受行业关注的电子制造高端盛会——一步步新技术研讨会&#xff0c;凭借卓越的智能制造与物流自动化技术在会议中大放异彩。作为一家引领行业发展的企业&#xff0c;厦门攸信技术不仅展示了其深厚的技术底蕴&#xff0…

算法系列--动态规划--背包问题(4)--完全背包拓展题目

&#x1f495;"这种低水平质量的攻击根本就不值得我躲&#xff01;"&#x1f495; 作者&#xff1a;Lvzi 文章主要内容&#xff1a;算法系列–动态规划–背包问题(4)–完全背包拓展题目 大家好,今天为大家带来的是算法系列--动态规划--背包问题(4)--完全背包拓展题目…

计算机图形学和OpenGL

一.计算机图形学和OpenGL的推荐书籍&#xff1a; 1. **OpenGL编程指南&#xff08;第九版&#xff09;**&#xff08;OpenGL Programming Guide, Ninth Edition&#xff09;&#xff1a;这本书是学习OpenGL编程的经典教材&#xff0c;详细介绍了OpenGL的基础知识和高级技术&…

《web应用技术》第一次课后练习

上机任务&#xff08;利用好chatgpt&#xff0c;文心一言等工具。&#xff09;&#xff1a; 1、下载软件&#xff0c;并安装。相关安装文件已上传至群文件。 JDK,TOMCAT&#xff0c;IDEA 2、学会用记事本编写jsp文件&#xff0c;并放进tomcat的相关目录下&#xff0c;运行。 …

使用Windows自带服务(BitLocker)加密U盘

第一步&#xff1a;启用 BitLocker 服务 1.1快捷键&#xff1a;WinR 调出运行框&#xff0c;输入services.msc 1.2找到服务列表中的BitLocker Drive Encryption Service&#xff0c;启动此项 第二步&#xff1a;加密U盘 把你的U盘插入电脑&#xff0c;打开“我的电脑”&#…

EFCore的空迁移(EFCore操作已存在的数据库表,不影响其中的数据)

背景&#xff1a;EFCore默认的会自动创建数据表&#xff0c;但是有时又是DBFirst&#xff0c;数据库写好了要用现成的表。这个时候就需要进行一些特殊的操作了 1、写出跟要对接数据库的实体类 比如我的表是这样创建的 create table mail_test (user_id bigint auto_increment …

java普通类和抽象类有哪些区别?

普通类和抽象类在面向对象编程中有一些显著的区别&#xff0c;这些区别主要体现在类的定义、使用方式和实例化等方面。以下是它们之间的一些主要区别&#xff1a; 定义与实现&#xff1a; 普通类&#xff1a;普通类可以包含非抽象方法和抽象方法&#xff08;如果有的话&#x…

OSCP靶场--Twiggy

OSCP靶场–Twiggy 考点(CVE-2020-11651[RCE]) 1.nmap扫描 ## ┌──(root㉿kali)-[~/Desktop] └─# nmap 192.168.216.62 -sV -sC -Pn --min-rate 2500 -p- Starting Nmap 7.92 ( https://nmap.org ) at 2024-03-30 06:43 EDT Nmap scan report for 192.168.216.62 Host i…

MongoDB聚合运算符:$ln

文章目录 MongoDB聚合运算符&#xff1a;$ln语法使用举例 MongoDB聚合运算符&#xff1a;$ln $ln聚合运算符计算数字的自然对数ln&#xff08;即 log e&#xff09;并将结果作为双精度值返回。 语法 { $ln: <number> }<number>表达式可以是任何有效的表达式&…