WDF驱动开发-电源策略(三)

2024-06-11 07:36
文章标签 wdf 电源 驱动 策略 开发

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

多组件设备的 KMDF 驱动程序只能将请求发送到处于活动状态的组件。 通常,驱动程序将 I/O 队列分配给组件或组件集。

首先考虑分配给单个组件的队列。 驱动程序在组件变为活动状态时启动队列,并在组件空闲时停止队列。 因此,当 KMDF 调用队列的请求处理程序时,设备完全处于 (D0) 状态,并且所需的组件处于活动状态。 请求处理程序可以安全地访问组件硬件。

相同的概念适用于分配给一组组件的队列。 在这种情况下,当集中的所有组件都处于活动状态时,驱动程序将启动队列。 当任一组件处于空闲状态时,驱动程序将停止队列。

下面介绍在涉及需要不同组件组合的多个请求类型的情况下,多组件设备的 KMDF 驱动程序如何实现此类支持。

示例

对于驱动程序支持的每个请求类型,请标识所需的组件。 例如,假设某个设备具有三个组件:0、1 和 2,驱动程序会接收三种类型的请求:A、B 和 C。请求的组件要求如下:

请求类型   所需组件
A0,2
B1
C0,1,2

在此示例中,有三组不同的组件,每个请求类型各有一组。 驱动程序为设备提供一个默认的电源托管 I/O 队列,以及对应于每组组件的额外电源管理队列。 在上面的示例中,驱动程序创建一个主队列和三个辅助队列,其中一个对应于每个组件集。 此队列配置如下图所示: 

驱动程序为每个组件集维护一个位掩码。 位掩码中的每个位表示其中一个组件的活动/空闲状态。 如果设置了位,则组件处于活动状态。 如果清除该位,则组件处于空闲状态。

请求到达时,顶级队列 的请求处理程序 将确定请求需要哪些组件,并为每个组件调用 PoFxActivateComponent 。 然后,请求处理程序将请求转发到与该组件集对应的辅助 I/O 队列。

组件变为活动状态时,电源管理框架 (PoFx) 调用驱动程序的 ComponentActiveConditionCallback 例程。 在此回调中,驱动程序在表示该组件的每个位掩码中设置与指定组件对应的位。 如果设置了给定位掩码中的所有位,则相应集中的所有组件都处于活动状态。 对于每个完全处于活动状态的组件集,驱动程序会调用 WdfIoQueueStart 来启动相应的辅助 I/O 队列。

例如,请考虑上面的假设设备。 假设组件 0 处于活动状态,而组件 1 和组件 2 处于空闲状态。 当组件 2 变为活动状态时,PoFx 将调用该组件的 ComponentActiveConditionCallback 例程。 请求类型 A 和 C 使用组件 2,因此驱动程序会操作这两种请求类型的位掩码。 由于现在已设置请求类型 A 的位掩码中的所有位,因此驱动程序会启动请求类型 A 的队列。但是,并非所有位都为请求类型 C 设置, (组件 1 仍为空闲) 。 驱动程序不会启动请求类型 C 的队列。

启动辅助 I/O 队列时,框架开始传递存储在队列中的请求。 在辅助 I/O 队列 的请求处理程序 中,驱动程序可以安全地处理请求,因为组件处于活动状态,并且已针对每个请求的组件使用电源引用。

驱动程序处理完请求后,会针对请求使用的每个组件调用 PoFxIdleComponent ,然后完成请求。 当没有更多使用组件的请求时,Power Framework 将调用驱动程序的 ComponentIdleConditionCallback 例程。

在此回调中,驱动程序在表示该组件的每个位掩码中清除与指定组件对应的位。 如果给定的位掩码指示组件是相应集中的第一个组件,以转换为空闲条件,则驱动程序将调用 WdfIoQueueStop 来停止相应的辅助 I/O 队列。 这样,驱动程序可确保队列不会调度请求,除非相应集中的所有组件都处于活动状态。

再次考虑上述示例。 假设所有组件都处于活动状态,因此所有队列都已启动。 组件 1 空闲时,PoFx 会调用组件 1 的 ComponentIdleConditionCallback 例程。 在此回调中,驱动程序操作请求类型 B 和 C 的位掩码,因为它们使用组件 1。 由于组件 1 是这两种请求类型的第一个空闲组件,因此驱动程序会停止请求类型 B 和 C 的队列。

假设此时组件 0 处于空闲状态。 在组件 0 的 ComponentIdleConditionCallback 中,驱动程序操作请求类型 A 和 C 的位掩码。由于组件 0 是请求类型 A 的第一个空闲组件, (组件 2 在) 仍处于活动状态,因此驱动程序停止请求类型 A 的队列。但是,对于请求类型 C,组件 0 不是第一个处于空闲状态的组件。 驱动程序不会停止请求类型 C 的队列, (它之前) 这样做。

若要使用此示例中所述的技术,驱动程序还必须为其每个辅助队列注册 EvtIoCanceledOnQueue 回调函数。 如果在辅助队列中取消请求,驱动程序可以使用此回调为每个相应的组件调用 PoFxIdleComponent 。 这样做会释放请求处理程序在将请求转发到辅助队列之前调用 PoFxActivateComponent 时获取的电源引用。

