WDF驱动开发-特定于KMDF的技术(一)

2024-06-24 08:28

本文主要是介绍WDF驱动开发-特定于KMDF的技术(一),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

这部分的技术是一些零散的记录知识点,它们主要是在WDF框架中特定于KMDF的部分。

将内核模式驱动程序框架和非 PnP 驱动程序配合使用

如果要为不支持 即插即用 (PnP) 的设备编写驱动程序,则驱动程序必须:

  • 在 WDF_DRIVER_CONFIG 结构的 DriverInitFlags 成员中设置 WdfDriverInitNonPnpDriver 标志;
  • 提供 EvtDriverUnload 事件回调函数;
  • 创建仅表示控制设备对象的框架 设备对象;

如果设备不支持 PnP,则驱动程序 不提供EvtDriverDeviceAdd 回调函数。 相反,驱动程序必须确定其设备是否存在。

安装非 PnP 驱动程序

如果 KMDF 驱动程序支持Windows 10上的非即插即用 (PnP) 设备,请使用与非 PnP 驱动程序示例中所示相同的方法,但删除对 INF 文件和共同安装程序的引用。 例如,不需要以下内容:

#define NONPNP_INF_FILENAME  L"\\nonpnp.inf"
#define WDF_SECTION_NAME L"nonpnp.NT.Wdf"LoadWdfCoInstaller
UnloadWdfCoInstallerPFN_WDFPREDEVICEINSTALLEX pfnWdfPreDeviceInstallEx;
PFN_WDFPOSTDEVICEINSTALL   pfnWdfPostDeviceInstall;
PFN_WDFPREDEVICEREMOVE     pfnWdfPreDeviceRemove;
PFN_WDFPOSTDEVICEREMOVE   pfnWdfPostDeviceRemove;

对于非 PnP KMDF 驱动程序,只需调用 SCM API 来创建服务。 

保证向前推进 I/O 操作

某些驱动程序(例如系统分页设备的存储驱动程序)必须至少执行一些受支持的 I/O 操作,而不会失败,以避免丢失关键系统数据。 驱动程序故障的一个潜在原因是内存不足的情况。 如果框架或驱动程序无法分配足够的内存来处理 I/O 请求,则其中一个或另一个可能需要通过错误状态值 完成 I/O 请求来使 I/O 请求失败。

在版本 1.9 之前的 KMDF 版本中,如果框架无法为 I/O 请求数据包分配框架请求对象, I/O 管理器已发送到驱动程序的 I/O,框架始终会失败 I/O 请求。 为了使驱动程序能够在内存不足的情况下处理 I/O 请求,框架版本 1.9 及更高版本为 I/O 队列提供了 有保证的向前进度 功能。

此功能使框架和驱动程序能够分别为请求对象集和与请求相关的驱动程序上下文缓冲区预先分配内存。 仅当系统内存量较低时,框架和驱动程序才使用此预分配的内存。

保证向前进度的功能

通过使用框架保证的 I/O 队列向前进度,驱动程序可以:

  • 要求框架预先分配一组请求对象,以在内存不足的情况下用于特定的 I/O 队列;
  • 提供一个回调函数,用于预分配特定于请求的资源,驱动程序在内存不足的情况下从框架接收预分配的请求对象时可以使用这些资源;
  • 提供另一个回调函数,用于在 未检测到内存不足的情况时为 I/O 请求分配特定于驱动程序的资源。 如果此回调函数的分配由于内存不足而失败,它可以指示框架是否应使用其预分配的请求对象之一;
  • 指定哪些 I/O 请求需要使用预分配的请求对象。 选项包括为所有 IRP 使用预分配的对象、仅在分页 I/O 操作正在进行时使用它们,或让其他驱动程序回调函数检查每个 IRP 以确定是否使用预分配的对象;

如果驱动程序对其一个或多个 I/O 队列实现了有保证的向前进度,则驱动程序将能够更好地在内存不足的情况下成功 处理 I/O 请求 。 你可以为设备的默认 I/O 队列以及驱动程序通过调用 WdfDeviceConfigureRequestDispatching 配置的任何 I/O 队列实现有保证的向前进度。

