串口通信原理 想必玩过一点单片机的人都懂一点串口通信,而恐怕我们印象中的串口通信不过如此,两个芯片通过串口通信,需要共地、Tx(发送)和Rx(接收)相互连接即可:4 u( c3 o. t! y( u- G- O
) n" y. b# {/ E5 W而实际上,串口通讯种类繁多,即使是通用异步收发器UART也有RS-232和RS-485等不同接口,有兴趣可以深入学习。
% V4 G! ?4 ~% d5 `首先串口通信指的是一种设备通信的协议而不是指接口。串口通信的概念非常简单,串口按位(bit)发送和接收字节。而我们所使用的USART和UART的区别是可以同步,你在STM32的datasheet里可以看到USART2_CK,而实际上我们很少使用,我们完全可以把USART当UART来用。
/ R* x, k' `% N) X' c
HC06蓝牙模块- @/ `* P8 A$ j7 G
无线蓝牙透传模块。 % g2 w# N" b. [" ?0 v
默认名字HC06、波特率9600bps、配对密码00000。/ _! D# C8 A! L
; b) ^, ]; }5 y$ g1 V, r) d 9 q% w' `* a3 U; B9 y
用Arduino的时候想必是很傻瓜式的,如果用STM32来做还是要了解一番的。
* h, S$ u4 w2 W* w! h; F& ` U2 d6 `" N t' b0 [9 h
首先好好看一下datasheet,看一下AT指令集,然后用一个USB转TTL的模块(这里用的是CH340)和电脑连接:
# s$ d; b" F& G {$ A7 c
3 B7 i3 s+ I: J6 C& X, T; E+ D/ w/ P- J: R6 {' Z* _" I
可以看到HC06的指示灯一直在闪,说明蓝牙未连接。打开串口助手,试一试AT指令是否有效,然后你就可以设置波特率(建议就选默认的9600,据说快了会降低通信的稳定性),改名字和密码,详见datasheet。2 Y% ^# l, t) L
5 Q" C% @5 o7 J- V2 P" X" I6 P* R1 |: `' m5 b& H% c( O
1 E( `. g' h% Q" H& d
* ~# R; R( @: A% s0 E- A5 N |打开蓝牙串口APP,会自动连接,连接以后指示灯就从闪烁变成常亮了。在串口助手里把HEX显示给勾上,然后用APP发几条指令,会发现收到这样: ! E$ F: X" f8 K( Y+ j" M% B
?. y* K: r3 ], S/ S' }
- Z4 ]5 E/ u; H4 @) j+ r* d# ?那就说明蓝牙这边不会出问题了,专心调试STM32吧 - -‘
" b7 b" [0 l; |! ]8 T5 e STM32F407的USART2配置
& @+ H% V u; F9 B
) ~* s/ P/ a, D8 s4 T首先查datasheet表,找USART2对应的是哪个GPIO:
: X& P0 l: J, B8 E0 x3 G
! Y8 f6 c7 d9 h" G1 C$ |& M3 X! X/ I, C1 `/ v7 e
千万不要忘记查看总线!& Z1 r1 n0 }% T' i& Y/ ]5 N3 V9 D
( G( `, g- O9 p0 D8 o. C- r9 @
4 m/ @0 m, [& \) A: ]
& x6 Y% e' a" G% Q* a4 r/ \
4 f7 d/ c5 _( d! L/ `( I; h看了系统架构就会知道,F103和407的GPIO挂在不同的总线上。103挂在APB2上,而407挂在AHB1上,所以初始化时钟的时候千万要注意~ 3 d2 R/ c3 V( K$ y+ b* j
- RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA,ENABLE);
复制代码 0 V! k6 a9 A K# i" D* x
另外在配置GPIO时的参数也有不同:9 J% x7 v m! Q
我们查看GPIO_InitTypeDef的定义:
. _/ {! @7 b7 p2 E, l6 f
- //STM32F103中的定义:5 k. [- m8 W* C! v; T! @
- typedef struct h2 [. b. n1 F) w) h" b9 c
- {
3 p; g0 T& K' b: v - uint16_t GPIO_Pin;
( z) q$ ~6 V1 e# F5 G - GPIOSpeed_TypeDef GPIO_Speed;# Y& q0 y! T- C. U: C' g% @# T
- GPIOMode_TypeDef GPIO_Mode; & f) V! w* E! N2 j; w8 _# I
- }GPIO_InitTypeDef;
/ u4 M) h5 n9 k/ z! G5 r: N - ! m) _$ i7 ?7 n! R
- //STM32F407中的定义:) c& Y4 N( H; P+ B5 b' A
- typedef struct
6 B+ e! e/ C h! _- {* n H - {
' I2 s" t; Z0 K- |: l0 D - uint32_t GPIO_Pin; ( I4 S4 T7 F- p
- GPIOMode_TypeDef GPIO_Mode;
8 x" w" i& ]3 H% j, R - GPIOSpeed_TypeDef GPIO_Speed;
( t0 `5 E6 p7 l5 N, Q6 r, w! d - GPIOOType_TypeDef GPIO_OType;
# |* g& X- [. ^ {7 H: G! R6 a - GPIOPuPd_TypeDef GPIO_PuPd; 9 }1 H$ U" K' P# g1 ?$ s6 w7 i
- }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_H1 b# w" f% \. r7 I+ j" a6 P
- #define __USART2_H
4 \3 v& U3 k5 a4 i2 B; g - #include "stdio.h"
9 P+ h# i$ E* Z& ?, Z; I1 } - #include "stm32f4xx_conf.h"
$ U" X/ b" G5 I. K7 s( ~6 q3 p - #include "sys.h" " C, l7 l: D; Z: V
1 ]& y) Q4 S( U3 B' x0 Z# x- #define USART2_REC_LEN 200
" y; e8 r# Z/ x- K2 ] - #define EN_USART2_RX 1
, ?8 W- |# n9 }; o% h4 c - - S6 |, z/ c r2 t' O
- extern u8 USART2_RX_BUF[USART2_REC_LEN];7 r/ B" _1 P0 Q4 r5 U. l
- extern u16 USART2_RX_STA;
0 n2 R. |; l# H w6 u' H
. t, _: C8 _# b" i$ E& S1 y# O; O- void bluetooth_init(u32 bound);% Q& ?& n; x1 k8 R/ v; A
- #endif</font>
复制代码bluetooth.c: - <font face="Tahoma" size="3">#include "sys.h"6 i) n$ R1 E' n3 i
- #include "bluetooth.h": v6 g* o4 t/ K' N; H
- //这个是一个舵机控制板的驱动,感兴趣的可以看我下一篇博客 W. C* ~9 P1 Q0 V# u7 N
- #include "pcf8574.h"* e) `, K- W4 o
. Z4 x! P& r; }$ H- #if SYSTEM_SUPPORT_OS5 U: S) b( g; A( S5 r
- #include "includes.h"
! Z8 [+ x" m1 m - #endif' P# Y$ P% @) H' T+ x/ h
2 b, m) Q$ g$ W. f- #if EN_USART2_RX
3 u x5 t" a' W4 u
3 D, c& [- q: U! O- u8 USART2_RX_BUF[USART2_REC_LEN];/ l3 b% N* f% ]1 S" t
* U( {* l1 a9 x4 Z, t- u16 USART2_RX_STA=0;1 ]8 E( ~0 h% n r$ y- r) T
; r' v, H$ ~3 E' V8 I" s- void bluetooth_init(u32 bound){# X+ I$ o- p" [* \' e/ g p5 p
- ( S: G# E+ `% |; y0 \7 H& o
- GPIO_InitTypeDef GPIO_InitStructure;, q* f6 a3 f9 i5 u- A( P
- USART_InitTypeDef USART_InitStructure;
/ I0 b% e' v; _/ U - NVIC_InitTypeDef NVIC_InitStructure;- a2 N+ B: V7 N# \
- , n6 |) c1 a- N
- RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA,ENABLE);
2 }& w m% G! ~ - RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2,ENABLE);
7 x! C$ @7 S! q9 O0 L" V7 H& Q
2 W. d) z/ i, y- GPIO_PinAFConfig(GPIOA,GPIO_PinSource2,GPIO_AF_USART2);! C0 P8 b) G, M1 e
- GPIO_PinAFConfig(GPIOA,GPIO_PinSource3,GPIO_AF_USART2);" r! Q/ T7 u* Q) D7 X
( \" r+ Y2 \7 O9 O- //GPIOA2,3->USART2RxTx
4 n( k- R' I. d. U, M; Z" T( V1 T - GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2 | GPIO_Pin_3;9 y8 y w7 y' M9 J7 |1 J
- GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;4 S0 A% E! d) `8 k3 [/ x3 r( r; Z
- GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;
: l: g1 Q9 L5 S$ C2 t# z$ Y - GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; 3 \" j4 y8 u; b: V( N4 [
- GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;
# {3 J, [, u" W# @ - GPIO_Init(GPIOA,&GPIO_InitStructure);+ r5 F& M Y& A' S9 l' E) i9 b; A
- 0 s2 I" Q) ^; ^
- //USART2 init
# ~! c2 Q7 j0 Q/ ^& i - USART_InitStructure.USART_BaudRate = bound;3 P& R! D7 U( {9 l
- USART_InitStructure.USART_WordLength = USART_WordLength_8b;' P2 @. s6 E: Y0 g+ b r
- USART_InitStructure.USART_StopBits = USART_StopBits_1;
$ k; V9 G8 A7 @4 M9 ` p, q - USART_InitStructure.USART_Parity = USART_Parity_No;% F) c6 j7 X. v
- USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;4 N# m3 b/ G# T j' _- B
- USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;8 R4 L& N8 q/ E3 l5 a
- USART_Init(USART2, &USART_InitStructure);9 C! c5 Y# _% u3 @
- 2 [4 c! `7 U4 w6 A0 h; b
- USART_Cmd(USART2, ENABLE); % ~/ P3 a$ U9 N+ K: j& t3 j
" d! x5 B+ J) M* Q# e0 G- //USART2_ClearFlag(USART2, USART2_FLAG_TC);
" Y, c4 B; a; W) ^0 S - $ T: X/ N4 F5 a) b% |) q2 W! b
- #if EN_USART2_RX 8 J$ X6 a7 A6 T6 e
- USART_ITConfig(USART2, USART_IT_RXNE, ENABLE);//interrupt5 R( R# A% `; v/ m
- ) q, U( Z- y' i. `7 \
- //USART2 NVIC9 A; J7 e# a' n d1 ^
- NVIC_InitStructure.NVIC_IRQChannel = USART2_IRQn;//USART2 interrupt channel" S& ]( B9 Y7 o
- NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3;
7 x1 }/ ]$ n" _) t$ X - NVIC_InitStructure.NVIC_IRQChannelSubPriority =3;; ^ ?! @; l8 g4 R/ B9 U q' Z! j
- NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IQR enable
( L7 K5 J) U& E* P2 a, H9 E! q - NVIC_Init(&NVIC_InitStructure);; U! O" ^/ j% V3 s" h4 i
2 i1 n6 F9 B" L, Q0 p2 H% e$ k- #endif! g! d5 q. I+ F' E
- } q, A, P9 _" Q% }
/ V! \ X6 Q4 O; z
. W1 f$ f- G5 P' A+ m) [+ ]- void USART2_IRQHandler(void)
" }# ^# {, i9 U# C - {
4 d4 Q+ p0 ~& p+ b, ~2 o - u8 Res;; x6 p3 ]; {6 K+ \* Q1 P
- #if SYSTEM_SUPPORT_OS
6 ` q, S q3 o+ z1 E4 W - OSIntEnter();
! d3 ^. \' M. ]# t: G. j9 M - #endif$ d3 u& q7 `+ n `$ @5 E
- if(USART_GetITStatus(USART2, USART_IT_RXNE) != RESET); {, G4 _; g2 [* D
- {0 B3 S5 e( \/ S% `+ d
- Res =USART_ReceiveData(USART2);//(USART2->DR)- `1 {; x% q6 _' A
- // USART_SendData(USART2,Res);
" {3 g3 B% }# h - if(Res == 0x00)
8 e, [* V$ c: V% Z3 D - down();7 g R2 l6 w* x+ J% ^0 q
- if(Res == 0x01)
: ^& V% S" @3 C- k, } - up();
8 Y2 d/ k. U/ ]! K' }) J" A2 U" M - } # w. P+ h) m- A- c
- #if SYSTEM_SUPPORT_OS# ~$ d- `$ |: |8 [; @/ {' G1 z/ g
- OSIntExit();
2 B. l& A* ?7 i) l- _" z/ |' U - #endif9 Y; f5 @) m ]9 r( k- y
- }
: q4 T1 _: Y6 ^* t6 a r - #endif </font>
复制代码0 C( @& I& i5 {! @
3 q* S1 \0 a* q( @+ ~
4 F9 S/ }1 R' ?$ u6 l, ^6 |! m/ }% E
# t, } {5 A% a( c n |