串口通信原理 想必玩过一点单片机的人都懂一点串口通信,而恐怕我们印象中的串口通信不过如此,两个芯片通过串口通信,需要共地、Tx(发送)和Rx(接收)相互连接即可:' \$ o1 b* {# g) A. T
( q" `7 `# s" v4 b. G2 {% n* f3 E5 y
而实际上,串口通讯种类繁多,即使是通用异步收发器UART也有RS-232和RS-485等不同接口,有兴趣可以深入学习。 9 b H- k6 Q1 [7 N- m8 ?6 M
首先串口通信指的是一种设备通信的协议而不是指接口。串口通信的概念非常简单,串口按位(bit)发送和接收字节。而我们所使用的USART和UART的区别是可以同步,你在STM32的datasheet里可以看到USART2_CK,而实际上我们很少使用,我们完全可以把USART当UART来用。
/ s8 Y; ] w- }- P9 ?& j+ F HC06蓝牙模块
7 q/ Y, e4 l" d0 h, x* k无线蓝牙透传模块。 : N& d0 c& H% Y1 P4 T0 H
默认名字HC06、波特率9600bps、配对密码00000。
0 r8 `) l- A) n2 [: @4 {& z$ e# n$ y
% t: S- G7 g1 o; ?# m7 W ; w1 A& ^# n9 o- l
用Arduino的时候想必是很傻瓜式的,如果用STM32来做还是要了解一番的。
. _$ q' y( U' I$ Q( M! {! f) H F- D# ^+ R
首先好好看一下datasheet,看一下AT指令集,然后用一个USB转TTL的模块(这里用的是CH340)和电脑连接:
; R2 `$ A6 O6 A5 V. e) {* Q$ B& R# U, u. R. M
$ n0 g3 }+ u s# V/ w1 _% ~' E$ c
可以看到HC06的指示灯一直在闪,说明蓝牙未连接。打开串口助手,试一试AT指令是否有效,然后你就可以设置波特率(建议就选默认的9600,据说快了会降低通信的稳定性),改名字和密码,详见datasheet。1 \9 p( k* N0 {2 U
' v4 K( U; K0 `/ i) J
7 C" t8 T1 `' e8 u& E/ h+ w* G% A( f( v- }
/ B: w8 B& z9 ~8 ~! J打开蓝牙串口APP,会自动连接,连接以后指示灯就从闪烁变成常亮了。在串口助手里把HEX显示给勾上,然后用APP发几条指令,会发现收到这样:
0 b( ~/ i+ v' Q
4 m' t1 p$ d, j' V) y' P- e' U
9 F' ^* C: t" r) q+ ~4 `& S5 n; r那就说明蓝牙这边不会出问题了,专心调试STM32吧 - -‘
1 \+ Y1 R0 z4 s3 E: S+ A3 i
STM32F407的USART2配置
. `5 d6 u- G1 V/ J0 M: w* d
, _( r1 O6 a) \7 [% i首先查datasheet表,找USART2对应的是哪个GPIO:; E1 w |3 [$ b2 z0 z4 m5 U
/ Z! e1 S$ }8 s# `& B9 A& W' f
千万不要忘记查看总线!
( j2 ^- D' S9 M, F' r- k% C& w5 r G# I
/ N' r! ~1 p; Q6 M, A, N4 S
! r3 |' i C: B2 z3 x/ c
. G, q8 }3 [& o2 H
看了系统架构就会知道,F103和407的GPIO挂在不同的总线上。103挂在APB2上,而407挂在AHB1上,所以初始化时钟的时候千万要注意~ - |0 M) D- ^3 p
- RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA,ENABLE);
复制代码 " S9 O. d; u5 l2 ^+ s1 R8 p- ^
另外在配置GPIO时的参数也有不同:
; G& }) Z, t5 T' z4 _7 ~我们查看GPIO_InitTypeDef的定义:
0 n/ v8 k" @, L1 h2 G8 t
- //STM32F103中的定义:
4 ]- ?7 r) s N4 t$ L0 X; H - typedef struct
8 t4 D" I( M% U. O( y - {: f& S6 T# ?+ d! ^" E
- uint16_t GPIO_Pin;
" `* L1 L: ~, U" I3 b+ ^; J/ q$ T' s - GPIOSpeed_TypeDef GPIO_Speed;
* G! k* S* d) F: y& y8 b - GPIOMode_TypeDef GPIO_Mode;
& m' |9 ?; o- @8 P - }GPIO_InitTypeDef;
) a- D& E; O4 A# { - 3 F2 D& x4 \. I. `7 q S, n& w
- //STM32F407中的定义:
1 u, K7 p. W' ]! H' o; m: ` - typedef struct
?1 \ S* Q1 H8 m - {
, y- Y1 g, ^9 J+ V! | - uint32_t GPIO_Pin; ) o( r" f H1 F5 S: d ?- Q, |, P- {
- GPIOMode_TypeDef GPIO_Mode;
; ]8 ]* k% G; o8 {% E7 e - GPIOSpeed_TypeDef GPIO_Speed;
, N1 z2 R8 P6 ] ^7 m- ^; x - GPIOOType_TypeDef GPIO_OType; * s# Y3 v/ J( ^
- GPIOPuPd_TypeDef GPIO_PuPd; $ N$ L' d, c6 c1 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_H
# w7 N, Y3 V& b1 d7 i0 g - #define __USART2_H
' b2 G' W" J2 W+ j6 _8 @$ R( } - #include "stdio.h" ! i- U0 { |: m
- #include "stm32f4xx_conf.h"* p1 w! w: J2 {1 u7 `
- #include "sys.h" - p: c: d3 u2 g
. j: i% D1 u3 T( j$ F7 v- #define USART2_REC_LEN 200
o, }2 l& ^. Y. v0 t - #define EN_USART2_RX 1
5 j# |+ Y* A: Z - / _- _* F. {8 u# Q( L
- extern u8 USART2_RX_BUF[USART2_REC_LEN];+ Z) p, H! v, v- G: N7 }( u
- extern u16 USART2_RX_STA;
' \1 I# Z6 O$ A% k! T
9 l: {9 x+ K3 w% J% W- void bluetooth_init(u32 bound);
+ d& I. N7 Y$ m1 `2 C - #endif</font>
复制代码bluetooth.c: - <font face="Tahoma" size="3">#include "sys.h"2 Y; m8 F' V ^& q4 }
- #include "bluetooth.h"
. }6 C; C- c# v& z+ K - //这个是一个舵机控制板的驱动,感兴趣的可以看我下一篇博客
, v1 l- Q9 N& D2 J1 y - #include "pcf8574.h"* Z0 C2 R j, w9 K
$ q& a* J/ k9 K2 ~8 P6 s- #if SYSTEM_SUPPORT_OS0 y7 ^1 ]8 g" n, j+ ~
- #include "includes.h"
: h8 [0 \ W6 b& f" \/ {$ L - #endif7 z: P8 R& O0 V
- $ y+ z9 p8 ^8 o" s2 A
- #if EN_USART2_RX
3 X) e% ^" A/ d5 H) Q( r
/ l; w+ m @$ b- u8 USART2_RX_BUF[USART2_REC_LEN];
! N* G' W" Q. J% @$ X& E - 4 e9 O( K& Y$ E# s8 a B2 [
- u16 USART2_RX_STA=0;( ]+ r* m; L/ \9 l" Y8 K9 Z
) L3 l1 F* E" `, {9 h- L- void bluetooth_init(u32 bound){
6 r' w: r% c- V, {# l - 6 A9 ]. J( K/ T# @5 d+ b
- GPIO_InitTypeDef GPIO_InitStructure;
- Z) a5 P" C: x8 r9 o$ E% [ C - USART_InitTypeDef USART_InitStructure;
4 X4 u- C, d0 r( N( a - NVIC_InitTypeDef NVIC_InitStructure;3 {5 a8 h2 K: ? }, r# [3 `
- + W$ L+ A4 `+ o, l. e5 R$ C$ b
- RCC_AHB1PeriphClockCmd(RCC_AHB1Periph_GPIOA,ENABLE);
& ]: s9 u+ W. o9 U* x: A R v( g - RCC_APB1PeriphClockCmd(RCC_APB1Periph_USART2,ENABLE);( n) i* s4 j- i1 Z4 r; r
4 E9 x2 ~4 P& O. X- n. \' Q- GPIO_PinAFConfig(GPIOA,GPIO_PinSource2,GPIO_AF_USART2);
- ?, W: l8 C9 G# N. T) G$ H - GPIO_PinAFConfig(GPIOA,GPIO_PinSource3,GPIO_AF_USART2);
~, J5 r; M! q6 U% j# N
8 o0 ~& Y# L6 k6 [8 g- //GPIOA2,3->USART2RxTx
# L4 r4 r% x7 S! K) @4 ^9 K - GPIO_InitStructure.GPIO_Pin = GPIO_Pin_2 | GPIO_Pin_3;
/ @' R I: _1 s4 w. G, ?1 G - GPIO_InitStructure.GPIO_Mode = GPIO_Mode_AF;! ?* |: q; ]/ c) A% g
- GPIO_InitStructure.GPIO_Speed = GPIO_Speed_50MHz;; i" T( }8 d" c5 ~* E0 |
- GPIO_InitStructure.GPIO_OType = GPIO_OType_PP; # `. j- a9 c4 J$ g2 t
- GPIO_InitStructure.GPIO_PuPd = GPIO_PuPd_UP;
$ H3 o+ b7 M- U8 l T `# P - GPIO_Init(GPIOA,&GPIO_InitStructure);
9 m- G5 l, t7 o% P) N! R - % @" w B/ I0 M' M9 n" r
- //USART2 init6 r, Q: w5 T1 {$ Z8 B1 S4 `
- USART_InitStructure.USART_BaudRate = bound;
# `6 v+ h4 A( r) z - USART_InitStructure.USART_WordLength = USART_WordLength_8b;, U! h/ h' s% F S
- USART_InitStructure.USART_StopBits = USART_StopBits_1;# p4 k/ V8 v" E
- USART_InitStructure.USART_Parity = USART_Parity_No;; `7 e, `, e- V: _
- USART_InitStructure.USART_HardwareFlowControl = USART_HardwareFlowControl_None;0 _% C) |4 u' J% v: A S8 f/ X
- USART_InitStructure.USART_Mode = USART_Mode_Rx | USART_Mode_Tx;1 z: G d* n, o0 i
- USART_Init(USART2, &USART_InitStructure);
& M- O4 q0 j, F+ N) M( I' W/ B - / P+ g: J1 h4 O8 }
- USART_Cmd(USART2, ENABLE); & d" m4 X K0 |) A
# _$ T% N- Y3 E& c- //USART2_ClearFlag(USART2, USART2_FLAG_TC);/ ?! L* _! g5 g8 ?: \3 _! T
- 3 H9 P# q8 h! ~7 P% R9 V' K
- #if EN_USART2_RX
! D7 I4 } \+ J Q. Y - USART_ITConfig(USART2, USART_IT_RXNE, ENABLE);//interrupt+ b' s- l$ t' n# f
- 5 f( a3 G( R/ v7 ]' S
- //USART2 NVIC" A1 n9 l% L: q% ^- s% @: j; l
- NVIC_InitStructure.NVIC_IRQChannel = USART2_IRQn;//USART2 interrupt channel
* I; U* P& u' z! V) y) r - NVIC_InitStructure.NVIC_IRQChannelPreemptionPriority=3;
8 K# y5 T5 p# C - NVIC_InitStructure.NVIC_IRQChannelSubPriority =3;" p* w2 d& H# Q
- NVIC_InitStructure.NVIC_IRQChannelCmd = ENABLE; //IQR enable
. Z. |! x/ x& E* q Z1 {% U - NVIC_Init(&NVIC_InitStructure);5 Y8 p+ Z) D9 W6 h( |1 m% \. P
/ D6 p6 N+ ^1 | G) x9 y- #endif
6 ^) `# W7 n# {0 d3 r% S$ C - }/ `: G+ s3 E; d1 ^% w; I% R
/ W5 I2 D* i* X# q% S" }+ u" s' M- * P6 e* |/ R0 `- W: ~
- void USART2_IRQHandler(void)
* m+ w! }( s1 `, z2 ~; z - {
0 J. n" ], h, ~1 W' k - u8 Res;
; B( U8 c# W1 o - #if SYSTEM_SUPPORT_OS) p- k. T" j* z& E! q1 `' Z$ K
- OSIntEnter();
1 \) K% U% q( `" m - #endif
# `, r( n$ E' o a) B. r7 v0 f1 P# l - if(USART_GetITStatus(USART2, USART_IT_RXNE) != RESET)1 ^/ K( _" Y' I7 {$ {- R. N' c& d0 Z
- {
2 H( L8 I( @0 ^9 c1 j - Res =USART_ReceiveData(USART2);//(USART2->DR)5 ?2 W1 |2 P. P4 ^5 Q, Y8 S
- // USART_SendData(USART2,Res);' z2 b* C% M o; K# [/ ~9 H
- if(Res == 0x00)& w$ x4 R ~: z3 @7 L1 `, t
- down();
7 Q, J4 q, v0 \' p$ t5 w - if(Res == 0x01)3 Q; C. `$ A/ ?* O; {) C* I
- up();
: [9 `) Y [4 S9 u; H - } ( o1 V2 k; B+ C
- #if SYSTEM_SUPPORT_OS
$ G+ `2 {, T8 X - OSIntExit(); ' z9 j# X3 c3 e) ?
- #endif& w7 E" J5 K, L/ G6 ^8 J
- } 6 j1 D7 i, H9 | b9 e+ }) i
- #endif </font>
复制代码& p2 \& V% L* ?; k$ r: ^1 }
. r7 J9 u) l& P( Y E3 ^3 f
% f0 @) T! ^+ r5 w/ `0 q5 Y. n# m) K1 a
- b& r* ?* Z" M3 n5 l |