
本帖最后由 creep 于 2017-9-28 22:37 编辑 ! ]0 j0 A6 o3 q# j7 P. d6 G8 G 之前一直使用STM32的CDC虚拟串口和上位机进行数据通信,通常只枚举一个串口基本就能满足要求。但是STM32 USB提供了足够的的端点可以在需要的情况下枚举出多个串口供使用,最新跟着论坛大神飞哥 @wofei1314 玩了下多个串口的使用,然后我在STM32F769-DISCO 上测试了下多串口的移植,以便以后使用的时候可以参考。0 Q/ D' t4 I- C' f 2 @ C# W% U) `4 h0 S& ~ 关于USB的使用ST的库文件里面提供很多的例子可以参考,除此之外官方也进行了不少USB的培训,没法参加的小伙伴可以看下培训的文档对USB有个基本的了解,然后找个板子把USB库中例子都跑一遍看下代码应该就会有个基本的了解。 + [: I" y: P. T/ v* A9 Y* x 0、复合设备 % w2 J% s7 G; K+ D. A 6 {/ o* f3 s, h) z5 e5 t2 l4 S CDC 类设备在枚举过程中最主要的信息存储在配置描述符内:; w T0 E( A) x; q 9 r% M5 U: s9 h1 B/ V( g . [; o) U/ p1 e ![]() 7 |; V' w, H! t" ?" {, d - a7 D, [) f; h2 B/ S 如上图所示, CDC 类的配置描述符一般包含两个接口 (Interface 0),一个控制接口,另外一个是数据接口 (Interface 1 ), 除此之外,还有一个虚线指向的 IAD(Interface Association Description),这个表示这个是不是可选的, 得根据实际情况来确定其是否真实存在。我们使用的CDC多串口要使用到IAD进行设置,通过IAD可以设置相应的接口枚举的设备类型3 J* c. J t. e, t! k, p 和使用的接口。 ( n) @# w0 H+ \4 n7 f 一般来说把多CDC 串口 叫做复合设备,和一个CDC串口相比,USB库要把设备描述符、控制描述符进行修改 ,多串口新增的端点也要进行初始化 接收发送函数也要根据相应的端点进行处理,PC端加载的驱动也要适当修改。: v: G* v; G/ P) ^ & \9 y8 r, A6 G0 Y4 } 1、USB库修改 ! M6 @ ?) W/ H& F$ w + H3 |% L5 h3 ^ 不同的芯片型号和不同的USB库配置可能有不少的区别,但是知道了需要修改哪里应该可以根据具体的情况进行操作,下面的移植是在STM32F769-DISCO 上由官方的HAL USB 库进行修改。默认是在一个CDC串口可以使用的前提下进行,关于一个CDC的移植可以参考官方代码或者使用Cubemx。9 m/ q5 k5 r- r* ^ ) P5 C+ z0 Q( W 1)、2个VCP串口移植是在可以枚举出来一个VCP串口的工程上修改的,主要集中在USB CDC模式下的Class中文件及相应的头文件: ! C+ j% A8 o/ {* |, I" V+ D2 w ![]() : a5 v6 T. A/ s4 ^. U! W% _1 c: ~% F a)增加端点 一个VCP的时候使用3个非0端点(2个BULK,一个Interrupt),再增加3个非0端点用于枚举另一个VCP。4 I) ~4 T! i3 `, O% \ ![]() 设置增加的端点的FIFO ; P3 L- N) B8 _ ![]() ( k3 X# ?8 E: x B0 q# i b)对增加的端点进行相应的初始化操作,我测试的是STM32F769-DISCO 上面的USB 是高速接口' c7 k. C; ~, u z ![]() c)将设备描述符修改为复合设备模式/ T/ r1 E# l4 ?6 S! l ![]() T- h; E7 h" [ d2 O d)配置描述符,注意我使用的高速 USB模式,如果是全速或者其他速度模式选择相应的数组& ]5 U5 ?. n( R. |% E5 I$ p " l2 c* k; d2 L% ], B3 R. d* y) ` ![]() 此外还需要修改最大的接口数(好像2个VCP设置大于2即可) 因为USB库中使用了maclloc申请内存,要适当调整栈空间。 ![]() 下面是枚举的过程和安装好驱动之后的设备管理器显示: . F& T: y, z P/ u ![]() ![]() 8 [* Y' {" S3 l9 g2 O ![]() 测试代码如下:
![]() " y" v8 }* B( P/ m+ Z, Y) e7 O& E9 S6 X 具体发送方向为:& N' {3 U! I/ W( Z5 K" N6 X . _ Q+ j, w6 [2 O q6 [ ![]() * w% R s" X+ K1 O* l& L2 d, a 2)、3个VCP串口,: _: R/ G+ d- j8 l: G4 o6 I' [; d 9 R5 E+ D+ K7 y$ t9 M, {$ m" n 在上面的代码基础上简单修改之后可以很容易的枚举出来3个串口 ![]() 测试代码如下:! u& _$ q7 Y m0 \1 w
USB收发数据测试如下(点开查看大图):( Q( e0 N- t, Z9 ^ ![]() ![]() 2 _$ O: }% [! x f# e; E& x 2 、驱动INF文件) e3 D/ C+ g+ F- ^9 I' m3 u 2 l: y& O3 T* f: A# v 复合设备需要修改inf文件并手动加载驱动,此时要注意库文件中VID/PID和驱动里面的值一致。修改过的驱动可能会因为没有数字签名证书在某些系统上无法加载。如果是WIN10系统可以加载兼容驱动也可以正常通信。1 B4 a0 F2 M2 F2 g* J , {1 ?6 j: `- P8 k" q5 M$ q! o T 测试代码: ![]() |
最全USB HID开发资料,悉心整理一个月,亲自测试
实战经验 | 选择USBX模块生成USB CDC ACM无PD的项目
STM32 USB HID键盘例程
刘氓兔的杂谈【001】-片上USB 高速PHY
【经验分享】在进行 USB CDC 类开发时,无法发送 64整数倍的数据
【源码】STLINK-V3MINI 高速USB仿真器,成功改刷【高速CMSIS-DAP】
在线直播|无需编写任何代码即可在STM32上实现USB-C Power Delivery
圈圈发布USB图书第二版有感,以及分享一些我学习USB过程...
USB Audio设计与实现
【MCU实战经验】+STM32F107的USB使用
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 */* x( Y0 {7 m7 A7 m2 l* h; n
USBD_LL_PrepareReceive(pdev,- x9 m0 t% k9 Q! [' y" l/ S
CDC_OUT_EP1,
hcdc->RxBuffer,
CDC_DATA_HS_OUT_PACKET_SIZE); k; Y9 i. J9 ?9 b: V
这里怎么处理
1、 要保证STM32程序里面的用PID VID和驱动文件INF的PID VID保持一致,这里使用和官方的PID VID不一样是为了加载自己写的那个驱动,这样可以修改枚举出来的串口名称。
2、没有数字证书的问题我也搞不定,这个目前没有办法,WIN7 32位可以强制安装驱动,但是64位或者win10好像只能禁用签名验证才能安装驱动。win10也可以使用系统自动加载的兼容驱动好像也能正常通信。
可以先了解下USB复合设备。
好的
好的 !!!
谢谢大神分享!!
感谢飞哥带路!