只有当驱动程序和驱动程序的 I/O 目标 都实现有保证的前进进度时,框架的有保证向前进度功能才适用于驱动程序。 换句话说,如果驱动程序为设备实现有保证的向前进度,则设备驱动程序堆栈中的所有较低级别驱动程序也必须实现有保证的向前进度。

为 I/O 队列启用有保证的向前进度

若要为 I/O 队列启用有保证的向前进度,驱动程序会初始化 WDF_IO_QUEUE_FORWARD_PROGRESS_POLICY 结构,然后调用 WdfIoQueueAssignForwardProgressPolicy 方法。 如果驱动程序调用 WdfDeviceConfigureRequestDispatching 来配置 I/O 队列,则必须在调用 WdfIoQueueAssignForwardProgressPolicy 之前执行此操作。

当驱动程序调用 WdfIoQueueAssignForwardProgressPolicy 时,它可以指定以下三个事件回调函数,所有这些函数都是可选的:

  • EvtIoAllocateResourcesForReservedRequest:驱动程序的 EvtIoAllocateResourcesForReservedRequest 回调函数为框架在内存不足的情况下保留的请求对象分配和存储特定于请求的资源。框架每次创建保留请求对象时都会调用此回调函数。 驱动程序应为一个 I/O 请求分配特定于请求的资源,通常使用保留的请求对象的 上下文空间;
  • EvtIoAllocateRequestResources:驱动程序的 EvtIoAllocateRequestResources 回调函数分配特定于请求的资源以供立即使用。 在框架收到 IRP 并为 IRP 创建请求对象后,将立即调用它。如果回调函数分配资源的尝试失败,回调函数将返回错误状态值。 然后,框架删除新创建的请求对象,并使用其保留的请求对象之一。 反过来,驱动程序 的请求处理程序 使用其 EvtIoAllocateRequestResources 回调函数之前分配的特定于请求的资源;
  • EvtIoWdmIrpForForwardProgress:驱动程序的 EvtIoWdmIrpForForwardProgress 回调函数检查 IRP,并告知框架是使用 IRP 的保留请求对象,还是通过错误状态值完成 I/O 请求来使该请求失败;

仅当框架无法创建新的请求对象,并且你通过在驱动程序的 WDF_IO_QUEUE_FORWARD_PROGRESS_POLICY结构中 设置标志来指示, 希望驱动程序在内存不足的情况下检查 IRP 时,框架才会调用此回调函数。 换句话说,驱动程序可以评估每个 IRP,并确定它是否是即使在内存不足的情况下也必须处理的 IRP。

当驱动程序调用 WdfIoQueueAssignForwardProgressPolicy 时,它还指定你希望框架针对内存不足的情况预先分配的保留请求对象数。 可以选择适合你的设备和驱动程序的请求对象数。 为防止性能降低,驱动程序通常应指定一个数字,该数字近似于驱动程序和设备可以并行处理的 I/O 请求数。

但是,如果驱动程序调用 WdfIoQueueAssignForwardProgressPolicy 及其 EvtIoAllocateResourcesForReservedRequest 回调函数预先分配了过多的保留请求对象或过多特定于请求的资源内存,则驱动程序实际上可能会导致尝试处理的内存不足的情况。 应测试驱动程序和设备的性能,并包括低内存模拟,以确定要选择的最佳数字。

在 WdfIoQueueAssignForwardProgressPolicy 返回之前,框架会创建并保留驱动程序指定的请求对象数。 每次保留请求对象时,框架都会立即调用驱动程序的 EvtIoAllocateResourcesForReservedRequest 回调函数,以便在框架实际使用保留请求对象的情况下,驱动程序可以分配和保存特定于请求的资源。

当某个驱动程序 的请求处理程序 从 I/O 队列接收 I/O 请求时,它可以调用 WdfRequestIsReserved 方法,以确定请求对象是否是框架针对内存不足情况预先分配的请求对象。 如果此方法返回 TRUE,则驱动程序应使用其 EvtIoAllocateResourcesForReservedRequest 回调函数保留的资源。

