原文:C#与USB HID间的通信

C#与USBHID接口的通讯相对于与串口间的通讯较为复杂,其中需要多次调用到Windows的一些API。其原理编者尚未全部理清,以下提供简单的USBHID通讯流程。(参考网友资料)

一、获取所有连接HID的设备信息。

1.通过一个空的GUID来获取HID的全局GUID。

Guid
HIDGuid = Guid.Empty;

///

/// The
HidD_GetHidGuid routine returns the device interface GUID for
HIDClass devices.

///

/// a
caller-allocated GUID buffer that the routine uses to return the
device interface GUID for HIDClass devices.

[DllImport("hid.dll")]

private static extern
void HidD_GetHidGuid(ref Guid
HidGuid);

2.通过获取到的HID全局GUID来获取包含所有HID接口信息集合的句柄。

IntPtr
HIDInfoSet= SetupDiGetClassDevs(ref HIDGuid,0,IntPtr.Zero,DIGCF.DIGCF_PRESENT|DIGCF.DIGCF_DEVICEINTERFACE);

///

/// The
SetupDiGetClassDevs function returns a handle to a device
information set that contains requested device information elements
for a local machine.

///

/// GUID
for a device setup class or a device interface
class.

/// A
pointer to a NULL-terminated string that supplies the name of a PnP
enumerator or a PnP device instance identifier.

/// A
handle of the top-level window to be used for a user
interface

/// A
variable  that specifies control options that
filter the device information elements that are added to the device
information set.

///

/// a
handle to a device information set

[DllImport("setupapi.dll", SetLastError = true)]

private static extern
IntPtr
SetupDiGetClassDevs(ref
Guid ClassGuid, uint Enumerator, IntPtr HwndParent, USBHIDEnum.DIGCF Flags);

相关枚举:

public enum DIGCF

{

DIGCF_DEFAULT = 0x00000001,

DIGCF_PRESENT = 0x00000002,

DIGCF_ALLCLASSES = 0x00000004,

DIGCF_PROFILE = 0x00000008,

DIGCF_DEVICEINTERFACE = 0x00000010

}

3.获取接口信息。

///

/// The
SetupDiEnumDeviceInterfaces function enumerates the device
interfaces that are contained in a device information
set.

///

/// A
pointer to a device information set that contains the device
interfaces for which to return information

/// A
pointer to an SP_DEVINFO_DATA structure that specifies a device
information element in DeviceInfoSet

/// a
GUID that specifies the device interface class for the requested
interface

/// A
zero-based index into the list of interfaces in the device
information set

/// a
caller-allocated buffer that contains a completed
SP_DEVICE_INTERFACE_DATA structure that identifies an interface
that meets the search parameters

///

[DllImport("setupapi.dll", CharSet = CharSet.Auto, SetLastError = true)]

private static extern
Boolean
SetupDiEnumDeviceInterfaces(IntPtr deviceInfoSet, IntPtr deviceInfoData, ref Guid
interfaceClassGuid, UInt32
memberIndex, ref DEVICE_INTERFACE_DATA
deviceInterfaceData);

接口信息定义为:

public struct DEVICE_INTERFACE_DATA

{

public int cbSize;

public Guid interfaceClassGuid;

public int flags;

public int reserved;

}

4.获取接口详细信息,在第一次主要是读取缓存信息

int
requiredSize =0;

///

/// The
SetupDiGetDeviceInterfaceDetail function returns details about a
device interface.

///

/// A
pointer to the device information set that contains the interface
for which to retrieve details

/// A
pointer to an SP_DEVICE_INTERFACE_DATA structure that specifies the
interface in DeviceInfoSet for which to retrieve
details

/// A
pointer to an SP_DEVICE_INTERFACE_DETAIL_DATA structure to receive
information about the specified interface

/// The
size of the DeviceInterfaceDetailData buffer

/// A
pointer to a variable that receives the required size of the
DeviceInterfaceDetailData buffer

/// A
pointer buffer to receive information about the device that
supports the requested interface

///

[DllImport("setupapi.dll", SetLastError = true, CharSet = CharSet.Auto)]

private static extern
bool
SetupDiGetDeviceInterfaceDetail(IntPtr deviceInfoSet, ref DEVICE_INTERFACE_DATA deviceInterfaceData,
IntPtr
deviceInterfaceDetailData, int
deviceInterfaceDetailDataSize, ref
int requiredSize, DEVINFO_DATA deviceInfoData);

接口信息定义:

[StructLayout(LayoutKind.Sequential)]

public class DEVINFO_DATA

