一、开发板平台简介:! ]0 a6 o+ l% g8 b2 g
1、开发板资源简介4 J' G2 N/ P$ _5 @! m( Q
(1)开发板主芯片型号:STM32L431RCT6
$ y6 C% c5 ]6 s# ]' R- e(2)开发板主芯片封装:LQFP-64_10x10x05P
- k, V1 I7 z' _/ N ]3 Y(3)开发板主芯片内核:ARM® Cortex®-M4
6 t. D+ d7 n7 k+ l(4)开发板主芯片主频:80MHz9 N N7 _ O& b" L+ ~0 Q) ]
(5)开发板主芯片Flash大小:256KB" y% @$ A; [8 m- T
(6)开发板主芯片RAM大小:64KB
. y) w% p; n. u7 T3 J/ M4 J
8 c# U, H( P6 d+ i3 K) u5 Z7 P) v
4 [5 u8 u* G; s# G* S7 _4 H
7 g5 N8 \3 ?7 ]6 r
1 a6 b) e1 L- U. S9 Y. j; d1 Z
) G3 w% Y6 N1 x$ Z+ p# B/ R& x3 |
2、LED灯资源
6 {7 T1 ^: q/ ](1) STM32L431RCT6开发板共5个LED灯资源,其中一个红色LED为系统指示灯,指示开发板供电系统是否正常,如供电系统正常,红色LED为上电常亮状态,硬件原理图如下图所示:8 D) `7 ^% G2 Y( ?7 Z$ {% y
$ f" H% H, F) w% J1 s! \3 G- w
# A7 [( P) G( l/ D( p2 f# ^
# T. Z! \( G/ U) E5 v( ~) a& a$ e' B3 @, m; |; q
(2)其他四个LED灯为黄绿色可控LED,高电平点亮、低电平熄灭,计划用LED常亮验证看门狗的作用,硬件原理图如下图所示:/ n T! B, T- e2 Q
1 i) K# [- [% x/ G
7 f5 E4 M- K3 C4 t j
g+ ?& m% \! c ?, R: t3、串口DMA收发工作原理
" }( @+ K4 c3 g1 m/ A 串口全称为串行通讯接口,即数据在通信线上一次传输一位,按先后一定顺序传输。我们通常所说的单片机串口准确来说应该是串行异步收发传输器(Universal Asynchronous Receiver/Transmitter,UART),使用TTL电平,串口需要RXD、TXD、GND三根线进行通信。9 c% F1 g* B2 Z9 u) E$ [: h: D
- e; i3 J! A- \6 `9 N
(1)我们选用的STM32L431RCT6开发板串口1已通过USB转TLL串口芯片CH340G引出,使用时,只需要用公对公USB线连接电脑即可(注意也得需要安装CH340G驱动),后期验证试验也可使用该串口作为debug串口。) x8 @* `$ A7 \2 g5 r4 v
- U/ A% F0 Y1 _7 s+ L7 p$ V" J
(2)开发板上的其他串口已通过排针引出,为TTL电平,通信的时候需要注意选择对应的电平模块,如USB转TTL串口模块等。
3 U9 Y9 _7 e2 q0 v; f
; E t3 p% ?8 Q6 ?2 b5 l( N, {+ |TTL转CH340串口,硬件原理图如下所示:
9 y) r+ b/ a2 _. x6 U% M3 ^" w( N1 i- a6 u
+ _$ h/ d5 X, v' C) h9 D& [$ q# C
DMA(Direct Memory Access) :直接存储器存取,是单片机的一个外设,它的主要功能是用来搬数据,但是不需要占用 CPU,即在传输数据的时候, CPU 可以干其他的事情,好像是多线程一样。数据传输支持从外设到存储器或者存储器到存储器,这里的存储器可以是 SRAM 或者是 FLASH。
/ @+ @8 l7 z+ k C0 _" j
' }5 q+ M$ U, h, I. sDMA 控制器包含了 DMA1 和 DMA2,其中 DMA1 有 7 个通道, DMA2 有 5 个通道,这里的通道可以理解为传输数据的一种管道。要注意的是 DMA2 只存在于大容量的单片机中。6 A. N" G0 m @' Z) @4 D
5 g8 w9 Q# R; y0 u2 D
/ m8 ]8 Y1 ]2 b# n' W
1 ~2 F c8 ~% _, b 二、串口DMA收发实验过程
& S5 k+ @0 z F! T* W1、新建STM32CubeMX基础工程
& h: e5 `0 X! H# H' F0 @(1)打开STM32CubeMX,点击“File”-->"New Project"
( ] C7 X4 ~5 J2 o9 {( ]- ~$ R2 f! d g9 c/ q( o0 d9 s; {
, ~( Q4 D+ i! o+ H
; v* N7 y6 ^2 R& O* O) h(2)等待打开主芯片选项界面(大约1分钟时间)。
' q# }: U- \" ~9 ?- w% I9 l* @2 q0 \& P* e$ S( K
+ U% }5 e; D6 O/ u- M; V! h. U7 I# R0 ^
(3)昨天搜索框中输入(或选择)所需的主芯片型号(因为我们用的是STM32L431RCT6开发板,所以此处选择STM32L431RC),然后在右下角选择STM32L431RCTx(因为开发板主芯片是STM32L431RCT6),左键双击即可打开新建的项目。1 t/ F) F1 t$ ^7 o0 q( p1 a& I' G
: i/ x, g2 N8 B9 z0 A4 B. W7 V& c
% n0 Z1 V: b. X
2 t5 `) m e/ w4 C(4)选择时钟源。. f+ ^) H0 D4 V$ T+ Y
8 A# K2 u% c5 `# ^
因为开发板上有8M外部时钟,硬件原理图如下所示,所以此处选择使用外部高速时钟(HSE)。
* y( T. M! c/ K
( H* D7 l$ l8 p1 \. B# b
( k, }0 I. A9 f2 B" g$ x* y& |6 j0 i1 E: J9 [7 ^$ I# x- `+ i8 T, {! R* P
/ R9 X' o; F6 B7 ?
因为我们没有用到外部低速时钟(LSE),此处不做处理,如下图所示。
6 J' C, E; R7 }$ t# Y3 B% A& A3 W/ z, A% z5 T" @
" `9 k2 l5 y1 d4 E/ P+ K
- Z! Q6 C" i. q" N2、配置GPIO控制LED+ d& @" g3 ?; F0 s
(1)查开发板原理图得,LED1、LED2、LED3、LED4的控制引脚分别为:
5 j9 M j4 C5 ?1 L6 q6 K6 MLED1——PC0: d! \! Z6 ^- L$ }6 N" f( ^
LED2——PC1
; M" f+ y1 |2 \LED3——PC25 O/ }0 O! v( ~: K6 a. X8 v
LED4——PC3
; O8 ?4 E# x m- R/ n7 P7 t7 y8 q
/ w2 V7 R' D( }, L8 w8 C6 b/ e, @(2)配置LED的控制引脚为输出,输出频率、输出方式默认即可。
. z/ f0 \- [5 C/ x& H |# D5 E) j5 e4 F% V0 _1 ~
鼠标左键点击PC0,选择“GPIO_Output”,表示设置该引脚为输出模式
% q, t1 X0 W6 v# r Z鼠标左键点击PC1,选择“GPIO_Output”,表示设置该引脚为输出模式。
+ C+ j" W8 k' y$ ^. {& M鼠标左键点击PC2,选择“GPIO_Output”,表示设置该引脚为输出模式。
$ K# Z3 |3 L2 ^! {鼠标左键点击PC3,选择“GPIO_Output”,表示设置该引脚为输出模式。* n2 a; l% Z4 _- I6 t) I \
b: s8 {/ |# L4 X& c. @
0 x6 K1 ^- ^/ v% ?
3 A) F# c( I o) B4 c
' t+ k1 u' v p' R+ @& g' _6 |' W" }3 c) @8 w
(3)也根据自己的需求配置GPIO的参数,如输出方式、输出频率、上拉下拉等。因为GPIO控制LED的要求比较低,此处采用默认参数即可,不用修改。/ P0 F: |$ F! O6 }# T0 [/ ?
" [4 i* ]- z8 |1 i% ~) }8 B' M
' {: F0 y& W' i1 O+ `
" O- _) d- D0 n. i3、配置PA9、PA10为串口' L3 h6 L) h9 l# w& H7 X5 y
(1)查原理图得知,串口1使用STM32L431RCT6引脚为PA9-USART1_TX,PA10-USART1_RX,引脚设置如下:
' U. f. t( N" D# V3 l) T8 P. H
: L1 B& l+ R! |5 t" N) A
: D! r! h7 v+ [0 ~4 H8 Q4 {3 D% b( s$ T' k
序号1用来设置串口收发引脚的选择。
4 b& R% v+ ~# b4 J% s序号2-3-4-5-6设置串口参数,如波特率115200、8位、NONE无奇偶校验等。
1 c" r: [+ K8 a8 }/ y* _
* m) r! `# s5 d, v8 ^. \(2)设置NVIC settings 使能接收中断6 z5 i) V5 C3 C1 b9 j6 P
8 L/ H: ~0 i' \5 g8 `6 C7 _
7 P* r# W' S3 A
/ o9 d3 f! o1 X- F! t3、串口DMA设置 " Q4 B" q( O* Y) J/ T! B# B
3 K- ?. i8 }, N0 l" ^
% o1 }; X5 n8 T6 Y D$ z P7 I
: e, V! J2 j) K5 ?(1)根据DMA1通道预览可以得出,我们用的串口1的TX、RX分别为通道4、通道5:
) \% R" i0 _* D6 ]* |点击DMASettings 点击 Add 添加通道# A$ c* p6 x- h# m
选择USART_RX USART_TX 传输速率设置为中速
' F& h9 S/ T1 V9 M) E: T0 R8 \7 ADMA传输模式为正常模式,即发送一次就结束。6 D; m2 m) Q) m: H+ Q% k3 r
DMA内存地址自增,每次增加一个Byte(字节)9 m" \8 z/ K% s3 u g+ e, v: j
( s+ E, r+ r2 f/ P$ `* y
* t( y, b6 E& d1 F
* Z6 P+ @- h3 r# L( L* }
(2)DMA相关参数解析
5 G p" S& y3 e- KDirction : DMA传输方向有四类
) v# b# O1 S! F% V外设到内存 Peripheral To Memory. I9 ]2 j1 t7 {1 o, f: R( Z5 O
内存到外设 Memory To Peripheral
- V2 {: J& ^% C Y4 A内存到内存 Memory To Memory% H( w& R$ N: K0 W R) @
外设到外设 Peripheral To Peripheral& @8 Z, x7 G4 w4 X1 G
2 A* E8 g& e' k/ E+ P0 Y
Priority: DMA通信传输速度有四类0 f% X1 T |1 o% v6 }, ]' N( S8 ~
最高优先级 Very Hight
) u$ j' Z5 H' c/ R高优先级 Hight
7 _5 P1 \ \8 y) `+ `& s中等优先级 Medium
8 }# ^' a. p5 R4 \低优先级;Low$ C# a8 y5 r0 z
( k$ L; q, K9 T
Mode:DMA传输模式有两类
- t. p1 V; k- _# }3 q& u1 \1 ~/ r. @
, v* B* y( c2 |9 e& i) c! q
: L) h( l4 J5 b0 [5 Z' e9 MNormal正常模式:当一次DMA数据传输完后,停止DMA传送 ,也就是只传输一次。, h7 Q. B+ m: o, n
Circle循环模式:传输完成后又重新开始继续传输,不断循环永不停止。
" @6 u1 f- n% J( ]6 L. _8 ^% b9 M3 V: P& x6 j
Increment Address-DMA指针递增设置:$ y0 M ^% _( s9 X4 M
4 S$ Y! p7 {1 [0 ?1 _5 l' \
' }! E$ P4 D: D. x& ~% g
4 O# Q4 d0 R9 X' s. e; D; @0 a3 g: m, i& q ] T
4、配置项目工程参数
# g* S/ Z" w6 F9 w6 o+ n. ](1)配置时钟树,用于系统内部时钟,以及各个外设时钟等。此处选择外部8M晶振作为主时钟频率,内部最大倍频80MHz。7 h( u. q( C' R/ ~- |8 o
* q* p; X5 ~" B; R
/ H) ]% b- U* \9 a) [& H
- s( t7 }% L) s( T1 |(2)完成配置工程。2 ?! M8 u, q$ R3 P/ a' g, V
备注:需要注意代码生成过程中的继承关系,如图所示:需要保留开发者自己编写的代码时,请根据配置设置,不然生成代码后会删除自己编写的代码(从这个方面也可以看出开发者备份自己的代码是多么的重要。) u- _' w; i% e: _9 W2 q
" k; V: K5 u+ L- l& u$ y7 N" h% ~
' w, F" n# r+ v G. a$ I5 J- n8 ~6 x5 O" u$ |
9 e+ ?! ~% \% m; D5 I' P1 E5 A+ F( |0 w$ h1 X) w$ y' m
(3)生成代码。: N- g# S# N7 T* i. w+ X
# B6 n- w% ]& M* G# {
备注:使用Generate CODE生成工程代码前,请确保文件路径无中文,否则会生成项目失败。9 J' q/ V$ g8 B, m" U8 f/ L$ c
7 m! T. D5 R( S% g
) E3 E+ e7 {4 U( l6 h/ |
6 @5 M* |- `" h4 D( s2 j(4)工程代码生成成功。; E5 w/ m7 @4 T! w: q+ ^* H; s0 i1 t9 a
5 D6 D- F. Z& @' v, {4 Q
" M8 j) m# ^7 z5 V4 j5 g7 D- O
, \) q6 |, S" h- X l R6 M4 i' _
三、在KEIL 5中编写代码
% V/ W* I) `) ]2 f1、使用KEIL 5(MDK)打开项目工程文件
" x! k0 O* p' N8 r/ M+ q) b源码使用说明:使用前必须把项目工程复制到无中文路径的文件夹下使用。1 E5 z5 ^0 S* m- w
(1)找到刚才新建工程的存储路径,安装项目名称,打开项目工程.uvprojx。1 \9 N/ x' z0 k* u
& D7 l8 \: G# j( ]6 {1 y" `
) a! l$ e* S" W# B4 V. d( {* X& M6 j* X% R1 C
2、添加LED指示灯作为系统提示, s; g5 L& |0 I6 |1 @5 d5 q2 ^0 f
添加每隔100ms,LED1、LED2、LED3、LED4闪烁一次的系统提示,用于提示程序运行正常。. p+ ?3 C9 J" y& p: `: _( Y$ X
& A, h9 a( X, I
3 V/ R/ Z( L( z
) @7 U6 i9 Y0 ]
3、添加HAL库UART DMA 发送函数代码" Z1 ]9 i6 V0 I3 R% k5 J
(1)STM32 HAL函数库的串口DMA相关函数如下:, e$ c9 o9 F( f
- HAL_UART_Transmit();串口发送数据,使用超时管理机制! h% z* i. p: M1 B7 i: m
- HAL_UART_Receive();串口接收数据,使用超时管理机制
: G1 g. ?, X) S1 L1 Q, k2 W% N - HAL_UART_Transmit_IT();串口中断模式发送( M. b! M6 \0 P0 B# y
- HAL_UART_Receive_IT();串口中断模式接收4 J" h7 m Z! R6 F' R& `; ^
- HAL_UART_Transmit_DMA();串口DMA模式发送2 g5 e( J7 Y0 e
- HAL_UART_Transmit_DMA();串口DMA模式接收/ N; B: G. ?! O$ l: O
- HAL_UART_DMAPause() 暂停串口DMA/ X5 a! v, G) I% B0 Y/ C$ z2 V5 T
- HAL_UART_DMAResume(); 恢复串口DMA( `0 n0 H, S2 ]+ g7 W: h& L
- HAL_UART_DMAStop(); 结束串口DMA
复制代码
4 {- Z' p) W6 |$ `( @ Y(2)main.c添加串口DMA发送函数相关代码:' |3 x L, @* x: P' m/ ~5 E* K
- /* Private user code ---------------------------------------------------------*/
c g: Y/ z$ z) x - /* USER CODE BEGIN 0 */
+ P" V+ ^; t8 O4 G! Y& c - uart_rx_struct uart_rx_struct_t={0}; //串口接收数组% e" j% j0 J5 e$ v) h# j8 v
- /* USER CODE END 0 */
复制代码- int main(void)! d4 K' u8 Z7 i
- {: H8 ?2 H6 ~/ o4 ^7 C$ t3 w
- /* USER CODE BEGIN 1 */: K; \; j' G2 R2 ]
- uint8_t transmit_str[50]= {"hello world,this is uart dma function!\r\n"};
! X. G& w$ m! k# ^ - /* USER CODE END 1 */6 U: N5 t/ g9 b" W
-
, q& k% q- h4 X% V B6 V% P" s - /* MCU Configuration--------------------------------------------------------*/+ t8 l4 r! _8 l$ W" r; J |! c
-
- z% ^* G+ L% j, y% W% _# c7 x - /* Reset of all peripherals, Initializes the Flash interface and the Systick. */0 K& z1 v' r" I. H8 b
- HAL_Init();
, ~, q/ `8 g* A7 o/ l, }' d -
3 m: i7 X* R& ? - /* USER CODE BEGIN Init */
7 B* ]: ]4 \% }% Z _. L -
8 f( O6 g1 s- ]6 ~ - /* USER CODE END Init */+ Z' p3 q! d1 G6 L
-
6 H/ E7 s, E4 D: T: K9 X - /* Configure the system clock */
9 C. ]# g3 F! M# y$ Z# d- q - SystemClock_Config();% [/ G- l8 v: L5 K6 b
-
( o, }4 B2 t# P5 |* O+ P5 h9 x - /* USER CODE BEGIN SysInit */ o2 J& ?- t' {3 R
- , z! A2 z* f# b& u+ Y
- /* USER CODE END SysInit */
0 g+ f. k1 m& O( d7 w: h - $ g( R0 ^* ?) U+ {2 h
- /* Initialize all configured peripherals */+ F; h9 }% U$ p$ m
- MX_GPIO_Init();0 w0 q) I2 f+ j" O: t# u
- MX_DMA_Init();
; F6 I- c: W# S - MX_USART1_UART_Init();9 p9 ]& ]# y6 H7 I$ V
- /* USER CODE BEGIN 2 */
; n4 W6 G7 S: `. p; D/ n+ @ - HAL_GPIO_WritePin(GPIOC,GPIO_PIN_0,GPIO_PIN_SET); //初始化LED灯,默认点亮
( z! R7 C7 q' Y9 S L6 ^, F - HAL_GPIO_WritePin(GPIOC,GPIO_PIN_1,GPIO_PIN_SET); //初始化LED灯,默认点亮
: S: s8 o) P3 o/ [8 V' i9 k4 U - HAL_GPIO_WritePin(GPIOC,GPIO_PIN_2,GPIO_PIN_SET); //初始化LED灯,默认点亮! `) t9 |& i6 t& e
- HAL_GPIO_WritePin(GPIOC,GPIO_PIN_3,GPIO_PIN_SET); //初始化LED灯,默认点亮' l2 X; [9 h8 ]% X l# C3 T5 J
- HAL_UART_Transmit_DMA(&huart1, (uint8_t*)transmit_str, sizeof(transmit_str)); // + x1 q! J1 ^" g/ t( ?! P8 N9 ?/ H
- HAL_Delay(5);: d1 Q$ @; q: t
- HAL_UART_Receive_DMA(&huart1, uart_rx_struct_t.uart_rx_buf, UART_RX_LEN); // 启动DMA接收# P, M) \* s' G9 | x: s% P& {: z
- __HAL_UART_ENABLE_IT(&huart1, UART_IT_IDLE); // 使能空闲中断
- e- H" m) F2 Y* P2 }% ?7 R - 0 e& j5 ^. k) z
- /* USER CODE END 2 */
2 j) N3 V5 v! v; |4 h -
1 r5 k6 F: L) K& P* j - /* Infinite loop */8 g7 K3 d% H2 n; u; I# n& r5 R
- /* USER CODE BEGIN WHILE */8 T! C0 i1 H7 R. Z
- while (1)
$ M4 L# I. u8 e, \% n - {2 r* i; ^! P8 p. _
- /* USER CODE END WHILE */
# U& z% j& M$ B/ U -
; Q _9 ~ N' R8 Z: T @ - /* USER CODE BEGIN 3 */$ }* A; I1 d0 e/ ?$ J8 a
- HAL_Delay(100);. Y D3 L8 d8 b& P
- HAL_GPIO_TogglePin(GPIOC,GPIO_PIN_0); //反转LED点亮熄灭切换
1 a" A( _3 F* G& J! {" K! @# y - HAL_GPIO_TogglePin(GPIOC,GPIO_PIN_1); //反转LED点亮熄灭切换
* S* T# \ `3 m& u1 a0 V - HAL_GPIO_TogglePin(GPIOC,GPIO_PIN_2); //反转LED点亮熄灭切换
! X. V3 A" n+ f8 s; K* W/ k A - HAL_GPIO_TogglePin(GPIOC,GPIO_PIN_3); //反转LED点亮熄灭切换5 p0 m+ N8 w6 ^1 ?3 y
- if(uart_rx_struct_t.uart_rx_flag & 0X80) //DMA接收完成! N4 K' [ A* r, X1 [- X" ^% z
- {
4 v; Q6 m1 T7 l4 X) q/ F y1 G - HAL_UART_Transmit_DMA(&huart1, uart_rx_struct_t.uart_rx_buf, uart_rx_struct_t.uart_len); // 将接收到的数据发送回去4 a# a9 w8 o) u, O' H: V
- uart_rx_struct_t.uart_rx_flag = 0;" m9 x6 N9 B& R4 y& E3 m
- }
, Y. ]6 d/ U6 x9 }, L8 L: c -
8 G, D0 Q# l- N5 {8 m+ a5 P* a - }
6 F9 L9 B, E$ ~6 W3 r. U$ Z3 T$ H - /* USER CODE END 3 */
$ |1 k. g, ]/ e5 q- e8 Z - }
复制代码 $ X6 b0 \+ x) U
(3)main.h相关变量声明+ N8 L: T; Y* |. g" U0 d
- /* USER CODE BEGIN ET */0 ^& \1 b- D" W8 h
- #define UART_RX_LEN 1024 // 一次最大接收的数据量
# `! c1 c, H4 w% E; X E - typedef struct
{6 b; k+ L7 ^0 R/ |. h& d - {
8 R% M+ Q% r5 f3 f - uint8_t uart_rx_buf[UART_RX_LEN]; // DMA数据接收缓存数组9 e' r6 x" e' q" M
- uint16_t uart_len; //DMA数据接收长度/ n) T1 v3 n9 W& q
- uint8_t uart_rx_flag; //接收完成标志位+ ~- a9 B! I8 } [% f4 [; |
- }uart_rx_struct;
+ g3 P2 K7 h; f; w0 @ - extern uart_rx_struct uart_rx_struct_t;
- B3 ~3 z! \3 [5 V' |/ @1 T - ! K6 |+ J; B& V5 ]' o* c
- /* USER CODE END ET */
复制代码
; ?! Z# t# t! ?/ |" B4 W4、添加HAL库UART DMA 接收函数代码3 ]' M d s5 e* C& ?
(1)DMA接收设置前铺垫知识点:
: q$ c1 b+ R' y& @0 ^ STM32的IDLE的中断产生条件:在串口无数据接收的情况下,不会产生,当清除IDLE标志位后,必须有接收到第一个数据后,才开始触发,一但接收的数据断流,没有接收到数据,即产生IDLE中断停止。. f2 B! ~, h9 `7 S
4 n7 o% l4 p8 @* U* H(2)添加DMA接收处理代码0 L& v! e! c( A4 ]
( ]1 e# b& Q6 jstm32f1xx_it.c中添加代码如下:
3 | U9 T) j& b- void USART1_IRQHandler(void), `0 f4 ]% Z) s0 s
- {
9 [4 u$ G/ p1 P7 Z/ m' y3 Z - /* USER CODE BEGIN USART1_IRQn 0 */
* p, a5 x6 S4 i - if(__HAL_UART_GET_FLAG(&huart1, UART_FLAG_IDLE) != RESET) // 空闲中断标记被置位
1 j+ \* Y7 s1 t% j% ^+ A - {
8 p+ [- ^ y0 L1 }/ K9 i - __HAL_UART_CLEAR_IDLEFLAG(&huart1); // 清除中断标记& Y' t* o. p, {0 `6 d0 |
- HAL_UART_DMAStop(&huart1); // 停止DMA接收' W5 }" Z ^" E& B+ v% b1 S
- uart_rx_struct_t.uart_len = UART_RX_LEN - __HAL_DMA_GET_COUNTER(huart1.hdmarx); // 总数据量减去未接收到的数据量为已经接收到的数据量' I* Q" f0 [1 d/ ^
- uart_rx_struct_t.uart_rx_buf[uart_rx_struct_t.uart_len] = 0; // 添加结束符& O8 Z. G. A+ @- o& q* g' F8 _
- uart_rx_struct_t.uart_rx_flag |= 0x80; // 标记接收结束
! Q8 v7 {- J) S& I' @( z2 |7 T- U - HAL_UART_Receive_DMA(&huart1, uart_rx_struct_t.uart_rx_buf, UART_RX_LEN); // 重新启动DMA接收
' r( l1 \- r7 g) I' f' H. G+ h - }
; z) [) F* i% u }9 N& e% s - /* USER CODE END USART1_IRQn 0 */
' {! { a. z- c# N - HAL_UART_IRQHandler(&huart1);
2 Q3 {# T) K3 `, h K! E - /* USER CODE BEGIN USART1_IRQn 1 */+ F8 v; x4 N3 M3 Y: G$ n& }
- ; H. N# T: y% m9 t, p6 i
- /* USER CODE END USART1_IRQn 1 */5 c, Z+ `* y+ C! `5 e2 y
- }
复制代码 + K! f# e4 Z g% U) e
(3)至此,串口DMA收发数据需要添加的代码已经完成。; {4 p6 W. Y# \6 o' W q; A
5 v% j8 {9 s) T; P* e
5、设置编程仿真下载模式: g! T$ q* {4 }
(1)选择Options for target ...>>Debug>>J-Link/J-JTRACE Cortex,点击Settings>>选择Port(SW),可以看到搜索成功SW Device,表示芯片可用,可以下载。' U( d2 x0 S& Q$ V5 p) v5 g! X
: L1 \ Y+ q" L) @
: b# N- @* C ]$ G0 E4 r* k. n0 O0 t
5 W, N4 ^# w8 ?: l( @" q
(2)点击编译,完成后提示“0 error(s),0 warning(s)”。
$ p; W% n$ S0 P! k1 T8 Y* I) m$ Z9 U$ i& P
! Q% `8 X+ w8 N) g& N, {
4 o- s* H9 c" ~0 T" T3 |9 D1 K(3)点击Download(或者快捷键F8),即可下载程序。" G0 S1 y8 u1 e7 f
. Q/ Z* E e4 ^1 f& F3 I q
, z8 C d6 s' V; L. y! a, M) y; P
" ^. f; u# f( E) W0 A7 d
3 w' J0 A. _3 E# t% a |' S(4) 如果下载程序后,没有看到LED1、LED2、LED3、LED4闪烁,可以按下述方式设置一下(Reset and run表示下载后自动复位和重启运行)。或者重新彻底断电再次上电(或按开发板的Reset按键复位MCU即可)。# _% [9 @4 t/ u) T5 H+ T7 Z7 E
+ X. V9 O' N6 y* p$ B
6 k: g5 e1 m l- U
0 l4 i. J3 g ^& z) L
6、串口DMA收发实验效果展示+ h8 t3 I, P, z5 S, X
程序烧录到开发板后,即可看到LED1、LED2、LED3、LED4初始化后每隔100ms闪烁一次,并且打开串口助手后(串口参数:波特率115200、N、8、1),通过串口助手可以看到,开发板接收到发送的数据转发出来。4 y. H: O* s C( p+ n8 n( |( S
% c, J. a% q: m6 C+ P9 U5 ~% s
! ?: G. p- p% a% A
/ P4 o2 s% H2 Q) e: |) g$ K
————————————————
' ^8 G R( I T' |1 h. ?( P版权声明:智能小屋ZYXC
/ ]) |3 z+ Q, P" c m
& K, _7 b) O& D) K* o; ?+ o: d7 J, h7 [; {( t; L/ }% Q
|