广告

基于WinUSB实现的嵌入式USB免驱设备通信方式

时间:2018-11-27 阅读:
为了简化USB设备的开发和接入到PC系统,微软开发了WinUSB,可以将Winusb.sys作为设备功能驱动程序安装,并提供WinUSB API供应用程序访问设备。一直以来,除了USB HID设备,其他类型的设备在WINDOWS环境下需要安装驱动程序才能工作……
广告

为了简化USB设备的开发和接入到PC系统,微软开发了WinUSB,可以将Winusb.sys作为设备功能驱动程序安装,并提供WinUSB API供应用程序访问设备。一直以来,除了USB HID设备,其他类型的设备在WINDOWS环境下需要安装驱动程序才能工作。要实现USB设备免驱,就只能使用HID设备。而HID设备传输速度慢,在有些场合必须使用Bulk类型进行批量传输时,就必须使用第三方驱动,或者自己开发一个驱动,使得项目开发非常麻烦。现在好了,自从微软推出了WinUSB,在微软的最新操作系统上实现简单的Bulk类型批量传输也变得非常的方便快捷,在研发过程当中或者一些对于差异化要求不高的场合,是非常适用且容易实现的。本文致力于实现一个最简单的WinUSB通信系统,以满足此类需求。iNdEETC-电子工程专辑

如何让嵌入式设备枚举成WinUSB设备

系统通过USB描述符来确定以何种USB Class类型来工作。如果希望WINDOWS能够将嵌入式设备识别为WinUSB设备,则其描述符至少应当包含以下字段:iNdEETC-电子工程专辑

1、支持 OS 字符串描述符:iNdEETC-电子工程专辑

为了让 USB 驱动程序堆栈了解设备支持扩展的特征描述符,设备必须定义存储在字符串索引 0xEE 处的 OS 字符串描述符。在枚举过程中,驱动程序堆栈查询字符串描述符。如果存在描述符,驱动程序堆栈会假定设备包含一个或多个 OS 特征描述符和检索这些特征描述符所需要的数据。检索的字符串描述符具有 bMS_VendorCode 字段值。该值为1表示USB驱动程序堆栈必须用来检索扩展特征描述符的供应商代码。iNdEETC-电子工程专辑

#define bMS_VendorCode ( 0x01 )iNdEETC-电子工程专辑
// "MSFT100" : index : 0xEE : langId : 0x0000iNdEETC-电子工程专辑
const U8 OS_StringDescritpor[ ] =iNdEETC-电子工程专辑
{ 0x12, 0x03, 'M', 0, 'S', 0, 'F', 0, 'T', 0, '1', 0, '0', 0, '0', 0, bMS_VendorCode, 0 };iNdEETC-电子工程专辑

2、设置兼容ID特征描述符:iNdEETC-电子工程专辑

const U8 WINUSB_ExtendedCompatId_Descritpor[ ] =iNdEETC-电子工程专辑
{iNdEETC-电子工程专辑
0x28, 0x00, 0x00, 0x00, // dwLengthiNdEETC-电子工程专辑
0x00, 0x01, // bcdVersioniNdEETC-电子工程专辑
0x04, 0x00, // wIndexiNdEETC-电子工程专辑
0x01, // bCountiNdEETC-电子工程专辑
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // Reserved[7]iNdEETC-电子工程专辑
0x00, // bFirstInterfaceNumberiNdEETC-电子工程专辑
0x01, // RESERVED ( 0x01 )iNdEETC-电子工程专辑
'W', 'I', 'N', 'U', 'S', 'B', 0x00, 0x00, // compactiableID[8]iNdEETC-电子工程专辑
0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, // subCompactiableID[8]iNdEETC-电子工程专辑
0x00, 0x00, 0x00, 0x00, 0x00, 0x00 // Reserved[6]iNdEETC-电子工程专辑
}; iNdEETC-电子工程专辑
注:WinUSB还支持复合设备,对于单一传输类型最简系统,我们忽略复合设备的要求即可。compatibleID字段必须指定 "WINUSB" 作为字段值。其他可以根据需求更改。iNdEETC-电子工程专辑

3、注册设备接口 GUID描述符:iNdEETC-电子工程专辑

该描述符用于区分不同的WinUSB设备。iNdEETC-电子工程专辑

