赞
踩
符合通用串行总线 (USB) 规范的 USB 设备的电源管理功能具有一组丰富而复杂的电源管理功能。 请务必了解这些功能如何与 Windows 驱动程序模型 (WDM) 交互,特别是 Microsoft Windows 如何调整标准 USB 功能以支持系统唤醒体系结构。
基于内核模式驱动程序框架的 USB 客户端驱动程序 (KMDF) 和用户模式驱动程序框架 (UMDF) 应使用基本技术和相应框架支持的机制来管理 USB 设备的电源。
下面介绍用于通用串行总线 2.0 规范第 9.1 节规定的 USB 设备电源状态的 WDM 设备状态。
USB 设备电源状态(如通用串行总线 2.0 规范第 9.1 节所述)可分为三大类:
WDM 功率模型中定义的设备功率状态与 USB 标准中定义的设备功率状态之间没有直接关联。 例如,在 USB 规范中,术语 suspended 和 idle 具有非常明确的含义;但在 WDM 功率模型中,这些术语的用法往往不同。 Windows 客户端驱动程序可以将 USB 设备置于挂起状态。 客户机驱动程序准备挂起设备时,会指示总线驱动程序将其空闲。
WDM 模型中的设备功率状态可概括如下:
WDM 功率模型使用设备防护一词来实现远程唤醒。 防护是一项软件操作,通常(但不总是)会导致启用 USB 设备远程唤醒功能的硬件操作。 等待唤醒 IRP (IRP_MN_WAIT_WAKE) 是 WDM 软件对设备进行的远程唤醒操作。
USB 设备的电源策略管理器负责设置设备的电源状态。 电源策略管理器通过发出 WDM 电源 (IRP_MN_SET_POWER) IRP 来设置电源状态。总线驱动程序采取的行动取决于电源策略管理器要求的设备功率级别。 下面列出了总线驱动程序针对各级设置功率请求所采取的操作:
总线驱动程序执行以下任务
确保所有上游 USB 集线器都已通电并准备好接收请求。
如果设备的 USB 端口处于暂停状态,则通过清除 PORT_SUSPEND 功能恢复端口。
以 STATUS_SUCCESS 完成设备的空闲 IRP(如果有的话)。
如果设备已布防,则解除布防以便远程唤醒。
总线驱动程序执行以下任务
如果等待唤醒 IRP (IRP_MN_WAIT_WAKE) 待定,则启动设备进行远程唤醒。
通过设置 PORT_SUSPEND 功能暂停设备的 USB 端口。
总线驱动程序执行以下任务
复合设备上接口的客户端驱动程序必须与设备上其他接口的客户端驱动程序共享复合设备的电源状态。 因此,接口的客户端驱动程序无法在不影响设备上其他接口的情况下,将复合设备置于低功耗状态。 当接口的客户端驱动程序发送 IRP_MN_SET_POWER 请求时,USB 通用父驱动程序 (Usbccgp.sys) 会采取以下操作。
总线驱动程序执行以下任务
确保所有上游 USB 集线器都已通电并准备好接收请求。
如果设备的 USB 端口处于暂停状态,则通过清除 PORT_SUSPEND 功能恢复端口。
完成客户端驱动程序的空闲 IRP(如果有),并显示 STATUS_SUCCESS。
总线驱动程序没有采取任何行动。
总线驱动程序执行以下任务
当以下条件之一为真时,通用父驱动程序会暂停设备的 USB 端口:
USB 函数驱动程序通过实现 USB 选择性挂起支持运行时空闲检测。 下面是面向驱动程序开发人员的内容,介绍如何在基于 Windows® Driver Foundation (WDF) 的 USB 驱动程序中实现选择性挂起。
选择性挂起是能够关闭并稍后恢复空闲 USB 设备,而连接到该设备的计算机保持工作状态 (S0) 。 对于节能操作(尤其是在移动电脑上),所有 USB 设备和驱动程序都应支持选择性挂起。 在设备处于空闲状态时关闭设备,但当系统保持 S0 状态时,具有以下显著优势:
USB 设备的每个功能驱动程序都应实现积极的电源管理,以便在系统运行时挂起空闲设备。
USB 设备支持通过 USB 选择性挂起进行运行时空闲检测。 选择性挂起允许将空闲设备置于挂起状态,而不会影响连接到同一集线器的其他设备(如果是多功能设备),而不会影响设备中的其他功能。 当所有设备或功能都已挂起时,整个集线器或多功能设备可以关闭电源。
从硬件的角度来看,选择性挂起是 USB 端口上的一种物理状态。 当附加到端口的所有函数都空闲时,该端口可以进入选择性挂起。
为了符合 USB 规范,所有 USB 设备都必须支持选择性挂起。 当 USB 总线空闲时,设备必须能够关闭电源。 Microsoft 提供的 USB 集线器驱动程序在硬件级别实现选择性挂起。
USB 功能驱动程序应通过 WDF 为其各个设备功能实现选择性挂起,WDF 与总线驱动程序通信,并管理暂停和恢复设备功能的设备 I/O 控制请求。 WDF 使内核模式和用户模式驱动程序都支持选择性挂起。
函数驱动程序的 USB 选择性挂起代码的详细信息取决于驱动程序是在用户模式还是内核模式下运行。 请考虑以下准则:
设备堆栈的电源策略所有者 (PPO) 是确定设备在任何给定时间应处于哪个电源状态的驱动程序。 每个设备堆栈中只有一个驱动程序可以是 PPO。 函数驱动程序通常是其设备的 PPO。
如果 USB 驱动程序支持选择性挂起,并且其设备堆栈中的 PPO 之上分层,则驱动程序不得使用电源管理的队列。 这适用于 UMDF 和 KMDF 驱动程序。 如果在设备挂起时收到电源管理队列的请求,整个设备堆栈可能会停止。
图 1 显示了通过 USB 驱动程序的 I/O 队列向 USB 驱动程序发出的 I/O 请求流。
在图中,针对 USB 驱动程序的请求到达。 框架将请求添加到相应的队列。
如果队列不受电源管理,框架将根据驱动程序为队列配置的调度类型向驱动程序提供请求,该调度类型 (顺序、并行或手动) 。 然后,驱动程序处理请求。
如果队列受电源管理且设备未挂起,则框架会根据配置的调度类型向驱动程序提供请求。
但是,如果设备挂起,框架的操作取决于驱动程序是否是设备堆栈的 PPO。 如果驱动程序是 PPO,则框架将与 USB 父驱动程序通信以启动设备。 设备恢复后,框架向驱动程序提供请求。
如果驱动程序不是 PPO,则框架不会执行进一步的操作,因为只有 PPO 可以恢复设备。 请求保留在队列中。 如果 PPO 未收到导致它恢复设备的任何请求,则设备堆栈将停止。
重要的 API
UMDF 函数驱动程序可以通过以下两种方式之一支持 USB 选择性挂起:
这两种方法只需要少量代码。 WDK 中提供的 IdleWake 示例演示如何在 UMDF USB 驱动程序中支持选择性挂起。 可以在 %WinDDK%\BuildNumber\Src\Usb\OsrUsbFx2\ UMDF\Fx2_Driver\IdleWake 中找到此示例,文件夹包含示例的 PPO 和非 PPO 版本。
支持选择性挂起的 UMDF 驱动程序必须遵循以下准则:
默认情况下,WinUSB.sys 是包含 UMDF USB 驱动程序的设备堆栈的 PPO。 从 WDF 1.9 开始,基于 UMDF 的 USB 驱动程序可以声明电源策略所有权。 由于每个设备堆栈中只有一个驱动程序可以是 PPO,因此作为 PPO 的 UMDF USB 驱动程序必须在 WinUSB.sys 中显式禁用电源策略所有权。
1.调用 IWDFDeviceInitialize::SetPowerPolicyOwnership 并传递 TRUE,通常是从驱动程序回调对象上的 IDriverEntry::OnDeviceAdd 方法传递。 例如:
FxDeviceInit->SetPowerPolicyOwnership(TRUE);
2.在 WinUSB 中禁用电源策略所有权。 在驱动程序的 INF 文件中,包含 AddReg 指令,该指令将注册表中的 WinUsbPowerPolicyOwnershipDisabled 值设置为非零值。 AddReg 指令必须出现在 DDInstall.HW 节中。 例如:
- [MyDriver_Install.NT.hw]
- AddReg=MyDriver_AddReg
-
- [MyDriver_AddReg]
- HKR,,"WinUsbPowerPolicyOwnershipDisabled",0x00010001,1
对于支持选择性挂起的 UMDF 驱动程序,UMDF 驱动程序是否为其设备拥有电源策略决定了它可以使用的 I/O 队列的类型。 支持选择性挂起和 PDO 的 UMDF 驱动程序可以使用电源托管或非电源管理的队列。 支持选择性挂起但不是 PPO 的 UMDF USB 驱动程序不应使用任何电源管理的 I/O 队列。
如果在设备挂起时,电源管理队列的 I/O 请求到达,则框架不会显示该请求,除非驱动程序为 PPO,如 USB 驱动程序中的选择性挂起图像中所示。 如果 UMDF 驱动程序不是设备的 PPO,则框架无法代表设备启动。 因此,请求仍停滞在电源管理的队列中。 请求永远不会到达 WinUSB,因此 WinUSB 无法启动设备。 因此,设备堆栈可能会停止。
如果队列不受电源管理,则框架会向 UMDF 驱动程序提供 I/O 请求,即使设备关闭也是如此。 UMDF 驱动程序格式化请求,并按照通常的方式将请求在设备堆栈中向下转发到默认 I/O 目标。 不需要特殊代码。 当请求到达 PPO (WinUSB.sys) 时,WinUSB.sys 启动设备并执行所需的 I/O 操作。
%WinDDK%\BuildNumber\Src\Usb\OsrUsbFx2\umdf\Fx2_Driver\IdleWake 中的示例驱动程序定义在生成非 PPO 版本的驱动程序时_NOT_POWER_POLICY_OWNER_常量。 当驱动程序为读取和写入请求创建队列时,它会确定是否通过检查常量来创建电源管理的队列。
为了创建队列,驱动程序调用驱动程序定义的 CMyQueue::Initialize 方法,该方法采用以下三个参数:
以下代码片段演示驱动程序在创建读写队列过程中对 CMyQueue::Initialize 方法的调用:
- #if defined(_NOT_POWER_POLICY_OWNER_)
- powerManaged = false;
- #else
- powerManaged = true;
- #endif
- hr = __super::Initialize(WdfIoQueueDispatchParallel,
- true,
- powerManaged,
- );
CMyQueue::Initialize 然后调用 IWDFDevice::CreateIoQueue 来创建队列,如下所示:
- hr = m_FxDevice->CreateIoQueue(
- callback,
- Default,
- DispatchType,
- PowerManaged,
- FALSE,
- &fxQueue
- );
此代码序列生成一个并行调度请求的默认队列。 如果驱动程序是 PPO,则队列受电源管理,如果驱动程序不是 PPO,则队列不受电源管理。
若要支持选择性挂起,作为其设备堆栈的 PPO 的 UMDF USB 驱动程序必须执行以下操作:
声明设备堆栈的电源策略所有权,通常在其驱动程序回调对象上的 IDriverEntry::OnDeviceAdd 方法中,如前所述。通过在框架设备对象上调用 IWDFDevice2::AssignS0IdleSettings 方法来启用选择性挂起。
通常从设备回调对象的 OnPrepareHardware 方法调用 IWDFDevice2::AssignS0IdleSettings。 将参数设置为 AssignS0IdleSettings,如下所示:
IdleCaps 到 IdleUsbSelectiveSuspend。
DxState 到框架将空闲设备转换到的设备睡眠状态。 对于 USB 选择性挂起,请指定 PowerDeviceMaximum,指示框架应使用总线驱动程序指定的值。
IdleTimeout 为在框架将其转换为 DxState 之前设备必须处于空闲状态的毫秒数。
UserControlOfIdleSettings 为 IdleAllowUserControl (如果驱动程序允许用户管理空闲设置),否则为 IdleDoNotAllowUserControl。
已启用WdfUseDefault 以默认启用选择性挂起,但允许用户设置替代默认值。
以下示例演示IdleWake_PPO驱动程序如何在其内部 CMyDevice::SetPowerManagement 方法中调用此方法:
- hr = m_FxDevice->CreateIoQueue(
- callback,
- Default,
- DispatchType,
- PowerManaged,
- FALSE,
- &fxQueue
- );
如果设备硬件可以生成唤醒信号,则 UMDF 驱动程序还可以支持来自 S1、S2 或 S3 的系统唤醒。
非 PPO 的 UMDF 函数驱动程序可以使用基础 WinUSB.sys 驱动程序的功能支持选择性挂起。 UMDF 驱动程序必须通知 WinUSB 设备和驱动程序支持选择性挂起,并且必须在 INF 文件中或通过在 USB 目标设备对象上设置电源策略来启用选择性挂起。
如果 UMDF 函数驱动程序启用选择性挂起,则基础 WinUSB.sys 驱动程序将确定设备何时处于空闲状态。 WinUSB 在没有挂起的传输或中断或批量端点上唯一挂起的传输为 IN 传输时,WinUSB 会启动空闲超时计数器。 默认情况下,空闲超时为 5 秒,但 UMDF 驱动程序可以更改此默认值。
当 WinUSB.sys 确定设备处于空闲状态时,它会发送请求,将设备挂起内核模式设备堆栈。 总线驱动程序会根据需要更改硬件的状态。 如果端口上的所有设备功能都已挂起,该端口将进入 USB 选择性挂起状态。
如果 I/O 请求在设备挂起时到达 WinUSB.sys,WinUSB.sys 恢复设备操作(如果设备必须通电才能为请求提供服务)。 当系统保留在 S0 中时,UMDF 驱动程序不需要任何代码即可恢复设备。 如果设备硬件可以生成唤醒信号,则 UMDF 驱动程序还可以支持来自 S1、S2 或 S3 的系统唤醒。
非 PPO 的 UMDF 驱动程序可以通过执行以下两个步骤来支持选择性挂起:
此外,驱动程序还可以选择:
有关如何在非 PPO 的 UMDF USB 函数驱动程序中实现 USB 选择性挂起的示例,请参阅 WDK 中的 Fx2_Driver 示例。 此示例位于 %WinDDK%\BuildNumber\Src\Usb\OsrUsbFx2\Umdf\Fx2_Driver\ IdleWake_Non-PPO。
若要通知 WinUSB.sys 设备可以支持 USB 选择性挂起,设备 INF 必须将 DeviceIdleEnabled 值添加到设备的硬件密钥,并将该值设置为 1。 以下示例演示Fx2_Driver示例如何在 WUDFOsrUsbFx2_IdleWakeNon-PPO.Inx 文件中添加和设置此值:
- [OsrUsb_Device_AddReg]
- ...
- HKR,,"DeviceIdleEnabled",0x00010001,1
UMDF USB 驱动程序可以在运行时或在 INF 中安装期间启用 USB 选择性挂起。
- BOOL AutoSuspend = TRUE;
- hr = m_pIUsbTargetDevice->SetPowerPolicy( AUTO_SUSPEND,
- sizeof(BOOL),
- (PVOID) &AutoSuspend );
HKR,,"DefaultIdleState",0x00010001,1
默认情况下,如果没有任何传输挂起,或者唯一挂起的传输是中断或批量终结点上的 IN 传输,WinUSB 会在 5 秒后暂停设备。 UMDF 驱动程序可以在安装 INF 时或在运行时更改此空闲超时值。
HKR,,"DefaultIdleTimeout",0x00010001,7000
- HRESULT hr;
- ULONG value;
- value = 10 * 1000;
- hr = m_pIUsbTargetDevice->SetPowerPolicy( SUSPEND_DELAY,
- sizeof(ULONG),
- (PVOID) &value );
使用 WinUSB 选择性挂起支持的 UMDF USB 驱动程序可以选择性地允许用户启用或禁用选择性挂起。 为此,请在 INF 中包含 AddReg 指令,该指令将 UserSetDeviceIdleEnabled 值添加到设备的硬件密钥,并将值设置为 1。 下面显示了要用于 AddReg 指令的字符串:
HKR,,"UserSetDeviceIdleEnabled",0x00010001,1
如果设置了 UserSetDeviceIdleEnabled,则设备的“属性”对话框包含一个“电源管理”选项卡,允许用户启用或禁用 USB 选择性挂起。
在 UMDF 驱动程序中,对系统唤醒的支持独立于对选择性挂起的支持。 UMDF USB 驱动程序可以同时支持系统唤醒和选择性挂起、系统唤醒和选择性挂起,或者系统唤醒或选择性挂起。 支持系统唤醒的设备可以将系统从睡眠状态唤醒 (S1、S2 或 S3) 。
UMDF USB PPO 驱动程序可以通过为框架的驱动程序对象提供唤醒信息来支持系统唤醒。 当外部事件触发系统唤醒时,框架会将设备返回到工作状态。
USB 非 PPO 驱动程序可以使用 WinUSB.sys 驱动程序实现的系统唤醒支持。
使用以下参数在框架的设备对象上调用 IWDFDevice2::AssignSxWakeSettings 方法:
以下示例演示IdleWake_PPO驱动程序如何在其内部 CMyDevice::SetPowerManagement 方法中调用此方法:
- hr = m_FxDevice->AssignSxWakeSettings( PowerDeviceMaximum,
- WakeAllowUserControl,
- WdfUseDefault);
若要通过 WinUSB 启用系统唤醒,驱动程序的 INF 会将注册表值 SystemWakeEnabled 添加到设备的硬件密钥,并将其设置为 1。 IdleWake_Non-PPO 示例启用系统唤醒,如下所示:
- [OsrUsb_Device_AddReg]
- ...
- HKR,,"SystemWakeEnabled",0x00010001,1
通过设置此值,驱动程序将启用系统唤醒并允许用户控制设备唤醒系统的能力。 在 设备管理器 中,设备的电源管理设置属性页包括一个检查框,用户可以使用该框启用或禁用系统唤醒。
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。