STM32 串口详解 1. USART的特点 USART是通用异步收发传输器(UniversalAsynchronousReceiver/Transmitter),通常称作UART,是一种异步收发传输器,是设备间进行异步通信的关键模块。UART负责处理数据总线和串行口之间的串/并、并/串转换,并规定了帧格式;通信双方只要采用相同的帧格式和波特率,就能在未共享时钟信号的情况下,仅用两根信号线(Rx和Tx)就可以完成通信过程,因此也称为异步串行通信。
# A" b B( f4 p# T) K
(1) 全双工异步通信。 (2) 小数波特率发生器系统,提供精确的波特率。 (3) 可配置的16倍过采样或8倍过采样,因而为速度容差与时钟容差的灵活配置提供了可能。 (4) 可编程的数据字长度(8位或者9位); (5) 可配置的停止位(支持1或者2位停止位); (6) 可配置的使用DMA多缓冲器通信。 (7) 单独的发送器和接收器使能位。 (8) 检测标志:①接受缓冲器 ②发送缓冲器空 ③传输结束标志 (9) 多个带标志的中断源。触发中断。 (10) 其他:校验控制,四个错误检测标志。
6 Y8 _! C& ~/ x6 |4 Y; G
通信结构 ( B( p ^( j4 A" v+ o
2. USART简介 2.1、数据传输模型 - L' g) @- V, X2 {$ m
2.2、帧结构 串口异步通信需要定义的参数 ① 起始位 ② 数据位(8位或者9位) ③ 奇偶校验位(第9位) ④ 停止位(1,15,2位) ⑤ 波特率设置 带奇偶校验的数据为就是9位 ( J7 S3 h3 Y% T6 {- D
(1) 数据包 串口通讯的数据包由发送设备通过自身的TXD接口传输到接收设备得RXD接口,在协议层中规定了数据包的内容,具体包括起始位、主体数据(8位或9位)、校验位以及停止位,通讯的双方必须将数据包的格式约定一致才能正常收发数据。 9 Q8 x1 Y) k9 d
(2) 波特率 由于异步通信中没有时钟信号,所以接收双方要约定好波特率,即每秒传输的码元个数,以便对信号进行解码,常见的波特率有4800、9600、115200等。STM32中波特率的设置通过串口初始化结构体来实现。
& y. I- A- n$ k' I& X% W
(3) 起始和停止信号 数据包的首尾分别是起始位和停止位,数据包的起始信号由一个逻辑0的数据位表示,停止位信号可由0.5、1、1.5、2个逻辑1的数据位表示,双方需约定一致。STM32中起始和停止信号的设置也是通过串口初始化结构体来实现。 + [ i/ h2 p1 F' h2 F& }0 q) \
(4) 有效数据 有效数据规定了主题数据的长度,一般为8或9位,其在STM32中也是通过串口初始化结构体来实现的。 W8 K! }4 A( {/ ^# X! M7 g
(5) 数据校验 在有效数据之后,有一个可选的数据校验位。由于数据通信相对更容易受到外部干扰导致传输数据出现偏差,可以在传输过程加上校验位来解决这个问题。校验方法有奇校验(odd)、偶校验(even)、0 校验(space)、1 校验(mark)以及无(noparity)。这些也都可以在串口初始化结构体中实现的。 y: E2 J, `# g; ~4 W
2.3、波特率 $ Z3 Z/ S4 [ n* _& t0 e+ o+ W
OVER8,用于配置过采样,通常情况下,OVER8设置为0。 4 n, h/ z# b" ^- z+ ]
如果时钟时84M USARTDIV = 84000000/(115200*16) = 45.572 那么得到: DIV_Fraction = 16*0.572 = 0x09; DIV_Mantissa = 45 = 0x2D; 8 D8 Q; C6 \2 U {, F
3. STM32的USART 根据STM32F207数据手册,STM32F207一共6个串口 ( `5 t# n% _1 L) w" j. x5 f2 Y
下文我们以USART1为例讲解 从STM32F207数据手册的Table10.Alternate functionmapping图中看到USART1的对应管脚,下文我们选择PA9和PA10作为USART1的管脚。
! P1 ?3 w/ l$ d& p# Z2 ]$ k, U1 m4. 代码配置 配置中断优先级。 - /* Enable the USARTx Interrupt */ l9 A- z; Z5 D) _5 o
- NVIC_InitStructure.NVIC_IRQChannel = USART1_IRQn;: `9 i# q( d% R" B& q+ y# h
- NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority =1;5 A. q5 F4 n8 c0 y
- NVIC_InitStructure.NVIC_IRQChannelSubPriority = 0;
% k+ A4 h0 O8 k - NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE;
1 r4 B- G T1 L8 m, C U - NVIC_Init(&NVIC_InitStructure);
复制代码
* x: m$ o* \4 e6 ~5 \7 N* o/ E打开串口与相应的GPIO引脚,配置好相应串口信息与GPIO引脚的工作模式。 - /* Enable GPIO clock */5 r6 p: s/ ~2 w
- RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA, ENABLE);* h. r! P' t1 }% v, ~# G% B$ W* v) O
- /* Enable UART1 clock */
6 i# ~ C7 k2 A/ B - RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1, ENABLE);; d1 ~4 a# D' s1 e; D. P5 Z
- /* Connect PXx to USARTx_Tx*/* g2 {$ C7 l" P
- GPIO_PinAFConfig(GPIOA, 9, GPIO_AF_USART1);
1 z2 ^& A9 T" N -
. y( q) F" Z7 s& z( D4 F# m- @: c - /* Connect PXx to USARTx_Rx*/! a( o& M5 Z0 b: w+ P U5 R. V! p
- GPIO_PinAFConfig(GPIOA, 10, GPIO_AF_USART1);6 ?) l) c# ^: y' e+ b+ P! t
- ' H, O/ [7 g7 }% \% m
- /* Configure USART Tx as alternate function */
7 ~$ v2 T" j6 Y9 }2 s - GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;
6 n9 R% \ T: C& `; o7 i/ D - GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;
* Q& V( H! G0 N* b/ m& L( b/ j% o - GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
+ I+ s# X& R# }# o - 2 f) v7 p5 u/ J7 _
- GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9;
' t4 f1 x3 ^) G s, r% K: K - GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
. \) q8 g2 x+ f0 ~, c4 [# ] - GPIO_Init(GPIOA, &GPIO_InitStructure);
1 X; `+ z% w/ u; ]5 {" T1 A0 J -
0 [' A: t; l2 d. C) ] - /* Configure USART Rx as alternate function */
p3 { `4 J) [5 c% Q - GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
0 Z, _; `6 w# |! s - GPIO_InitStructure.GPIO_Pin = GPIO_Pin_10;6 t& p6 ^% G( x! `) c" {
- GPIO_Init(GPIOA, &GPIO_InitStructure);
复制代码 & K' U& M1 }1 O+ g
配置USART1。 - USART_InitStructure.USART_BaudRate = 115200;//配置波特率
' M: P% b* T% t8 Z/ @& x - USART_InitStructure.USART_WordLength = USART_WordLength_8b;//配置数据字长; K7 N5 u' {, b
- USART_InitStructure.USART_StopBits = USART_StopBits_1;//配置停止位
1 c8 m3 y" X; u - USART_InitStructure.USART_Parity = USART_Parity_No;//配置校验位
: Z3 ^6 z# \+ p8 F% U" _7 y3 \ - USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;//配置硬件流控制
" U3 \, P6 O+ A* H - USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;//配置工作模式,收发一起) l. e) L6 X$ U7 u: F3 z
-
/ H2 I' ]( Y3 |7 C. z* _ - /* USART configuration */' ?% I j6 ]$ x9 @0 D
- USART_Init(USART1, &USART_InitStructure);// 完成串口的初始化配置
复制代码
* T/ v3 Y' m0 x3 W6 g N) l u使能中断配置。 - USART_ITConfig(USART1, USART_IT_TC, ENABLE);/ v& K; D) i" G0 N* ~9 i- f% s
- USART_ITConfig(USART1, USART_IT_RXNE, ENABLE)
复制代码 D* E5 o/ p# o) i# M- `8 N
我们配置了发送传输完成中断和接收数据寄存器非空中断。我们可以配置很多类型中断,在ST提供的标准库函数中看到。 - /** [, H z3 s% R7 x0 v0 ?3 [2 O
- * @brief Enables or disables the specified USART interrupts.! Z' ~. ?) E7 E2 L7 k& c
- * @param USARTx: where x can be 1, 2, 3, 4, 5 or 6 to select the USART or ' j+ p4 n7 R8 ~) C# H9 l
- * UART peripheral.3 `& ~# _# Q0 l& E3 m( ^
- * @param USART_IT: specifies the USART interrupt sources to be enabled or disabled.
3 s* c, e2 X- {) c% W: p - * This parameter can be one of the following values:, N r6 V) _5 C5 e m6 w/ n
- * @arg USART_IT_CTS: CTS change interrupt
b2 s' k M2 }2 X/ n2 K/ ? - * @arg USART_IT_LBD: LIN Break detection interrupt+ T/ n3 q2 P- [6 t* t
- * @arg USART_IT_TXE: Transmit Data Register empty interrupt- R1 D G. F! J, q; o" m
- * @arg USART_IT_TC: Transmission complete interrupt
: |- p q% {2 f5 S - * @arg USART_IT_RXNE: Receive Data register not empty interrupt* Z M! [" _* U/ ]: x
- * @arg USART_IT_IDLE: Idle line detection interrupt5 d4 }! C' T& a% p3 {
- * @arg USART_IT_PE: Parity Error interrupt# N: ~4 g: x: F' K9 P& S, f
- * @arg USART_IT_ERR: Error interrupt(Frame error, noise error, overrun error)6 o+ g& h4 F8 G# t& _7 i$ E6 c
- * @param NewState: new state of the specified USARTx interrupts.
7 c3 A2 E. Q: A+ R; J - * This parameter can be: ENABLE or DISABLE.( n1 L) X/ b k% c
- * @retval None7 k7 ^, ^$ N& M, N$ q! P
- */
复制代码
+ v0 ~& M8 Y1 B/ Y: W2 ?! P最后使能串口。 - /* Enable USART */0 N3 j, G. g: i5 _' ^
- USART_Cmd(USART1, ENABLE);
复制代码 - y, J9 h( P, D8 I/ T
main主函数,功能是LCD显示串口接收的10个字符(如果不是ascii码则不显示),串口倒序返回接收到的10个字节。 - int main(void)% s) G+ F8 t2 c7 B
- {* G# ]3 q' c- ~ s8 {% I
- while (1)# f) K4 |6 ]+ f6 e
- {
. q2 }. Q0 O" S- y - if(LCD_refresh_flg){
# B$ a ^9 E8 I9 [) u - LCD_refresh_flg = 0;
4 k+ l% k6 R' I) K: U C7 ^& x5 M. b - LCD_ShowString(0,16,receive_data);
" X. p' C/ ~* D$ n2 h$ u1 s - receive_num--;# C& K2 \* ]: e2 c6 g
- USART_SendData(USART1, receive_data[receive_num--]);
* V: [3 Z+ P4 s: j - send_flg = 1;
, u- ^5 X ~1 D* H K. } m" S - }( t9 N7 M' d$ B( a
- }
: Q% |% C2 w+ \' ^6 P/ C, M: [) D - }
复制代码 + d# b! R9 _9 j, I/ X3 k
因为使能了中断,我们还需要编写中断函数。 - void USART1_IRQHandler(void)
5 G% o! y; {: Z7 }+ w. T - {
) O/ E3 |0 g3 k2 I) y. o& J - if(USART_GetFlagStatus(USART1, USART_FLAG_TC))" g X' r/ u* B* D
- {
& I( z- c9 }7 P$ ]. e - if(send_flg == 1){+ y; l* B8 o/ H4 B3 g% d2 `/ W
- if(receive_num==0){
* e9 K9 ~; k" M% D3 B) x6 o6 M, [- V- R - USART_SendData(USART1, receive_data[receive_num]);) a5 }- W; l5 y4 a
- send_flg = 0;% X& {# E( |- x# J2 J: N8 J: W# _
- receive_flg = 1;
! @+ J( H- O' K2 w' p - }else{5 C* S- V8 ^: X
- USART_SendData(USART1, receive_data[receive_num--]);8 b; e0 z% X3 Y
- }
3 @; o+ S' L- D0 P% i - }
# I" U3 F7 v' `2 Y! q H% F - USART_ClearFlag(USART1, USART_FLAG_TC);
2 ?9 Z$ Q4 e1 y5 P: w' Z - }, Z* L* N( l' m1 g
- if(USART_GetFlagStatus(USART1, USART_FLAG_RXNE))
% t, T1 W: C" _& j% N/ a% q- I6 z7 H. F - {
) x5 o. t& I2 a: e2 |- F$ z. y - if((receive_flg)&&(send_flg == 0)){
$ S' f: l* R) ~5 { - receive_data[receive_num++] = USART_ReceiveData(USART1);
% U9 d4 y& Z+ K7 Z* s' ` - if(receive_num==10){" j) o' P; @9 }7 H7 f0 m4 s
- receive_flg = 0;
0 I) f W! H% T. I! P! O: l1 v - LCD_refresh_flg = 1;
% G2 A, ^" A5 L+ \- `0 n& y - }
, s; x1 H+ C+ f; e% L - }
' ~& P/ K8 f: b1 {9 ^1 r/ ~ - USART_ClearFlag(USART1, USART_FLAG_RXNE);
8 X8 j9 m4 U% Q- Z. w - }
" ] w! V% C) Y - }
复制代码 7 r- @. V/ T( H+ V! H. N4 r
下载验证 LCD显示屏可以显示接收的10个字符,且PC以100ms为间隔发送数据,发送977包收到977包回复,测试demo就够健壮,没有出现丢包。
; ]+ C* D9 x0 R2 n ; J0 F% D9 l2 V; m5 N3 a
文章出处: 知晓编程 6 y5 y- Z2 K: u4 ^, t; H$ N, T
|