本帖最后由 creep 于 2017-9-28 22:37 编辑 3 X3 x# k! |) i4 |/ I b 之前一直使用STM32的CDC虚拟串口和上位机进行数据通信,通常只枚举一个串口基本就能满足要求。但是STM32 USB提供了足够的的端点可以在需要的情况下枚举出多个串口供使用,最新跟着论坛大神飞哥 @wofei1314 玩了下多个串口的使用,然后我在STM32F769-DISCO 上测试了下多串口的移植,以便以后使用的时候可以参考。4 Q/ z$ s: [0 Z; m7 x- \ 7 z' D" k) k# E2 m: i 关于USB的使用ST的库文件里面提供很多的例子可以参考,除此之外官方也进行了不少USB的培训,没法参加的小伙伴可以看下培训的文档对USB有个基本的了解,然后找个板子把USB库中例子都跑一遍看下代码应该就会有个基本的了解。 - j I- J' S* G* H6 h5 t# u 0、复合设备 8 J# j% n3 e4 F+ V- B* K$ h CDC 类设备在枚举过程中最主要的信息存储在配置描述符内:) k$ {" O- S5 b+ E0 i / Y, T0 a+ O8 _. { + W& e# s' s3 v. _. ` 如上图所示, CDC 类的配置描述符一般包含两个接口 (Interface 0),一个控制接口,另外一个是数据接口 (Interface 1 ), 除此之外,还有一个虚线指向的 IAD(Interface Association Description),这个表示这个是不是可选的, 得根据实际情况来确定其是否真实存在。我们使用的CDC多串口要使用到IAD进行设置,通过IAD可以设置相应的接口枚举的设备类型( K& I9 Z- w- G" b 和使用的接口。 & N6 P2 b8 W. [ 一般来说把多CDC 串口 叫做复合设备,和一个CDC串口相比,USB库要把设备描述符、控制描述符进行修改 ,多串口新增的端点也要进行初始化 接收发送函数也要根据相应的端点进行处理,PC端加载的驱动也要适当修改。+ C6 s8 h$ l: r * o! M7 [% H8 S* D! S, g 1、USB库修改2 O) W- x y8 b" I( @/ s4 e( f) ? ' s! \4 o3 O' M$ _ 7 \) k/ o% K+ d2 p, } {/ ~! ^ 不同的芯片型号和不同的USB库配置可能有不少的区别,但是知道了需要修改哪里应该可以根据具体的情况进行操作,下面的移植是在STM32F769-DISCO 上由官方的HAL USB 库进行修改。默认是在一个CDC串口可以使用的前提下进行,关于一个CDC的移植可以参考官方代码或者使用Cubemx。 1)、2个VCP串口移植是在可以枚举出来一个VCP串口的工程上修改的,主要集中在USB CDC模式下的Class中文件及相应的头文件: : L; a$ e- F5 p) i4 W' ] ' J4 U7 p& q% x. v1 h% q. j a)增加端点& r7 A0 B% F: t! _! {" v& Y( M 一个VCP的时候使用3个非0端点(2个BULK,一个Interrupt),再增加3个非0端点用于枚举另一个VCP。 4 G }' R3 |- d" }4 `" h * ^4 ]' Q3 ?0 h: Z% J3 C 设置增加的端点的FIFO7 [% j) f) [, z5 g, u3 A/ F9 q ) c) F$ e. e7 Q! M/ l b)对增加的端点进行相应的初始化操作,我测试的是STM32F769-DISCO 上面的USB 是高速接口$ H3 n. N! H4 F% @, e3 w" o , b3 |5 ]0 v3 ] c)将设备描述符修改为复合设备模式 + I! O: \. r* s3 B* @1 n/ ~ : E- D/ S) q1 B! S " ?& m" x: t+ m! ^0 m% i1 l d)配置描述符,注意我使用的高速 USB模式,如果是全速或者其他速度模式选择相应的数组 此外还需要修改最大的接口数(好像2个VCP设置大于2即可) 因为USB库中使用了maclloc申请内存,要适当调整栈空间。 , a6 v* x v ` 2 P) G( Q; e( ~1 c4 r- t! D 下面是枚举的过程和安装好驱动之后的设备管理器显示:& `/ c; } B: V8 V- f2 B" \ ! H. v: f& N- A: j m; i" r; I " X. \7 I) O- E + z, X7 _; t, [& Y 测试代码如下:
z5 O; q; y) }% o 具体发送方向为: ) k8 | a' o7 [. j1 ~+ k! j 2)、3个VCP串口, 在上面的代码基础上简单修改之后可以很容易的枚举出来3个串口' H" v3 [4 I& P: J & M4 x$ T8 L( V/ k% e* f% I 测试代码如下:: L6 N( C: v; r* X3 L. k
M% h% V' u! y2 T \9 a USB收发数据测试如下(点开查看大图):' z, y) G$ N- v1 J! W% r 7 D2 \3 ?- O. J. x, M( A" q 7 @* d8 i4 c) Y 2 、驱动INF文件+ u7 P4 ?8 L- ~9 f' f & ~+ d. k7 M9 L 复合设备需要修改inf文件并手动加载驱动,此时要注意库文件中VID/PID和驱动里面的值一致。修改过的驱动可能会因为没有数字签名证书在某些系统上无法加载。如果是WIN10系统可以加载兼容驱动也可以正常通信。 测试代码:+ A8 ~, _- _# G4 c * k! Z0 V; j$ W0 K% z% z |
最全USB HID开发资料,悉心整理一个月,亲自测试
USB Audio设计与实现
【MCU实战经验】+STM32F107的USB使用
圈圈发布USB图书第二版有感,以及分享一些我学习USB过程...
STM32F4-DISC 实现USB主机(U盘)和USB设备(虚拟串口)自动切换
STM32 USB-HID通信移植步骤STM32 USB HID键盘例程
【经验分享】在进行 USB CDC 类开发时,无法发送 64整数倍的数据
如何让CDC类USB设备批量接收64字节以上数据
用STM32F4实现的USB摄像头UVC,配合上位机可识别车牌
基于STM32的USB HID基础经验分享
endpoint_control_address = (UX_DCD_OTG_FS_DIEPCTL0 + (ed -> ux_dcd_otg_ed_index * UX_DCD_OTG_ENDPOINT_SPACE_SIZE));! y6 X! O9 N, ~+ E: A1 h- V, o
endpoint_size_address = (UX_DCD_OTG_FS_DIEPTSIZ0 + (ed -> ux_dcd_otg_ed_index * UX_DCD_OTG_ENDPOINT_SPACE_SIZE));
endpoint_dma_address = (UX_DCD_OTG_FS_DIEPDMA0 + (ed -> ux_dcd_otg_ed_index * UX_DCD_OTG_ENDPOINT_SPACE_SIZE));
就像这样,在一个函数里就设置了端点0和非0端点的所有端点的IN设置,可根据ed -> ux_dcd_otg_ed_index来判断是哪一个端点,所以,相对来说,如果我添加了端点,就是不是不需要再添加我加的端点的初始化函数呢
/* Prepare Out endpoint to receive next packet */
USBD_LL_PrepareReceive(pdev,
CDC_OUT_EP1,
hcdc->RxBuffer,
CDC_DATA_HS_OUT_PACKET_SIZE);
+ N" w3 z3 M2 u/ K& g, ^7 Q
这里怎么处理
1、 要保证STM32程序里面的用PID VID和驱动文件INF的PID VID保持一致,这里使用和官方的PID VID不一样是为了加载自己写的那个驱动,这样可以修改枚举出来的串口名称。 K5 ?, N4 t; M' \
2、没有数字证书的问题我也搞不定,这个目前没有办法,WIN7 32位可以强制安装驱动,但是64位或者win10好像只能禁用签名验证才能安装驱动。win10也可以使用系统自动加载的兼容驱动好像也能正常通信。
可以先了解下USB复合设备。
好的
好的 !!!
: t/ h! `- m* q. g8 k( i/ r
谢谢大神分享!!
感谢飞哥带路!