如果框架使用其保留请求对象之一,则会在驱动程序完成请求后将对象返回到其保留对象集。 框架保存请求对象以及驱动程序通过调用 WdfDeviceInitSetRequestAttributes 或 WdfObjectAllocateContext 创建的任何上下文空间,以便在出现另一个内存不足的情况时重复使用。

框架和驱动程序支持如何保证向前推进

以下是驱动程序和框架为支持 I/O 队列的有保证向前进度而执行的步骤:

1. 驱动程序调用 WdfIoQueueAssignForwardProgressPolicy。

作为响应,框架分配并存储驱动程序指定的请求对象数。 如果驱动程序以前调用 了 WdfDeviceInitSetRequestAttributes,则每个分配包括 WdfDeviceInitSetRequestAttributes 指定的上下文空间。

此外,如果驱动程序提供了 EvtIoAllocateResourcesForReservedRequest 回调函数,则框架会在每次分配和存储请求对象时调用回调函数。

2. 框架接收 I/O 请求数据包 (IRP) I/O 管理器发送到驱动程序。

框架尝试为 IRP 分配请求对象。 如果驱动程序为请求类型创建的 I/O 队列支持保证向前进度,则下一步取决于分配是成功还是失败:

请求对象分配成功:如果驱动程序提供了 EvtIoAllocateRequestResources 回调函数,框架将调用它。 如果回调函数返回STATUS_SUCCESS,框架会将请求添加到 I/O 队列。 如果回调函数返回错误状态值,框架将删除它刚刚创建的请求对象,并使用其预分配的请求对象之一。 当驱动程序的请求处理程序收到请求对象时,它会确定请求对象是否已预先分配,因此是否应使用驱动程序的预分配资源。如果驱动程序 未 提供 EvtIoAllocateRequestResources 回调函数,框架会将请求添加到 I/O 队列,就像驱动程序未启用有保证的向前进度一样。

请求对象分配失败:框架接下来执行的操作取决于驱动程序为 WDF_IO_QUEUE_FORWARD_PROGRESS_POLICY 结构的 ForwardProgressReservedPolicy 成员提供的值。 此成员通知框架何时使用保留请求:始终,仅当 I/O 请求是分页 I/O 操作时,或仅当 EvtIoWdmIrpForwardProgress 回调函数指示应使用保留请求时。

在所有情况下,驱动程序的请求处理程序都可以调用 WdfRequestIsReserved 来确定框架是否使用了保留的请求对象。 如果是这样,驱动程序应使用其 EvtIoAllocateResourcesForReservedRequest 回调函数分配的请求资源。

保证向前进度方案

你正在为可能包含系统分页文件的存储设备编写驱动程序。 从分页文件读取操作和写入操作成功非常重要。

你决定为读取和写入操作创建单独的 I/O 队列,并为这两个 I/O 队列启用有保证的向前进度。 你决定为所有其他请求类型创建第三个 I/O 队列,但不启用有保证的向前进度。

驱动程序堆栈和设备能够并行处理四个写入操作,因此,在调用 WdfIoQueueAssignForwardForwardProgressPolicy 之前,请将 WDF_IO_QUEUE_FORWARD_PROGRESS_POLICY 结构的 TotalForwardProgressRequests 成员设置为 4。

你决定仅当驱动程序的设备是分页设备时才保证向前进度很重要,因此驱动程序将WDF_IO_QUEUE_FORWARD_PROGRESS_POLICY结构的 ForwardProgressReservedPolicy 成员设置为 WdfIoForwardProgressReservedPolicyPagingIO。

由于驱动程序需要每个读取请求和每个写入请求的框架内存对象,因此你决定驱动程序应预先分配一些内存对象,以便在内存不足的情况下用于调用 WdfIoTargetFormatRequestForRead 和 WdfIoTargetFormatRequestForWrite 。