支持多组件设备空闲时关闭电源

多组件设备的 KMDF 驱动程序可以支持 空闲关闭 电源和功能电源状态。 由于在这种情况下,驱动程序直接注册到电源管理框架 (PoFx) ,因此驱动程序必须与 PoFx 协调生成的 Dx 状态更改。

调用 WdfDeviceAssignS0IdleSettings 时,驱动程序必须在WDF_DEVICE_POWER_POLICY_IDLE_SETTINGS结构中将 IdleTimeoutType 设置为 DriverManagedIdleTimeout。 此外,驱动程序必须将 PowerUpIdleDeviceOnSystemWake 设置为 WdfTrue, 并将 IdleCaps 设置为 IdleCannotWakeFromS0,如以下示例所示。

WDF_DEVICE_POWER_POLICY_IDLE_SETTINGS s0IdleSettings;WDF_DEVICE_POWER_POLICY_IDLE_SETTINGS_INIT(&s0IdleSettings, IdleCannotWakeFromS0);
s0IdleSettings.IdleTimeoutType = DriverManagedIdleTimeout;
s0IdleSettings.PowerUpIdleDeviceOnSystemWake = WdfTrue;
s0IdleSettings.IdleTimeout = 1;
status = WdfDeviceAssignS0IdleSettings(device, &s0IdleSettings);
从 Working (D0) 转换为 Low-Power (Dx) 状态

在 EvtDeviceSelfManagedIoInit 中,驱动程序调用 WdfDeviceStopIdle 来获取电源参考,从而阻止 WDF 将设备置于低功耗状态。

驱动程序通过从其 DevicePowerRequiredCallback 回调例程调用 WdfDeviceResumeIdle 来释放电源引用。

驱动程序通常指定非常短的空闲超时,以便 WDF 在所有电源参考释放后不久将设备置于低功耗状态。

从 Low-Power (Dx) 转换为工作状态 (D0) 状态

在 DevicePowerRequiredCallback 中,驱动程序必须将设备置于其工作 (D0) 状态。 为此,它必须遵循对 WdfDeviceStopIdle 的调用,并将 WaitForD0 参数设置为 TRUE。 对 WdfDeviceStopIdle 的阻止调用不得从 DevicePowerRequiredCallback中进行。

相反,驱动程序必须延迟对在被动级别运行的工作线程的阻塞调用,并保证不会在电源管理的队列的 I/O 调度例程的上下文中进行 WdfDeviceStopIdle 调用。

如果驱动程序之前已调用 WdfDeviceInitSetPowerPageable (这意味着它可以在电源转换期间访问可分页数据) ,则驱动程序可以调用 WdfWorkItemCreate 来创建框架工作项。 如果驱动程序尚未设置 power-pageable,则驱动程序必须创建自己的系统线程。 

WdfDeviceStopIdle 返回后,即使该方法返回错误,驱动程序也必须调用 PoFxReportDevicePoweredOn。

这篇关于WDF驱动开发-电源策略(三)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

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

服务器雪崩的应对策略之----SQL优化

SQL语句的优化是数据库性能优化的重要方面,特别是在处理大规模数据或高频访问时。作为一个C++程序员,理解SQL优化不仅有助于编写高效的数据库操作代码,还能增强对系统性能瓶颈的整体把握。以下是详细的SQL语句优化技巧和策略: SQL优化 1. 选择合适的数据类型2. 使用索引3. 优化查询4. 范式化和反范式化5. 查询重写6. 使用缓存7. 优化数据库设计8. 分析和监控9. 调整配置1、

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

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

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

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

设置Nginx缓存策略

详细信息 Nginx服务器的缓存策略设置方法有两种:add_header或者expires。 1. add_header 1)语法:add_header name value。 2)默认值:none。 3)使用范围:http、server、location。 配置示例如下: add_header cache-control "max-age=86400";#设置缓存时间为1天。add

手把手教你入门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

ROS2从入门到精通4-4:局部控制插件开发案例(以PID算法为例)

目录 0 专栏介绍1 控制插件编写模板1.1 构造控制插件类1.2 注册并导出插件1.3 编译与使用插件 2 基于PID的路径跟踪原理3 控制插件开发案例(PID算法)常见问题 0 专栏介绍 本专栏旨在通过对ROS2的系统学习,掌握ROS2底层基本分布式原理,并具有机器人建模和应用ROS2进行实际项目的开发和调试的工程能力。 🚀详情:《ROS2从入门到精通》 1 控制插

JavaWeb 学习笔记 spring+jdbc整合开发初步

JdbcTemplate类是Spring的核心类之一,可以在org.springframework.jdbc.core中找到它。JdbcTemplate类在内部已经处理数据库的建立和释放,可以避免一些常见的错误。JdbcTemplate类可直接通过数据源的应用实例化,然后在服务中使用,也可在xml配置中作为JavaBean应用给服务使用直接上一个实例步骤1.xml配置 <?xml version