串口通信原理 想必玩过一点单片机的人都懂一点串口通信,而恐怕我们印象中的串口通信不过如此,两个芯片通过串口通信,需要共地、Tx(发送)和Rx(接收)相互连接即可:) B& l! ?+ E/ O3 Z2 A
3 h% P& z( U. {2 J
而实际上,串口通讯种类繁多,即使是通用异步收发器UART也有RS-232和RS-485等不同接口,有兴趣可以深入学习。
, k/ f/ o" N5 Z! \6 A: k0 _; T首先串口通信指的是一种设备通信的协议而不是指接口。串口通信的概念非常简单,串口按位(bit)发送和接收字节。而我们所使用的USART和UART的区别是可以同步,你在STM32的datasheet里可以看到USART2_CK,而实际上我们很少使用,我们完全可以把USART当UART来用。
9 R& R1 s _4 _" z/ d HC06蓝牙模块4 p; E5 I% r7 J$ G" f: ?9 @
无线蓝牙透传模块。 6 M: [: H! G, m# U
默认名字HC06、波特率9600bps、配对密码00000。
) ^0 K3 g2 R) E8 i0 k$ R
. J' z5 X) M8 y# Z5 l& S % u7 ^1 P& u+ R! i3 D# @
用Arduino的时候想必是很傻瓜式的,如果用STM32来做还是要了解一番的。
9 [1 A& f @# U2 X7 P6 j4 ], I
: s: @ F$ {. ]0 t+ Q首先好好看一下datasheet,看一下AT指令集,然后用一个USB转TTL的模块(这里用的是CH340)和电脑连接:: ?5 Y; q2 r1 L3 e8 X$ [( r
/ w9 O$ Y+ P% f) L& c. @3 f9 X1 A% z1 G; }$ J) Q, p
可以看到HC06的指示灯一直在闪,说明蓝牙未连接。打开串口助手,试一试AT指令是否有效,然后你就可以设置波特率(建议就选默认的9600,据说快了会降低通信的稳定性),改名字和密码,详见datasheet。
. ~( R2 K7 j' R5 }' A8 F' ]* h5 A* n; [( U! d) L
; h' e3 c/ p. y, n8 T
2 s, {: H2 G' a8 l m' r/ ~
* R$ a8 g2 [5 C& N
打开蓝牙串口APP,会自动连接,连接以后指示灯就从闪烁变成常亮了。在串口助手里把HEX显示给勾上,然后用APP发几条指令,会发现收到这样: 1 }2 ]5 y0 Y* ]+ y
6 B: m. A+ \. B% K+ u* y, c! ]
- ~& k) X2 |* Q! p1 {2 {3 L
那就说明蓝牙这边不会出问题了,专心调试STM32吧 - -‘ / |6 p' `, B. ^! U: h2 h( H. z
STM32F407的USART2配置
- [3 F& ~( K( m' Y, T
& F/ }0 [5 t5 }. e% i: X首先查datasheet表,找USART2对应的是哪个GPIO:
6 ]0 J5 [$ ^7 d. x7 U0 J1 R- q0 H2 z8 t0 y* k* R$ t! \+ | C0 w
9 X S( G3 |7 k! ? 千万不要忘记查看总线!) T/ b) \8 l5 ] L1 D# B
; ^: k @6 O; A. O1 g+ D
. O* ?: O' F. I: F. [
6 Q" N" o3 J1 Z4 H+ }7 B$ R+ ]4 q8 D" w( L/ x5 a4 X( I& H. g' U# c
看了系统架构就会知道,F103和407的GPIO挂在不同的总线上。103挂在APB2上,而407挂在AHB1上,所以初始化时钟的时候千万要注意~ 8 p/ s- b' r& h
- RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA,ENABLE);
复制代码 / {. n1 K$ {% A$ N3 X/ N
另外在配置GPIO时的参数也有不同:
' `8 i6 {7 Y+ v0 {' |! m我们查看GPIO_InitTypeDef的定义: 9 O% X: L! w4 y9 v' y
- //STM32F103中的定义:( w, g7 Z. F% a' v
- typedef struct2 w2 O; ~+ A2 ~! ]6 [8 a p+ g
- {
+ ^( [. p0 h' r - uint16_t GPIO_Pin; 8 {$ Y' t8 v3 A7 o
- GPIOSpeed_TypeDef GPIO_Speed;
4 @8 R: k, \& c+ K5 S) x# S9 I8 E9 c - GPIOMode_TypeDef GPIO_Mode;
5 Z; \# F( t' e! I! D3 [- d - }GPIO_InitTypeDef;: _ |& ]" n# t; U, S
- 1 [' a2 a/ |3 Z: L# l2 l( W, _
- //STM32F407中的定义:
7 o; W1 ?5 ^/ H+ p - typedef struct7 A" e+ ~: S5 V
- {+ H9 `- G* s; K9 g) J0 d. k# l
- uint32_t GPIO_Pin;
" C5 @. J$ V6 s; P% W - GPIOMode_TypeDef GPIO_Mode;
7 C1 J5 U3 F& e9 f% o8 l; l - GPIOSpeed_TypeDef GPIO_Speed;) I% n$ q* J+ o2 b# q& X2 v
- GPIOOType_TypeDef GPIO_OType; 7 B* V# n! e& b+ F5 o4 z
- GPIOPuPd_TypeDef GPIO_PuPd;
% g" v$ ?! d4 D, p$ D - }GPIO_InitTypeDef;
复制代码可以看到F407多出了两个参数,即GPIO_OType和GPIO_PuPd。 在F103需要分别设置两个引脚的Mode为复用推挽输出(GPIO_Mode_AF_PP)和浮空输入(GPIO_Mode_IN_FLOATING);但是在F407,两个引脚可以一起设置为GPIO_Mode_AF,并且都设置为推挽输出即可(这个我研究了好半天,据说是因为F1和F4的GPIO复用功能不同,F4复用功能不是在配置IO口模式这儿去配置,而是需要单独配置复用模式,然后再映射,总之就是F407配置串口复用的时候不管Rx还是Tx统统设置为复用推挽输出就好了。) 剩下的USART2初始化、中断初始化等都和F1一样。 最后详细代码如下: bluetooth.h: - <font face="Tahoma" size="3">#ifndef __USART2_H* J+ o) c. r2 G& j$ |* ^6 G# g3 n
- #define __USART2_H
( J6 n5 O( F0 z$ \ - #include "stdio.h" ; a6 J7 ^$ E5 Z! M c1 J) s ?
- #include "stm32f4xx_conf.h"1 r* q. u; N5 q
- #include "sys.h" , o0 r7 N% G$ o" m# L
- y0 G# w2 V6 ?1 H
- #define USART2_REC_LEN 200
# G% \0 q" E3 S1 d - #define EN_USART2_RX 1
1 i8 c* B. j* ]4 b% `9 O& f
1 `* h3 x2 ^; v( N q, ^9 u- extern u8 USART2_RX_BUF[USART2_REC_LEN];+ Q% T( t2 I+ _7 }; S+ h( `; O
- extern u16 USART2_RX_STA;
9 Z2 v$ M+ K8 t6 E6 R
2 y; k F# }8 i) L: I- void bluetooth_init(u32 bound);, m' N' b% {7 W) Q
- #endif</font>
复制代码bluetooth.c: - <font face="Tahoma" size="3">#include "sys.h"
0 Y- u+ Q+ [5 p3 f' l0 t- ? - #include "bluetooth.h"& U3 j# u. `4 K
- //这个是一个舵机控制板的驱动,感兴趣的可以看我下一篇博客
1 m! r% e+ u6 \* S* {+ z - #include "pcf8574.h"5 Y2 G! p( D0 _9 I& I* k4 j
- , K$ Z9 E7 l* S. T e8 ^$ S
- #if SYSTEM_SUPPORT_OS
/ a4 F$ H- e) ^& K5 h - #include "includes.h" ! ~& u8 E# U- Q G s% R3 n
- #endif
; R! ^' d: H5 ~% M7 n - " @; [2 B( _6 |
- #if EN_USART2_RX
$ V/ K* \# R0 q9 \3 `* g - 5 P; O( I) z: [# [" i: u) w
- u8 USART2_RX_BUF[USART2_REC_LEN];5 ~) a# t) g. M. ~
- ; ]+ w, K! R6 P9 @9 Y
- u16 USART2_RX_STA=0;0 n% X1 l5 }' l, G3 w$ u: N
- " P. p8 }8 X2 @; V9 q& o: T
- void bluetooth_init(u32 bound){: G6 l: e+ {) a$ ^
- s ]1 }) S/ j* h! @- `4 d
- GPIO_InitTypeDef GPIO_InitStructure;( e* C6 p0 {% {, F# L3 P
- USART_InitTypeDef USART_InitStructure;; u+ O0 d! Y! M# [( ?5 t
- NVIC_InitTypeDef NVIC_InitStructure;
: @, J0 o$ ~3 ^' o) q7 {" G - 7 H4 c! O; u) y1 p; M7 K/ }
- RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA,ENABLE);
6 e" }) k. t' l3 I6 c1 w2 H" I' H V - RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2,ENABLE);) B: K" I% s+ }. t. b5 }. _# G
% b* G& q. l5 Y. Y; C! o- GPIO_PinAFConfig(GPIOA,GPIO_PinSource2,GPIO_AF_USART2);
% ]1 A* b( e& Q: Y" X3 } L4 W - GPIO_PinAFConfig(GPIOA,GPIO_PinSource3,GPIO_AF_USART2);
- f5 ^) x5 |/ n. ~ - 2 k1 n5 m, d; F( B
- //GPIOA2,3->USART2RxTx/ _( C4 ~1 t: k j% n( a
- GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2 | GPIO_Pin_3;
0 V$ f [; s0 m# l - GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;
7 t0 [2 j3 ]" N' h7 _# g# [# L: b" V - GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
2 {; }% M/ _2 O3 R0 c8 ]4 a& e - GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; : B0 j2 A) m. U' A/ {
- GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;
6 A2 p& C3 g9 V" L: v - GPIO_Init(GPIOA,&GPIO_InitStructure);) j. q1 G/ j4 p6 ]) I; R& @6 h
( c2 b/ Y: X0 z6 ^2 u4 m- //USART2 init
3 V. H. D8 o$ Q6 `5 L - USART_InitStructure.USART_BaudRate = bound;
/ S/ A* L! W" d# F0 O& K* ` - USART_InitStructure.USART_WordLength = USART_WordLength_8b;
! l3 ~8 ]- Q0 d) q4 S - USART_InitStructure.USART_StopBits = USART_StopBits_1;( a- _! v p8 g u. k
- USART_InitStructure.USART_Parity = USART_Parity_No;
! T2 c% |$ G. w3 Z- h" V* |- u1 k - USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;0 ]; f! z1 W! ]: y+ F
- USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
G0 f0 ?; i3 N/ l - USART_Init(USART2, &USART_InitStructure);
6 K6 c; x. u$ o8 t% n3 O
~6 `2 u3 @( ?1 c- USART_Cmd(USART2, ENABLE);
# V9 ^: {( F/ l& I/ i0 z
& h+ D9 E j1 }! b- //USART2_ClearFlag(USART2, USART2_FLAG_TC);
1 f4 N: ^# O8 h Z# X - Q* R! N; W: n3 b
- #if EN_USART2_RX
$ K; N6 X: F: [- _- S - USART_ITConfig(USART2, USART_IT_RXNE, ENABLE);//interrupt' o0 x' s- v9 x; h# j4 P, P
/ h+ `& n& ~5 J2 E; e1 {9 w# q- //USART2 NVIC; q' z; G7 c/ P% z% w
- NVIC_InitStructure.NVIC_IRQChannel = USART2_IRQn;//USART2 interrupt channel! F) h* d4 ^# w2 }& S. W& t$ w
- NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3;& s* \& F9 \0 N7 \1 Q
- NVIC_InitStructure.NVIC_IRQChannelSubPriority =3;
8 g6 l; Q' k: u( G8 y - NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IQR enable* s, O" ]5 c* T. W, ?, E" @" Z: d
- NVIC_Init(&NVIC_InitStructure);& O5 g0 W: ]0 c5 i" F6 r- J- l
5 i! u) u5 S1 G- #endif8 `. h0 `' p2 T8 O2 A6 e( d
- }+ s" j: ?% B* e$ b; s
- 1 a& d0 \1 P: H
- 8 s, y1 W/ i" O) [/ c# N
- void USART2_IRQHandler(void)' X6 Z) t, \' L2 r, i
- {
1 p! h, m, z! y8 N' O- u0 O - u8 Res;! ]9 l1 p, v! X9 I& d7 X
- #if SYSTEM_SUPPORT_OS* q$ n! C8 q) R* v! G- V4 ^
- OSIntEnter(); " R# F6 n5 u# E' _* U5 x- w: H
- #endif" h: i( X! b1 g+ e+ s6 I- d
- if(USART_GetITStatus(USART2, USART_IT_RXNE) != RESET)
c. @5 d4 `3 ~4 K) ~/ n: r - {8 e5 i; ^4 s: J$ N( T/ k$ ~
- Res =USART_ReceiveData(USART2);//(USART2->DR)# ]/ c0 y: Z3 [- @( r/ X3 ~/ l
- // USART_SendData(USART2,Res);) p; {) _9 d6 J! C% u
- if(Res == 0x00)6 V( W+ Z6 `5 E+ g; L
- down();
; S0 _% Y$ T, q$ Y* d$ B/ g - if(Res == 0x01)' T( I% _- b/ A5 V+ k
- up();
3 L4 X2 f9 m6 ]7 C( Y& G9 w - } ( `, x6 P0 [! J! K& D
- #if SYSTEM_SUPPORT_OS: h- ~6 T7 Z5 j
- OSIntExit();
& g$ G6 d' V: E! r _ - #endif" P6 A8 b$ m! w( e, T( U: I g+ Z6 n
- } , r! K& Q7 j' f6 T) C
- #endif </font>
复制代码
7 h' {& _% G1 I. w
0 k" K! N2 m% L, S* u
# H" b3 `0 `0 h' ~. Q9 v: s
: F( K* K. u2 _) k! Z
- U; ]! w9 C& r1 B& b, V3 i( ~, z
|