因此,驱动程序为读取队列提供 EvtIoAllocateResourcesForReservedRequest 回调函数,为写入队列提供另一个回调函数。 每次框架调用其中一个回调函数时,回调函数都会调用 WdfMemoryCreate 并保存返回的对象句柄,以应对内存不足的情况。 因为回调函数接收预分配的请求对象的句柄,所以它可以将内存对象父级给请求对象。 DMA 设备的驱动程序也可能预先分配 框架 DMA 对象。

读取和写入队列 的请求处理程序 必须确定每个接收的请求对象是否为框架为内存不足的情况保留的对象。 请求处理程序可以调用 WdfRequestIsReserved,也可以将请求对象句柄与 EvtIoAllocateResourcesForReservedRequest 回调函数之前收到的句柄进行比较。

该驱动程序还为读取队列提供 EvtIoAllocateRequestResources 回调函数,并为写入队列提供另一个回调函数。 框架在收到来自 I/O 管理器的读取或写入请求并成功创建请求对象时调用其中一个回调函数。 其中每个回调函数调用 WdfMemoryCreate 为请求分配内存对象。 如果分配失败,回调函数将返回错误状态值,以通知框架刚刚出现内存不足的情况。 框架检测错误返回值,删除刚刚创建的请求对象,并使用其预分配的对象之一。

此驱动程序不提供 EvtIoWdmIrpForForwardProgress 回调函数,因为它不需要在框架将其添加到 I/O 队列之前检查单个读取或写入 IRP。

请记住,当驱动程序为设备实现有保证的向前进度时,设备驱动程序堆栈中的所有较低级别驱动程序也必须实现有保证的向前进度。

完成 I/O 请求时指定优先级提升

当驱动程序完成 I/O 请求时,它可以调用 WdfRequestCompleteWithPriorityBoost ,以指定系统用于提高请求 I/O 操作的线程的运行时优先级的值。

如果驱动程序调用 WdfRequestComplete 或 WdfRequestCompleteWithInformation 而不是 WdfRequestCompleteWithPriorityBoost,框架将使用基于设备类型的默认优先级提升值。 下表列出了框架使用的默认优先级提升值。 设备类型和优先级提升常量在 Wdm.h 中定义。

