前言
. u9 Z& y, S3 M9 C最近用了NUCLEO的板子做一些东西,用到了串口通讯,结合一些资料再加上自己整合基本实现了发送和接收任意长度数据段(第一次写博客有点小紧张哈哈)。" q4 q3 d/ s- V
1 w; E. J( L7 N% i9 |* @( u) x
CubeMx软件的配置过程
, `* p8 A+ H1 S8 K+ g首先,建立新的工程,选择芯片类型,这里我用的是stm32l073rz。
& M. I; ?7 Z( N1 `) Z8 S# ~* T9 \! k* z
, L' K2 U, c, z; C6 V' G$ c* ~下面选择sys部分,勾选Debug Serial Wire (这个好像是下载调试 st-link,也不是很清楚)。2 U, b' J/ P+ n4 K; D0 ~. J G
% S8 E) J! J1 Q! t' r1 e
" e X6 T4 I) E m; w+ M' |8 h8 t, T$ E9 j4 U
选择USART2,选择模式Asynchronous(异步通信)模式。
5 Y- W, a) n8 t2 r+ q2 u
. p- P/ S P9 V; T3 P# A0 o
0 b3 N" r8 v4 S# ?) e# h7 |& D- O7 r' a- k" \5 ]/ A& ?+ U' u
选择下面DMA模式 ,添加rx tx,并设置tx为循环模式。
* J, c" `) ?1 o. a. i9 {* y8 u" C4 [( q7 y' | B
* F- [2 ^, u' Z+ x
, E7 M7 {9 k l" U' x勾选USART2全局中断。
4 o5 @+ ?9 ]' h. h9 l5 N0 N, Q$ v' n5 _+ D% |: H: Z
: W* ?( O5 _- N+ s8 V3 k
N7 D7 D/ {7 H0 n# S' @) E保存工程,选择相应的选项,生成代码即可。
3 W4 _. e0 k0 ^, K; t
, ~7 r$ H) c3 T& d: k5 X
- D" h' G3 l. S2 g. z/ w3 f0 R' x" W* E
代码部分
& H) i$ z- o# m, wC语言的重定向fputc函数 v P* P5 ^2 H c& @7 K; K, m/ X
- int fputc(int ch,FILE *f)
6 ?1 q* {" I- ]- v [$ h" w! O* ? - {
+ \1 R5 a( [1 ?5 D1 ?0 ^- e; { `9 m - HAL_UART_Transmit(&huart2, (uint8_t *)&ch, 1, 0xFFFF);
8 e* L; F6 d. v; B3 T/ C. K - return ch;$ {8 g4 N$ K! \/ O; P1 ^% N
- }
复制代码 " E, H9 @8 x3 y% ~+ P9 l# ^
ch为要写入的参数,fp指向FILE结构的指针;这样重新定向后就可以使用printf直接输出。
! m* q& S# T/ o' R3 G2 Y+ c/ P2 l1 T+ d3 V
变量定义
9 c9 P; E1 K$ m; V9 I: R9 w& m1 Q- #define RX_LEN 1024 //定义总的数据长度" ?& q, t2 C* q. W8 y. d
- volatile uint8_t RX_flag=0; //IDLE 接收标志% i+ m4 B/ z+ U5 W2 v
- volatile uint8_t RX_Size=0; //接收的数据段长度
. i2 o3 u9 [( g" j - uint8_t RX_pData[RX_LEN]; //DMA 接收的数组
复制代码 ! y5 _+ W; z0 u# m- {: T4 C! ]
volatile 关键字提醒编译器定义的变量是易变的,编译后的程序每次需要存储或读取该变量时,会直接从变量地址读取数据。在中断或多线程中使用volatile关键字可以避免不同优化等级时程序出错,提高程序的鲁棒性。2 V1 f" t1 J _# x/ T
h& K5 ]- e( J% `: q- e2 s( \( c, G中断函数" k Y o* }5 U
- void UsartReceive_IDLE(UART_HandleTypeDef *huart)
- T: _; w) m- P; ]. \ - {
4 I$ H7 f5 c6 V% a7 O; o1 O) ] - uint32_t temp; , W: E3 Z9 D7 d7 @9 V% } k' _0 d3 e
- uint32_t tmp_flag=0;6 k0 |9 n- E! G* ?0 o9 N
- tmp_flag =__HAL_UART_GET_FLAG(huart,UART_FLAG_IDLE); // USART_IT_IDLE中断,是串口收到一帧数据(也可以叫做一包数据)后,发生的中断。
3 W3 k f4 h' {( F, U - if((tmp_flag != RESET))//tmp_flag不等于0
& ?5 j( L ?& U- m" {1 \- S; a - { 6 N5 e- I" t6 \. T3 I1 L1 M( s
- __HAL_UART_CLEAR_IDLEFLAG(huart); //tmp_flag=0;' Z3 n# T& b; A# t
- HAL_UART_DMAStop(huart); 0 A( ]; Q/ t/ g
- temp = hdma_usart2_rx.Instance->CNDTR; //cndtr为未传输的数据, l% x8 ^! X; \" B
- RX_Size = RX_LEN - temp; //总数减去未传输的,得到发送的数据长度
# @/ @$ o4 d% D5 ?, k - RX_flag=1;
6 q4 g" E/ G0 h/ T1 p: L - }
6 i# _* m, n0 B2 i+ O - }
复制代码
* r4 x3 j- P/ @: Q) Q$ p) }; Y
8 \$ p( Q1 E! r& J1 y2 }6 J* [! B( t在中断请求函数加入自定义的中断函数。4 j, H* i' p& @, k5 c5 i' k$ U" D3 v
5 v3 N+ G: e) x' {- void USART2_IRQHandler(void)//串口中断请求% M2 v- N/ {/ w9 |' {9 }1 Y
- {! B. P( E N; i% z: b
- /* USER CODE BEGIN USART2_IRQn 0 */; H/ c' w# C' K' T: R
- UsartReceive_IDLE(&huart2);
0 Y; @+ j, x' c9 i. V2 m - /* USER CODE END USART2_IRQn 0 */
7 R% g1 u% A8 s# i: c, C! d) l) @! p - HAL_UART_IRQHandler(&huart2);% I2 K+ u1 U9 z! I$ f5 u! S
* x8 l Z0 h" q/ ]3 U2 [- }
" S" S+ r6 E2 _7 F& S - static void MX_USART2_UART_Init(void)//在这个中间加入使能函数 切记切记
4 T3 e! S, o) V - {
/ l) K( I, A$ h; G0 P' b - /* USER CODE BEGIN USART2_Init 2 */
: t; U+ Y" c+ e - __HAL_UART_ENABLE_IT(&huart2, UART_IT_IDLE);//使能idle中断1 R" Z: K8 z B, ~5 w- f( y7 ~
- 3 g: C5 O/ _& N; P" C8 T. i
- /* USER CODE END USART2_Init 2 */0 T! O& i. V4 Z6 w' n; [4 X
- 2 l6 a5 w- t$ Y* ~% p
- }
复制代码 1 b$ ~# N [' n6 l5 l/ n9 n0 a
发送主函数 ^9 M/ d& o9 p6 q; g0 k6 V
- void uart_pc_com (UART_HandleTypeDef *huart)6 Z7 z3 _/ Y. f4 r2 e
- {
0 h9 u8 T7 V+ H2 w g - if(RX_flag ==1)
R \* c: B& r# ^3 ~" X) z y - {4 p3 D) t g! C9 w5 X7 E0 v
- printf("rx_len=%d\r\n",RX_Size);//\r是回车符,\n是换行符
- S7 l' n X" s" J G5 O. M' ` - HAL_UART_Transmit(&huart ,RX_pData, RX_Size,200);
8 o/ Z" L" F# a1 t - for(uint8_t i=0;i<RX_Size;i++)
* |+ H* Q! |7 n7 [9 u - {* A; c# {0 K1 Y6 l0 n
- RX_pData<i>=0;//清除缓存
7 P) v, G$ |1 I5 }/ ] - </i> }/ D0 x" S0 Y% r! ` ?8 k
- RX_Size=0;0 Z' A7 c; ]" v4 O
- RX_flag=0;//清除结束标志位
* B5 q8 f9 r- r1 T6 W( ~: Z - }& t2 X D) a6 \( n# \% m
- HAL_UART_Receive_DMA(huart,RX_pData,RX_LEN );//打开dma数据存入rx——RX_pData1 K6 r) J& n _' u- b2 Z, Z9 u
- //加在 MX_USART2_UART_Init中 pc串口助手只能接受到一次数据 只有处于main函数while循环才能一直接受发送3 I0 S/ }, w- W6 D0 W, [7 o6 _
- }
复制代码 # M) d: K$ F( `4 G: O
在main函数的while循环里面加入此函数,并注明使用的uart。# G" ?% U; q* R1 n, b! J4 R2 T5 s4 ~
2 U" E# V- V! w' I
对于程序的理解# d2 @, {+ A+ S* E* l
首先uart接受数据,触发idle中断,进入中断请求函数,获取接受到的数据以及数据长度,并将标志位置1,进入主函数输出到串口助手, p2 f( k* S! T8 S. t, w
' ^4 O0 e! S" L串口助手效果; [& _" w3 V, C# V2 y5 t/ p- Y7 I# f# |
3 v6 ~3 `+ Q9 H9 j0 M* U基本可以实现本次的目的,首次写,表达不够清楚,望理解。3 l& y! U5 ~3 `3 Z- K0 t
: S8 P2 U9 P! g Z3 R% w9 ^6 R
* J1 C7 i \ r3 I* w0 b8 Z' O
" _% v v! U9 I8 E7 K# c
|