用于 USB 设备的 Windows 桌面应用
在本文中,你将了解应用程序如何调用 WinUSB 函数 来与 USB 设备通信。 对于此类应用程序, WinUSB (Winusb.sys)必须安装为设备的函数驱动程序。 设备内核模式堆栈中的 WinUSB。 此驱动程序包含在 \Windows\System32\drivers 文件夹中的 Windows 中。
如果使用 Winusb.sys 作为 USB 设备的函数驱动程序,则可以从应用程序调用 WinUSB 函数 以与设备通信。 用户模式 DLL Winusb.dll公开的这些函数简化了通信过程。 应用程序无需构造设备 I/O 控制请求来执行标准 USB 操作(例如配置设备、发送控制请求以及向设备传输数据),而是调用等效的 WinUSB 函数。
Winusb.dll使用应用程序提供的数据来构造适当的设备 I/O 控制请求,然后将请求发送到Winusb.sys进行处理。 为了与 USB 堆栈通信,WinUSB 函数使用与应用程序请求关联的相应 IOCTL 调用 DeviceIoControl 函数。 请求完成后,WinUSB 函数会将Winusb.sys(例如读取请求中的数据)返回的任何信息传回调用进程。 如果对 DeviceIoControl 的调用成功,它将返回非零值。 如果调用失败或挂起(未立即处理),DeviceIoControl 将返回零值。 如果发生错误,应用程序可以调用 GetLastError 以获取更详细的错误消息。
与实现驱动程序相比,使用 WinUSB 函数与设备通信更简单。 但是,请注意以下限制:
WinUSB 函数允许一个应用程序一次与设备通信。 如果需要多个应用程序与设备并发通信,则必须实现函数驱动程序。
在Windows 8.1之前,WinUSB 函数不支持向或传出异时序终结点流式传输数据。
WinUSB 函数不支持已具有内核模式支持的设备。 此类设备的示例包括电话 API (TAPI) 和 NDIS 支持的调制解调器和网络适配器。
对于多功能设备,可以使用设备的 INF 文件单独为每个 USB 函数指定内置内核模式驱动程序或Winusb.sys。 但是,只能为特定函数指定其中一个选项,而不是同时指定这两个选项之一。
注意
WinUSB 函数需要 Windows XP 或更高版本。 可以在 C/C++应用程序中使用这些函数与 USB 设备通信。 若要编写使用 WinUSB API 的 UWP 应用,请参阅 适用于 USB 设备的 UWP 应用。
入门
获取为设备编写 Windows 桌面应用所需的工具
- 按照下载 Windows 驱动程序工具包中的说明进行操作。
获取测试 USB 设备及其硬件规格。
编写获取设备的句柄的框架应用。
编写第一个应用程序有两种方法:
基于 Visual Studio 中包含的 WinUSB 模板进行写入。 有关详细信息,请参阅 基于 WinUSB 模板编写 Windows 桌面应用。
调用 SetupAPI 例程以获取设备的句柄,并通过调用 WinUsb_Initialize将其打开。 有关详细信息,请参阅 如何使用 WinUSB 函数访问 USB 设备。
为设备安装Winusb.sys。
如果使用 Visual Studio,请使用 Visual Studio 部署在目标计算机上安装驱动程序包。 有关说明,请参阅 基于 WinUSB 模板编写 Windows 桌面应用。 否则,请编写自定义 INF,在设备管理器中手动安装驱动程序。 有关详细信息,请参阅 WinUSB (Winusb.sys) 安装。
获取有关设备的信息并查看其描述符。
有关概念信息,请参阅 所有 USB 开发人员的概念。 通过读取每个受支持的备用设置的配置描述符、接口描述符及其终结点描述符来获取有关设备功能的信息。 有关详细信息,请参阅 查询设备以获取 USB 描述符。
发送 USB 控制传输。
向设备发送标准控制请求和供应商指令。 有关详细信息,请参阅 将控制传输发送到默认终结点。
发送批量传输或中断传输。
对设备支持的批量、中断和不连续终结点执行读取和写入操作。 有关详细信息,请参阅 问题 I/O 请求。
发送时序传输。
发送时序读取和写入请求,主要用于流式处理数据。 此功能仅适用于Windows 8.1及更高版本。 有关详细信息,请参阅 从 WinUSB 桌面应用发送 USB 时序传输。