设备类型默认优先级提升
FILE_DEVICE_UNDEFINEDIO_NO_INCREMENT
FILE_DEVICE_BEEPIO_NO_INCREMENT
FILE_DEVICE_CD_ROMIO_CD_ROM_INCREMENT
FILE_DEVICE_CD_ROM_FILE_SYSTEMIO_CD_ROM_INCREMENT
FILE_DEVICE_CONTROLLERIO_NO_INCREMENT
FILE_DEVICE_DATALINKIO_NO_INCREMENT
FILE_DEVICE_DFSIO_NO_INCREMENT
FILE_DEVICE_DISKIO_DISK_INCREMENT
FILE_DEVICE_DISK_FILE_SYSTEMIO_DISK_INCREMENT
FILE_DEVICE_FILE_SYSTEMIO_NO_INCREMENT
FILE_DEVICE_INPORT_PORTIO_NO_INCREMENT
FILE_DEVICE_KEYBOARDIO_KEYBOARD_INCREMENT
FILE_DEVICE_MAILSLOTIO_MAILSLOT_INCREMENT
FILE_DEVICE_MIDI_INIO_SOUND_INCREMENT
FILE_DEVICE_MIDI_OUTIO_SOUND_INCREMENT
FILE_DEVICE_MOUSEIO_MOUSE_INCREMENT
FILE_DEVICE_MULTI_UNC_PROVIDERIO_NO_INCREMENT
FILE_DEVICE_NAMED_PIPEIO_NAMED_PIPE_INCREMENT
FILE_DEVICE_NETWORKIO_NETWORK_INCREMENT
FILE_DEVICE_NETWORK_BROWSERIO_NETWORK_INCREMENT
FILE_DEVICE_NETWORK_FILE_SYSTEMIO_NETWORK_INCREMENT
FILE_DEVICE_NULLIO_NO_INCREMENT
FILE_DEVICE_PARALLEL_PORTIO_PARALLEL_INCREMENT
FILE_DEVICE_PHYSICAL_NETCARDIO_NETWORK_INCREMENT
FILE_DEVICE_PRINTERIO_NO_INCREMENT
FILE_DEVICE_SCANNERIO_NO_INCREMENT
FILE_DEVICE_SERIAL_MOUSE_PORTIO_SERIAL_INCREMENT
FILE_DEVICE_SERIAL_PORTIO_SERIAL_INCREMENT
FILE_DEVICE_SCREENIO_VIDEO_INCREMENT
FILE_DEVICE_SOUNDIO_SOUND_INCREMENT
FILE_DEVICE_STREAMSIO_SOUND_INCREMENT
FILE_DEVICE_TAPEIO_NO_INCREMENT
FILE_DEVICE_TAPE_FILE_SYSTEMIO_NO_INCREMENT
FILE_DEVICE_TRANSPORTIO_NO_INCREMENT
FILE_DEVICE_UNKNOWNIO_NO_INCREMENT
FILE_DEVICE_VIDEOIO_VIDEO_INCREMENT
FILE_DEVICE_VIRTUAL_DISKIO_DISK_INCREMENT
FILE_DEVICE_WAVE_INIO_SOUND_INCREMENT
FILE_DEVICE_WAVE_OUTIO_SOUND_INCREMENT
FILE_DEVICE_8042_PORTIO_KEYBOARD_INCREMENT
FILE_DEVICE_NETWORK_REDIRECTORIO_NETWORK_INCREMENT
FILE_DEVICE_BATTERYIO_NO_INCREMENT
FILE_DEVICE_BUS_EXTENDERIO_NO_INCREMENT
FILE_DEVICE_MODEMIO_SERIAL_INCREMENT
FILE_DEVICE_VDMIO_NO_INCREMENT
FILE_DEVICE_MASS_STORAGEIO_DISK_INCREMENT
FILE_DEVICE_SMBIO_NETWORK_INCREMENT
FILE_DEVICE_KSIO_SOUND_INCREMENT
FILE_DEVICE_CHANGERIO_NO_INCREMENT
FILE_DEVICE_SMARTCARDIO_NO_INCREMENT
FILE_DEVICE_ACPIIO_NO_INCREMENT
FILE_DEVICE_DVDIO_NO_INCREMENT
FILE_DEVICE_FULLSCREEN_VIDEOIO_VIDEO_INCREMENT
FILE_DEVICE_DFS_FILE_SYSTEMIO_NO_INCREMENT
FILE_DEVICE_DFS_VOLUMEIO_NO_INCREMENT
FILE_DEVICE_SERENUMIO_SERIAL_INCREMENT
FILE_DEVICE_TERMSRVIO_NO_INCREMENT
FILE_DEVICE_KSECIO_NO_INCREMENT
FILE_DEVICE_FIPSIO_NO_INCREMENT
FILE_DEVICE_INFINIBANDIO_NO_INCREMENT

 

 

这篇关于WDF驱动开发-特定于KMDF的技术(一)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1089629

相关文章

乐鑫 Matter 技术体验日|快速落地 Matter 产品,引领智能家居生态新发展

随着 Matter 协议的推广和普及,智能家居行业正迎来新的发展机遇,众多厂商纷纷投身于 Matter 产品的研发与验证。然而,开发者普遍面临技术门槛高、认证流程繁琐、生产管理复杂等诸多挑战。  乐鑫信息科技 (688018.SH) 凭借深厚的研发实力与行业洞察力,推出了全面的 Matter 解决方案,包含基于乐鑫 SoC 的 Matter 硬件平台、基于开源 ESP-Matter SDK 的一

一份LLM资源清单围观技术大佬的日常;手把手教你在美国搭建「百万卡」AI数据中心;为啥大模型做不好简单的数学计算? | ShowMeAI日报

👀日报&周刊合集 | 🎡ShowMeAI官网 | 🧡 点赞关注评论拜托啦! 1. 为啥大模型做不好简单的数学计算?从大模型高考数学成绩不及格说起 司南评测体系 OpenCompass 选取 7 个大模型 (6 个开源模型+ GPT-4o),组织参与了 2024 年高考「新课标I卷」的语文、数学、英语考试,然后由经验丰富的判卷老师评判得分。 结果如上图所

