你的浏览器版本过低,可能导致网站不能正常访问!
为了你能正常使用网站功能,请使用这些浏览器。

STM32 USB CDC 虚拟多串口  

[复制链接]
creep 发布时间:2017-9-28 22:30
本帖最后由 creep 于 2017-9-28 22:37 编辑   q5 Q5 j& Z2 o1 ]4 W+ g

6 z- P" W( Z! W1 l- f5 m' Q5 B- M/ s% V$ ^0 p# R) G7 k1 B
         之前一直使用STM32的CDC虚拟串口和上位机进行数据通信,通常只枚举一个串口基本就能满足要求。但是STM32 USB提供了足够的的端点可以在需要的情况下枚举出多个串口供使用,最新跟着论坛大神飞哥  @wofei1314 玩了下多个串口的使用,然后我在STM32F769-DISCO 上测试了下多串口的移植,以便以后使用的时候可以参考。: _! A' E. X: V
- W) `0 O& B: V9 T) X% }5 W0 t
         关于USB的使用ST的库文件里面提供很多的例子可以参考,除此之外官方也进行了不少USB的培训,没法参加的小伙伴可以看下培训的文档对USB有个基本的了解,然后找个板子把USB库中例子都跑一遍看下代码应该就会有个基本的了解。
) Z5 g4 b- @$ j# z. z9 U/ i3 r1 \8 [& @/ z, o8 ~% A
4 Z! S3 U9 Y* U* @: N/ _
8 q" j* s" H. J
0、复合设备: H! ]0 ?+ ~" ^( |) ~3 t2 R
% @' @# Y  w7 G0 F' N7 }" q/ L3 f

% {4 Q+ P  k9 `/ ~5 F5 I: \% gCDC 类设备在枚举过程中最主要的信息存储在配置描述符内:
: a  ]+ n! U/ Y, j; B* j% p  ~4 t4 ]! K2 J& Q1 E
6 ^' `+ r: m, B
abfdf189-a154-49fc-bd00-ab9c51382987.png 7 U3 p0 z6 W% _2 j% ]2 V

4 Z6 H4 Y; u6 {  z; R4 c! \0 u6 L3 T) V# y# [
  如上图所示, CDC 类的配置描述符一般包含两个接口 (Interface 0),一个控制接口,另外一个是数据接口 (Interface 1 ), 除此之外,还有一个虚线指向的 IAD(Interface Association Description),这个表示这个是不是可选的, 得根据实际情况来确定其是否真实存在。我们使用的CDC多串口要使用到IAD进行设置,通过IAD可以设置相应的接口枚举的设备类型! n) e$ h6 @* V: b& R2 T  H+ f" d
