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