const U8 WINUSB_ExtendedProperty_InterfaceGUID_Descritpor[ ] =iNdEETC-电子工程专辑
{iNdEETC-电子工程专辑
0x8E, 0x00, 0x00, 0x00, // dwTotalSize = Header + All sectionsiNdEETC-电子工程专辑
0x00, 0x01, // bcdVersioniNdEETC-电子工程专辑
0x05, 0x00, // wIndexiNdEETC-电子工程专辑
0x01, 0x00, // wCountiNdEETC-电子工程专辑
0x84, 0x00, 0x00, 0x00, // dwSize -- this sectioniNdEETC-电子工程专辑
0x01, 0x00, 0x00, 0x00, // dwPropertyDataTypeiNdEETC-电子工程专辑
0x28, 0x00, // wPropertyNameLength 'D',0,'e',0,'v',0,'i',0,'c',0,'e',0,'I',0,'n',0x00,'t',0,'e',0,'r',0,'f',0,'a',0,'c',0,'e',0, 'G',0,'U',0,'I',0,'D',0,0,0,iNdEETC-电子工程专辑
0x4E, 0x00, 0x00, 0x00, // dwPropertyDataLength : 78 Bytes = 0x0000004EiNdEETC-电子工程专辑
'{',0,'1',0,'2',0,'3',0,'4',0, '5',0,'6',0,'7',0,'8',0,'-',0,'1',0,'2',0,'3',0,'4',0,'-',0,'1',0,'3',0,'4',0,'4',0,'-',0,'1',0,'2',0,'3',0,'4',0,'-',0,'1',0,'2',0,'3',0,'4',0,'5',0,'6',0,'7',0,'8',0,'9',0,'A',0,'B',0,'C',0,'}',0,0,0iNdEETC-电子工程专辑
};// bPropertyData : WCHAR : L"{12345678-1234-1234-1234-123456789ABC}"iNdEETC-电子工程专辑

4、端点描述符:iNdEETC-电子工程专辑

按实际的需求的配置端点数量和类型,即可完成嵌入式设备的描述符配置了。iNdEETC-电子工程专辑

一般固件程序可以通过MCU厂家提供的范例程序进行修改,这里省略USB固件功能的说明。只要包含以上三个描述符中的必须的字段,就可以成功枚举成USB Device。枚举成功后在设备WINDOWS设备管理器中可看到类似设备,如下图1所示。iNdEETC-电子工程专辑
20181127-windows.gifiNdEETC-电子工程专辑
图1 成功枚举为USB DeviceiNdEETC-电子工程专辑

如何编写PC应用程序与嵌入式设备进行USB通信

PC机软件相对来说比较简单,并且微软官方也给出了示例代码。唯一需要注意的是,对应的软件程序获取WinUSB设备句柄的GUID参数,需要与嵌入式设备的描述符中的GUID保持一致。GUID是WinUSB用以区分设备的唯一标志。GUID,是Globally Unique Identifier的简称,翻译为全局唯一标识符,是一种由算法生成的二进制数据,长度为128位的数字标识符。iNdEETC-电子工程专辑

具体实现步骤如下:iNdEETC-电子工程专辑

1、创建设备的文件句柄:

调用 SetupDiGetClassDevs 获取设备信息集的句柄;调用 SetupDiEnumDeviceInterfaces 枚举设备信息集中的设备接口并获取有关设备接口的信息;调用 SetupDiGetDeviceInterfaceDetail 获取设备接口的详细信息,所获取的信息通过SP_DEVICE_INTERFACE_DETAIL_DATA结构返回。由于该结构大小无法提前获取,故需连续两次调用该函数,第二次调用时接口详细信息将填充到根据第一次调用返回值所确定大小的该缓冲区,通过缓冲内该结构的DevicePath成员中可获得“设备路径”。iNdEETC-电子工程专辑

2、获取设备的 WinUSB 接口句柄:

调用 WinUsb_Initialize通过传递在创建设备的文件句柄中创建的文件句柄。iNdEETC-电子工程专辑

3、查询设备以获取 USB 描述符:

接下来,查询设备以获取特定于 USB 的信息,如设备速度、接口描述符、相关端点及其管道。调用 WinUsb_QueryDeviceInformation 从设备的设备描述符请求信息。调用 WinUsb_QueryInterfaceSettings 并传递设备的接口句柄,以获得对应的接口描述符。调用 WinUsb_QueryPipe 获取有关每个接口每个终结点的信息。此步骤不是必须的,因为端点方向及传输特性由嵌入式设备描述符决定,是已知的。iNdEETC-电子工程专辑