和使用的接口。
' f& T! O7 w- Y9 P) u  一般来说把多CDC 串口 叫做复合设备,和一个CDC串口相比,USB库要把设备描述符、控制描述符进行修改 ,多串口新增的端点也要进行初始化 接收发送函数也要根据相应的端点进行处理,PC端加载的驱动也要适当修改。: ~9 y; U  C1 A6 ~0 q5 L
7 h; N5 g9 D( f$ b' }5 ~

) u! A; b" @3 r  V. ?: W1、USB库修改
( U: B4 ~% O, |" B. {8 X
, U& W4 E! l' H6 j" {0 X& s7 M# j- ~# G$ Z' f: o8 v7 B

5 Z/ N2 v' q- D0 e' n不同的芯片型号和不同的USB库配置可能有不少的区别,但是知道了需要修改哪里应该可以根据具体的情况进行操作,下面的移植是在STM32F769-DISCO
- Y; `: D# ~! m$ E( V8 F, B* e3 X上由官方的HAL USB 库进行修改。默认是在一个CDC串口可以使用的前提下进行,关于一个CDC的移植可以参考官方代码或者使用Cubemx。
+ g" y  [2 b9 P$ Z2 Y0 G: D$ J
$ }: s, g* t! D+ l1)、2个VCP串口移植是在可以枚举出来一个VCP串口的工程上修改的,主要集中在USB CDC模式下的Class中文件及相应的头文件:( G+ j& K+ }0 k

1 c  J6 f3 R. d# W+ u3 E% q" } 5ec90026-3c9b-4786-87e0-06cadbef4265.png & w, g) M# E0 G
7 r/ l0 i5 g7 w0 X  m7 r- v3 V: R
7 c4 Z0 K2 h# Y$ G8 K/ `* j3 W9 u  h
a)增加端点
- \& v1 g0 @& n" V( g 一个VCP的时候使用3个非0端点(2个BULK,一个Interrupt),再增加3个非0端点用于枚举另一个VCP。
4 ?9 G, P& \; V0 [5 o2 Q
' |5 }: c# x# S+ F- L4 B 5b8ef82c-10a1-4d37-946b-ae60da254152.png
( }$ {' C5 |" D
0 J. ^7 Y, D. ]9 x设置增加的端点的FIFO
4 b1 d2 x% X" Z
( z" g  d3 s" G
# I" P8 r( Y$ t 0ddf6de2-e026-485b-8070-04514e83e56c.png
' ]. D( m7 B8 D: o
' ~4 Q* F/ F8 c/ y' U; S5 R% ^9 A0 Y: a
b)对增加的端点进行相应的初始化操作,我测试的是STM32F769-DISCO 上面的USB 是高速接口9 j+ x( T1 I7 v& U% J
2 g2 f, ~; k2 l. Y! Z6 _/ b

# X4 L& O& A; H1 ~' a8 |. s+ L 48c4af14-a790-4617-8015-6c8f1131cdd4.png
% U. e1 @4 C0 ?- I% U3 |, d# U5 N9 _1 s% p+ V: Y. s* B
  K1 e* H  z, C, |: ]( y: v
c)将设备描述符修改为复合设备模式
5 v6 ]$ x7 P9 H0 h3 X: w9 w: ?
, F# S' a, E" Y6 t+ t! V- W. ?3 b) Q' }
98db37ba-b3dc-45be-b228-1590afb41c85.png
* N% F1 G7 ]% B% J8 J+ V8 @! `

# S6 w" U2 _5 n+ }% rd)配置描述符,注意我使用的高速 USB模式,如果是全速或者其他速度模式选择相应的数组# y- G) k5 r( [. a2 R1 f

' K- l& {6 u2 x( D- F8 g" ~+ i TIM图片20170928215619.png
- @0 N  o, t, _9 J% a$ G* B此外还需要修改最大的接口数(好像2个VCP设置大于2即可)
/ ]( t7 t/ V9 S. D* u因为USB库中使用了maclloc申请内存,要适当调整栈空间。' U, }2 v3 R$ A" R

2 T& p8 N) e$ F 2a0963ce-d11f-4a48-aff2-f502028d0e1f.png
4 I8 u, G7 O+ i! ]4 a' ]( U  e1 E7 [2 n  v1 T& M" V7 M# V8 S
下面是枚举的过程和安装好驱动之后的设备管理器显示:
$ Q/ o' R1 t4 ^3 K0 r: H; c
  F9 p# M& ^9 V. _  P& a$ J6 I0 @1 j8 r! U
b1c85265-21c9-4197-9d29-b539741de625.jpg 6 f& B$ i) m9 H5 Q+ T  Y
5294a9d3-f04b-4c63-a85f-e3edd6e56495.jpg
, p6 R  o3 X  q5 ~, x4 }! k
; v0 N# [+ c  K/ p  h6 W) W$ o, V5 t
ed917e76-1d1c-4323-8bbc-dccf431f929a.png
4 f% r% O. H8 c7 k& Q+ u
7 t2 Q& A1 j. w4 c  S$ u! f9 C8 j1 s: y3 ?# l! [# S. p
测试代码如下:- ^( \! L0 J( V5 s
  1.   while (1)
    2 X3 W* {3 o! }9 \' c' F) Y
  2.                 {1 D+ S5 R5 S# S) |& [5 `3 d
  3.             memcpy(TestBuff,"This message from VCP0\n",23);+ ^8 j+ _7 |1 s' `7 z# Q6 ^4 V
  4.                         USB2PC(TestBuff,23,CDC_IN_EP1);' U( T1 S. u* {4 T2 e
  5.             HAL_Delay(1000);
    $ w: _% A' y) Q
  6.             memcpy(TestBuff,"This message from VCP1\n",23);1 J( a7 ?+ J* c3 T+ [" a
  7.                         USB2PC(TestBuff,23,CDC_IN_EP3);
    # v! u$ U& ]' d/ U
  8.             HAL_Delay(1000);      ' J6 w4 `: o0 J5 [
  9.                 }
复制代码
USB收发测试效果如下
(点开查看大图)
# h% v9 b$ F* a8 a5 v test1.gif 4 R4 y; H, T+ l* ?+ M% a6 u! L7 O

- m8 y" ~/ @( e7 B$ O, s$ u9 s) Z  k* |3 q3 v  d
具体发送方向为:6 n1 e! q# H4 k
7 r  j, N8 x7 A, F3 q( b
. u+ K% J' M1 g5 }
a5af1022-17e9-4a9d-9f0f-d7efb9f29422.png & D- w5 T; H; ?

' e* U" M6 a+ ^' N+ n1 U+ q% e, i7 I1 Q# l- C, B
2)、3个VCP串口,
' Q3 l+ G% U7 k$ i2 z2 H5 i: k
0 E% B2 m$ ]! h; t* T' N6 i在上面的代码基础上简单修改之后可以很容易的枚举出来3个串口6 S9 A6 a- x  V

