本帖最后由 creep 于 2017-9-28 22:37 编辑 ! j2 y; a% J7 |: k3 Y; N2 x 之前一直使用STM32的CDC虚拟串口和上位机进行数据通信,通常只枚举一个串口基本就能满足要求。但是STM32 USB提供了足够的的端点可以在需要的情况下枚举出多个串口供使用,最新跟着论坛大神飞哥 @wofei1314 玩了下多个串口的使用,然后我在STM32F769-DISCO 上测试了下多串口的移植,以便以后使用的时候可以参考。9 o/ n; k2 J! D' N N, w; P0 c 关于USB的使用ST的库文件里面提供很多的例子可以参考,除此之外官方也进行了不少USB的培训,没法参加的小伙伴可以看下培训的文档对USB有个基本的了解,然后找个板子把USB库中例子都跑一遍看下代码应该就会有个基本的了解。 ~; T' J( T( J* z; R6 E5 l ) K3 Q+ @6 y# B( V5 A( n X& H 9 a1 w: C" D( D4 s1 X) d: g) } ' ~7 x6 m, o8 j: m& f% _9 u/ B: t 0、复合设备: t" ]8 `/ Q# u* I5 ]6 p" V m 0 U8 Z, x' F9 R* g c& ^ E8 V n7 h" H* Z' O7 l, \/ C CDC 类设备在枚举过程中最主要的信息存储在配置描述符内: 如上图所示, CDC 类的配置描述符一般包含两个接口 (Interface 0),一个控制接口,另外一个是数据接口 (Interface 1 ), 除此之外,还有一个虚线指向的 IAD(Interface Association Description),这个表示这个是不是可选的, 得根据实际情况来确定其是否真实存在。我们使用的CDC多串口要使用到IAD进行设置,通过IAD可以设置相应的接口枚举的设备类型 和使用的接口。 / S% D$ y( K% N' }% _ 一般来说把多CDC 串口 叫做复合设备,和一个CDC串口相比,USB库要把设备描述符、控制描述符进行修改 ,多串口新增的端点也要进行初始化 接收发送函数也要根据相应的端点进行处理,PC端加载的驱动也要适当修改。9 f; Q- r+ s9 Q1 |& \+ ^9 | 5 w% r# [8 ?1 t# c$ | 1、USB库修改 ; m: t$ N" M2 M ( b: r5 y! a r" x 不同的芯片型号和不同的USB库配置可能有不少的区别,但是知道了需要修改哪里应该可以根据具体的情况进行操作,下面的移植是在STM32F769-DISCO7 p m; U0 a$ j! ~3 W6 I) I3 W 上由官方的HAL USB 库进行修改。默认是在一个CDC串口可以使用的前提下进行,关于一个CDC的移植可以参考官方代码或者使用Cubemx。# p7 n( u# c1 r) w6 V 1)、2个VCP串口移植是在可以枚举出来一个VCP串口的工程上修改的,主要集中在USB CDC模式下的Class中文件及相应的头文件: a)增加端点5 C& h( \# E; H0 G1 S$ x/ I 一个VCP的时候使用3个非0端点(2个BULK,一个Interrupt),再增加3个非0端点用于枚举另一个VCP。. Z3 W4 K7 {* p# W7 B6 h( v( B0 J , `: U8 K1 F) [9 D1 A" C 设置增加的端点的FIFO * P5 w0 c: f1 O b)对增加的端点进行相应的初始化操作,我测试的是STM32F769-DISCO 上面的USB 是高速接口% S! a; y& x0 O c)将设备描述符修改为复合设备模式 & ~( a U% M. K& T( Q' f & _ N" H% } q2 B$ f# r6 } d)配置描述符,注意我使用的高速 USB模式,如果是全速或者其他速度模式选择相应的数组! |: P$ o: r" u. R , [! l4 s5 d8 Y0 j, s) P 此外还需要修改最大的接口数(好像2个VCP设置大于2即可)! o s; A9 r5 Z( d 因为USB库中使用了maclloc申请内存,要适当调整栈空间。' E! Z9 i, g* R6 n2 S" X# O. F6 G4 D ) q' w7 U/ w1 r$ e0 M5 D$ i 下面是枚举的过程和安装好驱动之后的设备管理器显示: ' R( `0 Z& p. N/ p. ~/ ? / I4 b2 t1 W$ u I' D5 n' }) ? 4 ^' A; D$ F, `$ u" v7 Q 测试代码如下:* j- K4 y9 F$ m0 }0 U4 k: A
9 l7 ]- Q. d8 ~3 J* u. U# r0 k 具体发送方向为: & M: A7 \! r2 S " T: @# x; G# L" d3 F7 h % i& _ Z6 P) R! B1 ` 2)、3个VCP串口,+ `% M! ]) M+ D 在上面的代码基础上简单修改之后可以很容易的枚举出来3个串口 测试代码如下:* l& r- t' Q% |" K1 c6 w+ E
) d% k1 T- O+ o USB收发数据测试如下(点开查看大图):/ Y( u% w& `" d# p! _ & }1 f* a5 H7 }* b: Q+ S 2 、驱动INF文件 ' O) i" h# U0 e6 [ 复合设备需要修改inf文件并手动加载驱动,此时要注意库文件中VID/PID和驱动里面的值一致。修改过的驱动可能会因为没有数字签名证书在某些系统上无法加载。如果是WIN10系统可以加载兼容驱动也可以正常通信。 测试代码: |
最全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));
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 */2 L8 t ^4 N7 ^& E
USBD_LL_PrepareReceive(pdev,7 ~7 c& W( E8 L+ F6 W
CDC_OUT_EP1,# u- n- k6 o9 `. K" ?3 I4 D# u T( o
hcdc->RxBuffer,, o& D& K( W/ F; D+ e
CDC_DATA_HS_OUT_PACKET_SIZE);
这里怎么处理
1、 要保证STM32程序里面的用PID VID和驱动文件INF的PID VID保持一致,这里使用和官方的PID VID不一样是为了加载自己写的那个驱动,这样可以修改枚举出来的串口名称。
2、没有数字证书的问题我也搞不定,这个目前没有办法,WIN7 32位可以强制安装驱动,但是64位或者win10好像只能禁用签名验证才能安装驱动。win10也可以使用系统自动加载的兼容驱动好像也能正常通信。
可以先了解下USB复合设备。
好的
好的 !!!
谢谢大神分享!!
感谢飞哥带路!