概述
Y+ d/ T/ |. O4 }直接存储器访问(DMA)用于在外设与存储器之间以及存储器与存储器之间提供高速数据传输。可以在无需任何CPU操作的情况下通过DMA快速传输传输。这样节省的CPU资源可供其它操作使用。+ |" y5 A$ T6 }
DMA允许在后台执行数据传输,无需Cortex-MO处理器干预。在此操作过程中,主处理器可以执行其它任务,仅当整个数据块需要处理时,才会中断主处理器。这样即使传输大量数据也不会对系统性能造成太大影响。
& `. O) D3 t2 ~! {) FDMA主要用于为不同的外设模块实现集中数据缓冲存储(通常在系统SRAM中)。与分布式解决方案(其中每个外设都需要实现自己的本地数据存储)相比,DMA解决方案在硅片成本和功耗方面的成本较低。
- ^5 W) y& L" K) p) F& N7 w8 _5 Y根据使用的产品型号的不同,有一个或两个DMA模块。
/ T0 k9 q9 F% m: e1 J9 pSTM32F0XX DMA控制器总共有5个通道用于DMA1,每个通道都专门管理来自一个或多个外设的存储器访问请求。它具有一个仲裁器,用于处理不同的DMA请求的优先级。/ ~0 y1 t& j+ V) H q
本篇文章主要介绍如何使用STM32CubeMX实现ADC在DMA下快速采集,并且打印出去。: n8 \8 l# H2 c% S. B9 ]. @
6 N3 b: `+ |. I* b2 K
$ u" G, e$ [+ a% Q% Y$ B: F C' L# P0 q' Q: O
硬件准备. Z% D3 ~5 ?% Q
首先需要准备一个开发板,这里我准备的是NUCLEO-F030R8的开发板:2 ?/ r2 s( J+ Y& I) `% e
) J+ j* n7 U% H2 u' N
" J; k; R/ Q5 X+ ]+ s3 Q, l, [
( S6 O+ o) p# G) Q选择芯片型号 ~3 Z3 _6 Y0 e1 x; O
使用STM32CUBEMX选择芯片stm32f030r8,如下所示:
6 `4 I& P# r8 w0 x! Z" z, @6 ^: v+ q
% i0 p$ r5 O+ l9 a! z/ X" @
7 k. S! U5 s) _4 x- ]( g0 p* a' |配置时钟源6 N( n2 e% f' u. A. w% a
HSE与LSE分别为外部高速时钟和低速时钟,在本文中使用内置的时钟源,故都选择Disable选项,如下所示:4 W+ K- E+ _: H+ S$ r5 q, B
3 h6 K5 V& y: A
% F" L* _3 D: ]+ ?
( E+ p: B0 d$ U) S2 x9 P0 d配置时钟树
; { p# ?5 n3 d( {STM32F0的最高主频到48M,所以配置48即可:1 t, J. `$ B0 }
2 x1 n7 M/ C& U: `6 ~8 Z8 m
; q" [: r7 e' k
; q! H! L% g% q4 f: H串口配置! z2 _6 i6 l) H/ Y+ K. y, w( [& X
本次实验使用的串口1进行串口通信,波特率配置为115200。
0 T' R0 |6 j+ Q) J1 Y2 R# C1 S
/ |8 z8 K+ _# f1 @$ T
) Z, Q0 e+ `2 q% ?
! L# w) j2 c3 R r, w0 @ADC配置
/ a" W- i4 }1 \: `' eSTM32f030中,有一个ADC(模拟/数字转换器),每个 ADC 有 12 位、 10 位、 8 位和 6 位可选,每个ADC有16个外部通道、2个内部通道和一个VBAT 通道的信号。' n' Z9 C0 F* \4 G
本文将开ADC的IN0、IN1、IN4一共三个通道,来分别读取ADC,由于串口2和IN2,IN3复用,故不使用IN2、IN3。配置如下:
$ \+ t/ \! _4 t; [8 A1 `
y' B* R- M' Z% O2 ?
1 u$ f6 R, N2 }4 R. q9 A* b
9 b# }& O, r- b3 W8 ?! [开启DMA:0 J1 W# l/ u$ O/ J) [4 x2 Z. g
2 j3 o1 H+ E2 A$ c/ L' h) B, }& w
# I A- [( X5 e- m3 h7 b9 o5 f: j8 Z, A
打开中断: [: l0 g. _1 S2 Y
, ]6 r0 H1 O" c' `) G: U$ x" b2 f: T5 w
e$ N5 J0 W6 u7 N- U4 w3 b8 r& E3 i4 P# B u6 k
生成工程设置2 p/ i2 B3 }! l6 w+ I0 |# ~7 t
最后设置生成独立的初始化文件:; v: ?2 i: ~& |! `7 [
, X) @6 n) {' B, J l! D
; B# k1 d& J& y a9 u2 N
, q/ @8 V9 ^) n, P生成代码( w, {% y1 ^$ L% F) z
; m4 c C2 L) d0 o0 {6 Y% \2 }
5 ?; X1 x; C) l! |/ g4 ]
8 J+ P$ e; s: X3 [2 c配置keil) b" J2 S" f+ P: y/ t; y/ s- w
; L( K( {5 Q( T: ^" F0 m
) G3 E3 G1 D! j* i. i V4 W9 H. v" ~
代码" K G! z- m6 B3 y% h
在main.c中,添加头文件,若不添加会出现 identifier “FILE” is undefined报错。
& R8 I; G8 v+ c0 \3 Y* g
5 \7 ~' E" f1 A! U' h- /* USER CODE BEGIN Includes */# D% W# [# K+ b0 d( `
- #include "stdio.h"( w ^# G6 l9 D F, s3 ?- K3 C
- /* USER CODE END Includes */
复制代码
4 E5 \2 y O( v* a/ c函数声明和串口重定向:" l0 y$ g) E' R4 f8 Q
. c2 t* G( R( K# @3 {, d+ I- /* USER CODE BEGIN PFP */
" x& r& Y& _" n1 P# @9 ]- y - #ifdef __GNUC__
1 m, l% h( G9 u) m% d" R }. D, W& a, y - #define PUTCHAR_PROTOTYPE int __io_putchar(int ch)* x) k0 ]3 y/ G, R/ E
- #else
. k' t/ A' {6 ?+ l: E' V( `8 n+ N - #define PUTCHAR_PROTOTYPE int fputc(int ch, FILE *f)
; O( v3 v, T: k( K4 C: N, ?* u - #endif /* __GNUC__ */
& N: L$ ^, L- [. r - /* USER CODE END PFP */
复制代码- /* USER CODE BEGIN 0 */
( _( n* ]" `! Z% s - PUTCHAR_PROTOTYPE6 g u/ n) s5 H' b( x
- {$ X' N( m. ^! q* w7 @
- HAL_UART_Transmit(&huart1 , (uint8_t *)&ch, 1, 0xFFFF);8 C( v: f, |; Q6 a
- return ch;
5 e1 _' G4 }1 h - }
+ y: a ?: f3 d' ^: n/ R( U - /* USER CODE END 0 */
复制代码 0 A' W5 G, f. T- h7 _4 M
变量定义:# _. p! {# e- i) |( g" |
% @3 B. U) r: ?+ r) ~
- /* USER CODE BEGIN 1 */+ p; A P ?0 |7 I, C
- uint32_t ADC_1, ADC_2,ADC_3;
; B: ]) J5 b7 v. y1 n# h, M - //各采样30次,故30*3为90! y- O* C0 r: {- Z% c
- uint32_t ADC_Value[90];5 J2 e$ i- L1 M5 L7 H$ k2 u
- uint8_t i;
- G. H1 P" K2 D% o Z7 F - /* USER CODE END 1 */
复制代码
( v# ^1 J* X8 J& w3 v2 c( q使用DMA传输:
# r5 j) k, `1 h' D4 x4 q- i1 I# y. h/ r# A* c) t* a4 h
- /* USER CODE BEGIN 2 */
A7 R; Q/ ~+ m7 Z - HAL_ADC_Start_DMA(&hadc,(uint32_t*)&ADC_Value,90); //使用DMA传输7 R( D& b- F* ]. \
- /* USER CODE END 2 */
复制代码 0 b* J/ F) b6 m0 W0 Y5 `8 r/ Z
主循环:
) l. F( Y ]2 S& M7 ^5 Y
& P+ I- _) b: @9 s$ Q7 p- /* USER CODE BEGIN WHILE */
; Z! y% O2 k8 j - while (1)9 B, C9 B! y. \- e' X" d
- {% O# m3 W5 e; v+ r3 `4 s$ l9 G
- //放个延迟,防止程序运行第一次读出数据有误5 O+ A* a5 l- H0 h
- HAL_Delay(100);0 x, p+ V3 Y7 ~3 `
- ADC_1=0;8 T) X1 C" F5 M* I6 o
- ADC_2=0;
4 F" J3 E6 m+ _/ V2 _% I - ADC_3=0;) s. u5 c& x) Z
- for(i=0,ADC_1=0,ADC_2=0;i<90;)& I3 @1 }# u8 C3 N9 q8 k
- {
4 B6 S8 ~9 l8 G' X - ADC_1+=ADC_Value[i++];
3 Y& l$ h. X: Q" b2 T - ADC_2+=ADC_Value[i++];
6 X3 t$ s! |: A# [* r( } - ADC_3+=ADC_Value[i++];
0 z r8 g1 H9 W& U+ u( G - }
4 E1 f2 k4 Z6 d" \! `, `$ n - printf("ADC数据如下\n");6 R* r" u: x5 [5 ^' J( B1 y
- //除以30为求30次平均ADC值,乘以3.3为以3.3电压为基准,除以4096为ADC配置为12位4 H+ x$ r @+ ]% G3 [3 n% q
- ( `$ w, x4 M5 M7 v9 g
- printf("ADC_IN0=%1.4f\r\n",ADC_1/30*3.3f/4096);
( D& A# l. M/ s2 ` - printf("ADC_IN1=%1.4f\r\n",ADC_2/30*3.3f/4096);$ M! p1 Z" C9 J2 e! I6 e+ ]
- printf("ADC_IN2=%1.4f\r\n",ADC_3/30*3.3f/4096);' I. h6 y+ V! O' j* j
- /* USER CODE END WHILE */
& d! ]; O# I7 D; r$ b; ]5 ~# k5 J
], p/ G) E$ m9 t+ _: |' E E E- 3 n9 j9 X; X) w1 n2 x0 Y
- /* USER CODE BEGIN 3 */- o+ u: @6 y, k7 O- p
- HAL_Delay(900);
2 s1 Z" m8 E: T& r" O4 z - }
% {8 N1 Y1 j7 v7 J; f- a# n - /* USER CODE END 3 */
复制代码
3 G; C' D1 m9 j. f: K演示效果
. p' S! s. F# a; E& p$ } L5 E设定adc1口接3.3V,adc2口接GND,adc口接1.5V,输出如下。
. Y i p6 @( H0 W# \# s" X. U+ |' L" B; M' J& i
r' ]1 W! ~; C$ V, V( @
4 P) E A7 h) Y0 E
; U+ Y. ]- Z" I$ b
|