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

【经验分享】STM32的bulk双缓冲传输速度的讨论,硬件的坑永远填不完

[复制链接]
STMCU小助手 发布时间:2022-2-7 21:06
USB 1.0的最高12Mbps.
- [+ Y3 g0 b  O/ n9 m  oUSB 2.0的高速模式480Mbps,全速模式12Mbps,低速模式1.5Mbps
% t/ {* G5 ]0 C5 I
而是设置STM32端的USART的波特率。PC与STM32传输速度是以USB1.1的理论速度传输的,是不能设置的。

. f" w/ h* v0 N接收到数据,置NAK->将缓冲区数据拷贝到用户区(用户处理过程)->发ACK通知主机完成了完整的接收可以发送下一个->主机发送下一个,
; P- g5 ~7 M2 H4 V* n% V
: J4 y! i! U7 M/ h0 ^- o2 y6 U' B 前天测试自己编写的USB驱动程序时候发现从主机到STM32的OUT传输(主机到设备)速率竟然只有最高33KB/S,实在是晕死了。经过研究后发现是 驱动程序中设置的PIPE MaxTransferSize参数的关系,原先设置64只能33KB/S,后参考其他USB设备驱动程序的值,设置成了65535,再测试USB OUT的速度,达到了500KB/S,终于解决了驱动程序的瓶颈。不过算下USB 2.0全速的通讯速率是12Mb/S,排除掉CRC、令牌、SOF等等开销怎么也应该不止最大500KB/S啊。到网上看了看,基本上应该能达到 600KB/S~700KB/S以上,我现在的速度应该还有很大的提升才是。
& `' B0 y- R! ~0 ~" R    看看程序,发现! E1 g% k* u6 {: v) s0 I
void EP3_OUT_Callback(void)//EP3 OUT的回调函数,当EP3接收到数据时候中断调用该函数
5 d3 V" c7 w" k9 N- H{
! A4 U" l$ ~# e) ?  ]  count_out = GetEPRxCount(ENDP3);//获得接收到的数据长度
) K0 k% G  f: ]& H& W  PMAToUserBufferCopy(buffer_out, ENDP3_RXADDR, count_out);//将数据从USB EP3 RX的缓冲区拷贝到用户指定的数组中7 x) W4 Y1 ~; V) }* T. c
  SetEPRxValid(ENDP3); //完成拷贝后置有效状态,从而EP3发送ACK主机可以进行下一个数据包的发送
1 D& [2 q0 e: \0 X1 t}% `2 ^0 P! {2 Z% a
    试着将PMAToUserBufferCopy这句注释掉(这样STM32就不处理接收到的数据了)后再测试速度,惊奇地发现速度竟然达到了 997KB/S!晚上仔细想了想,数据肯定是要使用的,这个数据拷贝的过程的时间消费总是少不了的;由于通常情况下USB设备BULK数据接收的步骤就 是:接收到数据,置NAK->将缓冲区数据拷贝到用户区(用户处理过程)->发ACK通知主机完成了完整的接收可以发送下一个->主机 发送下一个,按照以上的步骤USB接收一步步的进行,只要STM32不完成数据处理,状态就一直是NAK,主机就会不停地发送该数据包,浪费了带宽,因此 就会导致我上面最大速度500KB/S难以再增加的情况!不甘心啊~~
5 ?/ o! m' C8 e* |# B& \8 d    昨天晚上又仔细研究了STM32的技术参考手册的USB章节内容,里面提到BULK可以采用双缓冲机制(PING-PONG)进行处理,正好可以解决上面 的情况。双缓冲机制的原理就是分配2块接收缓冲,STM32的用户处理和USB接口可以分别交替占用2个缓冲区,当USB端点接收数据写其中一个缓冲区的 时候,用户的应用程序可以同时处理另一个缓冲区,这样缓冲区依次交换占有者,只要用户处理程序在USB端点接收的时间片段内完成处理,就能够完全不影响 USB的通讯速度!
+ {% b3 U; j' O# f+ [    程序部分修改! Z# j8 P1 W: ?/ S1 C
一、EP3_OUT的设置修改,
: r/ V( n2 z$ F: |. E# j$ d/ G$ Q//ZYP:修改EP3为BULK双缓冲方式-------------------------2 b2 q; e- Z7 O& o
  SetEPType(ENDP3, EP_BULK);9 z1 W$ i9 [5 ~, K3 V9 w) X
  SetEPDoubleBuff(ENDP3);
- A7 ~& _% P, ^) G( U9 E  SetEPDblBuffAddr(ENDP3, ENDP3_BUF0Addr, ENDP3_BUF1Addr);# j- k. Q5 h$ C% x5 F
  SetEPDblBuffCount(ENDP3, EP_DBUF_OUT, VIRTUAL_COM_PORT_DATA_SIZE);9 y2 _- }( k) x) \  Q' l7 k# D9 O
  ClearDTOG_RX(ENDP3);0 G# g, }& }. J. e7 D. X5 F( u
  ClearDTOG_TX(ENDP3);
+ P( u+ S, H3 `1 L  ToggleDTOG_TX(ENDP3);
$ J9 D$ T2 _0 s; X& S3 @4 w  SetEPRxStatus(ENDP3, EP_RX_VALID);
9 ]! B5 Q9 P2 u  w' x' g) u  SetEPTxStatus(ENDP3, EP_TX_DIS);
5 s: G6 a! l; `8 I! v( @$ i9 M//------------------------------------------------------  m( v1 V0 P. y/ h
二、EP3_OUT回调函数的修改
9 S. ?+ N. w+ ^( k  C) {void EP3_OUT_Callback(void)* g- d0 O; Q  O, t) @. [
{2 o% ~' X3 W) `
//ZYP:以下是修改成EP3双缓冲OUT后的处理函数! d8 K; s5 j( U1 c" E
  if (GetENDPOINT(ENDP3) & EP_DTOG_TX)//先判断本次接收到的数据是放在哪块缓冲区的
, r+ A' x0 ^* _7 l; s  {
& B# e/ S* u( `4 k5 |    FreeUserBuffer(ENDP3, EP_DBUF_OUT); //先释放用户对缓冲区的占有,这样的话USB的下一个接收过程可以立刻进行,用另一块缓冲区,同时用户并行进行下面处理,在另一块接收完之前,处理完用户数据就行,否则缓冲区竞争。- f, H/ J0 [6 C: I7 }/ x
    count_out = GetEPDblBuf0Count(ENDP3);//读取接收到的字节数,, e. b3 {, K6 y; W# x
    PMAToUserBufferCopy(buffer_out, ENDP3_BUF0Addr, count_out);
! Z; [9 x! x  z  }
+ `) B; ]( B, W! S  else9 ]% ?6 o9 m  @  m) Q
  {* K  t; N  p4 X8 t" i  f7 b' n
    FreeUserBuffer(ENDP3, EP_DBUF_OUT);* |  p* [8 J( F6 `! E; j4 g
    count_out = GetEPDblBuf1Count(ENDP3);
8 m5 ~1 p2 B7 [' z7 C& X5 ^! ^    PMAToUserBufferCopy(buffer_out, ENDP3_BUF1Addr, count_out);' l! J1 j) a5 G; E& l( ]
  }
& v# Q4 A- s4 _$ v: c" \}
# C8 i5 s9 }7 R5 w# C. q+ d    经过上面的修改,终于解决了STM32在处理接收数据时导致主机等待的情况,用BUS HOUND软件测试了下1 s1 t% I; A  H" d( i/ D

5 k& S: a5 W% \# \ PS:上面的FreeUserBuffer(ENDP3, EP_DBUF_OUT); 这句话的上下位置是关键,如果放到函数的后面,则仍旧会有主机等待STM32处理数据的情况,速度仍然是500KB/S!
/ O1 d$ Z2 I, E9 {$ A8 x/ G) ^    把这句话放在拷贝函数的前面的话就真正把双缓冲PING-PONG机制用起来了。大致算了下 PMAToUserBufferCopy(buffer_out, ENDP3_BUF1Addr, count_out);这句话当count_out为最大值64的时候STM32执行需要302个周期,72MHZ情况下约4.2微秒执行时间,而USB 传输按照12Mb/s的线速度传输64字节的数据至少也得40微秒,因此只要PMAToUserBufferCopy的时间不超过40微秒,就不会导致缓 冲区竞争的情况。 : g7 @0 s5 J# [+ G

0 ~2 D7 m3 o+ ^+ I# D! J3 @
& ]6 n$ c6 `% s% C: p
收藏 评论0 发布时间:2022-2-7 21:06

举报

0个回答

所属标签

相似分享

官网相关资源

关于
我们是谁
投资者关系
意法半导体可持续发展举措
创新与技术
意法半导体官网
联系我们
联系ST分支机构
寻找销售人员和分销渠道
社区
媒体中心
活动与培训
隐私策略
隐私策略
Cookies管理
行使您的权利
官方最新发布
STM32N6 AI生态系统
STM32MCU,MPU高性能GUI
ST ACEPACK电源模块
意法半导体生物传感器
STM32Cube扩展软件包
关注我们
st-img 微信公众号
st-img 手机版