windows驱动开发-WDF对象

WDF封装了大量的WDF对象,不过,和应用层不一样,不用去尝试从WDF框架对象类上派生和改写原有的WDF类,本意WDF就是希望我们使用这些对象和类,而不是创造新的奇怪的类。

每个WDF对象都代表着对一项驱动需要使用的子功能的封装;WDF对象本身存在是和WDF框架导出的函数接口对应的;另外,每个版本的WDF可能会有细微的差别。

WDF对象

WDF提供给驱动程序的接口是基于对象的, 框架会定义多个对象和接口。 这些对象导出方法 (函数) 和 属性 (驱动程序可以访问的数据) 。 框架对象还会启动事件,驱动程序可以通过提供事件回调函数来支持这些事件。

基于框架的驱动程序永远不会直接访问框架对象,这是因为不同版本之间对象的内存布局有细微的差别。 驱动程序按句柄引用对象,驱动程序将其作为输入传递给对象的方法。下面列出了所有的WDF对象:

所有框架对象具有以下特征:

引用计数:框架维护对每个对象的引用数。 当框架创建对象时,它将对象的引用计数设置为 1。 当框架使用完对象后,它会递减引用计数。 在引用计数减为零之前,框架无法删除对象,因此驱动程序可以通过递增对象的引用计数来阻止删除该对象。

上下文空间:基于框架的驱动程序可以为驱动程序接收或创建的每个框架对象创建特定于对象的上下文空间。 驱动程序应将所有特定于对象的数据存储在对象的上下文空间中。 

删除回调函数:驱动程序可以注册框架在删除对象时调用的回调函数。 回调函数可以删除驱动程序分配的资源,例如特定于对象的内存分配。 

父对象:所有框架对象都可以具有父对象。 WDF框架为大多数对象指定默认父对象。 当驱动程序创建对象时,它可以指定替代该对象的默认父对象的父对象。 若要指定对象的父对象,驱动程序设置对象的WDF_OBJECT_ATTRIBUTES结构的 ParentObject 成员。 对于少数对象类型,驱动程序无法替代默认的父对象, 框架或驱动程序删除父对象时,框架也会删除父对象的子对象。

生命周期

WDF框架对象的“生命周期”跨越从创建对象到删除对象的时间,对象的引用计数控制何时将其删除。下面是一些生命周期节点中的特点:

创建框架对象:大多数框架对象都是通过驱动程序调用对象的创建方法创建的。 例如,每个框架驱动程序必须调用 WdfDriverCreate 来创建框架驱动程序对象。

其他框架对象由框架创建。 例如,当用户应用程序打开设备进行读取或写入操作时,框架会创建一个框架文件对象,并将其传递给驱动程序的 EvtDeviceFileCreate 回调函数。

框架或驱动程序可以创建一些框架对象。 例如,当 I/O 管理器将 I/O 请求传递到驱动程序时,框架会创建一个框架请求对象并将其传递给驱动程序,通常通过调用驱动程序的请求处理程序之一。 驱动程序还可以创建框架请求对象并将其传递给其他驱动程序。

使用引用计数: 框架为每个对象维护引用计数。 创建对象时,框架将其引用计数设置为 1。 如果引用计数变为零,框架将删除对象。

驱动程序可以通过调用 WdfObjectReference 来递增引用计数或 调用 WdfObjectDereference 来递减引用计数来修改对象的引用计数。

在大多数情况下,驱动程序不必递增或递减对象的引用计数。 框架在将对象的句柄传递给驱动程序之前递增计数,并在驱动程序不再需要该对象时递减计数。

驱动程序调用 WdfObjectReference 以确保在驱动程序使用完对象之前,不会由框架或驱动程序线程删除对象。 

删除框架对象: 对象被删除是因为驱动程序调用 WdfObjectDelete 或框架调用内部删除例程,但仅当对象的引用计数为零时才被删除。 驱动程序或框架尝试删除对象后,对象的句柄将保持有效,直到引用计数变为零。 驱动程序无法通过调用 WdfObjectDereference 将对象的引用计数减少到零来删除对象-驱动程序还必须调用 WdfObjectDelete。