8 S5 e: @$ r+ Z2 l7 i0 |+ H$ L d79eea4b-4d3d-42d4-b2ed-3c461a007d97.png ; H" [2 h! E0 M9 W
测试代码如下:  B) o8 T6 w0 N4 C
  1. while (1)( l, ]# a+ O- O, t7 V
  2.                 {
    ' a9 O9 A# E& Q, E# f$ B
  3.             memcpy(TestBuff,"This message from VCP0\n",23);/ Q( P$ S1 v, U2 G! r
  4.                         USB2PC(TestBuff,23,CDC_IN_EP1);
    , X  s9 [" }5 {, O8 u
  5.             HAL_Delay(1000);
    + Z& c0 C1 p7 n2 o% h
  6.             memcpy(TestBuff,"This message from VCP1\n",23);2 g; S) ]$ D. _" w( c
  7.                         USB2PC(TestBuff,23,CDC_IN_EP3);1 `, I9 E8 U) |+ ~$ U' v, d; ~. d
  8.             HAL_Delay(1000);     
    ! V" h) T" Z, a4 H9 S- Q) I1 ~) b3 V
  9.             memcpy(TestBuff,"This message from VCP2\n",23);
    . Z4 {: U5 L+ ^1 W7 @+ Q
  10.                         USB2PC(TestBuff,23,CDC_IN_EP5);7 O) m( Z1 H1 m' k/ P( A# X! S
  11.             HAL_Delay(1000);      
    - j3 ^/ p" g$ f9 d! n( c9 W
  12.                 }
复制代码
  o3 t1 q" X# L+ {, k
4 f4 U/ F7 z# f7 O; l$ l) o  e9 J4 |
USB收发数据测试如下
(点开查看大图); o$ d/ U1 [- N6 _9 o/ g. v, t
test6.gif
1 z% r" z; }  P* |! Z, P$ A7 k& h8 h' E/ @" x7 i3 l
373b1865-5fcd-493b-9972-f02d2b90dd98.png ! A6 t. C, Q+ e, G" t3 x

0 Y$ S: s+ d! }: U' [) Y2 、驱动INF文件  W* v* P! z# W
) O+ w$ L6 o# d% `8 W) p& f! B0 U
复合设备需要修改inf文件并手动加载驱动,此时要注意库文件中VID/PID和驱动里面的值一致。修改过的驱动可能会因为没有数字签名证书在某些系统上无法加载。如果是WIN10系统可以加载兼容驱动也可以正常通信。. x9 R) r' ~/ M! [! M" d
6 G4 m- u- o% h% g# \  B7 H/ u
测试代码:, h& p; [# F# q5 j/ H

# l2 e5 ?, f8 L( q) f Two-VCP-HS.rar (1.19 MB, 下载次数: 1802)

点评

厉害,研究得比较深入  发表于 2018-1-11 14:31

评分

参与人数 5 ST金币 +73 收起 理由
零壹 + 1 赞一个!
kylongmu + 20 很给力!
g921002 + 10 很给力!
wofei1314 + 22 神马都是浮云
zero99 + 20

查看全部评分

4 收藏 51 评论133 发布时间:2017-9-28 22:30

举报

133个回答
wylew 回答时间:2018-2-6 09:25:22
你好,你写的教程基本都能理解,就是我想问一下,我研究的源码里,它没有分别对某一个端点进行初始化的函数,而是统一用一个函数来初始化:) n3 q/ W1 B5 L8 G4 Y
endpoint_control_address = (UX_DCD_OTG_FS_DIEPCTL0 + (ed -> ux_dcd_otg_ed_index * UX_DCD_OTG_ENDPOINT_SPACE_SIZE));6 D* Q* C2 G, n' E
endpoint_size_address = (UX_DCD_OTG_FS_DIEPTSIZ0 + (ed -> ux_dcd_otg_ed_index * UX_DCD_OTG_ENDPOINT_SPACE_SIZE));# J  ~( m: N2 t4 @1 o- ?9 G
endpoint_dma_address = (UX_DCD_OTG_FS_DIEPDMA0 + (ed -> ux_dcd_otg_ed_index * UX_DCD_OTG_ENDPOINT_SPACE_SIZE));6 l2 O+ A( i/ Y, c4 S
就像这样,在一个函数里就设置了端点0和非0端点的所有端点的IN设置,可根据ed -> ux_dcd_otg_ed_index来判断是哪一个端点,所以,相对来说,如果我添加了端点,就是不是不需要再添加我加的端点的初始化函数呢
深圳老刘 回答时间:2019-9-4 01:01:07
还是不行,感觉大神的代码有错& ~( r* K" y% d
      /* Prepare Out endpoint to receive next packet */
) S# i/ R$ N3 k& a, o      USBD_LL_PrepareReceive(pdev,
0 |6 F  m2 x) Z' I8 v* k                             CDC_OUT_EP1,
% p3 Z* I. K1 B+ f  U$ s                             hcdc->RxBuffer,5 M6 ]8 w% `' P2 i* G) ], s
                             CDC_DATA_HS_OUT_PACKET_SIZE);
0 u$ m0 o& @: N$ {  f: A6 F) F  F: u8 c+ H9 \
这里怎么处理
creep 回答时间:2018-1-11 16:06:23
MaggiGunner 发表于 2018-1-11 15:35
6 h7 t( [' O' l: _现在用CUBEMX生成单VCP发送没有问题。我看网上说VID和PID也很重要,我在您例程里看用的是0x03EB/0x6133, ...
8 i& I  C5 w& A+ _) H# X: p& c
1、 要保证STM32程序里面的用PID VID和驱动文件INF的PID VID保持一致,这里使用和官方的PID VID不一样是为了加载自己写的那个驱动,这样可以修改枚举出来的串口名称。6 `* \0 N* O8 K7 Z
2、没有数字证书的问题我也搞不定,这个目前没有办法,WIN7 32位可以强制安装驱动,但是64位或者win10好像只能禁用签名验证才能安装驱动。win10也可以使用系统自动加载的兼容驱动好像也能正常通信。
anny 回答时间:2017-9-29 08:55:31
多谢分享!!!
gujiamao 回答时间:2017-9-29 09:09:50
版主,能不能直观的给我讲下,为何可以虚拟成多个串口,我一直理解成只能虚拟成一个
creep 回答时间:2017-9-29 09:52:37
gujiamao 发表于 2017-9-29 09:09
2 Q* O/ P  X7 ]' b2 ]* F& ?版主,能不能直观的给我讲下,为何可以虚拟成多个串口,我一直理解成只能虚拟成一个 ...
9 y6 w1 d( X( f" z6 g
可以先了解下USB复合设备。
七哥 回答时间:2017-9-29 10:12:38
这个比较有用,以前没搞明白端点怎么加,枚举描述等跟我之前理解一致。
zero99 回答时间:2017-9-29 10:30:09
膜拜一下大佬  
gujiamao 回答时间:2017-9-29 11:12:51
creep 发表于 2017-9-29 09:52
8 s" p9 a/ V2 b- X% z& H5 S8 f可以先了解下USB复合设备。
! [- p( Z7 r6 L. Y# d) p: [
好的
gujiamao 回答时间:2017-9-29 11:13:41
creep 发表于 2017-9-29 09:52
# ?2 d3 k7 R6 r- w6 H5 ~可以先了解下USB复合设备。

- d/ r: J8 A+ q2 O4 o2 S, P) V( k好的 !!!
adlu 回答时间:2017-9-29 11:24:17
厉害!
斜阳 回答时间:2017-9-29 11:45:05
大神之作。mark备用
党国特派员 回答时间:2017-9-29 13:07:29
这个绝对要收藏...
wofei1314 回答时间:2017-10-10 22:55:38
膜拜~
( b# g7 [# H6 V5 c$ y5 ^$ l- V; q) B, _+ J" v, E7 P, y
谢谢大神分享!!
creep 回答时间:2017-10-11 08:50:29
wofei1314 发表于 2017-10-10 22:55# N+ N$ l. ]3 \% F8 s2 K
膜拜~' v, D/ e6 o, a; {; @% g& @5 w1 u
2 K; `9 ~+ K$ n+ A, H0 `
谢谢大神分享!!
' \# p/ P. I; {
感谢飞哥带路!
wofei1314 回答时间:2017-10-13 09:36:15
又看了一遍,好详细!!!
唯有奋斗 回答时间:2017-12-30 08:58:42
您好,请问一下STM32F4能虚拟出几个VCP呢?
关于意法半导体
我们是谁
投资者关系
意法半导体可持续发展举措
创新和工艺
招聘信息
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
关注我们
st-img 微信公众号
st-img 手机版