持久层 技术选型如何决策?JPA,Hibernate,ibatis(mybatis)

转自:http://t.51jdy.cn/thread-259-1-1.html 持久层 是一个项目 后台 最重要的部分。他直接 决定了 数据读写的性能,业务编写的复杂度,数据结构(对象结构)等问题。 因此 架构师在考虑 使用那个持久层框架的时候 要考虑清楚。 选择的 标准: 1,项目的场景。 2,团队的技能掌握情况。 3,开发周期(开发效率)。 传统的 业务系统,通常业

亮相WOT全球技术创新大会,揭秘火山引擎边缘容器技术在泛CDN场景的应用与实践

2024年6月21日-22日,51CTO“WOT全球技术创新大会2024”在北京举办。火山引擎边缘计算架构师李志明受邀参与,以“边缘容器技术在泛CDN场景的应用和实践”为主题,与多位行业资深专家,共同探讨泛CDN行业技术架构以及云原生与边缘计算的发展和展望。 火山引擎边缘计算架构师李志明表示:为更好地解决传统泛CDN类业务运行中的问题,火山引擎边缘容器团队参考行业做法,结合实践经验,打造火山

Eclipse+ADT与Android Studio开发的区别

下文的EA指Eclipse+ADT,AS就是指Android Studio。 就编写界面布局来说AS可以边开发边预览(所见即所得,以及多个屏幕预览),这个优势比较大。AS运行时占的内存比EA的要小。AS创建项目时要创建gradle项目框架,so,创建项目时AS比较慢。android studio基于gradle构建项目,你无法同时集中管理和维护多个项目的源码,而eclipse ADT可以同时打开

Python应用开发——30天学习Streamlit Python包进行APP的构建(9)

st.area_chart 显示区域图。 这是围绕 st.altair_chart 的语法糖。主要区别在于该命令使用数据自身的列和指数来计算图表的 Altair 规格。因此,在许多 "只需绘制此图 "的情况下,该命令更易于使用,但可定制性较差。 如果 st.area_chart 无法正确猜测数据规格,请尝试使用 st.altair_chart 指定所需的图表。 Function signa

WDF驱动开发-WDF总线枚举(一)

支持在总线驱动程序中进行 PnP 和电源管理 某些设备永久插入系统,而其他设备可以在系统运行时插入和拔出电源。 总线驱动 必须识别并报告连接到其总线的设备,并且他们必须发现并报告系统中设备的到达和离开情况。 总线驱动程序标识和报告的设备称为总线的 子设备。 标识和报告子设备的过程称为 总线枚举。 在总线枚举期间,总线驱动程序会为其子 设备创建设备对象 。  总线驱动程序本质上是同时处理总线枚

JavaWeb系列六: 动态WEB开发核心(Servlet) 上

韩老师学生 官网文档为什么会出现Servlet什么是ServletServlet在JavaWeb项目位置Servlet基本使用Servlet开发方式说明快速入门- 手动开发 servlet浏览器请求Servlet UML分析Servlet生命周期GET和POST请求分发处理通过继承HttpServlet开发ServletIDEA配置ServletServlet注意事项和细节 Servlet注

手把手教你入门vue+springboot开发(五)--docker部署

文章目录 前言一、前端打包二、后端打包三、docker运行总结 前言 前面我们重点介绍了vue+springboot前后端分离开发的过程,本篇我们结合docker容器来研究一下打包部署过程。 一、前端打包 在VSCode的命令行中输入npm run build可以打包前端代码,出现下图提示表示打包完成。 打包成功后会在前端工程目录生成dist目录,如下图所示: 把

Sapphire开发日志 (十) 关于页面

关于页面 任务介绍 关于页面用户对我组工作量的展示。 实现效果 代码解释 首先封装一个子组件用于展示用户头像和名称。 const UserGrid = ({src,name,size,link,}: {src: any;name: any;size?: any;link?: any;}) => (<Box sx={{ display: "flex", flexDirecti