基于HAL库,实现最简单ADC采集。
3 h U5 B# x+ O4 N第一步,使用CubeMX配置时钟。9 v7 @4 J3 y0 o+ r, C$ q
/ F* C# s0 [" Q9 p
% K. O2 T" g8 _$ |$ t4 e0 D
' p3 \! `+ n4 H$ M然后是adc的配置:* \ }; T! Z, E
% }/ f% M4 n& O/ R1 q
9 x9 U4 e$ A& @! a. T- M B; Y) W m1 \, z
将,稍微修改一下风格,并手写头文件和源文件。* }2 J* m9 q/ r: [
. Z' q- y$ A4 \4 y8 ?" B, I# k$ r Tadc.h很简单,就不说了。# x7 e8 @, u! F1 @$ T2 t9 J! z3 _ d
- #ifndef __ADC_H
9 ^6 T9 t$ R- Q5 q% ] - #define __ADC_H: l* ~- ?5 r& K% I' Y) o
- #include "sys.h"5 Z/ Z# C% D8 x! S
7 q. M4 S' I! w A; H- ]$ P- void MY_ADC_Init(void); //ADC通道初始化8 C2 P- X- s) g
- u16 Get_Adc(void); //获得某个通道值 4 z) \6 h! t! q5 l3 v# m
- #endif
) Z5 v6 y& A6 n- Y
复制代码 % T2 C+ A0 F0 W5 l
其中,MY_ADC_Init和HAL_ADC_MspInit的内容完全是根据CubeMX生成的代码写的,就改了一个变量名字。Get_Adc则简单了,当我们需要获取adc的值的时候,先HAL_ADC_Start启动ADC,然后HAL_ADC_PollForConversion等待转换完成,最后HAL_ADC_GetValue返回结果。+ R8 F1 F6 [$ Q+ d, _
- #include "adc.h"
$ x# }/ d* Q) v& [/ {6 z- _) I8 T+ V - #include "delay.h"
. _% R* Y6 A- i# p- @8 r- Q+ A - * I! n& L3 o+ u: c! Z6 i
- 8 g. m$ s) {6 Z% P# X4 @
- ADC_HandleTypeDef ADC1_Handler; //ADC句柄
+ B% |: v2 x( ~* E0 a - ADC_ChannelConfTypeDef ADC1_ChanConf; //ADC通道配置句柄
2 g5 t5 S7 N, C2 s- n
7 q, i3 J& x: _, T0 S& b- : s4 J, J2 H6 W1 m# `, R
- //初始化ADC
; a3 L! j2 r+ W) f* S" I) E - void MY_ADC_Init(void)
, a: ?1 Q1 J8 ]2 G% ], P# W4 W& X - {: y- S/ M6 k. b' p
- ADC1_Handler.Instance = ADC1;
! R" L Q. e: q( M2 Y, L- I - ADC1_Handler.Init.ClockPrescaler = ADC_CLOCK_SYNC_PCLK_DIV4; //4分频,ADCCLK=PCLK2/4=108/4=27MHZ; A7 c" o( |, t) Q8 k6 B
- ADC1_Handler.Init.Resolution = ADC_RESOLUTION_12B; //12位模式' v" c/ m- ?, r1 r6 I9 V
- ADC1_Handler.Init.ScanConvMode = DISABLE; //非扫描模式' k7 ?4 k3 ]$ ^3 H' s
- ADC1_Handler.Init.ContinuousConvMode = DISABLE; //关闭连续转换9 y! ]* q4 U2 a- E* o! \
- ADC1_Handler.Init.DiscontinuousConvMode = DISABLE; //禁止不连续采样模式! A, m, I2 B6 m0 p
- ADC1_Handler.Init.ExternalTrigConvEdge = ADC_EXTERNALTRIGCONVEDGE_NONE; //使用软件触发
: w% J/ a2 q, U - ADC1_Handler.Init.ExternalTrigConv = ADC_SOFTWARE_START; //软件触发
3 j% s1 u! v' g7 u - ADC1_Handler.Init.DataAlign = ADC_DATAALIGN_RIGHT; //右对齐
1 d6 K3 P9 Z9 C+ Q - ADC1_Handler.Init.NbrOfConversion = 1; //1个转换在规则序列中 也就是只转换规则序列1
; i# v2 [; `( F; _/ X1 R+ m - ADC1_Handler.Init.NbrOfDiscConversion = 0; //不连续采样通道数为06 D8 J( B3 r9 S: w. @% z
- ADC1_Handler.Init.DMAContinuousRequests = DISABLE; //关闭DMA请求
. c. h8 s- ^4 @. D! Y1 ` - ADC1_Handler.Init.EOCSelection = DISABLE; //关闭EOC中断1 E6 p5 ~( b/ Q* @0 X2 ~
- HAL_ADC_Init(&ADC1_Handler); //初始化) x2 ^( c7 N! p8 [1 m$ l1 C& H
- ( |; n8 r# C; e2 w1 d$ n2 g
- ADC1_ChanConf.Channel = ADC_CHANNEL_5; //通道5 PA5
5 y# w( c% z4 E% O3 t) W - ADC1_ChanConf.Rank = 1; //1个序列" L7 X& N; {0 ^, T) s T: {, [0 P1 \
- ADC1_ChanConf.SamplingTime = ADC_SAMPLETIME_480CYCLES; //采样时间
) C; {7 @6 l: b; \ - ADC1_ChanConf.Offset = 0;
' k5 _( n% |; T - HAL_ADC_ConfigChannel(&ADC1_Handler, &ADC1_ChanConf); //通道配置" X1 [. c5 h4 [# {; U
0 F2 f/ J, d, n+ }' l U- }
8 x1 b G% ~) i( {5 y# }# Q! T - 3 e% l' ]; D* J6 y0 p3 f% b
- //ADC底层驱动,引脚配置,时钟使能
' x: a1 V$ O$ f( I8 Z6 L3 E - //此函数会被HAL_ADC_Init()调用: m% w ^& b0 k& @- m
- //hadc:ADC句柄
; @# P7 a% C# }9 K( [3 w - void HAL_ADC_MspInit(ADC_HandleTypeDef *hadc)
. C+ {' i8 ^8 P7 I/ {; t - {
3 ~7 I2 F1 Y- }) Y - GPIO_InitTypeDef GPIO_Initure;" t8 M* v0 M* ?" h2 Z6 \3 S( M
- __HAL_RCC_ADC1_CLK_ENABLE(); //使能ADC1时钟- B, a7 l0 O, h/ P M
- __HAL_RCC_GPIOA_CLK_ENABLE(); //开启GPIOA时钟- o. }, _4 x# c- v/ E* U$ @* s
- 9 y" Z/ ]- N4 }$ c' [8 H
- GPIO_Initure.Pin = GPIO_PIN_5; //PA5+ d% _* C/ J* ?+ C' J& o
- GPIO_Initure.Mode = GPIO_MODE_ANALOG; //模拟) t" H' m2 K' f# g
- GPIO_Initure.Pull = GPIO_NOPULL; //不带上下拉* f' k% C# s9 P; x( t& h W- x
- HAL_GPIO_Init(GPIOA, &GPIO_Initure);' Z) ]5 Z& W# Z
- }
) Z9 r y$ v( m3 C
/ m. V9 M; `5 D3 z% P4 O2 x4 P1 k9 F- //获得ADC值0 D" Y2 a3 }* M3 S4 N! y
- //ch: 通道值 0~16,取值范围为:ADC_CHANNEL_0~ADC_CHANNEL_16" b; Y6 I0 a7 Q0 ?/ K
- //返回值:转换结果( ?! F. y5 Z( x# S* a
- u16 Get_Adc()
: ^& E/ ?0 S+ Y - {
- `5 {6 S; L0 R) R& |1 \3 | - HAL_ADC_Start(&ADC1_Handler); //开启ADC
+ \3 K& a0 P" h0 F0 V - HAL_ADC_PollForConversion(&ADC1_Handler, 10); //轮询转换7 s5 x6 W* q+ A, Z
- return (u16)HAL_ADC_GetValue(&ADC1_Handler); //返回最近一次ADC1规则组的转换结果
# `6 R. C$ h% p; ?1 |% ]- W - }
# S% o3 b! c: P* V( q8 z: ~- x f4 C
复制代码
% ^& ^" _% p& f0 D* |# bmain.c中,主函数初始化之后,直接读取adc的值,并打印。1 o6 y4 t$ s" F# a) x# w
- #include "sys.h"
5 I5 g1 }) t- F7 V9 g( @ - #include "delay.h"5 j3 |9 q0 f' B5 [8 r+ C+ _
- #include "usart.h"* j* c k4 [1 w7 ~: x7 [
- #include "adc.h": W& K1 U( Z- d$ B! F
& F! y8 E a, |4 f( t- int main(void)1 _2 F3 A/ D6 q4 p+ F. J
- {
1 w& c. M6 y3 U' G& c% m$ u - u16 adcx;% L' @! D' G2 \- B+ w4 M
- Cache_Enable(); //打开L1-Cache
* z" D/ w0 z1 a& W& @: p+ F J* R - HAL_Init(); //初始化HAL库
) a1 B7 x' N/ q0 B - Stm32_Clock_Init(432, 8, 2, 9); //设置时钟,216Mhz( q6 i1 l7 c# N2 n5 y
- delay_init(216); //延时初始化; A- v6 X! N( _7 z3 q( W+ U
- uart_init(115200); //串口初始化5 Y* Q; q/ V, J- o8 ~' n \8 c
- MY_ADC_Init(); //初始化ADC1通道5
% F7 l. x1 |1 | -
9 I7 {5 s- e7 X. k" m# V( s; \' i2 q - while (1)
9 Z r# r5 d# E/ z: a$ J7 a - {
. p M- W! ~4 f7 n - adcx = Get_Adc(); //获取通道5的转换值,20次取平均
( B! w {1 e3 C& { - printf("%d\r\n", adcx);
: O" ^7 J* m6 p3 ?% ~" Q6 k - delay_ms(250);
9 | O3 g5 r1 r+ ?4 J/ I$ | - }
' Y& M5 ]: c* T2 [2 J - }0 W& X/ _* j+ B
复制代码
3 ]- ^ B% |. T1 w如果需要对多个通道进行轮询读取,可以稍微修改一下函数Get_Adc,在每次读取之前设置一下需要转换的通道:
) I4 V5 k( g1 K# D$ f- x, W- //获得ADC值
# N3 `/ v1 m1 g/ M4 e4 w9 v. Y O - //ch: 通道值 0~16,取值范围为:ADC_CHANNEL_0~ADC_CHANNEL_161 x8 o, J$ |% g8 S
- //返回值:转换结果
1 z4 k/ J. o/ g9 N - u16 Get_Adc(u32 ch)
1 O" w$ J) r. y% q5 L - {, B6 ^2 P$ e+ _& q% M
- ADC_ChannelConfTypeDef ADC1_ChanConf;4 C/ @" O$ H$ r5 u( p+ P* F" }' }: r
-
3 m( n. K2 J% p$ Z( P8 i - ADC1_ChanConf.Channel=ch; //通道
+ v# M& P1 a; g" D) v+ m - ADC1_ChanConf.Rank=1; //1个序列
: n- v2 r _% g4 Z/ n - ADC1_ChanConf.SamplingTime=ADC_SAMPLETIME_480CYCLES; //采样时间/ `1 b9 K- i {; w( p$ |4 V0 I3 U
- ADC1_ChanConf.Offset=0;
* M5 Q( i7 A) s/ Y7 a& O - HAL_ADC_ConfigChannel(&ADC1_Handler,&ADC1_ChanConf); //通道配置/ q; r+ E% a. `0 x" L1 s( T
-
6 l( R5 T* ^, t) z4 i - HAL_ADC_Start(&ADC1_Handler); //开启ADC: E1 O$ ]5 G9 f) B* x9 G3 V9 b
- * ?, V) i, @) O- y1 @" i
- HAL_ADC_PollForConversion(&ADC1_Handler,10); //轮询转换
6 A9 H; X0 p- `7 q( h3 z. ?& k -
3 p" Y. ]! N/ n - return (u16)HAL_ADC_GetValue(&ADC1_Handler); //返回最近一次ADC1规则组的转换结果
7 Z' ]4 d, f. {6 m" L8 {# H - }
复制代码 ————————————————
9 j, V% K' ]6 a$ N版权声明:小盼你最萌哒
( \* S: i; D+ h- U如有侵权请联系删除
. `0 ]: F3 V8 \7 \+ ?- p7 B( F* U
# Y1 i' k6 C) h4 k& ^' u
# a) z" w; A% w f7 i* O/ S |