串口通信原理 想必玩过一点单片机的人都懂一点串口通信,而恐怕我们印象中的串口通信不过如此,两个芯片通过串口通信,需要共地、Tx(发送)和Rx(接收)相互连接即可:
% {, l5 K5 `% {; S! u, {4 w+ ]6 Z' N1 \* b, H
而实际上,串口通讯种类繁多,即使是通用异步收发器UART也有RS-232和RS-485等不同接口,有兴趣可以深入学习。 / H) L& a& ^% N/ d. y
首先串口通信指的是一种设备通信的协议而不是指接口。串口通信的概念非常简单,串口按位(bit)发送和接收字节。而我们所使用的USART和UART的区别是可以同步,你在STM32的datasheet里可以看到USART2_CK,而实际上我们很少使用,我们完全可以把USART当UART来用。 + \1 }: e' n6 {
HC06蓝牙模块
% a4 z/ S ? L+ r, W无线蓝牙透传模块。
+ j. n& P; L& o9 v2 {+ W$ Q, @默认名字HC06、波特率9600bps、配对密码00000。
# M- H L" [$ P+ K" G- I) ~) v; K
$ d: S, Y. Z- c1 O" `4 A) |
5 u2 w8 T. T9 g( o# o4 a% ]( H4 {
用Arduino的时候想必是很傻瓜式的,如果用STM32来做还是要了解一番的。- D) D+ v5 t/ R" Y/ T# d, P
4 s+ j1 w; S" t+ `% k/ G
首先好好看一下datasheet,看一下AT指令集,然后用一个USB转TTL的模块(这里用的是CH340)和电脑连接:
* n2 K V2 I8 Q0 V
0 Q, e0 d: J V# z2 X" e. s+ V( _- V" }- p) A
可以看到HC06的指示灯一直在闪,说明蓝牙未连接。打开串口助手,试一试AT指令是否有效,然后你就可以设置波特率(建议就选默认的9600,据说快了会降低通信的稳定性),改名字和密码,详见datasheet。( E: `' n& |; z) S
4 D" y6 B2 d8 S) t7 Z4 C
: e% P2 c+ N* B- R7 i# X* W
. T( A- ^: @3 ?) r; S5 g* J& B. i3 K
打开蓝牙串口APP,会自动连接,连接以后指示灯就从闪烁变成常亮了。在串口助手里把HEX显示给勾上,然后用APP发几条指令,会发现收到这样:
% H& E) \2 \0 Y8 M8 T' r
9 G/ r5 Q7 C$ J; V; Q- j" Q2 P
# O: }: `* }/ r/ m& i8 e3 l那就说明蓝牙这边不会出问题了,专心调试STM32吧 - -‘
5 |5 |! t" P' D8 B; p! { STM32F407的USART2配置. R9 `7 z A* V8 c7 C/ l7 ~
+ n$ L4 h' p, d) T: B1 K首先查datasheet表,找USART2对应的是哪个GPIO:
8 a2 D! X# t2 ~; V9 W8 X
) S& |1 x! {% F# g
1 i7 ?; s. `/ @0 F t h5 G. J 千万不要忘记查看总线!9 k& r9 [7 d/ {! b
( W/ Q6 a9 I. k- ~
6 W. W; k& H! Y/ Z1 B6 r+ y* o& O; R. V# w& P' L% Y, R
6 ~5 H; ?* k: U7 i看了系统架构就会知道,F103和407的GPIO挂在不同的总线上。103挂在APB2上,而407挂在AHB1上,所以初始化时钟的时候千万要注意~ 7 e% D! B) V' \, Q' `
- RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA,ENABLE);
复制代码
) O( b8 l4 C5 q( Y* |, S" n另外在配置GPIO时的参数也有不同:
3 d% h, c& P! n3 S我们查看GPIO_InitTypeDef的定义: , W+ A& T" P. ?, ?$ v% V
- //STM32F103中的定义:+ v2 C% F) s7 \; t4 g' v, S# o
- typedef struct6 P2 J3 \/ ?# }- v
- {
" _5 s4 q; T; T3 ]) F - uint16_t GPIO_Pin; + u" Q5 ?2 [1 L+ a$ b
- GPIOSpeed_TypeDef GPIO_Speed;& v! c0 v$ @9 \
- GPIOMode_TypeDef GPIO_Mode;
/ a) {% J& F h: o - }GPIO_InitTypeDef;& r" z7 }- c7 ~$ g4 Z# R
- 5 r" P3 R$ U5 `) r5 K% a
- //STM32F407中的定义:' V5 C9 v5 l2 X7 C
- typedef struct: I8 M" I! T$ F& N7 u
- {! {7 e! |3 O) O% F
- uint32_t GPIO_Pin; 4 ~ e. u: i; c$ P0 W% ~1 D' {
- GPIOMode_TypeDef GPIO_Mode;
8 j8 X5 X P& G" k1 n$ N - GPIOSpeed_TypeDef GPIO_Speed;
7 Y, z% @% F1 N! { - GPIOOType_TypeDef GPIO_OType; 4 h: o4 C, \+ d0 c F) E
- GPIOPuPd_TypeDef GPIO_PuPd;
7 q9 Z' R4 ?: r1 z4 N - }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
& _4 l! B% a7 L: }" q - #define __USART2_H) U4 r( h) e: F6 ^3 ^
- #include "stdio.h" / w0 K, a% ]6 b1 x
- #include "stm32f4xx_conf.h"
* ?+ _, ^( F# R: ]5 H( x - #include "sys.h" ) B8 ?5 ]- J5 M; W P
& C; I( e! l/ T# |# q! t1 \, H- #define USART2_REC_LEN 200
/ B/ g/ y7 ~8 J3 q- } - #define EN_USART2_RX 1
; e8 p0 \6 d: c$ j, t; X0 Y - 0 ^* d: H6 k/ d0 u' i5 I+ y! J
- extern u8 USART2_RX_BUF[USART2_REC_LEN];$ Q" B, [# F/ K0 M/ v$ m4 B
- extern u16 USART2_RX_STA;+ S' W1 k+ N4 `/ D6 \
- 6 n! ~. @7 ~! f9 ]
- void bluetooth_init(u32 bound);& D- q3 T& ~, m8 i0 y P- h/ L3 A
- #endif</font>
复制代码bluetooth.c: - <font face="Tahoma" size="3">#include "sys.h"
4 [4 @" o% w) U( \2 p - #include "bluetooth.h"5 d- m' q7 G3 N2 N8 H$ k
- //这个是一个舵机控制板的驱动,感兴趣的可以看我下一篇博客
5 y( l# t1 ~/ G5 z0 _7 t - #include "pcf8574.h"
/ S1 p( w' N% G( }5 ^
$ r3 [5 {$ X( g. q- #if SYSTEM_SUPPORT_OS1 E; Y# T V5 \! v
- #include "includes.h" 0 h( n5 A; B: T( _$ [# Y
- #endif) t7 t- U- c+ V* ?) D: I9 o
- 2 A; I" h, a& T
- #if EN_USART2_RX- s1 T) w. v6 p$ b3 J
. I! I% o$ b4 E- u8 USART2_RX_BUF[USART2_REC_LEN];
8 V0 U5 r7 E/ J2 T! g
! f$ Z. Y; O4 A$ f# z F0 o: X- u16 USART2_RX_STA=0;
/ Z+ Z8 _ z1 I# l& c. t- M
* N( _) b) f4 w( B) D, L/ _+ h: G H- void bluetooth_init(u32 bound){2 }& z* D# K D
- q8 W7 ]2 l- r$ Z0 x! I. u- GPIO_InitTypeDef GPIO_InitStructure;5 k4 H, y0 s& E: q! s! N: `9 b5 N
- USART_InitTypeDef USART_InitStructure;$ y) ^4 _; w+ Q- F/ \0 a- Z9 x; A
- NVIC_InitTypeDef NVIC_InitStructure;
; s5 `% n8 U6 U1 S4 Y- _* o, S) W
4 G! Q8 z; o0 {. A% v6 R! f2 [- J- RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA,ENABLE);! L( k. S& A6 _, i
- RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2,ENABLE);3 j% f+ u8 n6 | o; b3 u; X) U% l
- 1 F, X* a& y) E5 j, M
- GPIO_PinAFConfig(GPIOA,GPIO_PinSource2,GPIO_AF_USART2);8 e; T7 l; \" |/ ^/ ~
- GPIO_PinAFConfig(GPIOA,GPIO_PinSource3,GPIO_AF_USART2);* t( m' Y8 N6 y9 e" b# Y$ V' j! r4 k
4 Q; H" x4 K/ D4 Q9 [4 y& N; z2 D- //GPIOA2,3->USART2RxTx; t+ w7 n3 [4 ^' o. W! T# t* Z# D
- GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2 | GPIO_Pin_3;, B1 b9 o6 k ^9 U# i% h
- GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;; l6 g T5 z- V# B% S6 i" @3 e
- GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;* W& {7 u& g! i* V; K" v
- GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; 9 {& L' Y5 ?2 D& `4 b; |
- GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;( L) L6 A$ e# |# Z, w6 h
- GPIO_Init(GPIOA,&GPIO_InitStructure);
) B. w& a( o5 E
- n$ I$ y9 s; j" I# l- //USART2 init
* T, x2 z. T% v7 E - USART_InitStructure.USART_BaudRate = bound;0 W- A, ]3 L6 E. m/ c) X- P' `2 J
- USART_InitStructure.USART_WordLength = USART_WordLength_8b;
; E/ B' {" g5 d9 n+ ?4 P9 P - USART_InitStructure.USART_StopBits = USART_StopBits_1;$ G6 {" C1 d9 B4 Z. Q# m9 [
- USART_InitStructure.USART_Parity = USART_Parity_No;' g/ T8 p, ?# l2 M
- USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;# |1 `1 t# V: g; Q' v* {
- USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;
, ?. ]$ i {$ T' i% a0 [- _+ E - USART_Init(USART2, &USART_InitStructure);
# P2 N" J% ~4 g+ ^& z
3 A: f9 s! z. g C: n5 c' I/ t- USART_Cmd(USART2, ENABLE); . K: V f7 g; C9 c# _ L
- & Z* @ [; M$ y& ~! ?/ c) T! \
- //USART2_ClearFlag(USART2, USART2_FLAG_TC);+ C+ f I, r8 u1 a
) W# o2 n2 k( t9 }2 ?$ ~. {7 T- #if EN_USART2_RX % t% B/ v- s2 M/ {4 d5 D
- USART_ITConfig(USART2, USART_IT_RXNE, ENABLE);//interrupt9 ]6 [' ?, K+ g' }* F! E( k9 \% \' \
- ( |" b! j- y' J' \" X# {
- //USART2 NVIC0 h7 z6 g8 h6 z B# `9 T
- NVIC_InitStructure.NVIC_IRQChannel = USART2_IRQn;//USART2 interrupt channel
/ g; v+ b7 x" c1 D/ z - NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3;
; K# H6 k% W) L9 `1 j - NVIC_InitStructure.NVIC_IRQChannelSubPriority =3;
& i/ z0 @( i! i+ Z - NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IQR enable9 N! ?& \1 L# {. a9 Y2 a, m* ~
- NVIC_Init(&NVIC_InitStructure);5 M/ v" b7 v% T
- D h7 S4 N4 I" b: X' |; l
- #endif% ` ^7 B9 ~' T9 V: }
- }; D! S: b$ Q" E( @3 z
8 h; g' d, {* i% Q$ Z( E6 o
/ v+ ?9 \8 x) u3 p- void USART2_IRQHandler(void)
" S4 O- {, p9 k- q4 k) y$ R4 N - {: r- r7 U3 ^# d4 |2 @7 U+ W
- u8 Res;
& E8 B6 V( p' F5 `7 g' _ - #if SYSTEM_SUPPORT_OS
+ Q) P" J' ~' \8 Q: s - OSIntEnter();
3 y' x( P) I0 t, n3 }. g- ?6 b - #endif3 F5 L1 X2 j# N$ B% ]) z
- if(USART_GetITStatus(USART2, USART_IT_RXNE) != RESET)
2 P/ P, Z& k4 ^5 C; I; z& O# Q - {
: l2 @' ?8 @% ?3 U. { - Res =USART_ReceiveData(USART2);//(USART2->DR)0 r. C6 G% j4 }3 x
- // USART_SendData(USART2,Res);; `, i* ?6 \. d; g+ ^
- if(Res == 0x00)( ^1 `# w& _ z- g
- down();
) n+ A& v+ [+ R& g4 z' M/ u - if(Res == 0x01)" Q0 @7 H7 y2 \0 {1 {
- up();
0 y; B- q5 H* P/ k. Z! B - } : V- ? Y- _$ O* i. G
- #if SYSTEM_SUPPORT_OS
' w9 _! \8 X5 S$ {7 q: `5 X - OSIntExit();
2 k$ R- A& q( o! } - #endif. J d5 ?) T& X+ M) Z) P
- } ) k( c& I3 ~ p3 t% U
- #endif </font>
复制代码0 |5 [( f1 V. o( u( @) M! I N7 p6 E
/ m( d/ a8 T# i" w% v( D9 K5 ?
& X; c9 Q$ r% m5 G$ T; Q
9 }1 b: h% T6 M& H$ I& [
# T9 _, X+ ^: u) \+ ^6 g5 @ |