如果框架对象是父对象的子对象,并且正在删除父对象,则框架会尝试先删除子对象,然后再删除父对象。 对象删除从离父级最远的对象开始,并针对根目录执行对象层次结构。

驱动程序可以注册驱动程序或框架删除对象时框架调用的以下两个回调函数:

EvtCleanupCallback 回调函数,框架调用该函数,以便驱动程序可以调用 WdfObjectDereference(如果它以前为要删除的对象调用 WdfObjectReference)。

EvtDestroyCallback 回调函数,框架在对象的引用计数减至零后调用该回调函数。

其中一个回调函数必须解除分配驱动程序在创建对象时分配的任何特定于对象的资源。

框架始终处理某些框架对象的删除,驱动程序不得尝试删除这些对象,例如驱动对象。 

框架对象的上下文

在WDM时代,驱动程序的设备对象有一个设备扩展,这个设备扩展绑定了设备对象,用于存储设备相关的信息,这使得每个设备对象都有自己的设备扩展部分,此时驱动程序导出的例程可以利用设备扩展来避免重入,这样几乎不需要为区分多个设备对象浪费额外的代码,例程天然就是可重入的。

这一良好的设计也延续到了WDF中,几乎所有WDF对象都可以分配上下文,对象上下文空间是驱动程序可以分配和分配给对象的额外、不可分页的内存空间。 每个基于框架的驱动程序都可以为驱动程序接收或创建的每个框架对象创建一个或多个特定于对象的上下文空间。

基于框架的驱动程序应在数据所属对象的上下文空间中按值或指针存储所有特定于对象的数据。

例如,USB 设备的驱动程序可能会为其框架设备对象创建上下文空间。 在上下文空间中,驱动程序可能会存储设备特定的信息,例如设备的 USB_DEVICE_DESCRIPTOR 和 USB_CONFIGURATION_DESCRIPTOR 结构,以及表示设备接口管道的 集合对象的 句柄。

框架不会将框架对象从一个驱动程序传递到另一个驱动程序,因此不能使用对象的上下文空间在两个驱动程序之间传递数据。

若要定义对象的上下文空间,必须创建一个或多个结构。 每个结构表示单独的上下文空间。 驱动程序将使用每个结构成员来存储一段特定于对象的信息。 此外,驱动程序必须要求框架为每个结构生成 访问器方法 。 此访问器方法接受对象句柄作为输入,并返回对象的上下文空间的地址。

每当驱动程序调用对象创建方法(如 WdfDeviceCreate)时,该方法会选择性地分配上下文空间。 所有对象创建方法都接受可选的 WDF_OBJECT_ATTRIBUTES 结构作为输入。 此结构描述希望框架为对象分配的上下文空间。

若要在驱动程序调用对象的创建方法后向对象添加额外的上下文空间,驱动程序可以调用 WdfObjectAllocateContext 方法,该方法与对象创建方法一样,接受 WDF_OBJECT_ATTRIBUTES 结构作为输入。

当框架为对象分配上下文空间时,它还对上下文空间进行零初始化。

当框架或驱动程序删除框架对象时,框架将删除该对象的所有上下文空间。

如果驱动程序使用上下文空间来存储指向驱动程序在创建对象时分配的缓冲区的指针,则驱动程序应提供 一个 EvtCleanupCallback 函数,该函数在删除对象时解除分配缓冲区。

若要为驱动程序创建的对象定义对象的上下文空间结构和访问器方法,驱动程序必须使用以下步骤:

1. 定义描述要存储的数据的结构。 例如,如果要为驱动程序的设备对象创建上下文数据,驱动程序可能会定义一个名为 MY_DEVICE_CONTEXT 的结构。

