前几天参加论坛活动人品小爆发抽中了一个NUCLEO-STM32F030和一个移动电源,感谢与非网,ST,沐紫和苏柚,特别要感谢苏柚的不辞辛劳。
) `' `, D$ [8 N& G4 Q
NUCLOE-STM32F030:3 F0 E4 Z2 X9 m* I% h
为了避免板子一到手就沦落到吃灰的命运,趁着新鲜感和热乎劲先写个测试程序,也是为继续给自己攒人品。
: k% W; A" d5 X. I% Y6 X3 ~ST的很多ARM系列的串口都有个自动检测波特率的功能,比如这个STM32F030,还有前端时间论坛搞活动送的STM32L476。9 |" s6 e+ o% }2 e$ n( _2 K6 d
使用自动识别波特率主要有2个方面的原因: 2 z6 l& d, ?; d& Z1 C; D) W. D0 N I! w
●:通信之前并不能确定波特率或者波特率可能不固定
9 }1 F4 f. S1 x2 }" _ ●:系统的时钟准确性较差,计算出来的波特率可能不准确& G* e x. b! h) _/ S X; t
自动识别的原理是MCU内部检测一个下降沿到上升沿的时间间隔进而算出来波特率,STM32030有2种模式:
+ |: l: N2 M2 L5 V1 y* eSTM32L476有4中模式:+ n3 U# H# G: O8 H! L
! i9 u- e: z: t5 ?# i! ]/ P
7 \2 y! @' t r- p1 h3 Y% G这些模式的原理基本是上面说的那样测跳变沿的时间来计算波特率,下面我们使用模式0来测试。模式0的要求是发送的一个字节的第一个bit是1,比如. s& F& o' D8 l
小写的a(0110 0001),MCU在接受到这个字节后会根据这个字节算出新波特率并更新相应的串口的波特率,下面的通信就使用这个波特率进行通信。
4 c) `; _- w% G9 h$ L9 |) J* P1 b1 g使用这个功能时要注意有的串口可能不带这功能,比如STM32030只有串口1带有自动识别功能,串口2并不到自动识别,有的单片机不同串口支持的模式也不太相同,下面是STM320X0系列的一些支持:- |4 E: }+ P- A' Z
$ Q8 B+ c. e) ?0 d+ J! @在初始化串口是默认设置波特率为为9600,然后用不同的波特率向MCU进行通信测试自动识别功能。NUCLEO测试时可以使用杜邦线将串口1的PA9/PA10连接到STLINK上面的TX/RX引脚,这样就可以借助STLINK的虚拟串口和串口1进行通信了,NUCLEO板子上默认STLINK虚拟串口是和串口2物理想连接的。STM030有标准库,所以我先使用标准库测试。
$ _# \2 f- l' b& e- T, \; E串口的初始化和平常一样:
2 J' O1 Y, g2 F* Q, N- void USART_Config(void)
8 W3 v; ]0 g" J9 F _# X - { ) Y9 i6 ]5 i9 {0 t/ ^
- USART_InitTypeDef USART_InitStructure;) j* W! I8 s4 }$ h0 f
- GPIO_InitTypeDef GPIO_InitStructure;
1 ^$ g! W+ d, V8 n( g) J% V - RCC_AHBPeriphClockCmd(RCC_AHBPeriph_GPIOA,ENABLE);
4 }/ ^. ~$ }- p1 t$ B6 f$ M - RCC_APB2PeriphClockCmd(RCC_APB2Periph_USART1,ENABLE);
( w, z8 ^3 l% f6 L -
# V4 b8 p. Y& l4 K+ r9 T1 ?; Q - GPIO_PinAFConfig(GPIOA,GPIO_PinSource9,GPIO_AF_1);
% |; v2 H4 A+ J% I - GPIO_PinAFConfig(GPIOA,GPIO_PinSource10,GPIO_AF_1);- C' I( Q1 }# [1 y# s: B0 A
- //U1_TX :PA9: P3 ?/ s) W4 O. {4 |
- //U1_RX :PA10
! J' g K i6 ?5 s" F) Y - GPIO_InitStructure.GPIO_Pin = GPIO_Pin_9 | GPIO_Pin_10;
% B5 E5 p: c B" |* } W. L - GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
3 O; h1 r8 f4 s) l# q' f - GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;; C7 p/ B! {/ i$ ?1 [
- GPIO_InitStructure.GPIO_OType = GPIO_OType_PP;, H. d9 ?3 P* ]& }: Y! n. T4 ]
- GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;
& V$ d/ |7 N( T6 @- l! n7 H- X - GPIO_Init(GPIOA, &GPIO_InitStructure);
; j3 S+ i, L- r' N -
6 `/ E9 s% l. B) e1 V - USART_InitStructure.USART_BaudRate = 9600;
' L' x' @4 p! }' f - USART_InitStructure.USART_WordLength = USART_WordLength_8b;9 _7 G0 o. }, Z0 B2 K2 q3 Q& L3 }. g
- USART_InitStructure.USART_StopBits = USART_StopBits_1;
0 p$ l, ^6 ^9 M& e3 O5 |7 d/ w - USART_InitStructure.USART_Parity = USART_Parity_No;
: D% y' W0 `' T, m6 m) W - USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;
. J, w& o; F7 W, w/ B( y' k3 E+ T - USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;, U, W( |& T' V" ~ [+ z/ q; I
- USART_Init(USART1, &USART_InitStructure);
6 T& T4 ^+ T' H2 ]+ Q3 m' E# } - USART_Cmd(USART1, ENABLE);! D* U7 |1 Z w0 v6 h
-
4 h! f: l* s3 e/ I2 i - }% e$ {* C6 M: h% w6 U
复制代码 然后打开自动识别功能,等待接受字符进行识别,然后计算识别出来的波特率;6 x. }, Y8 O) Q) V
- void AutoBauRate_StartBitMethod(void)
4 [& ?6 G: T6 V0 L$ n3 D# J; x: B - {
4 y7 D; Z5 J1 G1 o8 K. F h; \8 @ - uint32_t baudrateval = 0;
$ D( v. |( r6 x4 L8 P8 c" y - uint8_t tmp[25];
* s0 Q. R$ k+ \1 d. Y -
9 d5 k# e" a! q; {, S/ o7 |+ b - USART_AutoBaudRateConfig(USART1, USART_AutoBaudRate_StartBit); ! q! ?, Z! B" {5 T
- ' o6 \' ]& d2 m `. y" b
- USART_AutoBaudRateCmd(USART1, ENABLE); " E4 b* Z2 V1 J. v4 Y0 q
-
$ J& C, L D# G/ u+ r - while(USART_GetFlagStatus(USART1, USART_FLAG_REACK) == RESET)
* G' ?) U) F& c" O' @ - {}
) U! C" A, C5 }9 ?7 D -
% T; E' s$ q& ^6 g) `: Y - while(USART_GetFlagStatus(USART1, USART_FLAG_TEACK) == RESET), G) K+ q( G- h0 T, E
- {} # E% R2 {5 W; ]7 Y B' @3 N
- . H: ]/ n$ h4 Z
- /* Loop until the end of Autobaudrate phase */
w$ e' |) y1 w- t( B - while(USART_GetFlagStatus(USART1, USART_FLAG_ABRF) == RESET)
# @* N+ @$ j R3 Z - {} 5 m0 T0 v' q+ |8 G
- V) d# W5 V% L7 {! ~1 b
- /* If AutoBaudBate error occurred */* G* z0 O- {" R# r, }! W
- if (USART_GetFlagStatus(USART1, USART_FLAG_ABRE) != RESET)
P' j5 k- g2 `+ C! s- c - {# a# B# R0 i7 T/ ?: ^ f
- }: |0 f! ]8 _' h) G9 G
- else
2 H4 |; w# z' E% C0 R - {
2 ]; r/ O; ^8 b% s1 n# s. u" a$ w - & Y% }, E: P( A; M' Y
- /* Wait until RXNE flag is set */9 e* B5 F$ D. V" M0 W* u
- while(USART_GetFlagStatus(USART1, USART_FLAG_RXNE) == RESET)
+ ~0 |" w. {2 x, p - {}) G( t; x6 J. T% N) u
- / ^, V7 D3 n$ s9 a' l4 M( ?
- /* Wait until TXE flag is set */ . L- B; s4 a3 `
- while(USART_GetFlagStatus(USART1, USART_FLAG_TXE) == RESET)1 A: e7 T \6 |- ]
- {}
, @ `/ o' R! p I& w -
8 M+ V% ?$ X3 K2 f" r1 U, ] - //将收到的数据发送到串口. J/ G- M* T! U! [% B; U
- USART_SendData(USART1, USART_ReceiveData(USART1));
' c" \: \) w4 m9 S: {) R5 D - while (USART_GetFlagStatus(USART1, USART_FLAG_TC) == RESET)
% y! }# R% R8 E - {}
0 ^* Q5 R7 ?3 G/ o8 P$ o
& ]2 o8 e& D! o8 C7 Q% O$ ^- //计算波特率+ {5 y7 b- f" y7 @6 F2 Z" H3 [
- baudrateval = RCC_Clocks.SYSCLK_Frequency / USART1->BRR;
/ M1 C7 B/ l7 e) n - sprintf((char*)tmp,"\r\n检测到波特率为:%d\r\n",baudrateval);
: ^. S: a' ~1 R4 \/ p* y/ Y - USART1_Send(tmp,25); , q W( `# ?# H& x
- } 3 ]6 ~* m/ U2 D; r7 P2 p {, ]
- }
0 ^/ _ V/ g( \& b: Z( `( U
复制代码 分别测试几种波特率向MCU发送数据a:可以看到识别计算出来的波特率和实际有差别,因为这个差别在串口波特率的接受范围内,所以并不影响实际的通信。
. h7 ]' r; ]8 R9 ~- X9 {
# w2 q' D# H4 }) K/ p5 d
& q; ? D* Z+ M9 p, Q
, V% d6 E P {1 d# n( l4 w7 @, l
- u3 R4 ]7 n9 U, J- }$ x
! Q) k/ B. F$ _NUCLEO-STM32L476:5 @! q: w: n2 Y. F
上面使用的STM32F030可能很多小伙伴没有这个板子,下面用NUCLEO-STML476测试下,相应的使用的HAL库。使用HAL设置比较简单,在初始化串口时顺便打开自动识别高级功能即可。' |9 j6 L( k6 Z# d3 p
- void USART1_Init(void)7 |% s$ x0 J: F4 ^1 O3 s6 m
- {
7 q0 E0 V9 D9 N9 h R& D$ z - UartHandle.Instance = USART1; @# I- k$ W; k- g% R9 _
- UartHandle.Init.BaudRate = 9600;) ^, Q7 |5 j8 d4 Z% l
- UartHandle.Init.WordLength = UART_WORDLENGTH_8B;
3 E2 c0 {+ V u - UartHandle.Init.StopBits = UART_STOPBITS_1;
9 x) e4 ?4 A! c3 r* q* B - UartHandle.Init.Parity = UART_PARITY_NONE; {" U% B4 Q e m
- UartHandle.Init.HwFlowCtl = UART_HWCONTROL_NONE;
1 [$ |1 P. Y5 U- m6 E, G - UartHandle.Init.Mode = UART_MODE_TX_RX;
7 D1 e! r) X- c# s. } - UartHandle.AdvancedInit.AdvFeatureInit = UART_ADVFEATURE_AUTOBAUDRATE_INIT;$ \9 U" a h; V! P
- UartHandle.AdvancedInit.AutoBaudRateEnable = UART_ADVFEATURE_AUTOBAUDRATE_ENABLE;. W' I: N& Q1 q. i% A- _, E
- UartHandle.AdvancedInit.AutoBaudRateMode = UART_ADVFEATURE_AUTOBAUDRATE_ONSTARTBIT;
; W1 Q! ~7 D6 R- {& A* C2 X - USART1_MspInit();* C9 C! e. o( D) J X) J; U% [
- if(HAL_UART_Init(&UartHandle) != HAL_OK)
* W9 n r: X* O/ e& f - {1 S9 F& t/ ~5 o; f' Z
- while(1);
1 [. R3 I# h) q7 y; g+ k2 V - }
) [1 H3 s, O3 T - }
复制代码 波特率初始化为9600后BRR的值为0X208D(8333),此时的波特率为80000000/8333 = 9600.此时MCU主频为80MHZ% t9 x/ V6 `# W: p L) T
2 z* O3 i1 Y' z- ]! i0 x7 ^
使用460800的波特率和MCU通信后BRR的值为:0xAE(174) ,此时的波特率为 80000000/174 = 459770.误差率为0.22%% c7 o9 i* K w, f W8 l
4 @ p# b3 K) }: u5 R! S总体来说波特率自动识别还是很实用的,识别一次后就可以一直使用,如果有需要还可以继续识别。8 Q" C2 q% n9 n/ q0 t. C; ^. z
测试代码:
: R9 t! k e. k$ w& |
USART_AutoBaudRate_STM32F030æ ååºæ¨¡å¼.rar
(232.69 KB, 下载次数: 449)
|
你在每次测试的时候,UART有复位吗??# B B+ g6 E1 r+ c0 q0 K
或者说改变波特率的时候,UART不需要复位就可以自动适应这次的波特率吗??# s/ O. r1 E4 S7 ?$ Z3 S) R0 `3 {
(我的问题是通信过程中,改变主设备波特率的时候,从设备都可以自动适应吗?还是只有再上电后那一次能适应?)% f1 T7 s- L6 g7 D
不知道有没有问的清楚。有时间请帮忙回答一下。
因为我想不明白,波特率降为一半时,从设备(stm32)怎么知道收到的约定字符是所谓的约定字符还是有效data。
非常规波特率指的是那些不是常用的波特率吗,一定的范围内应该可以支持,MCU只是简单的测试外部发送的数据的波特率然后就设置为自己的波特率。有空我测试下你说的这种情况看行不行。
+ I, t# b8 G5 g( s& H- L
while(__HAL_UART_GET_FLAG(&huart5, UART_FLAG_ABRF) == RESET)( F- F5 t# n1 @1 _
{}
我也只会玩这种小儿科的东西,太复杂的玩不了