4、向默认端点发送控制传输:

此步骤也不是必须的。一般都不通过默认端点发送有效载荷。iNdEETC-电子工程专辑

5、发送 I/O 请求:

将数据发送到设备的批量输入和批量输出端点,这些端点点可分别用于读取请求和写入请求。调用 WinUsb_ReadPipe 从设备的批量输入端点读取数据。调用 WinUsb_WritePipe 通过批量输出端点将数据写入设备。在嵌入式设备的输出端点内写入数据之后,就可以在PC端读出数据。反之,如果在PC端对嵌入式设备的输入端点写入数据,则嵌入式设备会产生一个USB端点写入事件,具体如何捕捉该事件,则由MCU厂家的产品硬件决定,产生相应的中断信息,供中断服务程序来判断。一般而言,芯片厂家会提供MCU的USB通信基础范例程序,在其基础上做简单的修改和适配即可。iNdEETC-电子工程专辑

6、释放设备句柄

在完成对设备的所有必要的调用之后,释放设备的文件句柄和 WinUSB 接口句柄。CloseHandle 释放由 CreateFile 创建的句柄。iNdEETC-电子工程专辑

WinUsb_Free 释放由 WinUsb_Initialize 返回的设备的 WinUSB 接口句柄。iNdEETC-电子工程专辑

至此,已经完成了嵌入式设备端固件的USB代码移植和PC端应用程序的编写,就可以实现USB免驱设备的通信方式了。iNdEETC-电子工程专辑

本文为EET电子工程专辑 原创文章,禁止转载。请尊重知识产权,违者本司保留追究责任的权利。
  • 威盛宣布出售x86芯片组IP给上海兆芯集成电路 威盛(VIA)召开重大信息记者会表示,经过董事会同意,将出售部分芯片组产品相关技术、数据等知识产权 (不含专利权), 给予威盛间接持股合计达 14.75% 之上海兆芯集成电路有限公司,交易总金额为美金 138,974 仟元……
  • 拆解华为5G基站:美国产零部件占成本近3成 近日《日本经济新闻》(Nikkei)在专业调查公司Fomalhaut Techno Solutions的协助下,拆解并分析了华为的最新5G基站,确定了组件制造商并估算了其市场价格。并计算了组成要素的每个国家/地区的组成部分的总价值,以及这些国家/地区的份额。
  • AI芯片:技术发展方向及应用场景落地 经过几年的喧闹后,AI应用场景的落地成为最大难题。AI芯片的设计不是简单的高性能微处理器硬件设计,而是涉及应用场景特定需求和算法的软硬件一体化设计。那么,AI芯片的技术发展未来在哪里?如何真正实现AI场景落地实施和商用呢?
  • AI芯片细分市场的金字塔结构 如今,AI工作负载仅仅意味着运行深度学习,这是目前的市场需求所在。但市场需求是多变的。尽管大多数AI训练都在数据中心(包括超大规模云端)和工作站上进行,但AI推理却随处可见:在云端、在工作站、在边缘……尤其是边缘端。
  • CINNO:2024年中国55%的智能手机将采用后置四摄 半导体显示产业咨询顾问公司CINNO Research,日前发布了中国手机市场后置四摄渗透率的报告,数据显示2020年第二季度中国智能机市场中后置四摄占比为41%。报告还分别对中国后置四摄手机中长焦,ToF,潜望式和微距镜头的市场渗透率以及其手机价格分布进行监测……
  • 特立独行的柔宇,能否撼动LTPS主导的柔性屏赛道? 目前三星占据OLED市场份额90%以上,全球大部分拥有柔性显示屏产品的厂商都是跟随其低温多晶硅(LTPS)技术方案,比如京东方、华星光电等。而柔宇采用与三星不同的超低温非硅制程集成(ULT-NSSP)技术,从全球范围看,拥有自己核心的柔性屏技术和可折叠技术,且两者都能量产的,也只有柔宇、三星两家而已……
广告
热门推荐
广告
广告
广告
EE直播间
在线研讨会
广告
广告
面包芯语
广告
向右滑动:上一篇 向左滑动:下一篇 我知道了