2. 使用 WDF_DECLARE_CONTEXT_TYPE 宏或 WDF_DECLARE_CONTEXT_TYPE_WITH_NAME 宏。 这两个宏都执行以下操作:

  • 创建并初始化 WDF_OBJECT_CONTEXT_TYPE_INFO 结构。
  • 定义一个访问器方法,驱动程序稍后将使用该方法访问对象的上下文空间。 访问器方法的返回值是指向对象的上下文空间的指针。

WDF_DECLARE_CONTEXT_TYPE宏根据结构的名称创建访问器方法的名称。 例如,如果上下文结构的名称MY_DEVICE_CONTEXT,则宏将创建一个名为 WdfObjectGet_MY_DEVICE_CONTEXT 的访问器方法。

使用 WDF_DECLARE_CONTEXT_TYPE_WITH_NAME 宏可以指定访问器方法的名称。 例如,可以将 GetMyDeviceContext 指定为设备对象的上下文访问器方法的名称。

3. 调用 WDF_OBJECT_ATTRIBUTES_INIT 以初始化对象的 WDF_OBJECT_ATTRIBUTES 结构。

4. 使用 WDF_OBJECT_ATTRIBUTES_SET_CONTEXT_TYPE 宏将 WDF_OBJECT_ATTRIBUTES 结构的 ContextTypeInfo 成员设置为 WDF_OBJECT_CONTEXT_TYPE_INFO 结构的地址。

5. 调用对象创建方法,例如 WdfDeviceCreate。

驱动程序创建对象后,驱动程序可以随时调用 WdfObjectAllocateContext ,以向对象添加额外的上下文空间。

由于步骤 1 和步骤 2 定义全局数据结构并创建驱动程序可调用的例程,因此驱动程序必须在声明全局数据的驱动程序区域中完成这些步骤,通常为头文件。 这些步骤不得从驱动程序的例程中完成。

驱动程序必须从创建对象的驱动程序例程中完成步骤 3、4 和 5。

框架可以代表驱动程序创建两种类型的对象(框架请求对象和框架文件对象)。 驱动程序可以通过分别调用 WdfDeviceInitSetRequestAttributes 和 WdfDeviceInitSetFileObjectConfig 来注册这些对象的上下文空间。 驱动程序还可以调用 WdfObjectAllocateContext 来为这些对象分配上下文空间。

创建对象后,驱动程序可以通过使用以下任一技术获取指向对象的上下文空间的指针:

  • 使用 WDF_DECLARE_CONTEXT_TYPE 或 WDF_DECLARE_CONTEXT_TYPE_WITH_NAME 宏调用在上述过程中的步骤 2 中创建的上下文访问器方法。
  • 调用 WdfObjectGetTypedContext,提供驱动程序定义的上下文结构的名称。

如果驱动程序具有上下文空间指针,则可以通过调用 WdfObjectContextGetObject 找到上下文空间所属的对象。

创建失败的返回值

当驱动程序尝试创建框架对象失败时,对象创建方法将返回指示失败类型的 NTSTATUS 值。

如果驱动程序在 WDF_OBJECT_ATTRIBUTES 结构中指定无效信息,则框架可以返回:

STATUS_WDF_OBJECT_ATTRIBUTES_INVALID
驱动程序指定了对象上下文名称,但上下文大小为零。

驱动程序指定了上下文大小替代值,但未提供 WDF_OBJECT_CONTEXT_TYPE_INFO 结构。

驱动程序在 WDF_OBJECT_ATTRIBUTES 中指定了一个小于 WDF_OBJECT_CONTEXT_TYPE_INFO 结构的 ContextSize 成员的 ContextSizeOverride 值。

驱动程序在 WDF_OBJECT_ATTRIBUTES 中指定了不在有效值范围内的 ExecutionLevel 值。

驱动程序在 WDF_OBJECT_ATTRIBUTES 中指定了不在有效值范围内的 SynchronizationScope 值。

STATUS_WDF_PARENT_ASSIGNMENT_NOT_ALLOWED
驱动程序尝试将父级分配给对象,但框架不允许驱动程序将父级分配给对象类型。