{

public int cbSize = Marshal.SizeOf(typeof(DEVINFO_DATA));

public Guid classGuid = Guid.Empty;

public
int devInst = 0;

public int reserved = 0;

}

5.第二次获取详细信息,与第一相同。

若Windows
API SetupDiGetDeviceInterfaceDetail返回数值为true则添加设备信息

ListdeviceList=new List();

deviceList.Add(Marshal.PtrToStringAuto((IntPtr)((int)pDetail + 4)));

6.删除设备信息并释放内存。

///

/// The
SetupDiDestroyDeviceInfoList function deletes a device information
set and frees all associated memory.

///

/// A
handle to the device information set to delete.

/// returns TRUE if it is successful. Otherwise, it
returns FALSE

[DllImport("setupapi.dll", CharSet = CharSet.Auto, SetLastError = true)]

private static extern
Boolean
SetupDiDestroyDeviceInfoList(IntPtr HIDInfoSet);

到此便获取到所有的设备。

二、通过获取到的设备信息打开指定的HID设备。

1.创建,打开设备信息。

///

/// This
function creates, opens, or truncates a file, COM port, device,
service, or console.

///

/// a
null-terminated string that specifies the name of the
object

/// Type
of access to the object

/// Share
mode for object

/// Ignored; set to NULL

/// Action to take on files that exist, and which action
to take when files do not exist

/// File
attributes and flags for the file

/// Ignored

/// An
open handle to the specified file indicates
success

[DllImport("kernel32.dll", SetLastError = true)]

//private static extern IntPtr
CreateFile(string fileName, uint desiredAccess, uint shareMode,
uint securityAttributes, uint creationDisposition, uint
flagsAndAttributes, uint templateFile);

static extern IntPtr CreateFile(

string
FileName,               
//
文件名

uint
DesiredAccess,            
//
访问模式

uint
ShareMode,                
//
共享模式

uint
SecurityAttributes,       
//
安全属性

uint
CreationDisposition,      
//
如何创建

uint
FlagsAndAttributes,       
//
文件属性

int
hTemplateFile              
//
模板文件的句柄

);

其中文件名为相对应的设备名deviceList[index]

2.获取设备属性

///

/// The
HidD_GetAttributes routine returns the attributes of a specified
top-level collection.

///

/// Specifies an open handle to a top-level
collection

/// a
caller-allocated HIDD_ATTRIBUTES structure that returns the
attributes of the collection specified by
HidDeviceObject

///

[DllImport("hid.dll")]

private static extern
Boolean
HidD_GetAttributes(IntPtr
hidDevice, out HID_ATTRIBUTES attributes);

相关HID属性:

public struct
HID_ATTRIBUTES

{

public int Size;

public ushort VendorID;

public ushort ProductID;

public ushort VersionNumber;

}

3.Get PreparsedData

///

/// The
HidD_GetPreparsedData routine returns a top-level collection's
preparsed data.

///

/// Specifies an open handle to a top-level
collection.

/// Pointer to the address of a routine-allocated buffer
that contains a collection's preparsed data in a
_HIDP_PREPARSED_DATA structure.

/// HidD_GetPreparsedData returns TRUE if it succeeds;
otherwise, it returns FALSE.

[DllImport("hid.dll")]

private static extern
Boolean
HidD_GetPreparsedData(IntPtr
hidDeviceObject, out IntPtr PreparsedData);

4. GetCaps

[DllImport("hid.dll")]

private static extern
uint HidP_GetCaps(IntPtr PreparsedData, out HIDP_CAPS Capabilities);

5. FreePreparsedData

[DllImport("hid.dll")]

private static extern
Boolean
HidD_FreePreparsedData(IntPtr
PreparsedData);

6.获取长度:

outputReportLength = caps.OutputReportByteLength;

inputReportLength = caps.InputReportByteLength;

7.最终得到相应的设备

hidDevice = new FileStream(new SafeFileHandle(device, false), FileAccess.ReadWrite, inputReportLength,
true);

三、设备读取,写入

通过最终获取到的设备可对设备进行读取和写入。

BeginRead,Read,Write,BeginWrite等。

以上便能实现对想要的USBHID设备进行简单的操作。

简单串口例子:http://blog.sina.com.cn/s/blog_6267db160102v53m.html

示例代码下载地址:http://download.csdn.net/detail/zhezizhang/8155795

