基于HAL库,实现最简单ADC采集。
! O) O2 o3 G( J% k4 l4 f第一步,使用CubeMX配置时钟。
8 |3 g. ~3 \2 X$ Y" k
3 [' \# d4 P+ f1 X' `
2 [% d, t: t3 z5 A+ E7 e
' v7 W& W, x1 z" d* \ ?/ \: l然后是adc的配置:8 W, O4 S& ]2 A! J
& V7 B6 y! l: `+ p) k- B+ |
* K: g* H1 X) A1 \! O8 E3 V/ X
. r/ [' T9 q( q: p; P
将,稍微修改一下风格,并手写头文件和源文件。
' d& K2 y, \/ @6 }& U/ q
: a# Y! d f# e2 ?adc.h很简单,就不说了。
2 ]9 ?& ?- F: b$ Y! [- #ifndef __ADC_H
4 E( {9 c/ W* E# k- x - #define __ADC_H7 O4 n" u/ _# y, w! D
- #include "sys.h"- {5 l {+ m9 N6 d
" |( W" t5 \; m& \- void MY_ADC_Init(void); //ADC通道初始化
! A! d Q) S+ I( j - u16 Get_Adc(void); //获得某个通道值 " \% x8 N$ i/ ~$ T/ r0 T' ~
- #endif
" P9 f6 ]# I, C/ S4 ?
复制代码 ; h) F0 b B2 D, d5 }: E6 s
其中,MY_ADC_Init和HAL_ADC_MspInit的内容完全是根据CubeMX生成的代码写的,就改了一个变量名字。Get_Adc则简单了,当我们需要获取adc的值的时候,先HAL_ADC_Start启动ADC,然后HAL_ADC_PollForConversion等待转换完成,最后HAL_ADC_GetValue返回结果。
, P0 f- T2 Q/ s- #include "adc.h"( K: y- q) M# Q; t1 Q7 I. S
- #include "delay.h") b% k& Y% s% H" d6 q
- & x# R- F+ Q" C! E0 F3 n
- 0 C: q: x; E' m
- ADC_HandleTypeDef ADC1_Handler; //ADC句柄
+ ~5 S3 e1 J* }' H - ADC_ChannelConfTypeDef ADC1_ChanConf; //ADC通道配置句柄! U. N/ \* ~" }2 _/ f8 F9 _
- * _( q8 |6 z* z1 ]8 O
- : n/ ~- Z) b, a g& B2 o
- //初始化ADC/ t7 _1 z! E+ m# j. D) Z
- void MY_ADC_Init(void): t3 N) [% G; S
- {
# n0 K9 g1 b _3 \1 R - ADC1_Handler.Instance = ADC1;; L: R7 E I5 q! P# h6 M
- ADC1_Handler.Init.ClockPrescaler = ADC_CLOCK_SYNC_PCLK_DIV4; //4分频,ADCCLK=PCLK2/4=108/4=27MHZ
6 x+ ~$ T% t- p M% [ - ADC1_Handler.Init.Resolution = ADC_RESOLUTION_12B; //12位模式; }: _, G6 A* A! i4 I2 f; F
- ADC1_Handler.Init.ScanConvMode = DISABLE; //非扫描模式
# ~( ~; C* }+ L. r @$ z/ _ - ADC1_Handler.Init.ContinuousConvMode = DISABLE; //关闭连续转换' g4 n( V C5 d4 Q# J* Z# d; A; F
- ADC1_Handler.Init.DiscontinuousConvMode = DISABLE; //禁止不连续采样模式- o* q& c2 B% A9 S0 g1 D% ^
- ADC1_Handler.Init.ExternalTrigConvEdge = ADC_EXTERNALTRIGCONVEDGE_NONE; //使用软件触发
% l, U R& C1 i2 O% W - ADC1_Handler.Init.ExternalTrigConv = ADC_SOFTWARE_START; //软件触发# J8 n, R* R4 t% Y) ~
- ADC1_Handler.Init.DataAlign = ADC_DATAALIGN_RIGHT; //右对齐$ C7 R- V$ B* w% P3 K; v
- ADC1_Handler.Init.NbrOfConversion = 1; //1个转换在规则序列中 也就是只转换规则序列16 u2 i9 W, J/ J. p4 B8 B# D2 h
- ADC1_Handler.Init.NbrOfDiscConversion = 0; //不连续采样通道数为0* u6 `, |7 H2 O- G. u3 ^+ @( f: w% x
- ADC1_Handler.Init.DMAContinuousRequests = DISABLE; //关闭DMA请求
- t* F7 {! `8 u @0 L: D, Y# W - ADC1_Handler.Init.EOCSelection = DISABLE; //关闭EOC中断& k! c2 D0 Z# K4 k
- HAL_ADC_Init(&ADC1_Handler); //初始化; \) s$ }6 y& V6 H {1 X6 A H
- V2 H5 L6 c$ T1 M" t: s- ADC1_ChanConf.Channel = ADC_CHANNEL_5; //通道5 PA5: R6 w9 o$ N/ B
- ADC1_ChanConf.Rank = 1; //1个序列
; I7 Y: Z. e% h3 u% E5 T" r - ADC1_ChanConf.SamplingTime = ADC_SAMPLETIME_480CYCLES; //采样时间
* U! a3 j+ T; L4 }" {# q! F - ADC1_ChanConf.Offset = 0;& r* l7 u+ m1 I7 K% V- @
- HAL_ADC_ConfigChannel(&ADC1_Handler, &ADC1_ChanConf); //通道配置
o6 _+ w3 f0 H/ i" v1 v- J
$ i, z- U# p2 q+ \$ t- }2 B5 @& p9 S: t }/ l# A
/ a, V' x) q4 ?1 b4 E- //ADC底层驱动,引脚配置,时钟使能
0 B$ x+ X' B6 x2 W, H - //此函数会被HAL_ADC_Init()调用
8 k4 v+ C9 o( k) x# ^3 R0 N& i - //hadc:ADC句柄
% B% x' j5 p+ e& o0 T# i - void HAL_ADC_MspInit(ADC_HandleTypeDef *hadc)
+ W' Q) M1 T& z - {
7 @2 Q" b0 ?, o6 [- _# J - GPIO_InitTypeDef GPIO_Initure;) A" Z4 i4 \$ g9 R4 U) v/ U% E- |2 d
- __HAL_RCC_ADC1_CLK_ENABLE(); //使能ADC1时钟, o/ N* P7 K6 y
- __HAL_RCC_GPIOA_CLK_ENABLE(); //开启GPIOA时钟2 L$ I* o( \" Y) w# m& r. ?
- ' D7 f- K3 U0 t/ J: o! |* j
- GPIO_Initure.Pin = GPIO_PIN_5; //PA54 i5 v2 {7 K+ d$ J
- GPIO_Initure.Mode = GPIO_MODE_ANALOG; //模拟
2 Z n" w4 j; R; ?1 X - GPIO_Initure.Pull = GPIO_NOPULL; //不带上下拉
& ?4 P5 C! {* D" T8 O1 r, l - HAL_GPIO_Init(GPIOA, &GPIO_Initure);
0 J6 q! h: [$ ?) Y1 [ [ - }) W, a4 z0 u7 ]7 o, v3 T# |, _
" S0 Q) j0 v' H; L- h' c5 o- //获得ADC值/ d& J. }* v# N, I0 r
- //ch: 通道值 0~16,取值范围为:ADC_CHANNEL_0~ADC_CHANNEL_16( n* J/ y, d. H: x8 l# E
- //返回值:转换结果
# [/ I& i) ~7 r. W - u16 Get_Adc()1 V) [; i4 y/ u0 H
- {
5 X3 t! m6 r3 U( ~: F* s' H" W - HAL_ADC_Start(&ADC1_Handler); //开启ADC. U/ r9 \" [2 E! y/ G4 h# F
- HAL_ADC_PollForConversion(&ADC1_Handler, 10); //轮询转换& b' K8 @3 w: j5 n/ S* E* ]
- return (u16)HAL_ADC_GetValue(&ADC1_Handler); //返回最近一次ADC1规则组的转换结果
# }0 v: c0 v9 `! o* y - }
, }$ t6 Y' G( Q. w
复制代码 5 S1 [( ]8 Q# y- }
main.c中,主函数初始化之后,直接读取adc的值,并打印。
- P+ e, D0 G9 O* c, p- #include "sys.h"9 h' Q$ ~% Z- y" q' ]" W
- #include "delay.h"4 z2 n/ E9 ]$ @: N) z( n# L: ^& j
- #include "usart.h"
" I7 g* r: r+ w; B - #include "adc.h"# J8 ^# p. S. m% }* A0 N
& Y0 O+ A" P5 Y; T+ Q- int main(void)+ ^4 ~! r' ]' W" j3 ^% o: Z
- {
t& c5 _9 `7 E$ o2 e9 m - u16 adcx;) G; Z# X: s3 T
- Cache_Enable(); //打开L1-Cache+ }5 y+ |+ \0 r% e4 _
- HAL_Init(); //初始化HAL库
0 _. D* B. m+ N" `4 c( T; m - Stm32_Clock_Init(432, 8, 2, 9); //设置时钟,216Mhz
- ^2 |1 M" N8 G+ D5 l" r) X - delay_init(216); //延时初始化* n$ O8 W' M/ n1 m" g( V5 n
- uart_init(115200); //串口初始化
2 n) \4 ^0 L3 h% j - MY_ADC_Init(); //初始化ADC1通道5
# t- {% W# s' P n, R+ k# Y1 n -
! i6 _( G$ A X G - while (1)
2 h. d' o7 v# [! X9 W) @( s - {4 s4 c( J2 k% K; W" S/ ?# L
- adcx = Get_Adc(); //获取通道5的转换值,20次取平均
- m, @& O/ ]1 \5 z% ] f% \ - printf("%d\r\n", adcx);$ {0 w7 o& B1 G/ M" n0 K( m) o
- delay_ms(250);
4 h, b- @" w! [; m. J( l - }
0 T3 r. E7 |% H5 w H - }3 y0 P4 ]' v, Z
复制代码 ) M% q# x2 W# \3 T4 M& p8 F0 e4 V
如果需要对多个通道进行轮询读取,可以稍微修改一下函数Get_Adc,在每次读取之前设置一下需要转换的通道:
0 U( n+ n g2 A5 r3 S- //获得ADC值' O- v. ~1 ^( V+ M5 O2 U2 l% A
- //ch: 通道值 0~16,取值范围为:ADC_CHANNEL_0~ADC_CHANNEL_16
* |. ? {3 Q3 x6 S4 E - //返回值:转换结果
0 Y& z% E& r) j2 ~6 U/ B" I1 ~ - u16 Get_Adc(u32 ch) 7 y+ C" }7 ?4 y, g" z6 ~$ D# G8 @+ u
- {, W2 z6 y& u" u- a9 r% Q( S z" n
- ADC_ChannelConfTypeDef ADC1_ChanConf;
) m+ Q; l g* c; E& v( `2 B3 X -
, U: _2 A& S( U- E7 j+ F" q - ADC1_ChanConf.Channel=ch; //通道" y" R# U/ W; b, p
- ADC1_ChanConf.Rank=1; //1个序列" S' S' S1 [* [# H. f* \
- ADC1_ChanConf.SamplingTime=ADC_SAMPLETIME_480CYCLES; //采样时间
" S5 I! v/ L- f1 O4 V - ADC1_ChanConf.Offset=0;
$ J9 E" \9 o" Z F ~ - HAL_ADC_ConfigChannel(&ADC1_Handler,&ADC1_ChanConf); //通道配置2 v, N* Q* W) t+ m( F
- 1 B- G+ y* O" }( C8 }4 \
- HAL_ADC_Start(&ADC1_Handler); //开启ADC
& a0 s; P$ h; J - 0 S; H$ _' E/ }, S: ~9 i( y
- HAL_ADC_PollForConversion(&ADC1_Handler,10); //轮询转换
5 i2 B+ ~/ B7 q" t/ r -
7 G. p% _, j* w - return (u16)HAL_ADC_GetValue(&ADC1_Handler); //返回最近一次ADC1规则组的转换结果( R" S; h d( M6 B
- }
复制代码 ————————————————
, \* ^* R& Q$ S O5 T m% r! V& w. ?版权声明:小盼你最萌哒8 a8 a6 E" I# g6 D7 D& w, S" H
如有侵权请联系删除
' S2 G7 Y% `9 C% z+ j
: p: g% z+ o; \$ K2 O. f9 ~/ Z! q) j0 u B# i7 h
|