STATUS_WDF_PARENT_ALREADY_ASSIGNED
驱动程序尝试将父级分配给对象,但已将父级分配给对象。

STATUS_WDF_PARENT_IS_SELF
驱动程序尝试将对象设为自己的父对象。

STATUS_WDF_SYNCHRONIZATION_SCOPE_INVALID
驱动程序指定了一个 WDF_SYNCHRONIZATION_SCOPE类型的值,该值对对象类型无效。

STATUS_WDF_EXECUTION_LEVEL_INVALID
驱动程序指定了一个 WDF_EXECUTION_LEVEL类型的值,该值对对象类型无效。

如果任何框架定义的结构的 Size 成员与结构的实际大小不匹配,框架可以返回STATUS_INFO_LENGTH_MISMATCH。

如果框架无法为新对象分配内存,则可以返回STATUS_INSUFFICIENT_RESOURCES。

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

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

相关文章

vue学习的预备知识为学好vue打好基础

目录 Vue是什么 ?如何使用Vue ?Vue ApiVue入口apiVue实例apiVue函数api 无构建过程的渐进式增强静态HTMLVue模块化构建工具npmyarnWebpackvue-cliVite Vue是什么 ? 文章基于Vue3叙述。 Vue (发音为 /vjuː/,类似 view) 是一款用于…

179. 最大数(LeetCode)

文章目录 前言一、题目讲解二、算法原理三、代码编写1.仿函数写法2.lambda表达式 四、验证五.总结 前言 在本篇文章中,我们将会带着大家采用贪心的方法解决LeetCode中最大数这道问题!!! 一、题目讲解 一组非负整数,包…

【面试经典 150 | 图】被围绕的区域

文章目录 写在前面Tag题目来源解题思路方法一:深搜方法二:广搜 写在最后 写在前面 本专栏专注于分析与讲解【面试经典150】算法,两到三天更新一篇文章,欢迎催更…… 专栏内容以分析题目为主,并附带一些对于本题涉及到的…

03.Kafka 基本使用

Kafka 提供了一系列脚本用于命令行来操作 kafka。 1 Topic 操作 1.1 创建 Topic 创建一个名为 oldersix-topic 的 topic,副本数设置为3,分区数设置为2: bin/kafka-topics.sh \ --create \ --zookeeper 192.168.31.162:2181 \ --replication…

ROS1快速入门学习笔记 - 07话题消息的定义与使用