C#与USB HID间的通信的更多相关文章

  1. android usb Host模式下与usb Hid 设备的通信

    做android 与USB HID设备的通信有段时间了,总结一下遇到的问题和解决方法: 1,第一次遇到的问题:android 版本低不支持usb hid, 被要求做相关项目的时候,就从mUsbMana ...

  2. Windows与自定义USB HID设备通信说明.

    1 .   所使用的典型 Windows API CreateFile ReadFile WriteFile 以下函数是 DDK 的内容: HidD_SetFeature HidD_GetFeatur ...

  3. USB HID介绍【转】

    本文转载自:http://blog.csdn.net/leo_wonty/article/details/6721214 HID是一种USB通信协议,无需安装驱动就能进行交互,在学习HID之前,先来复 ...

  4. USB HID 协议入门

    转载请注明来源:cuixiaolei的技术博客 USB HID设备类的应用场合 USB HID类是USB设备的一个标准设备类,包括的设备非常多.HID类设备定义它属于人机交互操作的设备,用于控制计算机 ...

  5. USB HID介绍

    HID是一种USB通信协议,无需安装驱动就能进行交互,在学习HID之前,先来复习一下USB协议的相关内容. USB设备描述符-概述 当插入USB设备后,主机会向设备请求各种描述符来识别设备.那什么是设 ...

  6. PIC32MZ 通过USB在线升级 -- USB HID bootloader

    了解 bootloader 的实现, 请加QQ: 1273623966(验证填bootloader); 欢迎咨询或定制bootloader; 我的博客主页 www.cnblogs.com/geekyg ...

  7. USB HID设备报告描述符详解(转)

    转自:http://group.ednchina.com/93/198.aspx. 参考:USB HID usage table 概述:   报告在这里意思是数据传输(data transfer),而 ...

  8. 史上最全USB HID开发资料

    史上最全USB HID开发资料 史上最全USB HID开发资料,悉心整理一个月,亲自测试. 涉及STM32 C51 8051F例子都有源码,VC上位机例子以及源码,USB协议,HID协议,USB抓包工 ...

  9. STC8H开发(九): STC8H8K64U模拟USB HID外设

    目录 STC8H开发(一): 在Keil5中配置和使用FwLib_STC8封装库(图文详解) STC8H开发(二): 在Linux VSCode中配置和使用FwLib_STC8封装库(图文详解) ST ...

随机推荐

  1. [磁盘管理与分区]——关于分区、磁盘分区表、MBR

    磁盘连接与设备文件名的关系 1. 如下图所示:

  2. public、protect、private在父类子类中使用

    先贴出一张,直观的.估计大家都见过的关于public.protect.private的范围图 作用域 当前类 同一package 子孙类 其他package public     T         ...

  3. modernizr.js

    1.判断浏览器是否支持 h5 if(Modernizr.canvas){ alert(123); }else{ alert(321); } 2.判断浏览器是否支持 canvas function su ...

  4. foj 2044 1 M possible 二进制压缩

    题目链接: http://acm.fzu.edu.cn/problem.php?pid=2044 题意:  给出 一大堆数,找出2个出现次数模3 为1 的两个数字   题解: 把一个数分为几位拆开统计 ...

  5. bzoj 2038 莫队算法

    莫队算法,具体的可以看10年莫涛的论文. 大题思路就是假设对于区间l,r我们有了一个答案,那么对于区间l,r+1,我们 可以暴力的转移一个答案,那么对于区间l1,r1和区间l2,r2,需要暴力处理 的 ...

  6. SGU 185 Two shortest 最短路+最大流

    题目链接:http://acm.hust.edu.cn/vjudge/problem/viewProblem.action?id=21068 Yesterday Vasya and Petya qua ...

  7. Noip模拟考第三题——饥饿游戏

    饥饿游戏 (hungry.pas/c/cpp) [问题描述] Chanxer饿了,但是囊中羞涩,于是他去参加号称免费吃到饱的“饥饿游戏”. 这个游戏的规则是这样的,举办者会摆出一排 个食物,希望你能够 ...

  8. 非阻塞式JavaScript脚本介绍

    JavaScript 倾向于阻塞浏览器某些处理过程,如HTTP 请求和界面刷新,这是开发者面临的最显著的性能问题.保持JavaScript文件短小,并限制HTTP请求的数量,只是创建反应迅速的网页应用 ...

  9. IOS 提交审核,遇到Missing Push Notification Entitlement 问题。

    貌似不影响提交........还是有人提交成了. 昨天晚上提交软件审核,遇到了Missing Push Notification Entitlement 的问题. 问题起因:这个版本我添加了PUSH推 ...

  10. c++中的原子操作

    1. c/c++标准中没有定义任何操作符为原子的,操作符是否原子和平台及编译器版本有关 2. GCC提供了一组内建的原子操作,这些操作是以函数的形式提供的,这些函数不需要引用任何头文件 2.1 对变量 ...