目录 一、话题模型 二、自定义话题消息 1. 在功能包下创建msg目录用于存储话题文件 2. 在package.xml文件中添加功能包依赖; 3. 在CMakeLists.txt增加编译选项; 4. 完成编译 5. 配置CMakeLists.txt中的编译规则(增加发布者和订阅者&am…

卫浴品牌商家做展示预约小程序的作用是什么

卫浴品牌类别多、普通/智能、场景化等,无论企业还是经销商市场门店都比较饱满,虽然市场需求度高,但同样需要商家不断拓宽销售渠道和挖掘客户价值,破圈增长。 线上多平台发展尤为重要,而小程序作为连接点,对…

ctf web-部分

** web基础知识 ** *一.反序列化 在PHP中,反序列化通常是指将序列化后的字节转换回原始的PHP对象或数据结构的过程。PHP中的序列化和反序列化通过serialize()和unserialize()函数实现。 1.序列化serialize() 序列化说通俗点就是把一个对象变成可以传输的字符串…

就业班 第三阶段(nginx) 2401--4.26 day5 nginx5 nginx https部署实战

三、HTTPS 基本原理 1、https 介绍 HTTPS(全称:HyperText Transfer Protocol over Secure Socket Layer),其实 HTTPS 并不是一个新鲜协议,Google 很早就开始启用了,初衷是为了保证数据安全。 近些年&…

ArcGIS小技巧—模型构建器快速提取河网

上篇文章介绍的基于DEM的河网提取,需要使用多个工具,整体操作比较繁琐,在日常工作中,使用Arcgis提供的模型构建器可以帮助我们将多个工具整合在一起,在面对大量数据批量处理时,可以大大提高工作效率 利用模…

【题解】—— LeetCode一周小结17

【题解】—— 每日一道题目栏 上接:【题解】—— LeetCode一周小结16 22.组合总和 Ⅳ 题目链接:377. 组合总和 Ⅳ 给你一个由 不同 整数组成的数组 nums ,和一个目标整数 target 。请你从 nums 中找出并返回总和为 target 的元素组合的个数…

基于SSM的“个性化电子相册”的设计与实现(源码+数据库+文档+PPT)

基于SSM的“个性化电子相册”的设计与实现(源码数据库文档PPT) 开发语言:Java 数据库:MySQL 技术:SSM 工具:IDEA/Ecilpse、Navicat、Maven 系统展示 个性化电子相册功能结构图 系统后台界面 会员信息管理界面 相…

在网站源码后台增加响应式布局

一本教材上的网站源码,后台在手机上查看还是按照电脑的页面样式,不方便查看和发布新内容。教材上讲了响应式布局。对于页面结构简单的网站,可以利用响应式,使页面自动适用各种屏幕的分辨率。 今天在一个网站源码的后台使用了响应…

经典案例:学习 Java 异常处理的最佳实践

哈喽,各位小伙伴们,你们好呀,我是喵手。 今天我要给大家分享一些自己日常学习到的一些知识点,并以文字的形式跟大家一起交流,互相学习,一个人虽可以走的更快,但一群人可以走的更远。 我是一名后…

OpenCV如何模板匹配

返回:OpenCV系列文章目录(持续更新中......) 上一篇:OpenCV如何实现背投 下一篇 :OpenCV在图像中寻找轮廓 目标 在本教程中,您将学习如何: 使用 OpenCV 函数 matchTemplate()搜索图像贴片和输入图像之间…

为什么我的Mac运行速度变慢 mac运行速度慢怎么办 如何使用CleanMyMac X修复它

近些年伴随着苹果生态的蓬勃发展,越来越多的用户开始尝试接触Mac电脑。然而很多人上手Mac后会发现,它的使用逻辑与Windows存在很多不同,而且随着使用时间的增加,一些奇奇怪怪的文件也会占据有限的磁盘空间,进而影响使用…

电脑已经有了一个Windows10,再多装一个Windows10组成双系统

前言 前段时间已经讲过一次双Windows系统的安装教程,但是小白重新去看了一下,发现写的内容太多,怕小伙伴看了之后一脸萌。 所以今天咱们就重新再来讲讲:在同一台机器上安装Windows10双系统的教程。 注意哦!这里的Wi…

用来传输文件的协议-FTP

一.FTP协议--文件传输协议 1.了解FTP协议 (1)FTP服务是用来传输文件的协议 FTP(File Transfer Protocol,文件传输协议)是TCP/IP协议组中的协议之一,用于互联网上的控制文件的双向传输。是传输文件到Linu…

C++:string 类

在C中定义一个 std::string 字符串可以采用以下几种方式: 1.使用字符串字面量初始化: std::string str "Hello, world!"; 2.使用构造函数初始化: std::string szStringB("Hello wolven"); 3.使用重复字符初始化&am…

51单片机入门(一)

1. 51单片机的基础介绍 2. RAM和ROM的区别 总体而言,RAM和ROM在计算机系统中起着不同的角色,RAM用于临时存储运行时数据,而ROM用于存储永久性的固件和系统程序。 3. 为什么叫51单片机 因为51系列单片机都是使用Intel 8031指令系统的单片机…

【链表——数据结构】

文章目录 1.单链表1.定义2.基本操作2.1.不带头结点2.2后插2.3前插2.4删除2.5按位查找2.6按值查找2.7求单链表长度2.8 建表 2.双链表1.初始化2.插入(后插)3.删除(后删)4.遍历 3.循环链表1.循环单链表2.循环双链表3.代码问题 4.静态链表1.简述基本操作的实现1.初始化3.删除某个结…